Allow naming a runAsync call using a class literal. There are now
two overloads of GWT.runAsync, and the new overload takes an
extra argument supplying the class literal.
Review by: kprobst (SplitPointRecorder), scottb (the bulk of it)
git-svn-id: https://google-web-toolkit.googlecode.com/svn/trunk@6097 8db76d5a-ed1c-0410-87a9-c151d255dfc7
diff --git a/dev/core/src/com/google/gwt/core/ext/soyc/impl/SplitPointRecorder.java b/dev/core/src/com/google/gwt/core/ext/soyc/impl/SplitPointRecorder.java
index 7826574..597ad33 100644
--- a/dev/core/src/com/google/gwt/core/ext/soyc/impl/SplitPointRecorder.java
+++ b/dev/core/src/com/google/gwt/core/ext/soyc/impl/SplitPointRecorder.java
@@ -80,7 +80,7 @@
htmlOut.printRaw(curLine);
htmlOut.newline();
}
-
+
if (!jprogram.getSplitPointInitialSequence().isEmpty()) {
curLine = "<initialseq>";
htmlOut.printRaw(curLine);
@@ -92,7 +92,7 @@
htmlOut.printRaw(curLine);
htmlOut.newline();
}
-
+
htmlOut.indentOut();
curLine = "</initialseq>";
htmlOut.printRaw(curLine);
@@ -127,13 +127,19 @@
Map<String, Integer> counts = new HashMap<String, Integer>();
for (RunAsyncReplacement replacement : program.getRunAsyncReplacements().values()) {
int entryNumber = replacement.getNumber();
- String methodDescription = fullMethodDescription(replacement.getEnclosingMethod());
- if (counts.containsKey(methodDescription)) {
- counts.put(methodDescription, counts.get(methodDescription) + 1);
- methodDescription += "#"
- + Integer.toString(counts.get(methodDescription));
+ String methodDescription;
+ if (replacement.getName() != null) {
+ methodDescription = replacement.getName();
} else {
- counts.put(methodDescription, 1);
+ methodDescription = "@"
+ + fullMethodDescription(replacement.getEnclosingMethod());
+ if (counts.containsKey(methodDescription)) {
+ counts.put(methodDescription, counts.get(methodDescription) + 1);
+ methodDescription += "#"
+ + Integer.toString(counts.get(methodDescription));
+ } else {
+ counts.put(methodDescription, 1);
+ }
}
names.put(entryNumber, methodDescription);
diff --git a/dev/core/src/com/google/gwt/dev/jdt/FindDeferredBindingSitesVisitor.java b/dev/core/src/com/google/gwt/dev/jdt/FindDeferredBindingSitesVisitor.java
index e4baf2c..62e035b 100644
--- a/dev/core/src/com/google/gwt/dev/jdt/FindDeferredBindingSitesVisitor.java
+++ b/dev/core/src/com/google/gwt/dev/jdt/FindDeferredBindingSitesVisitor.java
@@ -59,8 +59,7 @@
public static final String REBIND_MAGIC_METHOD = "create";
public static final String ASYNC_MAGIC_METHOD = "runAsync";
- public static void reportRebindProblem(MessageSendSite site,
- String message) {
+ public static void reportRebindProblem(MessageSendSite site, String message) {
MessageSend messageSend = site.messageSend;
Scope scope = site.scope;
// Safe since CUS.referenceContext is set in its constructor.
@@ -111,11 +110,18 @@
}
} else {
assert asyncMagicMethod;
- if (args.length != 1) {
+ if (args.length != 1 && args.length != 2) {
reportRebindProblem(site,
- "GWT.runAsync() should take exactly one argument");
+ "GWT.runAsync() should take one or two arguments");
return;
}
+ if (args.length == 2) {
+ if (!(args[0] instanceof ClassLiteralAccess)) {
+ reportRebindProblem(site,
+ "Only class literals may be used to name a call to GWT.runAsync()");
+ return;
+ }
+ }
}
if (asyncMagicMethod) {
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 8b8a4a1..0c0d090 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
@@ -260,6 +260,74 @@
dependencyRecorder.close();
}
+ /**
+ * Find a split point as designated in the {@link #PROP_INITIAL_SEQUENCE}
+ * configuration property.
+ */
+ public static int findSplitPoint(String refString, JProgram program,
+ TreeLogger branch) throws UnableToCompleteException {
+ Map<JMethod, List<Integer>> methodToSplitPoint = reverseByEnclosingMethod(program.getRunAsyncReplacements());
+ Map<String, List<Integer>> nameToSplitPoint = reverseByName(program.getRunAsyncReplacements());
+
+ if (refString.startsWith("@")) {
+ JsniRef jsniRef = JsniRef.parse(refString);
+ if (jsniRef == null) {
+ branch.log(TreeLogger.ERROR, "Badly formatted JSNI reference in "
+ + PROP_INITIAL_SEQUENCE + ": " + refString);
+ throw new UnableToCompleteException();
+ }
+ final String lookupErrorHolder[] = new String[1];
+ HasEnclosingType referent = JsniRefLookup.findJsniRefTarget(jsniRef,
+ program, new JsniRefLookup.ErrorReporter() {
+ public void reportError(String error) {
+ lookupErrorHolder[0] = error;
+ }
+ });
+ if (referent == null) {
+ TreeLogger resolveLogger = branch.branch(TreeLogger.ERROR,
+ "Could not resolve JSNI reference: " + jsniRef);
+ resolveLogger.log(TreeLogger.ERROR, lookupErrorHolder[0]);
+ throw new UnableToCompleteException();
+ }
+
+ if (!(referent instanceof JMethod)) {
+ branch.log(TreeLogger.ERROR, "Not a method: " + referent);
+ throw new UnableToCompleteException();
+ }
+
+ JMethod method = (JMethod) referent;
+ List<Integer> splitPoints = methodToSplitPoint.get(method);
+ if (splitPoints == null) {
+ branch.log(TreeLogger.ERROR,
+ "Method does not enclose a runAsync call: " + jsniRef);
+ throw new UnableToCompleteException();
+ }
+ if (splitPoints.size() > 1) {
+ branch.log(TreeLogger.ERROR,
+ "Method includes multiple runAsync calls, "
+ + "so it's ambiguous which one is meant: " + jsniRef);
+ throw new UnableToCompleteException();
+ }
+
+ return splitPoints.get(0);
+ }
+
+ // Assume it's a raw class name
+ List<Integer> splitPoints = nameToSplitPoint.get(refString);
+ if (splitPoints == null || splitPoints.size() == 0) {
+ branch.log(TreeLogger.ERROR, "No runAsync call is labelled with class "
+ + refString);
+ throw new UnableToCompleteException();
+ }
+ if (splitPoints.size() > 1) {
+ branch.log(TreeLogger.ERROR,
+ "More than one runAsync call is labelled with class " + refString);
+ throw new UnableToCompleteException();
+ }
+
+ return splitPoints.get(0);
+ }
+
public static int getExclusiveFragmentNumber(int splitPoint,
int numSplitPoints) {
return splitPoint;
@@ -295,8 +363,6 @@
JProgram program, Properties properties) throws UnableToCompleteException {
TreeLogger branch = logger.branch(TreeLogger.TRACE,
"Looking up initial load sequence for split points");
- Map<JMethod, List<Integer>> reversedRunAsyncMap = reverse(program.getRunAsyncReplacements());
-
LinkedHashSet<Integer> initialLoadSequence = new LinkedHashSet<Integer>();
ConfigurationProperty prop;
@@ -314,8 +380,7 @@
}
for (String refString : prop.getValues()) {
- int splitPoint = findSplitPoint(refString, program, branch,
- reversedRunAsyncMap);
+ int splitPoint = findSplitPoint(refString, program, branch);
if (initialLoadSequence.contains(splitPoint)) {
branch.log(TreeLogger.ERROR, "Split point specified more than once: "
+ refString);
@@ -323,8 +388,6 @@
initialLoadSequence.add(splitPoint);
}
- // TODO(spoon) create an artifact in the aux dir describing the choice, so
- // that SOYC can use it
logInitialLoadSequence(logger, initialLoadSequence);
installInitialLoadSequenceField(program, initialLoadSequence);
program.setSplitPointInitialSequence(new ArrayList<Integer>(
@@ -395,63 +458,6 @@
return cfa;
}
- /**
- * Find a split point as designated in the {@link #PROP_INITIAL_SEQUENCE}
- * configuration property.
- *
- * TODO(spoon) accept a labeled runAsync call, once runAsyncs can be labeled
- */
- private static int findSplitPoint(String refString, JProgram program,
- TreeLogger branch, Map<JMethod, List<Integer>> reversedRunAsyncMap)
- throws UnableToCompleteException {
- if (refString.startsWith("@")) {
- JsniRef jsniRef = JsniRef.parse(refString);
- if (jsniRef == null) {
- branch.log(TreeLogger.ERROR, "Badly formatted JSNI reference in "
- + PROP_INITIAL_SEQUENCE + ": " + refString);
- throw new UnableToCompleteException();
- }
- final String lookupErrorHolder[] = new String[1];
- HasEnclosingType referent = JsniRefLookup.findJsniRefTarget(jsniRef,
- program, new JsniRefLookup.ErrorReporter() {
- public void reportError(String error) {
- lookupErrorHolder[0] = error;
- }
- });
- if (referent == null) {
- TreeLogger resolveLogger = branch.branch(TreeLogger.ERROR,
- "Could not resolve JSNI reference: " + jsniRef);
- resolveLogger.log(TreeLogger.ERROR, lookupErrorHolder[0]);
- throw new UnableToCompleteException();
- }
-
- if (!(referent instanceof JMethod)) {
- branch.log(TreeLogger.ERROR, "Not a method: " + referent);
- throw new UnableToCompleteException();
- }
-
- JMethod method = (JMethod) referent;
- List<Integer> splitPoints = reversedRunAsyncMap.get(method);
- if (splitPoints == null) {
- branch.log(TreeLogger.ERROR,
- "Method does not enclose a runAsync call: " + jsniRef);
- throw new UnableToCompleteException();
- }
- if (splitPoints.size() != 1) {
- branch.log(TreeLogger.ERROR,
- "Method includes multiple runAsync calls, "
- + "so it's ambiguous which one is meant: " + jsniRef);
- throw new UnableToCompleteException();
- }
-
- return splitPoints.get(0);
- }
-
- branch.log(TreeLogger.ERROR, "Unrecognized designation of a split point: "
- + refString);
- throw new UnableToCompleteException();
- }
-
private static String fullNameString(JField field) {
return field.getEnclosingType().getName() + "." + field.getName();
}
@@ -527,7 +533,7 @@
* Reverses a runAsync map, returning a map from methods to the split point
* numbers invoked from within that method.
*/
- private static Map<JMethod, List<Integer>> reverse(
+ private static Map<JMethod, List<Integer>> reverseByEnclosingMethod(
Map<Integer, RunAsyncReplacement> runAsyncMap) {
Map<JMethod, List<Integer>> revmap = new HashMap<JMethod, List<Integer>>();
for (RunAsyncReplacement replacement : runAsyncMap.values()) {
@@ -542,6 +548,23 @@
return revmap;
}
+ private static Map<String, List<Integer>> reverseByName(
+ Map<Integer, RunAsyncReplacement> runAsyncReplacements) {
+ Map<String, List<Integer>> revmap = new HashMap<String, List<Integer>>();
+ for (RunAsyncReplacement replacement : runAsyncReplacements.values()) {
+ String name = replacement.getName();
+ if (name != null) {
+ List<Integer> list = revmap.get(name);
+ if (list == null) {
+ list = new ArrayList<Integer>();
+ revmap.put(name, list);
+ }
+ list.add(replacement.getNumber());
+ }
+ }
+ return revmap;
+ }
+
/**
* Traverse all code in the program that is reachable via split point
* <code>splitPoint</code>.
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 8c341da..f47edb6 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
@@ -16,7 +16,9 @@
package com.google.gwt.dev.jjs.impl;
import com.google.gwt.core.ext.TreeLogger;
+import com.google.gwt.dev.jjs.InternalCompilerException;
import com.google.gwt.dev.jjs.ast.Context;
+import com.google.gwt.dev.jjs.ast.JClassLiteral;
import com.google.gwt.dev.jjs.ast.JClassType;
import com.google.gwt.dev.jjs.ast.JExpression;
import com.google.gwt.dev.jjs.ast.JField;
@@ -34,6 +36,8 @@
/**
* Replaces calls to
* {@link com.google.gwt.core.client.GWT#runAsync(com.google.gwt.core.client.RunAsyncCallback)}
+ * and
+ * {@link com.google.gwt.core.client.GWT#runAsync(Class, com.google.gwt.core.client.RunAsyncCallback)
* by calls to a fragment loader.
*/
public class ReplaceRunAsyncs {
@@ -44,12 +48,15 @@
public static class RunAsyncReplacement implements Serializable {
private final JMethod enclosingMethod;
private final JMethod loadMethod;
+ private final String name;
private final int number;
- RunAsyncReplacement(int number, JMethod enclosingMethod, JMethod loadMethod) {
+ RunAsyncReplacement(int number, JMethod enclosingMethod,
+ JMethod loadMethod, String name) {
this.number = number;
this.enclosingMethod = enclosingMethod;
this.loadMethod = loadMethod;
+ this.name = name;
}
/**
@@ -68,6 +75,15 @@
}
/**
+ * Return the name of this runAsync, which is specified by a class literal
+ * in the two-argument version of runAsync(). Returns <code>null</code> if
+ * there is no name for the call.
+ */
+ public String getName() {
+ return name;
+ }
+
+ /**
* The index of this runAsync, numbered from 1 to n.
*/
public int getNumber() {
@@ -87,16 +103,29 @@
@Override
public void endVisit(JMethodCall x, Context ctx) {
JMethod method = x.getTarget();
- if (method == program.getIndexedMethod("GWT.runAsync")) {
- assert (x.getArgs().size() == 1);
- JExpression asyncCallback = x.getArgs().get(0);
+ if (isRunAsyncMethod(method)) {
+ JExpression asyncCallback;
+ String name;
+ switch (x.getArgs().size()) {
+ case 1:
+ name = null;
+ asyncCallback = x.getArgs().get(0);
+ break;
+ case 2:
+ name = ((JClassLiteral) x.getArgs().get(0)).getRefType().getName();
+ asyncCallback = x.getArgs().get(1);
+ break;
+ default:
+ throw new InternalCompilerException(
+ "runAsync call found with neither 1 nor 2 arguments: " + x);
+ }
int entryNumber = entryCount++;
JClassType loader = getFragmentLoader(entryNumber);
JMethod loadMethod = getRunAsyncMethod(loader);
assert loadMethod != null;
runAsyncReplacements.put(entryNumber, new RunAsyncReplacement(
- entryNumber, currentMethod, loadMethod));
+ entryNumber, currentMethod, loadMethod, name));
JMethodCall methodCall = new JMethodCall(x.getSourceInfo(), null,
loadMethod);
@@ -113,6 +142,14 @@
currentMethod = x;
return true;
}
+
+ private boolean isRunAsyncMethod(JMethod method) {
+ /*
+ * The method is overloaded, so check the enclosing type plus the name.
+ */
+ return method.getEnclosingType() == program.getIndexedType("GWT")
+ && method.getName().equals("runAsync");
+ }
}
public static void exec(TreeLogger logger, JProgram program) {
@@ -133,6 +170,7 @@
}
private JProgram program;
+
private Map<Integer, RunAsyncReplacement> runAsyncReplacements = new HashMap<Integer, RunAsyncReplacement>();
private ReplaceRunAsyncs(JProgram program) {
diff --git a/dev/core/test/com/google/gwt/dev/javac/RunAsyncNameTest.java b/dev/core/test/com/google/gwt/dev/javac/RunAsyncNameTest.java
new file mode 100644
index 0000000..ad433e5
--- /dev/null
+++ b/dev/core/test/com/google/gwt/dev/javac/RunAsyncNameTest.java
@@ -0,0 +1,81 @@
+/*
+ * Copyright 2009 Google Inc.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License"); you may not
+ * use this file except in compliance with the License. You may obtain a copy of
+ * the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
+ * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
+ * License for the specific language governing permissions and limitations under
+ * the License.
+ */
+package com.google.gwt.dev.javac;
+
+import com.google.gwt.core.ext.TreeLogger;
+import com.google.gwt.core.ext.UnableToCompleteException;
+import com.google.gwt.dev.javac.impl.MockJavaResource;
+import com.google.gwt.dev.jjs.impl.OptimizerTestBase;
+import com.google.gwt.dev.util.UnitTestTreeLogger;
+
+/**
+ * This class tests naming of runAsync calls. Mostly it tests names that are
+ * invalid.
+ */
+public class RunAsyncNameTest extends OptimizerTestBase {
+ @Override
+ public void setUp() {
+ sourceOracle.addOrReplace(new MockJavaResource("test.CallRunAsync") {
+ @Override
+ protected CharSequence getContent() {
+ StringBuffer code = new StringBuffer();
+ code.append("package test;\n");
+ code.append("import com.google.gwt.core.client.GWT;\n");
+ code.append("public class CallRunAsync {\n");
+ code.append(" public static int notAmethod;");
+ code.append(" public static void call0() { }\n");
+ code.append(" public static void call1() {\n");
+ code.append(" GWT.runAsync(null);\n");
+ code.append(" }\n");
+ code.append(" public static void call2() {\n");
+ code.append(" GWT.runAsync(null);\n");
+ code.append(" GWT.runAsync(null);\n");
+ code.append(" }\n");
+ code.append("}\n");
+ return code;
+ }
+ });
+ }
+
+ /**
+ * Tests that it's an error to call the 2-argument version of GWT.runAsync
+ * with anything but a class literal.
+ */
+ public void testNonLiteralInCall() throws UnableToCompleteException {
+ UnitTestTreeLogger logger;
+ {
+ UnitTestTreeLogger.Builder builder = new UnitTestTreeLogger.Builder();
+ builder.setLowestLogLevel(TreeLogger.ERROR);
+ builder.expectError("Errors in /mock/test/EntryPoint.java", null);
+ builder.expectError(
+ "Line 5: Only class literals may be used to name a call to GWT.runAsync()",
+ null);
+ builder.expectError("Cannot proceed due to previous errors", null);
+ logger = builder.createLogger();
+ this.logger = logger;
+ }
+
+ addSnippetImport("com.google.gwt.core.client.GWT");
+ try {
+ compileSnippet("void", "GWT.runAsync((new Object()).getClass(), null);");
+ fail("Expected compilation to fail");
+ } catch (UnableToCompleteException e) {
+ // expected
+ }
+
+ logger.assertCorrectLogEntries();
+ }
+}
diff --git a/dev/core/test/com/google/gwt/dev/jjs/JavaAstConstructor.java b/dev/core/test/com/google/gwt/dev/jjs/JavaAstConstructor.java
index 2684691..8a6e49e 100644
--- a/dev/core/test/com/google/gwt/dev/jjs/JavaAstConstructor.java
+++ b/dev/core/test/com/google/gwt/dev/jjs/JavaAstConstructor.java
@@ -21,6 +21,7 @@
import com.google.gwt.dev.javac.impl.JavaResourceBase;
import com.google.gwt.dev.javac.impl.MockJavaResource;
import com.google.gwt.dev.jdt.BasicWebModeCompiler;
+import com.google.gwt.dev.jdt.FindDeferredBindingSitesVisitor;
import com.google.gwt.dev.jjs.CorrelationFactory.DummyCorrelationFactory;
import com.google.gwt.dev.jjs.ast.JDeclaredType;
import com.google.gwt.dev.jjs.ast.JMethod;
@@ -98,6 +99,8 @@
code.append("public final class GWT {\n");
code.append(" public boolean isClient() { return true; };\n");
code.append(" public boolean isScript() { return true; };\n");
+ code.append(" public static void runAsync(Object callback) { }\n");
+ code.append(" public static void runAsync(Class<?> name, Object callback) { }\n");
code.append("}\n");
return code;
}
@@ -131,6 +134,16 @@
//
JavaToJavaScriptCompiler.checkForErrors(logger, goldenCuds, false);
+ /*
+ * FindDeferredBindingSitesVisitor detects errors in usage of magic methods
+ * in the GWT class.
+ */
+ for (CompilationUnitDeclaration jdtCud : goldenCuds) {
+ jdtCud.traverse(new FindDeferredBindingSitesVisitor(), jdtCud.scope);
+ }
+
+ JavaToJavaScriptCompiler.checkForErrors(logger, goldenCuds, true);
+
CorrelationFactory correlator = new DummyCorrelationFactory();
JProgram jprogram = new JProgram(correlator);
JsProgram jsProgram = new JsProgram(correlator);
diff --git a/dev/core/test/com/google/gwt/dev/jjs/impl/OptimizerTestBase.java b/dev/core/test/com/google/gwt/dev/jjs/impl/OptimizerTestBase.java
index 9ef05dc..50cf016 100644
--- a/dev/core/test/com/google/gwt/dev/jjs/impl/OptimizerTestBase.java
+++ b/dev/core/test/com/google/gwt/dev/jjs/impl/OptimizerTestBase.java
@@ -77,8 +77,8 @@
/**
* Finds a type by name. The type name may be short, e.g. <code>"Foo"</code>,
- * or fully-qualified, e.g. <code>"com.google.example.Foo"</code>. If a
- * short name is used, it must be unambiguous.
+ * or fully-qualified, e.g. <code>"com.google.example.Foo"</code>. If a short
+ * name is used, it must be unambiguous.
*/
public static JDeclaredType findType(JProgram program, String typeName) {
JDeclaredType type = program.getFromTypeMap(typeName);
@@ -117,9 +117,9 @@
return TreeLogger.NULL;
}
- protected final MockResourceOracle sourceOracle = new MockResourceOracle();
+ protected TreeLogger logger = createTreeLogger();
- private final TreeLogger logger = createTreeLogger();
+ protected final MockResourceOracle sourceOracle = new MockResourceOracle();
private final Set<String> snippetClassDecls = new TreeSet<String>();
@@ -133,6 +133,15 @@
}
/**
+ * Adds a snippet of code, for example a field declaration, to the class that
+ * encloses the snippet subsequently passed to
+ * {@link #compileSnippet(String, String)}.
+ */
+ protected void addSnippetClassDecl(String fieldDecl) {
+ snippetClassDecls.add(fieldDecl);
+ }
+
+ /**
* Adds an import statement for any code subsequently passed to
* {@link #compileSnippet(String, String)}.
*/
@@ -141,14 +150,6 @@
}
/**
- * Adds a local field declaration for code subsequently passed to
- * {@link #compileSnippet(String, String)}.
- */
- protected void addSnippetClassDecl(String fieldDecl) {
- snippetClassDecls.add(fieldDecl);
- }
-
- /**
* Returns the program that results from compiling the specified code snippet
* as the body of an entry point method.
*
diff --git a/user/src/com/google/gwt/core/client/GWT.java b/user/src/com/google/gwt/core/client/GWT.java
index f58ce85..2028019 100644
--- a/user/src/com/google/gwt/core/client/GWT.java
+++ b/user/src/com/google/gwt/core/client/GWT.java
@@ -204,34 +204,20 @@
}
/**
+ * The same as {@link #runAsync(RunAsyncCallback)}, except with an extra
+ * parameter to provide a name for the call. The name parameter should be
+ * supplied with a class literal. No two runAsync calls in the same program
+ * should use the same name.
+ */
+ public static void runAsync(Class<?> name, RunAsyncCallback callback) {
+ runAsyncWithoutCodeSplitting(callback);
+ }
+
+ /**
* Run the specified callback once the necessary code for it has been loaded.
*/
public static void runAsync(RunAsyncCallback callback) {
- /*
- * By default, just call the callback. This allows using
- * <code>runAsync</code> in code that might or might not run in a web
- * browser.
- */
- if (isScript()) {
- /*
- * It's possible that the code splitter does not run, even for a
- * production build. Signal a lightweight event, anyway, just so that
- * there isn't a complete lack of lightweight events for runAsync.
- */
- AsyncFragmentLoader.BROWSER_LOADER.logEventProgress("noDownloadNeeded", "begin");
- AsyncFragmentLoader.BROWSER_LOADER.logEventProgress("noDownloadNeeded", "end");
- }
-
- UncaughtExceptionHandler handler = sUncaughtExceptionHandler;
- if (handler == null) {
- callback.onSuccess();
- } else {
- try {
- callback.onSuccess();
- } catch (Throwable e) {
- handler.onUncaughtException(e);
- }
- }
+ runAsyncWithoutCodeSplitting(callback);
}
/**
@@ -261,4 +247,36 @@
private static native String getVersion0() /*-{
return $gwt_version;
}-*/;
+
+ /**
+ * This implementation of runAsync simply calls the callback. It is only used
+ * when no code splitting has occurred.
+ */
+ private static void runAsyncWithoutCodeSplitting(RunAsyncCallback callback) {
+ /*
+ * By default, just call the callback. This allows using
+ * <code>runAsync</code> in code that might or might not run in a web
+ * browser.
+ */
+ if (isScript()) {
+ /*
+ * It's possible that the code splitter does not run, even for a
+ * production build. Signal a lightweight event, anyway, just so that
+ * there isn't a complete lack of lightweight events for runAsync.
+ */
+ AsyncFragmentLoader.BROWSER_LOADER.logEventProgress("noDownloadNeeded", "begin");
+ AsyncFragmentLoader.BROWSER_LOADER.logEventProgress("noDownloadNeeded", "end");
+ }
+
+ UncaughtExceptionHandler handler = sUncaughtExceptionHandler;
+ if (handler == null) {
+ callback.onSuccess();
+ } else {
+ try {
+ callback.onSuccess();
+ } catch (Throwable e) {
+ handler.onUncaughtException(e);
+ }
+ }
+ }
}
diff --git a/user/test/com/google/gwt/dev/jjs/InitialLoadSequence.gwt.xml b/user/test/com/google/gwt/dev/jjs/InitialLoadSequence.gwt.xml
index 5c8a752..7dcb746 100644
--- a/user/test/com/google/gwt/dev/jjs/InitialLoadSequence.gwt.xml
+++ b/user/test/com/google/gwt/dev/jjs/InitialLoadSequence.gwt.xml
@@ -21,5 +21,5 @@
<extend-configuration-property name="compiler.splitpoint.initial.sequence"
value="@com.google.gwt.dev.jjs.test.InitialLoadSequenceTest::callback1()" />
<extend-configuration-property name="compiler.splitpoint.initial.sequence"
- value="@com.google.gwt.dev.jjs.test.InitialLoadSequenceTest::callback2()" />
+ value="com.google.gwt.dev.jjs.test.InitialLoadSequenceTest$Callback2Marker" />
</module>
diff --git a/user/test/com/google/gwt/dev/jjs/test/InitialLoadSequenceTest.java b/user/test/com/google/gwt/dev/jjs/test/InitialLoadSequenceTest.java
index 478727e..39a2ff7 100644
--- a/user/test/com/google/gwt/dev/jjs/test/InitialLoadSequenceTest.java
+++ b/user/test/com/google/gwt/dev/jjs/test/InitialLoadSequenceTest.java
@@ -26,6 +26,13 @@
*/
public class InitialLoadSequenceTest extends GWTTestCase {
private static final int TIMEOUT = 10000;
+
+ /**
+ * This class is used to mark the second runAsync call.
+ */
+ public static class Callback2Marker {
+ }
+
/**
* The number of callbacks outstanding. When this gets to zero, the test
* finishes.
@@ -58,7 +65,7 @@
}
private void callback2() {
- GWT.runAsync(new RunAsyncCallback() {
+ GWT.runAsync(Callback2Marker.class, new RunAsyncCallback() {
public void onFailure(Throwable reason) {
fail(reason.toString());
}