This patch adds a better code-splitting strategy for implementing
GWT.runAsync.  Instead of lumping all non-exclusive code atoms into
the initial download, the initial download only includes the initially
needed code.  Additionally, a custom download is created for each
split point that can be used if that split point is the first one to
be reached.  After the first download, a "leftovers" fragment is
downloaded in the background.  So, each call to runAsync causes
three fragments to be created: the exclusive fragment as before,
a new custom download, and a "leftovers" fragment.


Review by: bobv (TBR)


git-svn-id: https://google-web-toolkit.googlecode.com/svn/trunk@3959 8db76d5a-ed1c-0410-87a9-c151d255dfc7
diff --git a/dev/core/src/com/google/gwt/dev/jjs/JavaToJavaScriptCompiler.java b/dev/core/src/com/google/gwt/dev/jjs/JavaToJavaScriptCompiler.java
index 15d18c1..0ab97c9 100644
--- a/dev/core/src/com/google/gwt/dev/jjs/JavaToJavaScriptCompiler.java
+++ b/dev/core/src/com/google/gwt/dev/jjs/JavaToJavaScriptCompiler.java
@@ -211,7 +211,7 @@
 
       // (10.5) Split up the program into fragments
       if (options.isAggressivelyOptimize()) {
-        CodeSplitter.exec(jprogram, jsProgram, postStringInterningMap);
+        CodeSplitter.exec(logger, jprogram, jsProgram, postStringInterningMap);
       }
 
       // (11) Perform any post-obfuscation normalizations.
diff --git a/dev/core/src/com/google/gwt/dev/jjs/ast/JVariable.java b/dev/core/src/com/google/gwt/dev/jjs/ast/JVariable.java
index a485615..b37e953 100644
--- a/dev/core/src/com/google/gwt/dev/jjs/ast/JVariable.java
+++ b/dev/core/src/com/google/gwt/dev/jjs/ast/JVariable.java
@@ -43,6 +43,10 @@
     }
     return null;
   }
+  
+  public JDeclarationStatement getDeclarationStatement() {
+    return declStmt;
+  }
 
   public JExpression getInitializer() {
     if (declStmt != null) {
diff --git a/dev/core/src/com/google/gwt/dev/jjs/impl/CodeSplitter.java b/dev/core/src/com/google/gwt/dev/jjs/impl/CodeSplitter.java
index 725804b..63cd8ea 100644
--- a/dev/core/src/com/google/gwt/dev/jjs/impl/CodeSplitter.java
+++ b/dev/core/src/com/google/gwt/dev/jjs/impl/CodeSplitter.java
@@ -15,6 +15,7 @@
  */
 package com.google.gwt.dev.jjs.impl;
 
+import com.google.gwt.core.ext.TreeLogger;
 import com.google.gwt.dev.jjs.ast.Context;
 import com.google.gwt.dev.jjs.ast.JClassLiteral;
 import com.google.gwt.dev.jjs.ast.JExpression;
@@ -25,6 +26,7 @@
 import com.google.gwt.dev.jjs.ast.JReferenceType;
 import com.google.gwt.dev.jjs.ast.JStringLiteral;
 import com.google.gwt.dev.jjs.ast.JVisitor;
+import com.google.gwt.dev.jjs.impl.FragmentExtractor.CfaLivenessPredicate;
 import com.google.gwt.dev.jjs.impl.FragmentExtractor.LivenessPredicate;
 import com.google.gwt.dev.jjs.impl.FragmentExtractor.NothingAlivePredicate;
 import com.google.gwt.dev.jjs.impl.FragmentExtractor.StatementLogger;
@@ -59,12 +61,21 @@
  * The precise way the program is fragmented is an implementation detail that is
  * subject to change. Whenever the fragment strategy changes,
  * <code>AsyncFragmentLoader</code> must be updated in tandem. That said, the
- * current fragmentation strategy is to create one fragment for each call to
- * <code>runAsync()</code>. Each such fragment holds the code that is
- * exclusively needed by that particular call to <code>runAsync()</code>. Any
- * code needed by two or more calls to <code>runAsync()</code> is placed in
- * the initial fragment.
+ * current fragmentation strategy is to create an initial fragment and then
+ * three more fragments for each split point. For each split point, there is:
  * </p>
+ * 
+ * <ul>
+ * <li>a secondary base fragment, which is downloaded if this split point is
+ * the first one reached. It contains enough code to continue running as soon as
+ * it downloads.
+ * <li>an exclusively live fragment, which is downloaded if this split point is
+ * reached but is not the first one. It includes only that code that is
+ * exclusively needed by this split point.
+ * <li>a leftovers fragment, which includes all code that is in none of: the
+ * initial download, any exclusive fragment, or the secondary base fragment for
+ * this split point.
+ * </ul>
  */
 public class CodeSplitter {
   /**
@@ -92,7 +103,8 @@
   }
 
   /**
-   * A map from program atoms to the fragment they should be placed in.
+   * A map from program atoms to the fragment they should be placed in. An entry
+   * of 0 means it did not go into any fragment in particular.
    */
   private static class FragmentMap {
     public Map<JField, Integer> fields = new HashMap<JField, Integer>();
@@ -148,6 +160,41 @@
   }
 
   /**
+   * A liveness predicate that checks two separate underlying predicates.
+   */
+  private static class UnionLivenessPredicate implements LivenessPredicate {
+    private final LivenessPredicate pred1;
+    private final LivenessPredicate pred2;
+
+    public UnionLivenessPredicate(LivenessPredicate pred1,
+        LivenessPredicate pred2) {
+      this.pred1 = pred1;
+      this.pred2 = pred2;
+    }
+
+    public boolean isLive(JField field) {
+      return pred1.isLive(field) || pred2.isLive(field);
+    }
+
+    public boolean isLive(JMethod method) {
+      return pred1.isLive(method) || pred2.isLive(method);
+    }
+
+    public boolean isLive(JReferenceType type) {
+      return pred1.isLive(type) || pred2.isLive(type);
+    }
+
+    public boolean isLive(String literal) {
+      return pred1.isLive(literal) || pred2.isLive(literal);
+    }
+
+    public boolean miscellaneousStatementsAreLive() {
+      return pred1.miscellaneousStatementsAreLive()
+          || pred2.miscellaneousStatementsAreLive();
+    }
+  }
+
+  /**
    * A Java property that causes the fragment map to be logged.
    * 
    * TODO(spoon) save the logging data to an auxiliary compiler output and, if
@@ -155,9 +202,14 @@
    */
   private static String PROP_LOG_FRAGMENT_MAP = "gwt.jjs.logFragmentMap";
 
-  public static void exec(JProgram jprogram, JsProgram jsprogram,
-      JavaToJavaScriptMap map) {
-    new CodeSplitter(jprogram, jsprogram, map).execImpl();
+  public static void exec(TreeLogger logger, JProgram jprogram,
+      JsProgram jsprogram, JavaToJavaScriptMap map) {
+    if (jprogram.entryMethods.size() == 1) {
+      // Don't do anything if there is no call to runAsync
+      return;
+    }
+
+    new CodeSplitter(logger, jprogram, jsprogram, map).execImpl();
   }
 
   private static <T> int getOrZero(Map<T, Integer> map, T key) {
@@ -182,20 +234,59 @@
   }
 
   private final Map<JField, JClassLiteral> fieldToLiteralOfClass;
+
+  private final FragmentExtractor fragmentExtractor;
+  /**
+   * Code that is initially live when the program first downloads.
+   */
+  private final ControlFlowAnalyzer initiallyLive;
   private JProgram jprogram;
   private JsProgram jsprogram;
+  private final TreeLogger logger;
   private final boolean logging;
   private JavaToJavaScriptMap map;
   private final int numEntries;
 
-  private CodeSplitter(JProgram jprogram, JsProgram jsprogram,
-      JavaToJavaScriptMap map) {
+  private CodeSplitter(TreeLogger logger, JProgram jprogram,
+      JsProgram jsprogram, JavaToJavaScriptMap map) {
+    this.logger = logger.branch(TreeLogger.TRACE,
+        "Splitting JavaScript for incremental download");
     this.jprogram = jprogram;
     this.jsprogram = jsprogram;
     this.map = map;
+
     numEntries = jprogram.entryMethods.size();
     logging = Boolean.getBoolean(PROP_LOG_FRAGMENT_MAP);
     fieldToLiteralOfClass = FragmentExtractor.buildFieldToClassLiteralMap(jprogram);
+    fragmentExtractor = new FragmentExtractor(jprogram, jsprogram, map);
+
+    initiallyLive = new ControlFlowAnalyzer(jprogram);
+    traverseEntry(initiallyLive, 0);
+    initiallyLive.finishTraversal();
+  }
+
+  /**
+   * Create a new fragment and add it to the list.
+   * 
+   * @param alreadyLoaded The code that should be assumed to have already been
+   *          loaded
+   * @param liveNow The code that needs to be live once this fragment loads
+   * @param statsToAppend Additional statements to append to the end of the new
+   *          fragment
+   * @param fragmentStats The list of fragments to append to
+   */
+  private void addFragment(LivenessPredicate alreadyLoaded,
+      LivenessPredicate liveNow, List<JsStatement> statsToAppend,
+      List<List<JsStatement>> fragmentStats) {
+    if (logging) {
+      System.out.println();
+      System.out.println("==== Fragment " + fragmentStats.size() + " ====");
+      fragmentExtractor.setStatementLogger(new EchoStatementLogger());
+    }
+    List<JsStatement> stats = fragmentExtractor.extractStatements(liveNow,
+        alreadyLoaded);
+    stats.addAll(statsToAppend);
+    fragmentStats.add(stats);
   }
 
   /**
@@ -203,17 +294,15 @@
    * traces every other split point.
    */
   private List<ControlFlowAnalyzer> computeAllButOneCfas() {
-    // Reusing initiallyLive for each entry gives a significant speedup
-    ControlFlowAnalyzer initiallyLive = new ControlFlowAnalyzer(jprogram);
-    traverseEntry(initiallyLive, 0);
-    initiallyLive.finishTraversal();
-
     List<ControlFlowAnalyzer> allButOnes = new ArrayList<ControlFlowAnalyzer>(
         numEntries - 1);
 
     for (int entry = 1; entry < numEntries; entry++) {
       ControlFlowAnalyzer cfa = new ControlFlowAnalyzer(initiallyLive);
       traverseAllButEntry(cfa, entry);
+      // Traverse leftoversFragmentHasLoaded, because it should not
+      // go into any of the exclusive fragments.
+      cfa.traverseFromLeftoversFragmentHasLoaded();
       cfa.finishTraversal();
       allButOnes.add(cfa);
     }
@@ -234,55 +323,66 @@
   }
 
   private void execImpl() {
-    if (numEntries == 1) {
-      // Don't do anything if there is no call to runAsync
-      return;
-    }
-
     PerfLogger.start("CodeSplitter");
 
-    // Map code to the fragments
     FragmentMap fragmentMap = mapFragments();
 
     List<List<JsStatement>> fragmentStats = new ArrayList<List<JsStatement>>(
-        numEntries);
+        3 * numEntries - 2);
 
-    // Save the extractor for reuse
-    FragmentExtractor fragmentExtractor = new FragmentExtractor(jprogram,
-        jsprogram, map);
-
-    // Extract the code for each fragment, according to fragmentMap
-    for (int i = 0; i < numEntries; i++) {
-      LivenessPredicate pred = new FragmentMapLivenessPredicate(fragmentMap, i);
-
-      LivenessPredicate alreadyLoaded;
-      if (i == 0) {
-        alreadyLoaded = new NothingAlivePredicate();
-      } else {
-        alreadyLoaded = new FragmentMapLivenessPredicate(fragmentMap, 0);
-      }
-
-      if (logging) {
-        System.out.println();
-        System.out.println("==== Fragment " + i + " ====");
-        fragmentExtractor.setStatementLogger(new EchoStatementLogger());
-      }
-
-      List<JsStatement> entryStats = fragmentExtractor.extractStatements(pred,
-          alreadyLoaded);
-
-      if (i > 0) {
-        /*
-         * The fragment mapper drops all calls to entry methods. Add them back.
-         */
-        fragmentExtractor.addCallsToEntryMethods(i, entryStats);
-      }
-
-      fragmentStats.add(entryStats);
+    {
+      /*
+       * Compute the base fragment. It includes everything that is live when the
+       * program starts.
+       */
+      LivenessPredicate alreadyLoaded = new NothingAlivePredicate();
+      LivenessPredicate liveNow = new CfaLivenessPredicate(initiallyLive);
+      List<JsStatement> noStats = new ArrayList<JsStatement>();
+      addFragment(alreadyLoaded, liveNow, noStats, fragmentStats);
     }
 
-    // Install the new statements in the program fragments
-    jsprogram.setFragmentCount(numEntries);
+    /*
+     * Compute the exclusively live fragments. Each includes everything
+     * exclusively live after entry point i.
+     */
+    for (int i = 1; i < numEntries; i++) {
+      LivenessPredicate alreadyLoaded = new FragmentMapLivenessPredicate(
+          fragmentMap, 0);
+      LivenessPredicate liveNow = new FragmentMapLivenessPredicate(fragmentMap,
+          i);
+      List<JsStatement> statsToAppend = fragmentExtractor.createCallsToEntryMethods(i);
+      addFragment(alreadyLoaded, liveNow, statsToAppend, fragmentStats);
+    }
+
+    /*
+     * Add secondary base fragments and their associated leftover fragments.
+     */
+    for (int base = 1; base < numEntries; base++) {
+      ControlFlowAnalyzer baseCfa = new ControlFlowAnalyzer(initiallyLive);
+      traverseEntry(baseCfa, base);
+      baseCfa.finishTraversal();
+      LivenessPredicate baseLive = new CfaLivenessPredicate(baseCfa);
+
+      // secondary base
+      List<JsStatement> baseStatsToAppend = fragmentExtractor.createCallsToEntryMethods(base);
+      addFragment(new CfaLivenessPredicate(initiallyLive), baseLive,
+          baseStatsToAppend, fragmentStats);
+
+      // leftovers
+      LivenessPredicate globalLeftoversLive = new FragmentMapLivenessPredicate(
+          fragmentMap, 0);
+      LivenessPredicate associatedExclusives = new FragmentMapLivenessPredicate(
+          fragmentMap, base);
+      // Be sure to add in anything in the exclusives for base that is
+      // not in its secondary base.
+      LivenessPredicate leftoversLive = new UnionLivenessPredicate(
+          globalLeftoversLive, associatedExclusives);
+      List<JsStatement> statsToAppend = fragmentExtractor.createCallToLeftoversFragmentHasLoaded();
+      addFragment(baseLive, leftoversLive, statsToAppend, fragmentStats);
+    }
+
+    // now install the new statements in the program fragments
+    jsprogram.setFragmentCount(fragmentStats.size());
     for (int i = 0; i < fragmentStats.size(); i++) {
       JsBlock fragBlock = jsprogram.getFragmentBlock(i);
       fragBlock.getStatements().clear();
@@ -310,18 +410,26 @@
    */
   private void fixUpLoadOrderDependenciesForClassLiterals(
       FragmentMap fragmentMap) {
+    int numClassLitStrings = 0;
+    int numFixups = 0;
     for (JField field : fragmentMap.fields.keySet()) {
       JClassLiteral classLit = fieldToLiteralOfClass.get(field);
       if (classLit != null) {
         int classLitFrag = fragmentMap.fields.get(field);
         for (String string : stringsIn(field.getInitializer())) {
+          numClassLitStrings++;
           int stringFrag = getOrZero(fragmentMap.strings, string);
           if (stringFrag != classLitFrag && stringFrag != 0) {
+            numFixups++;
             fragmentMap.strings.put(string, 0);
           }
         }
       }
     }
+    logger.log(TreeLogger.DEBUG, "Fixed up load-order dependencies by moving "
+        + numFixups
+        + " strings in class literal constructors to fragment 0, out of "
+        + numClassLitStrings);
   }
 
   /**
@@ -329,6 +437,7 @@
    * superclass.
    */
   private void fixUpLoadOrderDependenciesForTypes(FragmentMap fragmentMap) {
+    int numFixups = 0;
     Queue<JReferenceType> typesToCheck = new ArrayBlockingQueue<JReferenceType>(
         jprogram.getDeclaredTypes().size());
     typesToCheck.addAll(jprogram.getDeclaredTypes());
@@ -338,11 +447,15 @@
         int typeFrag = getOrZero(fragmentMap.types, type);
         int supertypeFrag = getOrZero(fragmentMap.types, type.extnds);
         if (typeFrag != supertypeFrag && supertypeFrag != 0) {
+          numFixups++;
           fragmentMap.types.put(type.extnds, 0);
           typesToCheck.add(type.extnds);
         }
       }
     }
+    logger.log(TreeLogger.DEBUG, "Fixed up load-order dependencies by moving "
+        + numFixups + " types to fragment 0, out of "
+        + jprogram.getDeclaredTypes().size());
   }
 
   /**
@@ -428,15 +541,15 @@
    * <code>frag</code>. This does not call
    * {@link ControlFlowAnalyzer#finishTraversal()}.
    */
-  private void traverseEntry(ControlFlowAnalyzer cfa, int entry) {
-    for (JMethod entryMethod : jprogram.entryMethods.get(entry)) {
+  private void traverseEntry(ControlFlowAnalyzer cfa, int splitPoint) {
+    for (JMethod entryMethod : jprogram.entryMethods.get(splitPoint)) {
       cfa.traverseFrom(entryMethod);
     }
-    if (entry == 0) {
+    if (splitPoint == 0) {
       /*
        * Include class literal factories for simplicity. It is possible to move
        * them out, if they are only needed by one fragment, but they are tiny,
-       * so it does not look like it is worth the complexity in the compiler.
+       * so it does not seem worth the complexity in the compiler.
        */
       cfa.traverseFromClassLiteralFactories();
     }
diff --git a/dev/core/src/com/google/gwt/dev/jjs/impl/ControlFlowAnalyzer.java b/dev/core/src/com/google/gwt/dev/jjs/impl/ControlFlowAnalyzer.java
index 8831aec..22ff39d 100644
--- a/dev/core/src/com/google/gwt/dev/jjs/impl/ControlFlowAnalyzer.java
+++ b/dev/core/src/com/google/gwt/dev/jjs/impl/ControlFlowAnalyzer.java
@@ -644,10 +644,6 @@
     rescuer.rescue(method);
   }
 
-  public void traverseFromReferenceTo(JReferenceType type) {
-    rescuer.rescue(type, true, false);
-  }
-
   /**
    * Trace all code needed by class literal constructor expressions except for
    * the string literals they include. At the time of writing, these would
@@ -658,8 +654,8 @@
       @Override
       public void endVisit(JStringLiteral stringLiteral, Context ctx) {
         ctx.replaceMe(program.getLiteralString(
-            stringLiteral.getSourceInfo().makeChild(ControlFlowAnalyzer.class, "remove string literals"),
-            ""));
+            stringLiteral.getSourceInfo().makeChild(ControlFlowAnalyzer.class,
+                "remove string literals"), ""));
       }
     }
 
@@ -677,4 +673,14 @@
 
     (new ClassLitTraverser()).accept(program);
   }
+
+  public void traverseFromLeftoversFragmentHasLoaded() {
+    if (program.entryMethods.size() > 1) {
+      traverseFrom(program.getIndexedMethod("AsyncFragmentLoader.leftoversFragmentHasLoaded"));
+    }
+  }
+
+  public void traverseFromReferenceTo(JReferenceType type) {
+    rescuer.rescue(type, true, false);
+  }
 }
diff --git a/dev/core/src/com/google/gwt/dev/jjs/impl/FragmentExtractor.java b/dev/core/src/com/google/gwt/dev/jjs/impl/FragmentExtractor.java
index 204e90f..3f9df05 100644
--- a/dev/core/src/com/google/gwt/dev/jjs/impl/FragmentExtractor.java
+++ b/dev/core/src/com/google/gwt/dev/jjs/impl/FragmentExtractor.java
@@ -38,6 +38,7 @@
 import com.google.gwt.dev.js.ast.JsVars.JsVar;
 
 import java.util.ArrayList;
+import java.util.Collections;
 import java.util.HashMap;
 import java.util.HashSet;
 import java.util.List;
@@ -185,16 +186,34 @@
   /**
    * Add direct calls to the entry methods of the specified entry number.
    */
-  public void addCallsToEntryMethods(int entry, List<JsStatement> stats) {
-    for (JMethod entryMethod : jprogram.entryMethods.get(entry)) {
+  public List<JsStatement> createCallsToEntryMethods(int splitPoint) {
+    List<JsStatement> callStats = new ArrayList<JsStatement>(
+        jprogram.entryMethods.size());
+    for (JMethod entryMethod : jprogram.entryMethods.get(splitPoint)) {
       JsName name = map.nameForMethod(entryMethod);
       assert name != null;
       SourceInfo sourceInfo = jsprogram.getSourceInfo().makeChild(
-          FragmentExtractor.class, "call to entry function " + entry);
+          FragmentExtractor.class, "call to entry function " + splitPoint);
       JsInvocation call = new JsInvocation(sourceInfo);
       call.setQualifier(name.makeRef(sourceInfo));
-      stats.add(call.makeStmt());
+      callStats.add(call.makeStmt());
     }
+    return callStats;
+  }
+
+  /**
+   * Create a call to
+   * {@link com.google.gwt.lang.AsyncFragmentLoader#leftoversFragmentHasLoaded()}.
+   */
+  public List<JsStatement> createCallToLeftoversFragmentHasLoaded() {
+    JMethod loadedMethod = jprogram.getIndexedMethod("AsyncFragmentLoader.leftoversFragmentHasLoaded");
+    JsName loadedMethodName = map.nameForMethod(loadedMethod);
+    SourceInfo sourceInfo = jsprogram.getSourceInfo().makeChild(
+        FragmentExtractor.class, "call to leftoversFragmentHasLoaded ");
+    JsInvocation call = new JsInvocation(sourceInfo);
+    call.setQualifier(loadedMethodName.makeRef(sourceInfo));
+    List<JsStatement> newStats = Collections.<JsStatement> singletonList(call.makeStmt());
+    return newStats;
   }
 
   /**
@@ -262,6 +281,13 @@
       assert name != null;
       entryMethodNames.add(name);
     }
+
+    JMethod leftoverFragmentLoaded = jprogram.getIndexedMethod("AsyncFragmentLoader.leftoversFragmentHasLoaded");
+    if (leftoverFragmentLoaded != null) {
+      JsName name = map.nameForMethod(leftoverFragmentLoaded);
+      assert name != null;
+      entryMethodNames.add(name);
+    }
   }
 
   /**
diff --git a/dev/core/src/com/google/gwt/dev/jjs/impl/GenerateJavaScriptAST.java b/dev/core/src/com/google/gwt/dev/jjs/impl/GenerateJavaScriptAST.java
index 8486fbd..c06573a 100644
--- a/dev/core/src/com/google/gwt/dev/jjs/impl/GenerateJavaScriptAST.java
+++ b/dev/core/src/com/google/gwt/dev/jjs/impl/GenerateJavaScriptAST.java
@@ -1020,14 +1020,26 @@
       }
 
       /*
-       * Add calls to all split points other than the initial ones. That way, if
-       * the code splitter does not run, the resulting code will still function.
+       * Add calls to all non-initial entry points. That way, if the code
+       * splitter does not run, the resulting code will still function.
+       * Likewise, add a call to
+       * AsyncFragmentLoader.leftoversFragmentHasLoaded().
        */
       List<JsFunction> nonInitialEntries = Arrays.asList(entryFunctions).subList(
           x.getEntryCount(0), entryFunctions.length);
+      if (!nonInitialEntries.isEmpty()) {
+        JMethod loadedMethod = program.getIndexedMethod("AsyncFragmentLoader.leftoversFragmentHasLoaded");
+        JsName loadedMethodName = names.get(loadedMethod);
+        SourceInfo sourceInfo = jsProgram.getSourceInfo().makeChild(
+            GenerateJavaScriptAST.class, "call to leftoversFragmentHasLoaded ");
+        JsInvocation call = new JsInvocation(sourceInfo);
+        call.setQualifier(loadedMethodName.makeRef(sourceInfo));
+        globalStmts.add(call.makeStmt());
+      }
       for (JsFunction func : nonInitialEntries) {
         if (func != null) {
-          SourceInfo sourceInfo = jsProgram.getSourceInfo().makeChild(GenerateJavaScriptAST.class,
+          SourceInfo sourceInfo = jsProgram.getSourceInfo().makeChild(
+              GenerateJavaScriptAST.class,
               "call to entry non-initial entry function");
           JsInvocation call = new JsInvocation(sourceInfo);
           call.setQualifier(func.getName().makeRef(sourceInfo));
diff --git a/dev/core/src/com/google/gwt/dev/jjs/impl/Pruner.java b/dev/core/src/com/google/gwt/dev/jjs/impl/Pruner.java
index 4b327ef..231e624 100644
--- a/dev/core/src/com/google/gwt/dev/jjs/impl/Pruner.java
+++ b/dev/core/src/com/google/gwt/dev/jjs/impl/Pruner.java
@@ -462,6 +462,7 @@
       for (JMethod method : program.getAllEntryMethods()) {
         livenessAnalyzer.traverseFrom(method);
       }
+      livenessAnalyzer.traverseFromLeftoversFragmentHasLoaded();
       livenessAnalyzer.finishTraversal();
 
       program.typeOracle.setInstantiatedTypes(livenessAnalyzer.getInstantiatedTypes());
diff --git a/dev/core/src/com/google/gwt/dev/jjs/impl/ReplaceRunAsyncs.java b/dev/core/src/com/google/gwt/dev/jjs/impl/ReplaceRunAsyncs.java
index f9dfa3c..1ceb55a 100644
--- a/dev/core/src/com/google/gwt/dev/jjs/impl/ReplaceRunAsyncs.java
+++ b/dev/core/src/com/google/gwt/dev/jjs/impl/ReplaceRunAsyncs.java
@@ -19,6 +19,7 @@
 import com.google.gwt.dev.jjs.ast.Context;
 import com.google.gwt.dev.jjs.ast.JClassType;
 import com.google.gwt.dev.jjs.ast.JExpression;
+import com.google.gwt.dev.jjs.ast.JField;
 import com.google.gwt.dev.jjs.ast.JMethod;
 import com.google.gwt.dev.jjs.ast.JMethodCall;
 import com.google.gwt.dev.jjs.ast.JModVisitor;
@@ -43,8 +44,8 @@
         JExpression asyncCallback = x.getArgs().get(0);
 
         int entryNumber = entryCount++;
-        logger.log(TreeLogger.TRACE, "Using island loader #" + entryNumber
-            + " in method " + currentMethod);
+        logger.log(TreeLogger.INFO, "Assigning split point #" + entryNumber
+            + " in method " + fullMethodDescription(currentMethod));
 
         JClassType loader = getFragmentLoader(entryNumber);
         JMethod loadMethod = getRunAsyncMethod(loader);
@@ -71,6 +72,10 @@
     return new ReplaceRunAsyncs(logger, program).execImpl();
   }
 
+  private static String fullMethodDescription(JMethod method) {
+    return (method.getEnclosingType().getName() + "." + JProgram.getJsniSig(method));
+  }
+
   private final TreeLogger logger;
   private JProgram program;
 
@@ -83,6 +88,7 @@
   private int execImpl() {
     AsyncCreateVisitor visitor = new AsyncCreateVisitor();
     visitor.accept(program);
+    setNumEntriesInAsyncFragmentLoader(visitor.entryCount);
     return visitor.entryCount;
   }
 
@@ -123,4 +129,9 @@
     }
     return null;
   }
+
+  private void setNumEntriesInAsyncFragmentLoader(int entryCount) {
+    JField field = program.getIndexedField("AsyncFragmentLoader.numEntries");
+    field.getDeclarationStatement().initializer = program.getLiteralInt(entryCount);
+  }
 }
diff --git a/user/src/com/google/gwt/core/client/AsyncFragmentLoader.java b/user/src/com/google/gwt/core/client/AsyncFragmentLoader.java
index 448472e..e991b89 100644
--- a/user/src/com/google/gwt/core/client/AsyncFragmentLoader.java
+++ b/user/src/com/google/gwt/core/client/AsyncFragmentLoader.java
@@ -15,6 +15,9 @@
  */
 package com.google.gwt.core.client;
 
+import java.util.LinkedList;
+import java.util.Queue;
+
 /**
  * <p>
  * Low-level support to download an extra fragment of code. This should not be
@@ -22,6 +25,22 @@
  * </p>
  * 
  * <p>
+ * The fragments are numbered as follows, assuming there are <em>m</em> split
+ * points:
+ * 
+ * <ul>
+ * <li> 0 -- the <em>base</em> fragment, which is initially downloaded
+ * <li> 1-m -- the <em>exclusively live</em> fragments, holding the code
+ * needed exclusively for each split point
+ * <li> m -- the <em>secondary base</em> fragment for entry point 1. It holds
+ * precisely that code needed if entry point 1 is the first one reached after
+ * the application downloads.
+ * <li> m+1 -- the <em>leftovers fragment</em> for entry point 1. It holds all
+ * code not in fragments 0-(m-1) nor in fragment m.
+ * <li> (m+2)..(3m) -- secondary bases and leftovers for entry points 2..m
+ * </ul>
+ * 
+ * <p>
  * Different linkers have different requirements about how the code is
  * downloaded and installed. Thus, when it is time to actually download the
  * code, this class defers to a JavaScript function named
@@ -30,23 +49,93 @@
  */
 public class AsyncFragmentLoader {
   /**
+   * The first entry point reached after the program started.
+   */
+  private static int base = -1;
+
+  /**
+   * Whether the secondary base fragment is currently loading.
+   */
+  private static boolean baseLoading = false;
+
+  /**
+   * Whether the leftovers fragment has loaded yet.
+   */
+  private static boolean leftoversLoaded = false;
+
+  /**
+   * Whether the leftovers fragment is currently loading.
+   */
+  private static boolean leftoversLoading = false;
+
+  /**
+   * The total number of split points in the program, counting the initial entry
+   * as a split point. This is changed to the correct value by
+   * {@link com.google.gwt.dev.jjs.impl.ReplaceRunAsyncs}.
+   */
+  private static int numEntries = 1;
+
+  /**
+   * Split points that have been reached, but that cannot be downloaded until
+   * the leftovers fragment finishes downloading.
+   */
+  private static Queue<Integer> waitingForLeftovers = new LinkedList<Integer>();
+
+  /**
    * Inform the loader that the code for an entry point has now finished
    * loading.
    * 
    * @param entry The entry whose code fragment is now loaded.
    */
   public static void fragmentHasLoaded(int entry) {
-    // There is nothing to do with the current fragmentation strategy
+    if (base < 0) {
+      // The base fragment has loaded
+      base = entry;
+      baseLoading = false;
+
+      // Go ahead and download the appropriate leftovers fragment
+      leftoversLoading = true;
+      startLoadingFragment(numEntries + 2 * (entry - 1) + 1);
+    }
   }
 
   /**
-   * Loads the specified fragment asynchronously.
+   * Loads the specified split point.
    * 
-   * @param fragment the fragment to load
+   * @param splitPoint the fragment to load
    */
-  public static void inject(int fragment) {
-    logEventProgress("download" + fragment, "begin");
-    startLoadingFragment(fragment);
+  public static void inject(int splitPoint) {
+    if (leftoversLoaded) {
+      /*
+       * A base and a leftovers fragment have loaded. Load an exclusively live
+       * fragment.
+       */
+      startLoadingFragment(splitPoint);
+      return;
+    }
+
+    if (baseLoading || leftoversLoading) {
+      /*
+       * Wait until the leftovers fragment has loaded before loading this one.
+       */
+      waitingForLeftovers.add(splitPoint);
+      return;
+    }
+
+    // Nothing has loaded or started to load. Treat this fragment as the base.
+    baseLoading = true;
+    startLoadingFragment(numEntries + 2 * (splitPoint - 1));
+  }
+
+  /**
+   * Inform the loader that the "leftovers" fragment has loaded.
+   */
+  public static void leftoversFragmentHasLoaded() {
+    leftoversLoaded = true;
+    leftoversLoading = false;
+    while (!waitingForLeftovers.isEmpty()) {
+      inject(waitingForLeftovers.remove());
+    }
   }
 
   /**
@@ -73,13 +162,18 @@
     };
   }-*/;
 
+  private static native void gwtStartLoadingFragment(int fragment) /*-{
+    __gwtStartLoadingFragment(fragment);
+  }-*/;
+
   private static native boolean isStatsAvailable() /*-{
     return !!$stats;
   }-*/;
 
-  private static native void startLoadingFragment(int fragment) /*-{
-    __gwtStartLoadingFragment(fragment);
-  }-*/;
+  private static void startLoadingFragment(int fragment) {
+    logEventProgress("download" + fragment, "begin");
+    gwtStartLoadingFragment(fragment);
+  }
 
   /**
    * Always use this as {@link isStatsAvailable} &amp;&amp;