 Chromium Code Reviews
 Chromium Code Reviews Issue 11453032:
  Reapply class/method/field minification  (Closed) 
  Base URL: https://dart.googlecode.com/svn/branches/bleeding_edge/dart
    
  
    Issue 11453032:
  Reapply class/method/field minification  (Closed) 
  Base URL: https://dart.googlecode.com/svn/branches/bleeding_edge/dart| Index: sdk/lib/_internal/compiler/implementation/js_backend/namer.dart | 
| diff --git a/sdk/lib/_internal/compiler/implementation/js_backend/namer.dart b/sdk/lib/_internal/compiler/implementation/js_backend/namer.dart | 
| index c4888b24b616c74b3b8c6622b1573bead56fc366..bf28c90e488770a7df38233817f8e5ec71969e11 100644 | 
| --- a/sdk/lib/_internal/compiler/implementation/js_backend/namer.dart | 
| +++ b/sdk/lib/_internal/compiler/implementation/js_backend/namer.dart | 
| @@ -8,8 +8,6 @@ part of js_backend; | 
| * Assigns JavaScript identifiers to Dart variables, class-names and members. | 
| */ | 
| class Namer { | 
| - final Compiler compiler; | 
| - | 
| static Set<String> _jsReserved = null; | 
| Set<String> get jsReserved { | 
| if (_jsReserved == null) { | 
| @@ -20,15 +18,22 @@ class Namer { | 
| return _jsReserved; | 
| } | 
| + final String CURRENT_ISOLATE = r'$'; | 
| + | 
| /** | 
| * Map from top-level or static elements to their unique identifiers provided | 
| * by [getName]. | 
| * | 
| * Invariant: Keys must be declaration elements. | 
| */ | 
| + final Compiler compiler; | 
| final Map<Element, String> globals; | 
| - final Map<String, int> usedGlobals; | 
| final Map<String, LibraryElement> shortPrivateNameOwners; | 
| + final Set<String> usedGlobalNames; | 
| + final Set<String> usedInstanceNames; | 
| + final Map<String, String> instanceNameMap; | 
| + final Map<String, String> globalNameMap; | 
| + final Map<String, int> popularNameCounters; | 
| /** | 
| * A cache of names used for bailout methods. We make sure two | 
| @@ -45,22 +50,27 @@ class Namer { | 
| Namer(this.compiler) | 
| : globals = new Map<Element, String>(), | 
| - usedGlobals = new Map<String, int>(), | 
| shortPrivateNameOwners = new Map<String, LibraryElement>(), | 
| bailoutNames = new Map<Element, String>(), | 
| usedBailoutInstanceNames = new Set<String>(), | 
| - constantNames = new Map<Constant, String>(); | 
| - | 
| - final String CURRENT_ISOLATE = r'$'; | 
| - final String ISOLATE = 'Isolate'; | 
| - final String ISOLATE_PROPERTIES = r"$isolateProperties"; | 
| + usedGlobalNames = new Set<String>(), | 
| + usedInstanceNames = new Set<String>(), | 
| + instanceNameMap = new Map<String, String>(), | 
| + globalNameMap = new Map<String, String>(), | 
| + constantNames = new Map<Constant, String>(), | 
| + popularNameCounters = new Map<String, int>(); | 
| + | 
| + String get isolateName => 'Isolate'; | 
| + String get isolatePropertiesName => r'$isolateProperties'; | 
| /** | 
| * Some closures must contain their name. The name is stored in | 
| * [STATIC_CLOSURE_NAME_NAME]. | 
| */ | 
| - final String STATIC_CLOSURE_NAME_NAME = r'$name'; | 
| - static const SourceString CLOSURE_INVOCATION_NAME = | 
| - Compiler.CALL_OPERATOR_NAME; | 
| + String get STATIC_CLOSURE_NAME_NAME => r'$name'; | 
| + SourceString get closureInvocationSelector => Compiler.CALL_OPERATOR_NAME; | 
| 
ngeoffray
2012/12/10 08:55:42
We have the Selector class in the compiler, so I'd
 
erikcorry
2012/12/10 12:37:40
We already have something else called closureInvoc
 | 
| + bool get shouldMinify => false; | 
| + | 
| + bool isReserved(String name) => name == isolateName; | 
| String constantName(Constant constant) { | 
| // In the current implementation it doesn't make sense to give names to | 
| @@ -69,16 +79,29 @@ class Namer { | 
| assert(!constant.isFunction()); | 
| String result = constantNames[constant]; | 
| if (result == null) { | 
| - result = getFreshGlobalName("CTC"); | 
| + String longName; | 
| + if (shouldMinify) { | 
| + if (constant.isString()) { | 
| + StringConstant stringConstant = constant; | 
| + // The minifier always constructs a new name, using the argument as | 
| + // input to its hashing algorithm. The given name does not need to be | 
| + // valid. | 
| + longName = stringConstant.value.slowToString(); | 
| + } else { | 
| + longName = "C"; | 
| + } | 
| + } else { | 
| + longName = "CONSTANT"; | 
| + } | 
| + result = getFreshName(longName, usedGlobalNames); | 
| constantNames[constant] = result; | 
| } | 
| return result; | 
| } | 
| String closureInvocationName(Selector selector) { | 
| - // TODO(floitsch): mangle, while not conflicting with instance names. | 
| - return instanceMethodInvocationName(null, CLOSURE_INVOCATION_NAME, | 
| - selector); | 
| + return | 
| + instanceMethodInvocationName(null, closureInvocationSelector, selector); | 
| 
ngeoffray
2012/12/10 08:55:42
The style we usually use is arguments on the new l
 
erikcorry
2012/12/10 12:37:40
Done.
 | 
| } | 
| String breakLabelName(LabelElement label) { | 
| @@ -104,25 +127,32 @@ class Namer { | 
| * mangles the [name] so that each library has a unique name. | 
| */ | 
| String privateName(LibraryElement lib, SourceString name) { | 
| + String result; | 
| if (name.isPrivate()) { | 
| String nameString = name.slowToString(); | 
| // The first library asking for a short private name wins. | 
| - LibraryElement owner = | 
| - shortPrivateNameOwners.putIfAbsent(nameString, () => lib); | 
| + LibraryElement owner = shouldMinify ? | 
| + shortPrivateNameOwners.putIfAbsent(nameString, () => lib) : | 
| + lib; | 
| // If a private name could clash with a mangled private name we don't | 
| // use the short name. For example a private name "_lib3_foo" would | 
| // clash with "_foo" from "lib3". | 
| - if (identical(owner, lib) && !nameString.startsWith('_$LIBRARY_PREFIX')) { | 
| - return nameString; | 
| + if (owner == lib && | 
| + !nameString.startsWith('_$LIBRARY_PREFIX') && | 
| + !shouldMinify) { | 
| + result = nameString; | 
| + } else { | 
| + String libName = getName(lib); | 
| + // If a library name does not start with the [LIBRARY_PREFIX] then our | 
| + // assumptions about clashing with mangled private members do not hold. | 
| + assert(shouldMinify || libName.startsWith(LIBRARY_PREFIX)); | 
| + // TODO(erikcorry): Fix this with other manglings to avoid clashes. | 
| + result = '_lib$libName\$$nameString'; | 
| } | 
| - String libName = getName(lib); | 
| - // If a library name does not start with the [LIBRARY_PREFIX] then our | 
| - // assumptions about clashing with mangled private members do not hold. | 
| - assert(libName.startsWith(LIBRARY_PREFIX)); | 
| - return '_$libName$nameString'; | 
| } else { | 
| - return name.slowToString(); | 
| + result = name.slowToString(); | 
| } | 
| + return result; | 
| } | 
| String instanceMethodName(FunctionElement element) { | 
| @@ -135,21 +165,27 @@ class Namer { | 
| FunctionSignature signature = element.computeSignature(compiler); | 
| String methodName = | 
| '${privateName(lib, name)}\$${signature.parameterCount}'; | 
| - if (!signature.optionalParametersAreNamed) { | 
| - return methodName; | 
| - } else if (!signature.optionalParameters.isEmpty) { | 
| + if (signature.optionalParametersAreNamed && | 
| + !signature.optionalParameters.isEmpty) { | 
| StringBuffer buffer = new StringBuffer(); | 
| signature.orderedOptionalParameters.forEach((Element element) { | 
| buffer.add('\$${JsNames.getValid(element.name.slowToString())}'); | 
| }); | 
| - return '$methodName$buffer'; | 
| + methodName = '$methodName$buffer'; | 
| } | 
| + if (name == closureInvocationSelector) return methodName; | 
| + return getMappedInstanceName(methodName); | 
| } | 
| String publicInstanceMethodNameByArity(SourceString name, int arity) { | 
| name = Elements.operatorNameToIdentifier(name); | 
| assert(!name.isPrivate()); | 
| - return '${name.slowToString()}\$$arity'; | 
| + var base = name.slowToString(); | 
| + // We don't mangle the closure invoking function name because it is | 
| + // generated in applyFunction. | 
| + var proposedName = '$base\$$arity'; | 
| + if (base == closureInvocationSelector) return proposedName; | 
| + return getMappedInstanceName(proposedName); | 
| } | 
| String instanceMethodInvocationName(LibraryElement lib, SourceString name, | 
| @@ -162,59 +198,108 @@ class Namer { | 
| buffer.add(r'$'); | 
| argumentName.printOn(buffer); | 
| } | 
| - return '${privateName(lib, name)}\$${selector.argumentCount}$buffer'; | 
| + if (name == closureInvocationSelector) { | 
| + // We don't mangle the closure invoking function name because it is | 
| + // generated in applyFunction. | 
| 
ngeoffray
2012/12/10 08:55:42
generated or used? What is applyFunction? Dart's F
 
erikcorry
2012/12/10 12:37:40
The name is generated (wiith string addition), the
 | 
| + return '$closureInvocationSelector\$${selector.argumentCount}$buffer'; | 
| + } | 
| + return getMappedInstanceName( | 
| + '${privateName(lib, name)}\$${selector.argumentCount}$buffer'); | 
| } | 
| String instanceFieldName(LibraryElement libraryElement, SourceString name) { | 
| String proposedName = privateName(libraryElement, name); | 
| - return safeName(proposedName); | 
| + return getMappedInstanceName(proposedName); | 
| } | 
| + // Construct a new name for the element based on the library and class it is | 
| + // in. The name here is not important, we just need to make sure it is | 
| + // unique. If we are minifying, we actually construct the name from the | 
| + // minified versions of the class and instance names, but the result is | 
| + // minified once again, so that is not visible in the end result. | 
| String shadowedFieldName(Element fieldElement) { | 
| + // Check for following situation: Native field ${fieldElement.name} has | 
| + // fixed JSName ${fieldElement.nativeName()}, but a subclass shadows this | 
| + // name. We normally handle that by renaming the superclass field, but we | 
| + // can't do that because native fields have fixed JSNames. In practice | 
| + // this can't happen because we can't inherit from native classes. | 
| + assert (!fieldElement.isNative()); | 
| + | 
| ClassElement cls = fieldElement.getEnclosingClass(); | 
| LibraryElement libraryElement = fieldElement.getLibrary(); | 
| String libName = getName(libraryElement); | 
| String clsName = getName(cls); | 
| String instanceName = instanceFieldName(libraryElement, fieldElement.name); | 
| - return safeName('$libName\$$clsName\$$instanceName'); | 
| + return getMappedInstanceName('$libName\$$clsName\$$instanceName'); | 
| } | 
| String setterName(LibraryElement lib, SourceString name) { | 
| // We dynamically create setters from the field-name. The setter name must | 
| // therefore be derived from the instance field-name. | 
| - String fieldName = safeName(privateName(lib, name)); | 
| + String fieldName = getMappedInstanceName(privateName(lib, name)); | 
| return 'set\$$fieldName'; | 
| } | 
| + String setterNameFromAccessorName(String name) { | 
| + // We dynamically create setters from the field-name. The setter name must | 
| + // therefore be derived from the instance field-name. | 
| + return 'set\$$name'; | 
| + } | 
| + | 
| String publicGetterName(SourceString name) { | 
| // We dynamically create getters from the field-name. The getter name must | 
| // therefore be derived from the instance field-name. | 
| - String fieldName = safeName(name.slowToString()); | 
| + String fieldName = getMappedInstanceName(name.slowToString()); | 
| return 'get\$$fieldName'; | 
| } | 
| + String getterNameFromAccessorName(String name) { | 
| + // We dynamically create getters from the field-name. The getter name must | 
| + // therefore be derived from the instance field-name. | 
| + return 'get\$$name'; | 
| + } | 
| + | 
| String getterName(LibraryElement lib, SourceString name) { | 
| // We dynamically create getters from the field-name. The getter name must | 
| // therefore be derived from the instance field-name. | 
| - String fieldName = safeName(privateName(lib, name)); | 
| + String fieldName = getMappedInstanceName(privateName(lib, name)); | 
| return 'get\$$fieldName'; | 
| } | 
| - String getFreshGlobalName(String proposedName) { | 
| - String name = proposedName; | 
| - int count = usedGlobals[name]; | 
| - if (count != null) { | 
| - // Not the first time we see this name. Append a number to make it unique. | 
| - do { | 
| - name = '$proposedName${count++}'; | 
| - } while (usedGlobals[name] != null); | 
| - // Record the count in case we see this name later. We | 
| - // frequently see names multiple times, as all our closures use | 
| - // the same name for their class. | 
| - usedGlobals[proposedName] = count; | 
| + String getMappedGlobalName(String proposedName) { | 
| + var newName = globalNameMap[proposedName]; | 
| + if (newName == null) { | 
| + newName = getFreshName(proposedName, usedGlobalNames); | 
| + globalNameMap[proposedName] = newName; | 
| } | 
| - usedGlobals[name] = 0; | 
| - return name; | 
| + return newName; | 
| + } | 
| + | 
| + String getMappedInstanceName(String proposedName) { | 
| + var newName = instanceNameMap[proposedName]; | 
| + if (newName == null) { | 
| + newName = getFreshName(proposedName, usedInstanceNames); | 
| + instanceNameMap[proposedName] = newName; | 
| + } | 
| + return newName; | 
| + } | 
| + | 
| + String getFreshName(String proposedName, Set<String> usedNames) { | 
| + var candidate; | 
| + proposedName = safeName(proposedName); | 
| + if (!usedNames.contains(proposedName)) { | 
| + candidate = proposedName; | 
| + } else { | 
| + var counter = popularNameCounters[proposedName]; | 
| + var i = counter == null ? 0 : counter; | 
| + while (usedNames.contains("$proposedName$i")) { | 
| + i++; | 
| + } | 
| + popularNameCounters[proposedName] = i + 1; | 
| + candidate = "$proposedName$i"; | 
| + } | 
| + usedNames.add(candidate); | 
| + return candidate; | 
| } | 
| static const String LIBRARY_PREFIX = "lib"; | 
| @@ -247,29 +332,38 @@ class Namer { | 
| } else { | 
| name = element.name.slowToString(); | 
| } | 
| - return safeName(name); | 
| + return name; | 
| } | 
| String getSpecializedName(Element element, Collection<ClassElement> classes) { | 
| + // This gets the minified name, but it doesn't really make much difference. | 
| + // the important thing is that it is a unique name. | 
| 
ngeoffray
2012/12/10 08:55:42
the -> The
 
erikcorry
2012/12/10 12:37:40
Done.
 | 
| StringBuffer buffer = new StringBuffer('${getName(element)}\$'); | 
| for (ClassElement cls in classes) { | 
| buffer.add(getName(cls)); | 
| } | 
| - return safeName(buffer.toString()); | 
| + return getMappedGlobalName(buffer.toString()); | 
| } | 
| String getBailoutName(Element element) { | 
| String name = bailoutNames[element]; | 
| if (name != null) return name; | 
| bool global = !element.isInstanceMember(); | 
| + // Despite the name of the variable, this gets the minified name when we | 
| + // are minifying, but it doesn't really make much difference. The | 
| + // important thing is that it is a unique name. We add $bailout and, if we | 
| + // are minifying, we minify the minified name and '$bailout'. | 
| String unminifiedName = '${getName(element)}\$bailout'; | 
| - name = unminifiedName; | 
| - if (!global) { | 
| + if (global) { | 
| + name = getMappedGlobalName(unminifiedName); | 
| + } else { | 
| + name = unminifiedName; | 
| int i = 0; | 
| while (usedBailoutInstanceNames.contains(name)) { | 
| name = '$unminifiedName${i++}'; | 
| } | 
| usedBailoutInstanceNames.add(name); | 
| + name = getMappedInstanceName(name); | 
| } | 
| bailoutNames[element] = name; | 
| return name; | 
| @@ -307,21 +401,29 @@ class Namer { | 
| String guess = _computeGuess(element); | 
| ElementKind kind = element.kind; | 
| - if (identical(kind, ElementKind.VARIABLE) || | 
| - identical(kind, ElementKind.PARAMETER)) { | 
| + if (kind == ElementKind.VARIABLE || | 
| + kind == ElementKind.PARAMETER) { | 
| // The name is not guaranteed to be unique. | 
| - return guess; | 
| + return safeName(guess); | 
| } | 
| - if (identical(kind, ElementKind.GENERATIVE_CONSTRUCTOR) || | 
| - identical(kind, ElementKind.FUNCTION) || | 
| - identical(kind, ElementKind.CLASS) || | 
| - identical(kind, ElementKind.FIELD) || | 
| - identical(kind, ElementKind.GETTER) || | 
| - identical(kind, ElementKind.SETTER) || | 
| - identical(kind, ElementKind.TYPEDEF) || | 
| - identical(kind, ElementKind.LIBRARY) || | 
| - identical(kind, ElementKind.MALFORMED_TYPE)) { | 
| - String result = getFreshGlobalName(guess); | 
| + if (kind == ElementKind.GENERATIVE_CONSTRUCTOR || | 
| + kind == ElementKind.FUNCTION || | 
| + kind == ElementKind.CLASS || | 
| + kind == ElementKind.FIELD || | 
| + kind == ElementKind.GETTER || | 
| + kind == ElementKind.SETTER || | 
| + kind == ElementKind.TYPEDEF || | 
| + kind == ElementKind.LIBRARY || | 
| + kind == ElementKind.MALFORMED_TYPE) { | 
| + bool isNative = false; | 
| + if (kind == ElementKind.CLASS) { | 
| + ClassElement class_elt = element; | 
| 
ngeoffray
2012/12/10 08:55:42
class_elt -> classElement
 
erikcorry
2012/12/10 12:37:40
Done.
 | 
| + isNative = class_elt.isNative(); | 
| + } | 
| + if (Elements.isInstanceField(element)) { | 
| + isNative = element.isNative(); | 
| + } | 
| + String result = isNative ? guess : getFreshName(guess, usedGlobalNames); | 
| globals[element] = result; | 
| return result; | 
| } | 
| @@ -331,13 +433,12 @@ class Namer { | 
| } | 
| String getLazyInitializerName(Element element) { | 
| - // TODO(floitsch): mangle while not conflicting with other statics. | 
| assert(Elements.isStaticOrTopLevelField(element)); | 
| - return "get\$${getName(element)}"; | 
| + return getMappedGlobalName("get\$${getName(element)}"); | 
| } | 
| String isolatePropertiesAccess(Element element) { | 
| - return "$ISOLATE.$ISOLATE_PROPERTIES.${getName(element)}"; | 
| + return "$isolateName.$isolatePropertiesName.${getName(element)}"; | 
| } | 
| String isolateAccess(Element element) { | 
| @@ -345,7 +446,8 @@ class Namer { | 
| } | 
| String isolateBailoutAccess(Element element) { | 
| - return '${isolateAccess(element)}\$bailout'; | 
| + String newName = getMappedGlobalName('${getName(element)}\$bailout'); | 
| + return '$CURRENT_ISOLATE.$newName'; | 
| } | 
| String isolateLazyInitializerAccess(Element element) { | 
| @@ -353,6 +455,7 @@ class Namer { | 
| } | 
| String operatorIs(Element element) { | 
| + // TODO(erikcorry): Reduce from is$x to ix when we are minifying. | 
| return 'is\$${getName(element)}'; | 
| } |