Have ControlFlowAnalyzer record fields that are written. Previously,
it only recorded fields that were read. This allows fixing a bug
in CodeSplitter where a field could end up being written before the
field itself was loaded.
Also, this adds a test case for runAsync.
Review by: kprobst
git-svn-id: https://google-web-toolkit.googlecode.com/svn/trunk@4358 8db76d5a-ed1c-0410-87a9-c151d255dfc7
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 f5014cb..a44816d 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
@@ -254,6 +254,13 @@
return (value == null) ? 0 : value;
}
+ private static <T> Set<T> union(Set<? extends T> set1, Set<? extends T> set2) {
+ Set<T> union = new HashSet<T>();
+ union.addAll(set1);
+ union.addAll(set2);
+ return union;
+ }
+
private static <T> void updateMap(int entry, Map<T, Integer> map,
Set<?> liveWithoutEntry, Iterable<T> all) {
for (T each : all) {
@@ -577,11 +584,13 @@
allMethods.add((JMethod) node);
}
}
+ allFields.addAll(everything.getFieldsWritten());
for (int entry = 1; entry < numEntries; entry++) {
ControlFlowAnalyzer allButOne = allButOnes.get(entry - 1);
- updateMap(entry, fragmentMap.fields, allButOne.getLiveFieldsAndMethods(),
- allFields);
+ Set<JNode> allLiveNodes = union(allButOne.getLiveFieldsAndMethods(),
+ allButOne.getFieldsWritten());
+ updateMap(entry, fragmentMap.fields, allLiveNodes, allFields);
updateMap(entry, fragmentMap.methods,
allButOne.getLiveFieldsAndMethods(), allMethods);
updateMap(entry, fragmentMap.strings, allButOne.getLiveStrings(),
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 a7eee8f..bcac220 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
@@ -107,6 +107,10 @@
@Override
public boolean visit(JBinaryOperation x, Context ctx) {
+ if (x.isAssignment() && x.getLhs() instanceof JFieldRef) {
+ fieldsWritten.add(((JFieldRef) x.getLhs()).getField());
+ }
+
// special string concat handling
if ((x.getOp() == JBinaryOperator.ADD || x.getOp() == JBinaryOperator.ASG_ADD)
&& x.getType() == program.getTypeJavaLangString()) {
@@ -211,6 +215,10 @@
* the variable is accessed, not when its declaration runs.
*/
accept(x.getInitializer());
+
+ if (x.getVariableRef().getTarget() instanceof JField) {
+ fieldsWritten.add((JField) x.getVariableRef().getTarget());
+ }
}
}
@@ -618,6 +626,7 @@
}
}
+ private Set<JField> fieldsWritten = new HashSet<JField>();
private Set<JReferenceType> instantiatedTypes = new HashSet<JReferenceType>();
private Set<JNode> liveFieldsAndMethods = new HashSet<JNode>();
private Set<String> liveStrings = new HashSet<String>();
@@ -643,6 +652,7 @@
public ControlFlowAnalyzer(ControlFlowAnalyzer cfa) {
program = cfa.program;
+ fieldsWritten = new HashSet<JField>(cfa.fieldsWritten);
instantiatedTypes = new HashSet<JReferenceType>(cfa.instantiatedTypes);
liveFieldsAndMethods = new HashSet<JNode>(cfa.liveFieldsAndMethods);
referencedTypes = new HashSet<JReferenceType>(cfa.referencedTypes);
@@ -659,6 +669,13 @@
}
/**
+ * Return the set of all fields that are written.
+ */
+ public Set<JField> getFieldsWritten() {
+ return fieldsWritten;
+ }
+
+ /**
* Return the complete set of types that have been instantiated.
*/
public Set<JReferenceType> getInstantiatedTypes() {
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 a3c49a4..7e33287 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
@@ -57,7 +57,8 @@
}
public boolean isLive(JField field) {
- return cfa.getLiveFieldsAndMethods().contains(field);
+ return cfa.getLiveFieldsAndMethods().contains(field)
+ || cfa.getFieldsWritten().contains(field);
}
public boolean isLive(JMethod method) {
diff --git a/user/test/com/google/gwt/dev/jjs/CompilerSuite.java b/user/test/com/google/gwt/dev/jjs/CompilerSuite.java
index b079d19..fea18ff 100644
--- a/user/test/com/google/gwt/dev/jjs/CompilerSuite.java
+++ b/user/test/com/google/gwt/dev/jjs/CompilerSuite.java
@@ -39,6 +39,7 @@
import com.google.gwt.dev.jjs.test.MiscellaneousTest;
import com.google.gwt.dev.jjs.test.NativeLongTest;
import com.google.gwt.dev.jjs.test.ObjectIdentityTest;
+import com.google.gwt.dev.jjs.test.RunAsyncTest;
import com.google.gwt.dev.jjs.test.UnstableGeneratorTest;
import com.google.gwt.dev.jjs.test.VarargsTest;
import com.google.gwt.junit.tools.GWTTestSuite;
@@ -78,6 +79,7 @@
suite.addTestSuite(MiscellaneousTest.class);
suite.addTestSuite(NativeLongTest.class);
suite.addTestSuite(ObjectIdentityTest.class);
+ suite.addTestSuite(RunAsyncTest.class);
suite.addTestSuite(UnstableGeneratorTest.class);
suite.addTestSuite(VarargsTest.class);
// $JUnit-END$
diff --git a/user/test/com/google/gwt/dev/jjs/test/RunAsyncTest.java b/user/test/com/google/gwt/dev/jjs/test/RunAsyncTest.java
new file mode 100644
index 0000000..8039a5b
--- /dev/null
+++ b/user/test/com/google/gwt/dev/jjs/test/RunAsyncTest.java
@@ -0,0 +1,75 @@
+/*
+ * Copyright 2008 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.jjs.test;
+
+import com.google.gwt.core.client.GWT;
+import com.google.gwt.core.client.RunAsyncCallback;
+import com.google.gwt.junit.client.GWTTestCase;
+import com.google.gwt.user.client.Window;
+
+/**
+ * Tests runAsync in various ways.
+ */
+public class RunAsyncTest extends GWTTestCase {
+ private static final int RUNASYNC_TIMEOUT = 10000;
+
+ @Override
+ public String getModuleName() {
+ return "com.google.gwt.dev.jjs.CompilerSuite";
+ }
+
+ private static final String HELLO = "hello";
+
+ private static String staticWrittenInBaseButReadLater;
+
+ public void testBasic() {
+ delayTestFinish(RUNASYNC_TIMEOUT);
+
+ GWT.runAsync(new RunAsyncCallback() {
+ public void onFailure(Throwable caught) {
+ throw new RuntimeException(caught);
+ }
+
+ public void onSuccess() {
+ finishTest();
+ }
+ });
+ }
+
+ /**
+ * Unlike with pruning, writing to a field should rescue it for code-splitting
+ * purposes.
+ */
+ public void testFieldWrittenButNotRead() {
+ delayTestFinish(RUNASYNC_TIMEOUT);
+
+ // This write happens in the base fragment
+ staticWrittenInBaseButReadLater = HELLO;
+
+ GWT.runAsync(new RunAsyncCallback() {
+ public void onFailure(Throwable caught) {
+ throw new RuntimeException(caught);
+ }
+
+ public void onSuccess() {
+ // This read happens later
+ assertEquals(HELLO, staticWrittenInBaseButReadLater);
+ finishTest();
+ }
+ });
+ }
+
+}