/* * Copyright (c) 2003, 2015, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it * under the terms of the GNU General Public License version 2 only, as * published by the Free Software Foundation. Oracle designates this * particular file as subject to the "Classpath" exception as provided * by Oracle in the LICENSE file that accompanied this code. * * This code is distributed in the hope that it will be useful, but WITHOUT * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License * version 2 for more details (a copy is included in the LICENSE file that * accompanied this code). * * You should have received a copy of the GNU General Public License version * 2 along with this work; if not, write to the Free Software Foundation, * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. * * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA * or visit www.oracle.com if you need additional information or have any * questions. */ package com.sun.java.util.jar.pack; import java.io.IOException; import java.io.InputStream; import java.io.PrintStream; import java.io.PrintWriter; import java.security.AccessController; import java.security.PrivilegedAction; import java.util.ArrayList; import java.util.Collection; import java.util.Comparator; import java.util.HashMap; import java.util.List; import java.util.Map; import java.util.Properties; import java.util.Set; import java.util.SortedMap; import java.util.TreeMap; import java.util.jar.Pack200; /** * Control block for publishing Pack200 options to the other classes. */ @SuppressWarnings({"removal"}) final class PropMap implements SortedMap { private final TreeMap theMap = new TreeMap<>();; // Override: public String put(String key, String value) { String oldValue = theMap.put(key, value); return oldValue; } // All this other stuff is private to the current package. // Outide clients of Pack200 do not need to use it; they can // get by with generic SortedMap functionality. private static Map defaultProps; static { Properties props = new Properties(); // Allow implementation selected via -Dpack.disable.native=true String propValue = getPropertyValue(Utils.DEBUG_DISABLE_NATIVE, "false"); props.put(Utils.DEBUG_DISABLE_NATIVE, String.valueOf(Boolean.parseBoolean(propValue))); // Set the DEBUG_VERBOSE from system int verbose = 0; try { verbose = Integer.decode(getPropertyValue(Utils.DEBUG_VERBOSE, "0")); } catch (NumberFormatException e) { } props.put(Utils.DEBUG_VERBOSE, String.valueOf(verbose)); // The segment size is unlimited props.put(Pack200.Packer.SEGMENT_LIMIT, "-1"); // Preserve file ordering by default. props.put(Pack200.Packer.KEEP_FILE_ORDER, Pack200.Packer.TRUE); // Preserve all modification times by default. props.put(Pack200.Packer.MODIFICATION_TIME, Pack200.Packer.KEEP); // Preserve deflation hints by default. props.put(Pack200.Packer.DEFLATE_HINT, Pack200.Packer.KEEP); // Pass through files with unrecognized attributes by default. props.put(Pack200.Packer.UNKNOWN_ATTRIBUTE, Pack200.Packer.PASS); // Pass through files with unrecognized format by default, also // allow system property to be set props.put(Utils.CLASS_FORMAT_ERROR, getPropertyValue(Utils.CLASS_FORMAT_ERROR, Pack200.Packer.PASS)); // Default effort is 5, midway between 1 and 9. props.put(Pack200.Packer.EFFORT, "5"); // Define certain attribute layouts by default. // Do this after the previous props are put in place, // to allow override if necessary. String propFile = "intrinsic.properties"; PrivilegedAction pa = () -> PackerImpl.class.getResourceAsStream(propFile); try (InputStream propStr = AccessController.doPrivileged(pa)) { if (propStr == null) { throw new RuntimeException(propFile + " cannot be loaded"); } props.load(propStr); } catch (IOException ee) { throw new RuntimeException(ee); } for (Map.Entry e : props.entrySet()) { String key = (String) e.getKey(); String val = (String) e.getValue(); if (key.startsWith("attribute.")) { e.setValue(Attribute.normalizeLayoutString(val)); } } @SuppressWarnings({"unchecked", "rawtypes"}) HashMap temp = new HashMap(props); // shrink to fit defaultProps = temp; } private static String getPropertyValue(String key, String defaultValue) { PrivilegedAction pa = () -> System.getProperty(key); String s = AccessController.doPrivileged(pa); return s != null ? s : defaultValue; } PropMap() { theMap.putAll(defaultProps); } // Return a view of this map which includes only properties // that begin with the given prefix. This is easy because // the map is sorted, and has a subMap accessor. SortedMap prefixMap(String prefix) { int len = prefix.length(); if (len == 0) return this; char nextch = (char)(prefix.charAt(len-1) + 1); String limit = prefix.substring(0, len-1)+nextch; //System.out.println(prefix+" => "+subMap(prefix, limit)); return subMap(prefix, limit); } String getProperty(String s) { return get(s); } String getProperty(String s, String defaultVal) { String val = getProperty(s); if (val == null) return defaultVal; return val; } String setProperty(String s, String val) { return put(s, val); } // Get sequence of props for "prefix", and "prefix.*". List getProperties(String prefix) { Collection values = prefixMap(prefix).values(); List res = new ArrayList<>(values.size()); res.addAll(values); while (res.remove(null)); return res; } private boolean toBoolean(String val) { return Boolean.valueOf(val).booleanValue(); } boolean getBoolean(String s) { return toBoolean(getProperty(s)); } boolean setBoolean(String s, boolean val) { return toBoolean(setProperty(s, String.valueOf(val))); } int toInteger(String val) { return toInteger(val, 0); } int toInteger(String val, int def) { if (val == null) return def; if (Pack200.Packer.TRUE.equals(val)) return 1; if (Pack200.Packer.FALSE.equals(val)) return 0; return Integer.parseInt(val); } int getInteger(String s, int def) { return toInteger(getProperty(s), def); } int getInteger(String s) { return toInteger(getProperty(s)); } int setInteger(String s, int val) { return toInteger(setProperty(s, String.valueOf(val))); } long toLong(String val) { try { return val == null ? 0 : Long.parseLong(val); } catch (java.lang.NumberFormatException nfe) { throw new IllegalArgumentException("Invalid value"); } } long getLong(String s) { return toLong(getProperty(s)); } long setLong(String s, long val) { return toLong(setProperty(s, String.valueOf(val))); } int getTime(String s) { String sval = getProperty(s, "0"); if (Utils.NOW.equals(sval)) { return (int)((System.currentTimeMillis()+500)/1000); } long lval = toLong(sval); final long recentSecondCount = 1000000000; if (lval < recentSecondCount*10 && !"0".equals(sval)) Utils.log.warning("Supplied modtime appears to be seconds rather than milliseconds: "+sval); return (int)((lval+500)/1000); } void list(PrintStream out) { PrintWriter outw = new PrintWriter(out); list(outw); outw.flush(); } void list(PrintWriter out) { out.println("#"+Utils.PACK_ZIP_ARCHIVE_MARKER_COMMENT+"["); Set> defaults = defaultProps.entrySet(); for (Map.Entry e : theMap.entrySet()) { if (defaults.contains(e)) continue; out.println(" " + e.getKey() + " = " + e.getValue()); } out.println("#]"); } @Override public int size() { return theMap.size(); } @Override public boolean isEmpty() { return theMap.isEmpty(); } @Override public boolean containsKey(Object key) { return theMap.containsKey(key); } @Override public boolean containsValue(Object value) { return theMap.containsValue(value); } @Override public String get(Object key) { return theMap.get(key); } @Override public String remove(Object key) { return theMap.remove(key); } @Override public void putAll(Map m) { theMap.putAll(m); } @Override public void clear() { theMap.clear(); } @Override public Set keySet() { return theMap.keySet(); } @Override public Collection values() { return theMap.values(); } @Override public Set> entrySet() { return theMap.entrySet(); } @Override public Comparator comparator() { return theMap.comparator(); } @Override public SortedMap subMap(String fromKey, String toKey) { return theMap.subMap(fromKey, toKey); } @Override public SortedMap headMap(String toKey) { return theMap.headMap(toKey); } @Override public SortedMap tailMap(String fromKey) { return theMap.tailMap(fromKey); } @Override public String firstKey() { return theMap.firstKey(); } @Override public String lastKey() { return theMap.lastKey(); } }