+ */
+public interface ClassResource extends Resource {
+}
diff --git a/teavm-platform/src/main/java/org/teavm/platform/metadata/ClassScopedMetadataGenerator.java b/teavm-platform/src/main/java/org/teavm/platform/metadata/ClassScopedMetadataGenerator.java
new file mode 100644
index 000000000..a884bfac2
--- /dev/null
+++ b/teavm-platform/src/main/java/org/teavm/platform/metadata/ClassScopedMetadataGenerator.java
@@ -0,0 +1,35 @@
+/*
+ * Copyright 2015 Alexey Andreev.
+ *
+ * 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 org.teavm.platform.metadata;
+
+import java.util.Map;
+import org.teavm.model.MethodReference;
+import org.teavm.platform.PlatformClass;
+
+/**
+ * Behaviour of this class is similar to {@link MetadataGenerator}. The difference is that method, marked with
+ * {@link ClassScopedMetadataProvider} must take one argument of type {@link PlatformClass}. It will
+ * return different resource for each given class, corresponding to map entries, produced by
+ * {@link #generateMetadata(MetadataGeneratorContext, MethodReference)}.
+ *
+ * @see ClassScopedMetadataProvider
+ * @see MetadataGenerator
+ *
+ * @author Alexey Andreev
+ */
+public interface ClassScopedMetadataGenerator {
+ Map generateMetadata(MetadataGeneratorContext context, MethodReference method);
+}
diff --git a/teavm-platform/src/main/java/org/teavm/platform/metadata/ClassScopedMetadataProvider.java b/teavm-platform/src/main/java/org/teavm/platform/metadata/ClassScopedMetadataProvider.java
new file mode 100644
index 000000000..617cc36fc
--- /dev/null
+++ b/teavm-platform/src/main/java/org/teavm/platform/metadata/ClassScopedMetadataProvider.java
@@ -0,0 +1,34 @@
+/*
+ * Copyright 2015 Alexey Andreev.
+ *
+ * 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 org.teavm.platform.metadata;
+
+import java.lang.annotation.ElementType;
+import java.lang.annotation.Retention;
+import java.lang.annotation.RetentionPolicy;
+import java.lang.annotation.Target;
+
+/**
+ * Binds a {@link ClassScopedMetadataGenerator} to a native method.
+ *
+ * @see MetadataProvider
+ *
+ * @author Alexey Andreev
+ */
+@Retention(RetentionPolicy.RUNTIME)
+@Target(ElementType.METHOD)
+public @interface ClassScopedMetadataProvider {
+ Class extends ClassScopedMetadataGenerator> value();
+}
diff --git a/teavm-platform/src/main/java/org/teavm/platform/metadata/MetadataGenerator.java b/teavm-platform/src/main/java/org/teavm/platform/metadata/MetadataGenerator.java
index 92c0ce73a..5ca4546d6 100644
--- a/teavm-platform/src/main/java/org/teavm/platform/metadata/MetadataGenerator.java
+++ b/teavm-platform/src/main/java/org/teavm/platform/metadata/MetadataGenerator.java
@@ -50,6 +50,8 @@ import org.teavm.model.MethodReference;
*
* All other types are not considered to be resources and therefore are not accepted.
*
+ * @see ClassScopedMetadataGenerator
+ *
* @author Alexey Andreev
*/
public interface MetadataGenerator {
diff --git a/teavm-platform/src/main/java/org/teavm/platform/metadata/MetadataGeneratorContext.java b/teavm-platform/src/main/java/org/teavm/platform/metadata/MetadataGeneratorContext.java
index 067b4fe26..e9644c9d2 100644
--- a/teavm-platform/src/main/java/org/teavm/platform/metadata/MetadataGeneratorContext.java
+++ b/teavm-platform/src/main/java/org/teavm/platform/metadata/MetadataGeneratorContext.java
@@ -17,7 +17,9 @@ package org.teavm.platform.metadata;
import java.util.Properties;
import org.teavm.common.ServiceRepository;
+import org.teavm.model.FieldReference;
import org.teavm.model.ListableClassReaderSource;
+import org.teavm.platform.Platform;
import org.teavm.vm.TeaVM;
/**
@@ -48,6 +50,18 @@ public interface MetadataGeneratorContext extends ServiceRepository {
*/
T createResource(Class resourceType);
+ /**
+ * Creates a new resource that represents class literal. Client code then may use
+ * {@link Platform#classFromResource(ClassResource)} to get actual class.
+ */
+ ClassResource createClassResource(String className);
+
+ /**
+ * Creates a new resource that represents static field. Client code then may use
+ * {@link Platform#objectFromResource(StaticFieldResource)} to get actual field value.
+ */
+ StaticFieldResource createFieldResource(FieldReference field);
+
/**
* Creates a new resource array.
*/
diff --git a/teavm-platform/src/main/java/org/teavm/platform/metadata/StaticFieldResource.java b/teavm-platform/src/main/java/org/teavm/platform/metadata/StaticFieldResource.java
new file mode 100644
index 000000000..5bfdcead5
--- /dev/null
+++ b/teavm-platform/src/main/java/org/teavm/platform/metadata/StaticFieldResource.java
@@ -0,0 +1,23 @@
+/*
+ * Copyright 2015 Alexey Andreev.
+ *
+ * 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 org.teavm.platform.metadata;
+
+/**
+ *
+ * @author Alexey Andreev
+ */
+public interface StaticFieldResource extends Resource {
+}
diff --git a/teavm-platform/src/main/java/org/teavm/platform/plugin/AsyncMethodGenerator.java b/teavm-platform/src/main/java/org/teavm/platform/plugin/AsyncMethodGenerator.java
new file mode 100644
index 000000000..d81151af1
--- /dev/null
+++ b/teavm-platform/src/main/java/org/teavm/platform/plugin/AsyncMethodGenerator.java
@@ -0,0 +1,85 @@
+/*
+ * Copyright 2015 Alexey Andreev.
+ *
+ * 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 org.teavm.platform.plugin;
+
+import java.io.IOException;
+import org.teavm.codegen.SourceWriter;
+import org.teavm.dependency.DependencyAgent;
+import org.teavm.dependency.DependencyPlugin;
+import org.teavm.dependency.MethodDependency;
+import org.teavm.javascript.spi.Generator;
+import org.teavm.javascript.spi.GeneratorContext;
+import org.teavm.model.*;
+import org.teavm.platform.async.AsyncCallback;
+
+/**
+ *
+ * @author Alexey Andreev
+ */
+public class AsyncMethodGenerator implements Generator, DependencyPlugin {
+ private static final MethodReference completeMethod = new MethodReference(AsyncCallback.class, "complete",
+ Object.class, void.class);
+ private static final MethodReference errorMethod = new MethodReference(AsyncCallback.class, "error",
+ Throwable.class, void.class);
+
+ @Override
+ public void generate(GeneratorContext context, SourceWriter writer, MethodReference methodRef) throws IOException {
+ MethodReference asyncRef = getAsyncReference(methodRef);
+ writer.append("var callback").ws().append("=").ws().append("function()").ws().append("{};").softNewLine();
+ writer.append("callback.").appendMethod(completeMethod).ws().append("=").ws().append("function(val)").ws()
+ .append("{").indent().softNewLine();
+ writer.append("return $return($rt_asyncResult(val));").softNewLine();
+ writer.outdent().append("};").softNewLine();
+ writer.append("callback.").appendMethod(errorMethod).ws().append("=").ws().append("function(e)").ws()
+ .append("{").indent().softNewLine();
+ writer.append("return $return($rt_asyncError(e));").softNewLine();
+ writer.outdent().append("};").softNewLine();
+ writer.append("try").ws().append("{").indent().softNewLine();
+ writer.append("return ").appendMethodBody(asyncRef).append('(');
+ ClassReader cls = context.getClassSource().get(methodRef.getClassName());
+ MethodReader method = cls.getMethod(methodRef.getDescriptor());
+ int start = method.hasModifier(ElementModifier.STATIC) ? 1 : 0;
+ for (int i = start; i <= methodRef.parameterCount(); ++i) {
+ writer.append(context.getParameterName(i));
+ writer.append(',').ws();
+ }
+ writer.append("callback);").softNewLine();
+ writer.outdent().append("}").ws().append("catch($e)").ws().append("{").indent().softNewLine();
+ writer.append("return $return($rt_asyncError($e));").softNewLine();
+ writer.outdent().append("}").softNewLine();
+ }
+
+ private MethodReference getAsyncReference(MethodReference methodRef) {
+ ValueType[] signature = new ValueType[methodRef.parameterCount() + 2];
+ for (int i = 0; i < methodRef.parameterCount(); ++i) {
+ signature[i] = methodRef.getDescriptor().parameterType(i);
+ }
+ signature[methodRef.parameterCount()] = ValueType.parse(AsyncCallback.class);
+ signature[methodRef.parameterCount() + 1] = ValueType.VOID;
+ return new MethodReference(methodRef.getClassName(), methodRef.getName(), signature);
+ }
+
+ @Override
+ public void methodAchieved(DependencyAgent checker, MethodDependency method, CallLocation location) {
+ MethodReference asyncRef = getAsyncReference(method.getReference());
+ MethodDependency asyncMethod = checker.linkMethod(asyncRef, location);
+ int paramCount = method.getReference().parameterCount();
+ for (int i = 0; i <= paramCount; ++i) {
+ method.getVariable(i).connect(asyncMethod.getVariable(i));
+ }
+ asyncMethod.use();
+ }
+}
diff --git a/teavm-platform/src/main/java/org/teavm/platform/plugin/AsyncMethodProcessor.java b/teavm-platform/src/main/java/org/teavm/platform/plugin/AsyncMethodProcessor.java
new file mode 100644
index 000000000..1a8aa9e19
--- /dev/null
+++ b/teavm-platform/src/main/java/org/teavm/platform/plugin/AsyncMethodProcessor.java
@@ -0,0 +1,67 @@
+/*
+ * Copyright 2015 Alexey Andreev.
+ *
+ * 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 org.teavm.platform.plugin;
+
+import org.teavm.dependency.PluggableDependency;
+import org.teavm.diagnostics.Diagnostics;
+import org.teavm.javascript.spi.Async;
+import org.teavm.javascript.spi.GeneratedBy;
+import org.teavm.javascript.spi.Sync;
+import org.teavm.model.*;
+import org.teavm.platform.async.AsyncCallback;
+
+/**
+ *
+ * @author Alexey Andreev
+ */
+public class AsyncMethodProcessor implements ClassHolderTransformer {
+ @Override
+ public void transformClass(ClassHolder cls, ClassReaderSource innerSource, Diagnostics diagnostics) {
+ for (MethodHolder method : cls.getMethods()) {
+ if (method.hasModifier(ElementModifier.NATIVE) &&
+ method.getAnnotations().get(Async.class.getName()) != null &&
+ method.getAnnotations().get(GeneratedBy.class.getName()) == null) {
+ ValueType[] signature = new ValueType[method.parameterCount() + 2];
+ for (int i = 0; i < method.parameterCount(); ++i) {
+ signature[i] = method.parameterType(i);
+ }
+ signature[method.parameterCount()] = ValueType.parse(AsyncCallback.class);
+ signature[method.parameterCount() + 1] = ValueType.VOID;
+ MethodDescriptor asyncDesc = new MethodDescriptor(method.getName(), signature);
+ MethodHolder asyncMethod = cls.getMethod(asyncDesc);
+ if (asyncMethod != null) {
+ if (asyncMethod.hasModifier(ElementModifier.STATIC) !=
+ method.hasModifier(ElementModifier.STATIC)) {
+ diagnostics.error(new CallLocation(method.getReference()), "Methods {{m0}} and {{m1}} must " +
+ "both be either static or non-static",
+ method.getReference(), asyncMethod.getReference());
+ }
+ AnnotationHolder annot = new AnnotationHolder(GeneratedBy.class.getName());
+ annot.getValues().put("value", new AnnotationValue(ValueType.parse(AsyncMethodGenerator.class)));
+ method.getAnnotations().add(annot);
+ annot = new AnnotationHolder(PluggableDependency.class.getName());
+ annot.getValues().put("value", new AnnotationValue(ValueType.parse(AsyncMethodGenerator.class)));
+ method.getAnnotations().add(annot);
+ }
+ } else if (method.getName().equals("")) {
+ if (method.getAnnotations().get(Sync.class.getName()) == null) {
+ AnnotationHolder annot = new AnnotationHolder(Sync.class.getName());
+ method.getAnnotations().add(annot);
+ }
+ }
+ }
+ }
+}
diff --git a/teavm-platform/src/main/java/org/teavm/platform/plugin/BuildTimeClassResource.java b/teavm-platform/src/main/java/org/teavm/platform/plugin/BuildTimeClassResource.java
new file mode 100644
index 000000000..2e9ac00dd
--- /dev/null
+++ b/teavm-platform/src/main/java/org/teavm/platform/plugin/BuildTimeClassResource.java
@@ -0,0 +1,41 @@
+/*
+ * Copyright 2015 Alexey Andreev.
+ *
+ * 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 org.teavm.platform.plugin;
+
+import java.io.IOException;
+import org.teavm.codegen.SourceWriter;
+import org.teavm.platform.metadata.ClassResource;
+
+/**
+ *
+ * @author Alexey Andreev
+ */
+class BuildTimeClassResource implements ClassResource, ResourceWriter {
+ private String className;
+
+ public BuildTimeClassResource(String className) {
+ this.className = className;
+ }
+
+ public String getClassName() {
+ return className;
+ }
+
+ @Override
+ public void write(SourceWriter writer) throws IOException {
+ writer.appendClass(className);
+ }
+}
diff --git a/teavm-platform/src/main/java/org/teavm/platform/plugin/BuildTimeResourceProxyBuilder.java b/teavm-platform/src/main/java/org/teavm/platform/plugin/BuildTimeResourceProxyBuilder.java
index a936fee0f..a1cda00fd 100644
--- a/teavm-platform/src/main/java/org/teavm/platform/plugin/BuildTimeResourceProxyBuilder.java
+++ b/teavm-platform/src/main/java/org/teavm/platform/plugin/BuildTimeResourceProxyBuilder.java
@@ -102,8 +102,8 @@ class BuildTimeResourceProxyBuilder {
private void scanIface(Class> iface) {
if (!Resource.class.isAssignableFrom(iface)) {
- throw new IllegalArgumentException("Error creating a new resource of type " + iface.getName() +
- ". This type does not implement the " + Resource.class.getName() + " interface");
+ throw new IllegalArgumentException("Error creating a new resource of type " + iface.getName() + "." +
+ " This type does not implement the " + Resource.class.getName() + " interface");
}
// Scan methods
diff --git a/teavm-platform/src/main/java/org/teavm/platform/plugin/BuildTimeResourceWriterMethod.java b/teavm-platform/src/main/java/org/teavm/platform/plugin/BuildTimeResourceWriterMethod.java
index dead2901c..b24c0547d 100644
--- a/teavm-platform/src/main/java/org/teavm/platform/plugin/BuildTimeResourceWriterMethod.java
+++ b/teavm-platform/src/main/java/org/teavm/platform/plugin/BuildTimeResourceWriterMethod.java
@@ -36,7 +36,7 @@ class BuildTimeResourceWriterMethod implements BuildTimeResourceMethod {
if (i > 0) {
writer.append(',').ws();
}
- ResourceWriterHelper.writeString(writer, propertyNames[i]);
+ ResourceWriterHelper.writeIdentifier(writer, propertyNames[i]);
writer.ws().append(':').ws();
ResourceWriterHelper.write(writer, proxy.data[i]);
}
diff --git a/teavm-platform/src/main/java/org/teavm/platform/plugin/BuildTimeStaticFieldResource.java b/teavm-platform/src/main/java/org/teavm/platform/plugin/BuildTimeStaticFieldResource.java
new file mode 100644
index 000000000..fa848b04c
--- /dev/null
+++ b/teavm-platform/src/main/java/org/teavm/platform/plugin/BuildTimeStaticFieldResource.java
@@ -0,0 +1,42 @@
+/*
+ * Copyright 2015 Alexey Andreev.
+ *
+ * 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 org.teavm.platform.plugin;
+
+import java.io.IOException;
+import org.teavm.codegen.SourceWriter;
+import org.teavm.model.FieldReference;
+import org.teavm.platform.metadata.StaticFieldResource;
+
+/**
+ *
+ * @author Alexey Andreev
+ */
+class BuildTimeStaticFieldResource implements StaticFieldResource, ResourceWriter {
+ private FieldReference field;
+
+ public BuildTimeStaticFieldResource(FieldReference field) {
+ this.field = field;
+ }
+
+ public FieldReference getField() {
+ return field;
+ }
+
+ @Override
+ public void write(SourceWriter writer) throws IOException {
+ writer.appendField(field);
+ }
+}
diff --git a/teavm-classlib/src/main/java/org/teavm/classlib/impl/ClassLookupDependencySupport.java b/teavm-platform/src/main/java/org/teavm/platform/plugin/ClassLookupDependencySupport.java
similarity index 90%
rename from teavm-classlib/src/main/java/org/teavm/classlib/impl/ClassLookupDependencySupport.java
rename to teavm-platform/src/main/java/org/teavm/platform/plugin/ClassLookupDependencySupport.java
index 4d4238300..a57d59b02 100644
--- a/teavm-classlib/src/main/java/org/teavm/classlib/impl/ClassLookupDependencySupport.java
+++ b/teavm-platform/src/main/java/org/teavm/platform/plugin/ClassLookupDependencySupport.java
@@ -13,10 +13,11 @@
* See the License for the specific language governing permissions and
* limitations under the License.
*/
-package org.teavm.classlib.impl;
+package org.teavm.platform.plugin;
import org.teavm.dependency.*;
import org.teavm.model.*;
+import org.teavm.platform.Platform;
/**
*
@@ -38,14 +39,14 @@ public class ClassLookupDependencySupport implements DependencyListener {
@Override
public void methodAchieved(final DependencyAgent agent, MethodDependency method, final CallLocation location) {
MethodReference ref = method.getReference();
- if (ref.getClassName().equals("java.lang.Class") && ref.getName().equals("forNameImpl")) {
+ if (ref.getClassName().equals(Platform.class.getName()) && ref.getName().equals("lookupClass")) {
allClasses.addConsumer(new DependencyConsumer() {
@Override public void consume(DependencyAgentType type) {
ClassReader cls = agent.getClassSource().get(type.getName());
if (cls == null) {
return;
}
- MethodReader initMethod = cls.getMethod(new MethodDescriptor("", ValueType.VOID));
+ MethodReader initMethod = cls.getMethod(new MethodDescriptor("", void.class));
if (initMethod != null) {
agent.linkMethod(initMethod.getReference(), location).use();
}
diff --git a/teavm-platform/src/main/java/org/teavm/platform/plugin/ClassScopedMetadataProviderNativeGenerator.java b/teavm-platform/src/main/java/org/teavm/platform/plugin/ClassScopedMetadataProviderNativeGenerator.java
new file mode 100644
index 000000000..eef273ead
--- /dev/null
+++ b/teavm-platform/src/main/java/org/teavm/platform/plugin/ClassScopedMetadataProviderNativeGenerator.java
@@ -0,0 +1,97 @@
+/*
+ * Copyright 2015 Alexey Andreev.
+ *
+ * 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 org.teavm.platform.plugin;
+
+import java.io.IOException;
+import java.lang.reflect.Constructor;
+import java.lang.reflect.InvocationTargetException;
+import java.util.Map;
+import org.teavm.codegen.SourceWriter;
+import org.teavm.javascript.Renderer;
+import org.teavm.javascript.spi.Generator;
+import org.teavm.javascript.spi.GeneratorContext;
+import org.teavm.model.*;
+import org.teavm.platform.metadata.ClassScopedMetadataGenerator;
+import org.teavm.platform.metadata.ClassScopedMetadataProvider;
+import org.teavm.platform.metadata.Resource;
+
+/**
+ *
+ * @author Alexey Andreev
+ */
+public class ClassScopedMetadataProviderNativeGenerator implements Generator {
+ @Override
+ public void generate(GeneratorContext context, SourceWriter writer, MethodReference methodRef) throws IOException {
+ // Validate method
+ ClassReader cls = context.getClassSource().get(methodRef.getClassName());
+ MethodReader method = cls.getMethod(methodRef.getDescriptor());
+ AnnotationReader providerAnnot = method.getAnnotations().get(ClassScopedMetadataProvider.class.getName());
+ if (providerAnnot == null) {
+ return;
+ }
+ if (!method.hasModifier(ElementModifier.NATIVE)) {
+ context.getDiagnostics().error(new CallLocation(methodRef), "Method {{m0}} is marked with " +
+ "{{c1}} annotation, but it is not native", methodRef, ClassScopedMetadataProvider.class.getName());
+ return;
+ }
+
+ // Find and instantiate metadata generator
+ ValueType generatorType = providerAnnot.getValue("value").getJavaClass();
+ String generatorClassName = ((ValueType.Object)generatorType).getClassName();
+ Class> generatorClass;
+ try {
+ generatorClass = Class.forName(generatorClassName, true, context.getClassLoader());
+ } catch (ClassNotFoundException e) {
+ context.getDiagnostics().error(new CallLocation(methodRef), "Can't find metadata provider class {{c0}}",
+ generatorClassName);
+ return;
+ }
+ Constructor> cons;
+ try {
+ cons = generatorClass.getConstructor();
+ } catch (NoSuchMethodException e) {
+ context.getDiagnostics().error(new CallLocation(methodRef), "Metadata generator {{c0}} does not have " +
+ "a public no-arg constructor", generatorClassName);
+ return;
+ }
+ ClassScopedMetadataGenerator generator;
+ try {
+ generator = (ClassScopedMetadataGenerator)cons.newInstance();
+ } catch (IllegalAccessException | InstantiationException | InvocationTargetException e) {
+ context.getDiagnostics().error(new CallLocation(methodRef), "Error instantiating metadata " +
+ "generator {{c0}}", generatorClassName);
+ return;
+ }
+ DefaultMetadataGeneratorContext metadataContext = new DefaultMetadataGeneratorContext(context.getClassSource(),
+ context.getClassLoader(), context.getProperties(), context);
+
+ Map resourceMap = generator.generateMetadata(metadataContext, methodRef);
+ writer.append("var p").ws().append("=").ws().append("\"" + Renderer.escapeString("$$res_" +
+ writer.getNaming().getNameFor(methodRef)) + "\"").append(";").softNewLine();
+ for (Map.Entry entry : resourceMap.entrySet()) {
+ writer.appendClass(entry.getKey()).append("[p]").ws().append("=").ws();
+ ResourceWriterHelper.write(writer, entry.getValue());
+ writer.append(";").softNewLine();
+ }
+ writer.appendMethodBody(methodRef).ws().append('=').ws().append("function(cls)").ws().append("{")
+ .softNewLine().indent();
+ writer.append("return cls.hasOwnProperty(p)").ws().append("?").ws().append("cls[p]").ws().append(":")
+ .ws().append("null;").softNewLine();
+ writer.outdent().append("}").softNewLine();
+ writer.append("return ").appendMethodBody(methodRef).append("(").append(context.getParameterName(1))
+ .append(");").softNewLine();
+ }
+}
diff --git a/teavm-platform/src/main/java/org/teavm/platform/plugin/DefaultMetadataGeneratorContext.java b/teavm-platform/src/main/java/org/teavm/platform/plugin/DefaultMetadataGeneratorContext.java
index 947202715..35a1b08eb 100644
--- a/teavm-platform/src/main/java/org/teavm/platform/plugin/DefaultMetadataGeneratorContext.java
+++ b/teavm-platform/src/main/java/org/teavm/platform/plugin/DefaultMetadataGeneratorContext.java
@@ -18,11 +18,9 @@ package org.teavm.platform.plugin;
import java.lang.reflect.Proxy;
import java.util.Properties;
import org.teavm.common.ServiceRepository;
+import org.teavm.model.FieldReference;
import org.teavm.model.ListableClassReaderSource;
-import org.teavm.platform.metadata.MetadataGeneratorContext;
-import org.teavm.platform.metadata.Resource;
-import org.teavm.platform.metadata.ResourceArray;
-import org.teavm.platform.metadata.ResourceMap;
+import org.teavm.platform.metadata.*;
/**
*
@@ -70,6 +68,16 @@ class DefaultMetadataGeneratorContext implements MetadataGeneratorContext {
return new BuildTimeResourceArray<>();
}
+ @Override
+ public ClassResource createClassResource(String className) {
+ return new BuildTimeClassResource(className);
+ }
+
+ @Override
+ public StaticFieldResource createFieldResource(FieldReference field) {
+ return new BuildTimeStaticFieldResource(field);
+ }
+
@Override
public ResourceMap createResourceMap() {
return new BuildTimeResourceMap<>();
diff --git a/teavm-classlib/src/main/java/org/teavm/classlib/impl/EnumDependencySupport.java b/teavm-platform/src/main/java/org/teavm/platform/plugin/EnumDependencySupport.java
similarity index 64%
rename from teavm-classlib/src/main/java/org/teavm/classlib/impl/EnumDependencySupport.java
rename to teavm-platform/src/main/java/org/teavm/platform/plugin/EnumDependencySupport.java
index 190defd64..96620653b 100644
--- a/teavm-classlib/src/main/java/org/teavm/classlib/impl/EnumDependencySupport.java
+++ b/teavm-platform/src/main/java/org/teavm/platform/plugin/EnumDependencySupport.java
@@ -13,14 +13,11 @@
* See the License for the specific language governing permissions and
* limitations under the License.
*/
-package org.teavm.classlib.impl;
+package org.teavm.platform.plugin;
import org.teavm.dependency.*;
-import org.teavm.model.CallLocation;
-import org.teavm.model.ClassReader;
-import org.teavm.model.MethodDescriptor;
-import org.teavm.model.MethodReader;
-import org.teavm.model.ValueType;
+import org.teavm.model.*;
+import org.teavm.platform.Platform;
/**
*
@@ -28,7 +25,6 @@ import org.teavm.model.ValueType;
*/
public class EnumDependencySupport implements DependencyListener {
private DependencyNode allEnums;
- private boolean unlocked;
@Override
public void started(DependencyAgent agent) {
@@ -42,21 +38,25 @@ public class EnumDependencySupport implements DependencyListener {
return;
}
allEnums.propagate(agent.getType(className));
- if (unlocked) {
- MethodReader method = cls.getMethod(new MethodDescriptor("values",
- ValueType.arrayOf(ValueType.object(cls.getName()))));
- if (method != null) {
- agent.linkMethod(method.getReference(), location).use();
- }
- }
+
}
@Override
- public void methodAchieved(DependencyAgent agent, MethodDependency method, CallLocation location) {
- if (method.getReference().getClassName().equals("java.lang.Class") &&
- method.getReference().getName().equals("getEnumConstantsImpl")) {
- unlocked = true;
+ public void methodAchieved(final DependencyAgent agent, MethodDependency method, CallLocation location) {
+ if (method.getReference().getClassName().equals(Platform.class.getName()) &&
+ method.getReference().getName().equals("getEnumConstants")) {
allEnums.connect(method.getResult().getArrayItem());
+ final MethodReference ref = method.getReference();
+ allEnums.addConsumer(new DependencyConsumer() {
+ @Override public void consume(DependencyAgentType type) {
+ ClassReader cls = agent.getClassSource().get(type.getName());
+ MethodReader method = cls.getMethod(new MethodDescriptor("values",
+ ValueType.arrayOf(ValueType.object(cls.getName()))));
+ if (method != null) {
+ agent.linkMethod(method.getReference(), new CallLocation(ref)).use();
+ }
+ }
+ });
method.getResult().propagate(agent.getType("[java.lang.Enum"));
for (String cls : agent.getAchievableClasses()) {
classAchieved(agent, cls, location);
diff --git a/teavm-platform/src/main/java/org/teavm/platform/plugin/MetadataProviderNativeGenerator.java b/teavm-platform/src/main/java/org/teavm/platform/plugin/MetadataProviderNativeGenerator.java
index 6ddda44ca..3d44de603 100644
--- a/teavm-platform/src/main/java/org/teavm/platform/plugin/MetadataProviderNativeGenerator.java
+++ b/teavm-platform/src/main/java/org/teavm/platform/plugin/MetadataProviderNativeGenerator.java
@@ -19,8 +19,8 @@ import java.io.IOException;
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;
import org.teavm.codegen.SourceWriter;
-import org.teavm.javascript.ni.Generator;
-import org.teavm.javascript.ni.GeneratorContext;
+import org.teavm.javascript.spi.Generator;
+import org.teavm.javascript.spi.GeneratorContext;
import org.teavm.model.*;
import org.teavm.platform.metadata.MetadataGenerator;
import org.teavm.platform.metadata.MetadataProvider;
@@ -41,8 +41,9 @@ public class MetadataProviderNativeGenerator implements Generator {
return;
}
if (!method.hasModifier(ElementModifier.NATIVE)) {
- throw new IllegalStateException("Method " + method.getReference() + " was marked with " +
- MetadataProvider.class.getName() + " but it is not native");
+ context.getDiagnostics().error(new CallLocation(methodRef), "Method {{m0}} is marked with " +
+ "{{c1}} annotation, but it is not native", methodRef, MetadataProvider.class.getName());
+ return;
}
// Find and instantiate metadata generator
@@ -52,23 +53,25 @@ public class MetadataProviderNativeGenerator implements Generator {
try {
generatorClass = Class.forName(generatorClassName, true, context.getClassLoader());
} catch (ClassNotFoundException e) {
- throw new RuntimeException("Can't find metadata generator class: " + generatorClassName, e);
+ context.getDiagnostics().error(new CallLocation(methodRef), "Can't find metadata provider class {{c0}}",
+ generatorClassName);
+ return;
}
Constructor> cons;
try {
cons = generatorClass.getConstructor();
} catch (NoSuchMethodException e) {
- throw new RuntimeException("Metadata generator " + generatorClassName + " does not have a public " +
- "no-arg constructor", e);
+ context.getDiagnostics().error(new CallLocation(methodRef), "Metadata generator {{c0}} does not have " +
+ "a public no-arg constructor", generatorClassName);
+ return;
}
MetadataGenerator generator;
try {
generator = (MetadataGenerator)cons.newInstance();
- } catch (IllegalAccessException | InstantiationException e) {
- throw new RuntimeException("Error instantiating metadata generator " + generatorClassName, e);
- } catch (InvocationTargetException e) {
- throw new RuntimeException("Error instantiating metadata generator " + generatorClassName,
- e.getTargetException());
+ } catch (IllegalAccessException | InstantiationException | InvocationTargetException e) {
+ context.getDiagnostics().error(new CallLocation(methodRef), "Error instantiating metadata " +
+ "generator {{c0}}", generatorClassName);
+ return;
}
DefaultMetadataGeneratorContext metadataContext = new DefaultMetadataGeneratorContext(context.getClassSource(),
context.getClassLoader(), context.getProperties(), context);
diff --git a/teavm-platform/src/main/java/org/teavm/platform/plugin/MetadataProviderTransformer.java b/teavm-platform/src/main/java/org/teavm/platform/plugin/MetadataProviderTransformer.java
index 3ad502b9e..0bf498544 100644
--- a/teavm-platform/src/main/java/org/teavm/platform/plugin/MetadataProviderTransformer.java
+++ b/teavm-platform/src/main/java/org/teavm/platform/plugin/MetadataProviderTransformer.java
@@ -16,8 +16,10 @@
package org.teavm.platform.plugin;
import org.teavm.diagnostics.Diagnostics;
-import org.teavm.javascript.ni.GeneratedBy;
+import org.teavm.javascript.spi.GeneratedBy;
import org.teavm.model.*;
+import org.teavm.platform.PlatformClass;
+import org.teavm.platform.metadata.ClassScopedMetadataProvider;
import org.teavm.platform.metadata.MetadataProvider;
/**
@@ -29,13 +31,26 @@ class MetadataProviderTransformer implements ClassHolderTransformer {
public void transformClass(ClassHolder cls, ClassReaderSource innerSource, Diagnostics diagnostics) {
for (MethodHolder method : cls.getMethods()) {
AnnotationReader providerAnnot = method.getAnnotations().get(MetadataProvider.class.getName());
- if (providerAnnot == null) {
- continue;
+ if (providerAnnot != null) {
+ AnnotationHolder genAnnot = new AnnotationHolder(GeneratedBy.class.getName());
+ genAnnot.getValues().put("value", new AnnotationValue(ValueType.object(
+ MetadataProviderNativeGenerator.class.getName())));
+ method.getAnnotations().add(genAnnot);
+ }
+ providerAnnot = method.getAnnotations().get(ClassScopedMetadataProvider.class.getName());
+ if (providerAnnot != null) {
+ ValueType[] params = method.getParameterTypes();
+ if (params.length != 1 && params[0].isObject(PlatformClass.class.getName())) {
+ diagnostics.error(new CallLocation(method.getReference()), "Method {{m0}} marked with {{c1}} " +
+ "must take exactly one parameter of type {{c2}}",
+ method.getReference(), ClassScopedMetadataProvider.class.getName(),
+ PlatformClass.class.getName());
+ }
+ AnnotationHolder genAnnot = new AnnotationHolder(GeneratedBy.class.getName());
+ genAnnot.getValues().put("value", new AnnotationValue(ValueType.object(
+ ClassScopedMetadataProviderNativeGenerator.class.getName())));
+ method.getAnnotations().add(genAnnot);
}
- AnnotationHolder genAnnot = new AnnotationHolder(GeneratedBy.class.getName());
- genAnnot.getValues().put("value", new AnnotationValue(ValueType.object(
- MetadataProviderNativeGenerator.class.getName())));
- method.getAnnotations().add(genAnnot);
}
}
}
diff --git a/teavm-classlib/src/main/java/org/teavm/classlib/impl/NewInstanceDependencySupport.java b/teavm-platform/src/main/java/org/teavm/platform/plugin/NewInstanceDependencySupport.java
similarity index 76%
rename from teavm-classlib/src/main/java/org/teavm/classlib/impl/NewInstanceDependencySupport.java
rename to teavm-platform/src/main/java/org/teavm/platform/plugin/NewInstanceDependencySupport.java
index e94490899..346500701 100644
--- a/teavm-classlib/src/main/java/org/teavm/classlib/impl/NewInstanceDependencySupport.java
+++ b/teavm-platform/src/main/java/org/teavm/platform/plugin/NewInstanceDependencySupport.java
@@ -13,10 +13,11 @@
* See the License for the specific language governing permissions and
* limitations under the License.
*/
-package org.teavm.classlib.impl;
+package org.teavm.platform.plugin;
import org.teavm.dependency.*;
import org.teavm.model.*;
+import org.teavm.platform.Platform;
/**
*
@@ -39,28 +40,31 @@ public class NewInstanceDependencySupport implements DependencyListener {
if (cls.hasModifier(ElementModifier.ABSTRACT) || cls.hasModifier(ElementModifier.INTERFACE)) {
return;
}
- MethodReader method = cls.getMethod(new MethodDescriptor("", ValueType.VOID));
+ MethodReader method = cls.getMethod(new MethodDescriptor("", void.class));
if (method != null) {
allClassesNode.propagate(agent.getType(className));
}
}
@Override
- public void methodAchieved(final DependencyAgent agent, MethodDependency method, final CallLocation location) {
+ public void methodAchieved(final DependencyAgent agent, MethodDependency method, CallLocation location) {
MethodReader reader = method.getMethod();
- if (reader.getOwnerName().equals("java.lang.Class") && reader.getName().equals("newInstance")) {
+ if (reader.getOwnerName().equals(Platform.class.getName()) && reader.getName().equals("newInstance")) {
allClassesNode.connect(method.getResult());
+ final MethodReference methodRef = reader.getReference();
method.getResult().addConsumer(new DependencyConsumer() {
@Override public void consume(DependencyAgentType type) {
- attachConstructor(agent, type.getName(), location);
+ attachConstructor(agent, type.getName(), new CallLocation(methodRef));
}
});
}
}
private void attachConstructor(DependencyAgent checker, String type, CallLocation location) {
- MethodReference ref = new MethodReference(type, new MethodDescriptor("", ValueType.VOID));
- checker.linkMethod(ref, location).use();
+ MethodReference ref = new MethodReference(type, "", ValueType.VOID);
+ MethodDependency methodDep = checker.linkMethod(ref, location);
+ methodDep.getVariable(0).propagate(checker.getType(type));
+ methodDep.use();
}
@Override
diff --git a/teavm-platform/src/main/java/org/teavm/platform/plugin/PlatformDependencyListener.java b/teavm-platform/src/main/java/org/teavm/platform/plugin/PlatformDependencyListener.java
new file mode 100644
index 000000000..9ff5ad491
--- /dev/null
+++ b/teavm-platform/src/main/java/org/teavm/platform/plugin/PlatformDependencyListener.java
@@ -0,0 +1,54 @@
+/*
+ * Copyright 2015 Alexey Andreev.
+ *
+ * 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 org.teavm.platform.plugin;
+
+import org.teavm.dependency.*;
+import org.teavm.model.CallLocation;
+import org.teavm.platform.Platform;
+
+/**
+ *
+ * @author Alexey Andreev
+ */
+public class PlatformDependencyListener implements DependencyListener {
+ private DependencyNode allClasses;
+
+ @Override
+ public void started(DependencyAgent agent) {
+ allClasses = agent.createNode();
+ }
+
+ @Override
+ public void classAchieved(DependencyAgent agent, String className, CallLocation location) {
+ allClasses.propagate(agent.getType(className));
+ }
+
+ @Override
+ public void methodAchieved(final DependencyAgent agent, MethodDependency method, CallLocation location) {
+ if (!method.getReference().getClassName().equals(Platform.class.getName())) {
+ return;
+ }
+ switch (method.getReference().getName()) {
+ case "objectFromResource":
+ allClasses.connect(method.getResult());
+ break;
+ }
+ }
+
+ @Override
+ public void fieldAchieved(DependencyAgent agent, FieldDependency field, CallLocation location) {
+ }
+}
diff --git a/teavm-platform/src/main/java/org/teavm/platform/plugin/PlatformGenerator.java b/teavm-platform/src/main/java/org/teavm/platform/plugin/PlatformGenerator.java
new file mode 100644
index 000000000..b5574cdd4
--- /dev/null
+++ b/teavm-platform/src/main/java/org/teavm/platform/plugin/PlatformGenerator.java
@@ -0,0 +1,233 @@
+/*
+ * Copyright 2015 Alexey Andreev.
+ *
+ * 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 org.teavm.platform.plugin;
+
+import java.io.IOException;
+import org.teavm.codegen.SourceWriter;
+import org.teavm.dependency.DependencyAgent;
+import org.teavm.dependency.DependencyPlugin;
+import org.teavm.dependency.MethodDependency;
+import org.teavm.javascript.spi.Generator;
+import org.teavm.javascript.spi.GeneratorContext;
+import org.teavm.javascript.spi.Injector;
+import org.teavm.javascript.spi.InjectorContext;
+import org.teavm.model.*;
+import org.teavm.platform.Platform;
+import org.teavm.platform.PlatformClass;
+import org.teavm.platform.PlatformRunnable;
+
+/**
+ *
+ * @author Alexey Andreev
+ */
+public class PlatformGenerator implements Generator, Injector, DependencyPlugin {
+ @Override
+ public void methodAchieved(DependencyAgent agent, MethodDependency method, CallLocation location) {
+ switch (method.getReference().getName()) {
+ case "asJavaClass":
+ method.getResult().propagate(agent.getType("java.lang.Class"));
+ return;
+ case "clone":
+ method.getVariable(1).connect(method.getResult());
+ break;
+ case "startThread":
+ case "schedule": {
+ MethodDependency launchMethod = agent.linkMethod(new MethodReference(Platform.class,
+ "launchThread", PlatformRunnable.class, void.class), null);
+ method.getVariable(1).connect(launchMethod.getVariable(1));
+ launchMethod.use();
+ break;
+ }
+ }
+ }
+
+ @Override
+ public void generate(InjectorContext context, MethodReference methodRef) throws IOException {
+ switch (methodRef.getName()) {
+ case "asJavaClass":
+ case "classFromResource":
+ case "objectFromResource":
+ context.writeExpr(context.getArgument(0));
+ return;
+ }
+ }
+
+ @Override
+ public void generate(GeneratorContext context, SourceWriter writer, MethodReference methodRef) throws IOException {
+ switch (methodRef.getName()) {
+ case "newInstance":
+ generateNewInstance(context, writer, methodRef);
+ break;
+ case "lookupClass":
+ generateLookup(context, writer);
+ break;
+ case "clone":
+ generateClone(context, writer);
+ break;
+ case "startThread":
+ generateSchedule(context, writer, false);
+ break;
+ case "schedule":
+ generateSchedule(context, writer, true);
+ break;
+ case "getEnumConstants":
+ generateEnumConstants(context, writer);
+ break;
+ }
+ }
+
+ private void generateNewInstance(GeneratorContext context, SourceWriter writer, MethodReference methodRef)
+ throws IOException {
+ writer.append("var c").ws().append("=").ws().append("'$$constructor$$';").softNewLine();
+ if (context.isAsync()) {
+ writer.append("function async(cls, init) {").indent().softNewLine();
+ writer.append("return function($return) {").indent().softNewLine();
+ writer.append("var r = new cls;").softNewLine();
+ writer.append("init(r, $rt_guardAsync(function($restore) {").indent().softNewLine();
+ writer.append("$restore();").softNewLine();
+ writer.append("$return($rt_asyncResult(r))").softNewLine();
+ writer.outdent().append("}));").softNewLine();
+ writer.outdent().append("};").softNewLine();
+ writer.outdent().append("}").softNewLine();
+
+ writer.append("function sync(cls, init) {").indent().softNewLine();
+ writer.append("return function($return) {").indent().softNewLine();
+ writer.append("var r = new cls;").softNewLine();
+ writer.append("try {").indent().softNewLine();
+ writer.append("init(r);").softNewLine();
+ writer.append("$return($rt_asyncResult(r));").softNewLine();
+ writer.outdent().append("} catch (e) {").indent().softNewLine();
+ writer.append("$return($rt_asyncError(e));").softNewLine();
+ writer.outdent().append("}").softNewLine();
+ writer.outdent().append("};").softNewLine();
+ writer.outdent().append("}").softNewLine();
+ }
+ for (String clsName : context.getClassSource().getClassNames()) {
+ ClassReader cls = context.getClassSource().get(clsName);
+ MethodReader method = cls.getMethod(new MethodDescriptor("", void.class));
+ if (method != null) {
+ writer.appendClass(clsName).append("[c]").ws().append("=").ws();
+ if (!context.isAsync()) {
+ writer.append(writer.getNaming().getNameForInit(method.getReference()));
+ } else {
+ String function = context.isAsync(method.getReference()) ? "async" : "sync";
+ String methodName = context.isAsync(method.getReference()) ?
+ writer.getNaming().getFullNameForAsync(method.getReference()) :
+ writer.getNaming().getFullNameFor(method.getReference());
+ writer.append(function).append("(").appendClass(clsName).append(',').ws()
+ .append(methodName).append(")");
+ }
+ writer.append(";").softNewLine();
+ }
+ }
+ String selfName = context.isAsync() ? writer.getNaming().getFullNameForAsync(methodRef) :
+ writer.getNaming().getFullNameFor(methodRef);
+ writer.append(selfName).ws().append("=").ws().append("function(cls");
+ if (context.isAsync()) {
+ writer.append(',').ws().append("$return");
+ }
+ writer.append(")").ws().append("{").softNewLine().indent();
+ writer.append("if").ws().append("(!cls.hasOwnProperty(c))").ws().append("{").indent().softNewLine();
+ if (!context.isAsync()) {
+ writer.append("return null;").softNewLine();
+ } else {
+ writer.append("return $return($rt_asyncResult(null));").softNewLine();
+ }
+ writer.outdent().append("}").softNewLine();
+ if (!context.isAsync()) {
+ writer.append("return cls[c]();").softNewLine();
+ } else {
+ writer.append("return cls[c]($return);").softNewLine();
+ }
+ writer.outdent().append("}").softNewLine();
+
+ writer.append("return ").append(selfName).append("(").append(context.getParameterName(1));
+ if (context.isAsync()) {
+ writer.append(',').ws().append("$return");
+ }
+ writer.append(");").softNewLine();
+ }
+
+ private void generateLookup(GeneratorContext context, SourceWriter writer) throws IOException {
+ String param = context.getParameterName(1);
+ writer.append("switch ($rt_ustr(" + param + ")) {").softNewLine().indent();
+ for (String name : context.getClassSource().getClassNames()) {
+ writer.append("case \"" + name + "\": ").appendClass(name).append(".$clinit(); ")
+ .append("return ").appendClass(name).append(";").softNewLine();
+ }
+ writer.append("default: return null;").softNewLine();
+ writer.outdent().append("}").softNewLine();
+ }
+
+ private void generateClone(GeneratorContext context, SourceWriter writer) throws IOException {
+ String obj = context.getParameterName(1);
+ writer.append("var copy").ws().append("=").ws().append("new ").append(obj).append(".constructor();")
+ .softNewLine();
+ writer.append("for").ws().append("(var field in " + obj + ")").ws().append("{").softNewLine().indent();
+ writer.append("if").ws().append("(!" + obj + ".hasOwnProperty(field))").ws().append("{").softNewLine().indent();
+ writer.append("continue;").softNewLine().outdent().append("}").softNewLine();
+ writer.append("copy[field]").ws().append("=").ws().append(obj).append("[field];")
+ .softNewLine().outdent().append("}").softNewLine();
+ writer.append("return copy;").softNewLine();
+ }
+
+ private void generateSchedule(GeneratorContext context, SourceWriter writer, boolean timeout) throws IOException {
+ MethodReference launchRef = new MethodReference(Platform.class, "launchThread",
+ PlatformRunnable.class, void.class);
+ String runnable = context.getParameterName(1);
+ writer.append("return window.setTimeout(function()").ws().append("{").indent().softNewLine();
+ boolean async = context.isAsyncFamily(launchRef);
+ String methodName = async ? writer.getNaming().getFullNameForAsync(launchRef) :
+ writer.getNaming().getFullNameFor(launchRef);
+ if (async) {
+ writer.append("$rt_rootInvocationAdapter(");
+ }
+ writer.append(methodName);
+ if (async) {
+ writer.append(")");
+ }
+ writer.append("(").append(runnable).append(");")
+ .softNewLine();
+ writer.outdent().append("},").ws().append(timeout ? context.getParameterName(2) : "0")
+ .append(");").softNewLine();
+ }
+
+ private void generateEnumConstants(GeneratorContext context, SourceWriter writer) throws IOException {
+ writer.append("var c").ws().append("=").ws().append("'$$enumConstants$$';").softNewLine();
+ for (String clsName : context.getClassSource().getClassNames()) {
+ ClassReader cls = context.getClassSource().get(clsName);
+ MethodReader method = cls.getMethod(new MethodDescriptor("values",
+ ValueType.arrayOf(ValueType.object(clsName))));
+ if (method != null) {
+ writer.appendClass(clsName).append("[c]").ws().append("=").ws();
+ writer.appendMethodBody(method.getReference());
+ writer.append(";").softNewLine();
+ }
+ }
+
+ String selfName = writer.getNaming().getFullNameFor(new MethodReference(Platform.class, "getEnumConstants",
+ PlatformClass.class, Enum[].class));
+ writer.append(selfName).ws().append("=").ws().append("function(cls)").ws().append("{").softNewLine().indent();
+ writer.append("if").ws().append("(!cls.hasOwnProperty(c))").ws().append("{").indent().softNewLine();
+ writer.append("return null;").softNewLine();
+ writer.outdent().append("}").softNewLine();
+ writer.append("return cls[c]();").softNewLine();
+ writer.outdent().append("}").softNewLine();
+
+ writer.append("return ").append(selfName).append("(").append(context.getParameterName(1))
+ .append(");").softNewLine();
+ }
+}
diff --git a/teavm-platform/src/main/java/org/teavm/platform/plugin/PlatformPlugin.java b/teavm-platform/src/main/java/org/teavm/platform/plugin/PlatformPlugin.java
index ad16bfc25..4eb92c96b 100644
--- a/teavm-platform/src/main/java/org/teavm/platform/plugin/PlatformPlugin.java
+++ b/teavm-platform/src/main/java/org/teavm/platform/plugin/PlatformPlugin.java
@@ -29,5 +29,10 @@ public class PlatformPlugin implements TeaVMPlugin {
host.add(new ResourceTransformer());
host.add(new ResourceAccessorTransformer(host));
host.add(new ResourceAccessorDependencyListener());
+ host.add(new AsyncMethodProcessor());
+ host.add(new NewInstanceDependencySupport());
+ host.add(new ClassLookupDependencySupport());
+ host.add(new EnumDependencySupport());
+ host.add(new PlatformDependencyListener());
}
}
diff --git a/teavm-classlib/src/main/java/org/teavm/classlib/java/lang/StringNativeGenerator.java b/teavm-platform/src/main/java/org/teavm/platform/plugin/PlatformQueueGenerator.java
similarity index 62%
rename from teavm-classlib/src/main/java/org/teavm/classlib/java/lang/StringNativeGenerator.java
rename to teavm-platform/src/main/java/org/teavm/platform/plugin/PlatformQueueGenerator.java
index c2bb09b26..1f358308c 100644
--- a/teavm-classlib/src/main/java/org/teavm/classlib/java/lang/StringNativeGenerator.java
+++ b/teavm-platform/src/main/java/org/teavm/platform/plugin/PlatformQueueGenerator.java
@@ -1,5 +1,5 @@
/*
- * Copyright 2013 Alexey Andreev.
+ * Copyright 2015 Alexey Andreev.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
@@ -13,37 +13,33 @@
* See the License for the specific language governing permissions and
* limitations under the License.
*/
-package org.teavm.classlib.java.lang;
+package org.teavm.platform.plugin;
import java.io.IOException;
import org.teavm.dependency.DependencyAgent;
import org.teavm.dependency.DependencyPlugin;
import org.teavm.dependency.MethodDependency;
-import org.teavm.javascript.ni.Injector;
-import org.teavm.javascript.ni.InjectorContext;
+import org.teavm.javascript.spi.Injector;
+import org.teavm.javascript.spi.InjectorContext;
import org.teavm.model.CallLocation;
import org.teavm.model.MethodReference;
+import org.teavm.platform.PlatformObject;
+import org.teavm.platform.PlatformQueue;
/**
*
- * @author Alexey Andreev
+ * @author Alexey Andreev
*/
-public class StringNativeGenerator implements Injector, DependencyPlugin {
+public class PlatformQueueGenerator implements Injector, DependencyPlugin {
@Override
public void methodAchieved(DependencyAgent agent, MethodDependency method, CallLocation location) {
- switch (method.getReference().getName()) {
- case "wrap":
- method.getVariable(1).connect(method.getResult());
- break;
- }
+ MethodDependency addMethod = agent.linkMethod(new MethodReference(PlatformQueue.class, "wrap",
+ Object.class, PlatformObject.class), null);
+ addMethod.getVariable(1).connect(method.getResult());
}
@Override
public void generate(InjectorContext context, MethodReference methodRef) throws IOException {
- switch (methodRef.getName()) {
- case "wrap":
- context.writeExpr(context.getArgument(0));
- break;
- }
+ context.writeExpr(context.getArgument(0));
}
}
diff --git a/teavm-platform/src/main/java/org/teavm/platform/plugin/ResourceAccessorDependencyListener.java b/teavm-platform/src/main/java/org/teavm/platform/plugin/ResourceAccessorDependencyListener.java
index 0e3462599..20e7da06a 100644
--- a/teavm-platform/src/main/java/org/teavm/platform/plugin/ResourceAccessorDependencyListener.java
+++ b/teavm-platform/src/main/java/org/teavm/platform/plugin/ResourceAccessorDependencyListener.java
@@ -36,6 +36,9 @@ class ResourceAccessorDependencyListener implements DependencyListener {
@Override
public void methodAchieved(DependencyAgent agent, MethodDependency method, CallLocation location) {
+ if (!method.getReference().getClassName().equals(ResourceAccessor.class.getName())) {
+ return;
+ }
switch (method.getReference().getName()) {
case "castToString":
method.getResult().propagate(agent.getType("java.lang.String"));
diff --git a/teavm-platform/src/main/java/org/teavm/platform/plugin/ResourceAccessorGenerator.java b/teavm-platform/src/main/java/org/teavm/platform/plugin/ResourceAccessorGenerator.java
index 62c267ad5..18337035c 100644
--- a/teavm-platform/src/main/java/org/teavm/platform/plugin/ResourceAccessorGenerator.java
+++ b/teavm-platform/src/main/java/org/teavm/platform/plugin/ResourceAccessorGenerator.java
@@ -18,8 +18,8 @@ package org.teavm.platform.plugin;
import java.io.IOException;
import org.teavm.javascript.ast.ConstantExpr;
import org.teavm.javascript.ast.Expr;
-import org.teavm.javascript.ni.Injector;
-import org.teavm.javascript.ni.InjectorContext;
+import org.teavm.javascript.spi.Injector;
+import org.teavm.javascript.spi.InjectorContext;
import org.teavm.model.MethodReference;
import org.teavm.model.ValueType;
diff --git a/teavm-platform/src/main/java/org/teavm/platform/plugin/ResourceAccessorTransformer.java b/teavm-platform/src/main/java/org/teavm/platform/plugin/ResourceAccessorTransformer.java
index e95ef207d..894ce2cdc 100644
--- a/teavm-platform/src/main/java/org/teavm/platform/plugin/ResourceAccessorTransformer.java
+++ b/teavm-platform/src/main/java/org/teavm/platform/plugin/ResourceAccessorTransformer.java
@@ -35,7 +35,9 @@ class ResourceAccessorTransformer implements ClassHolderTransformer {
if (cls.getName().equals(ResourceAccessor.class.getName())) {
ResourceAccessorGenerator generator = new ResourceAccessorGenerator();
for (MethodHolder method : cls.getMethods()) {
- vm.add(method.getReference(), generator);
+ if (!method.getName().equals("")) {
+ vm.add(method.getReference(), generator);
+ }
}
}
}
diff --git a/teavm-platform/src/main/java/org/teavm/platform/plugin/ResourceWriterHelper.java b/teavm-platform/src/main/java/org/teavm/platform/plugin/ResourceWriterHelper.java
index af816217f..715d248ba 100644
--- a/teavm-platform/src/main/java/org/teavm/platform/plugin/ResourceWriterHelper.java
+++ b/teavm-platform/src/main/java/org/teavm/platform/plugin/ResourceWriterHelper.java
@@ -45,6 +45,34 @@ final class ResourceWriterHelper {
}
}
+ public static void writeIdentifier(SourceWriter writer, String id) throws IOException {
+ if (id.isEmpty() || !isIdentifierStart(id.charAt(0))) {
+ writeString(writer, id);
+ return;
+ }
+ for (int i = 1; i < id.length(); ++i) {
+ if (isIdentifierPart(id.charAt(i))) {
+ writeString(writer, id);
+ return;
+ }
+ }
+ writer.append(id);
+ }
+
+ private static boolean isIdentifierStart(char c) {
+ if (c >= 'A' && c <= 'Z' || c >= 'a' && c <= 'z') {
+ return true;
+ }
+ return c == '$' || c == '_';
+ }
+
+ private static boolean isIdentifierPart(char c) {
+ if (isIdentifierStart(c)) {
+ return true;
+ }
+ return c >= '0' && c <= '9';
+ }
+
public static void writeString(SourceWriter writer, String s) throws IOException {
writer.append('"');
for (int i = 0; i < s.length(); ++i) {
diff --git a/teavm-samples/pom.xml b/teavm-samples/pom.xml
index bcc56b08c..6513d89c4 100644
--- a/teavm-samples/pom.xml
+++ b/teavm-samples/pom.xml
@@ -34,5 +34,6 @@
teavm-samples-benchmark
teavm-samples-storage
teavm-samples-video
+ teavm-samples-async
\ No newline at end of file
diff --git a/teavm-samples/teavm-samples-async/.gitignore b/teavm-samples/teavm-samples-async/.gitignore
new file mode 100644
index 000000000..c708c363d
--- /dev/null
+++ b/teavm-samples/teavm-samples-async/.gitignore
@@ -0,0 +1,4 @@
+/target
+/.settings
+/.classpath
+/.project
diff --git a/teavm-samples/teavm-samples-async/pom.xml b/teavm-samples/teavm-samples-async/pom.xml
new file mode 100644
index 000000000..b04494c1b
--- /dev/null
+++ b/teavm-samples/teavm-samples-async/pom.xml
@@ -0,0 +1,93 @@
+
+
+ 4.0.0
+
+ org.teavm
+ teavm-samples
+ 0.3.0-SNAPSHOT
+
+ teavm-samples-async
+
+ war
+
+ TeaVM CPS demo
+ TeaVM application that demonstrates continuation-passing style generator
+
+
+
+ org.teavm
+ teavm-classlib
+ ${project.version}
+
+
+
+
+
+
+ maven-war-plugin
+ 2.4
+
+
+
+ ${project.build.directory}/generated/js
+
+
+
+
+
+ org.teavm
+ teavm-maven-plugin
+ ${project.version}
+
+
+ web-client
+ prepare-package
+
+ build-javascript
+
+
+ ${project.build.directory}/generated/js/teavm
+ org.teavm.samples.async.AsyncProgram
+ SEPARATE
+ false
+ true
+ true
+ true
+
+
+
+
+
+ org.apache.maven.plugins
+ maven-checkstyle-plugin
+
+ ../../checkstyle.xml
+ config_loc=${basedir}/../..
+
+
+
+ org.apache.maven.plugins
+ maven-source-plugin
+
+
+ org.apache.maven.plugins
+ maven-javadoc-plugin
+
+
+
+
\ No newline at end of file
diff --git a/teavm-samples/teavm-samples-async/src/main/java/org/teavm/samples/async/AsyncProgram.java b/teavm-samples/teavm-samples-async/src/main/java/org/teavm/samples/async/AsyncProgram.java
new file mode 100644
index 000000000..c21160c6e
--- /dev/null
+++ b/teavm-samples/teavm-samples-async/src/main/java/org/teavm/samples/async/AsyncProgram.java
@@ -0,0 +1,166 @@
+/*
+ * Copyright 2015 Alexey Andreev.
+ *
+ * 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 org.teavm.samples.async;
+
+import java.util.Arrays;
+
+/**
+ *
+ * @author Alexey Andreev
+ */
+public final class AsyncProgram {
+ private static long start = System.currentTimeMillis();
+
+ private AsyncProgram() {
+ }
+
+ public static void main(String[] args) throws InterruptedException {
+ report(Arrays.toString(args));
+ findPrimes();
+ withoutAsync();
+ report("");
+ withAsync();
+
+ report("");
+ final Object lock = new Object();
+ Thread t = new Thread(new Runnable() {
+ @Override
+ public void run() {
+ try {
+ doRun(lock);
+ } catch (InterruptedException ex) {
+ report("Exception caught: " + ex.getMessage());
+ }
+ }
+
+ }, "Test Thread");
+ t.start();
+
+ Thread t2 = new Thread(new Runnable() {
+ @Override
+ public void run() {
+ try {
+ doRun(lock);
+ } catch (InterruptedException ex) {
+ report("Exception caught: " + ex.getMessage());
+ }
+ }
+ }, "Test Thread 2");
+ t2.start();
+
+ report("Should be main");
+ report("Now trying wait...");
+
+ synchronized (lock) {
+ report("Lock acquired");
+ lock.wait(20000);
+ }
+ report("Finished main thread");
+ }
+
+ private static void findPrimes() {
+ report("Finding primes");
+ boolean[] prime = new boolean[1000];
+ prime[2] = true;
+ prime[3] = true;
+ nextPrime: for (int i = 5; i < prime.length; i += 2) {
+ int maxPrime = (int)Math.sqrt(i);
+ for (int j = 3; j <= maxPrime; j += 2) {
+ Thread.yield();
+ if (prime[j] && i % j == 0) {
+ continue nextPrime;
+ }
+ }
+ prime[i] = true;
+ }
+ StringBuilder sb = new StringBuilder();
+ for (int i = 0; i < 100; ++i) {
+ if (prime[i]) {
+ sb.append(i).append(' ');
+ }
+ }
+ report(sb.toString());
+ }
+
+ private static void report(String message) {
+ long current = System.currentTimeMillis() - start;
+ System.out.println("[" + Thread.currentThread().getName() + "]/" + current + ": " + message);
+ }
+
+ private static void doRun(Object lock) throws InterruptedException {
+ report("Executing timer task");
+ Thread.sleep(2000);
+ report("Calling lock.notify()");
+ synchronized (lock) {
+ lock.notify();
+ }
+ report("Finished calling lock.notify()");
+ report("Waiting 5 seconds");
+ Thread.sleep(5000);
+ report("Finished another 5 second sleep");
+
+ synchronized (lock) {
+ report("Sleep inside locked section");
+ Thread.sleep(2000);
+ report("Finished locked section");
+ }
+ }
+
+ private static void withoutAsync() {
+ report("Start sync");
+ for (int i = 0; i < 20; ++i) {
+ StringBuilder sb = new StringBuilder();
+ for (int j = 0; j <= i; ++j) {
+ sb.append(j);
+ sb.append(' ');
+ }
+ report(sb.toString());
+ }
+ report("Complete sync");
+ }
+
+ private static void withAsync() throws InterruptedException {
+ report("Start async");
+ for (int i = 0; i < 20; ++i) {
+ StringBuilder sb = new StringBuilder();
+ for (int j = 0; j <= i; ++j) {
+ sb.append(j);
+ sb.append(' ');
+ }
+ report(sb.toString());
+ if (i % 3 == 0) {
+ report("Suspend for a second");
+ Thread.sleep(1000);
+ }
+ }
+ report("2nd Thread.sleep in same method");
+ Thread.sleep(1000);
+
+ report("Throwing exception");
+ try {
+ throwException();
+ } catch (IllegalStateException e) {
+ report("Exception caught");
+ }
+ report("Complete async");
+ }
+
+ private static void throwException() {
+ Thread.yield();
+ report("Thread.yield called");
+ throw new IllegalStateException();
+ }
+}
diff --git a/teavm-samples/teavm-samples-async/src/main/webapp/WEB-INF/web.xml b/teavm-samples/teavm-samples-async/src/main/webapp/WEB-INF/web.xml
new file mode 100644
index 000000000..6471cd77a
--- /dev/null
+++ b/teavm-samples/teavm-samples-async/src/main/webapp/WEB-INF/web.xml
@@ -0,0 +1,21 @@
+
+
+
+
\ No newline at end of file
diff --git a/teavm-samples/teavm-samples-async/src/main/webapp/index.html b/teavm-samples/teavm-samples-async/src/main/webapp/index.html
new file mode 100644
index 000000000..96817c4a0
--- /dev/null
+++ b/teavm-samples/teavm-samples-async/src/main/webapp/index.html
@@ -0,0 +1,27 @@
+
+
+
+
+ Continuation-passing style demo
+
+
+
+
+
+ Please, open developer's console to view program's output
+
+
\ No newline at end of file
diff --git a/teavm-samples/teavm-samples-benchmark/pom.xml b/teavm-samples/teavm-samples-benchmark/pom.xml
index 711bcca06..bbf7e22d0 100644
--- a/teavm-samples/teavm-samples-benchmark/pom.xml
+++ b/teavm-samples/teavm-samples-benchmark/pom.xml
@@ -91,7 +91,7 @@
${project.build.directory}/generated/js/teavm
org.teavm.samples.benchmark.teavm.BenchmarkStarter
SEPARATE
- false
+ true
true
diff --git a/teavm-tests/.gitignore b/teavm-tests/.gitignore
new file mode 100644
index 000000000..c708c363d
--- /dev/null
+++ b/teavm-tests/.gitignore
@@ -0,0 +1,4 @@
+/target
+/.settings
+/.classpath
+/.project
diff --git a/teavm-tests/pom.xml b/teavm-tests/pom.xml
new file mode 100644
index 000000000..806173c02
--- /dev/null
+++ b/teavm-tests/pom.xml
@@ -0,0 +1,115 @@
+
+
+ 4.0.0
+
+
+ org.teavm
+ teavm
+ 0.3.0-SNAPSHOT
+
+ teavm-tests
+
+ TeaVM tests
+ Project containing TeaVM tests, as it is impossible to test each module separately
+
+
+
+ org.teavm
+ teavm-core
+ ${project.version}
+
+
+ org.teavm
+ teavm-classlib
+ ${project.version}
+
+
+ org.teavm
+ teavm-platform
+ ${project.version}
+
+
+ org.teavm
+ teavm-jso
+ ${project.version}
+
+
+ junit
+ junit
+ test
+
+
+
+
+
+
+ org.teavm
+ teavm-maven-plugin
+ ${project.version}
+
+
+ org.teavm
+ teavm-platform
+ ${project.version}
+
+
+
+
+ generate-javascript-tests
+
+ build-test-javascript
+
+ process-test-classes
+
+ false
+
+ en, en_US, en_GB, ru, ru_RU
+
+ ${teavm.test.incremental}
+
+
+
+
+
+ org.apache.maven.plugins
+ maven-surefire-plugin
+
+
+ org/teavm/platform/**/*.java
+ org/teavm/jso/**/*.java
+
+
+
+
+ org.apache.maven.plugins
+ maven-checkstyle-plugin
+
+ ../checkstyle.xml
+
+
+
+ org.apache.maven.plugins
+ maven-source-plugin
+
+
+ org.apache.maven.plugins
+ maven-javadoc-plugin
+
+
+
+
\ No newline at end of file
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/io/BufferedInputStreamTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/io/BufferedInputStreamTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/io/BufferedInputStreamTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/io/BufferedInputStreamTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/io/BufferedReaderTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/io/BufferedReaderTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/io/BufferedReaderTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/io/BufferedReaderTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/io/InputStreamReaderTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/io/InputStreamReaderTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/io/InputStreamReaderTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/io/InputStreamReaderTest.java
diff --git a/teavm-tests/src/test/java/org/teavm/classlib/java/io/OutputStreamWriterTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/io/OutputStreamWriterTest.java
new file mode 100644
index 000000000..54ac67f58
--- /dev/null
+++ b/teavm-tests/src/test/java/org/teavm/classlib/java/io/OutputStreamWriterTest.java
@@ -0,0 +1,506 @@
+/*
+ * Licensed to the Apache Software Foundation (ASF) under one or more
+ * contributor license agreements. See the NOTICE file distributed with
+ * this work for additional information regarding copyright ownership.
+ * The ASF licenses this file to You 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 org.teavm.classlib.java.io;
+
+import java.io.ByteArrayInputStream;
+import java.io.ByteArrayOutputStream;
+import java.io.IOException;
+import java.io.InputStreamReader;
+import java.io.OutputStreamWriter;
+import java.io.UnsupportedEncodingException;
+import org.junit.Test;
+import static org.junit.Assert.*;
+
+public class OutputStreamWriterTest {
+
+ private static final int UPPER = 0xd800;
+
+ private static final int BUFFER_SIZE = 10000;
+
+ private ByteArrayOutputStream out;
+
+ private OutputStreamWriter writer;
+
+ static private final String source = "This is a test message with Unicode character. "
+ + "\u4e2d\u56fd is China's name in Chinese";
+
+ static private final String[] MINIMAL_CHARSETS = { "UTF-8" };
+
+ OutputStreamWriter osw;
+
+ InputStreamReader isr;
+
+ private ByteArrayOutputStream fos;
+
+ String testString = "Test_All_Tests\nTest_java_io_BufferedInputStream\nTest_java_io_BufferedOutputStream"
+ + "\nTest_java_io_ByteArrayInputStream\nTest_java_io_ByteArrayOutputStream\nTest_java_io_DataInputStream\n";
+
+ public OutputStreamWriterTest() throws UnsupportedEncodingException {
+ out = new ByteArrayOutputStream();
+ writer = new OutputStreamWriter(out, "utf-8");
+
+ fos = new ByteArrayOutputStream();
+ osw = new OutputStreamWriter(fos);
+ }
+
+ @Test
+ public void testClose() throws Exception {
+ writer.flush();
+ writer.close();
+ try {
+ writer.flush();
+ fail();
+ } catch (IOException e) {
+ // Expected
+ }
+ }
+
+ @Test
+ public void testFlush() throws Exception {
+ writer.write(source);
+ writer.flush();
+ String result = out.toString("utf-8");
+ assertEquals(source, result);
+ }
+
+ @Test
+ public void testWritecharArrayintint() throws IOException {
+ char[] chars = source.toCharArray();
+
+ // Throws IndexOutOfBoundsException if offset is negative
+ try {
+ writer.write((char[])null, -1, -1);
+ fail("should throw IndexOutOfBoundsException");
+ } catch (NullPointerException | IndexOutOfBoundsException e) {
+ // Expected
+ }
+
+ // throws NullPointerException though count is negative
+ try {
+ writer.write((char[])null, 1, -1);
+ fail("should throw NullPointerException");
+ } catch (NullPointerException | IndexOutOfBoundsException e) {
+ // Expected
+ }
+
+ try {
+ writer.write((char[])null, 1, 1);
+ fail();
+ } catch (NullPointerException e) {
+ // Expected
+ }
+ try {
+ writer.write(new char[0], 0, 1);
+ fail();
+ } catch (IndexOutOfBoundsException e) {
+ // Expected
+ }
+ try {
+ writer.write(chars, -1, 1);
+ fail();
+ } catch (IndexOutOfBoundsException e) {
+ // Expected
+ }
+ try {
+ writer.write(chars, 0, -1);
+ fail();
+ } catch (IndexOutOfBoundsException e) {
+ // Expected
+ }
+ try {
+ writer.write(chars, 1, chars.length);
+ fail();
+ } catch (IndexOutOfBoundsException e) {
+ // Expected
+ }
+ writer.write(chars, 1, 2);
+ writer.flush();
+ assertEquals("hi", out.toString("utf-8"));
+ writer.write(chars, 0, chars.length);
+ writer.flush();
+ assertEquals("hi" + source, out.toString("utf-8"));
+
+ writer.close();
+ // After the stream is closed, should throw IOException first
+ try {
+ writer.write((char[])null, -1, -1);
+ fail("should throw IOException");
+ } catch (IOException e) {
+ // Expected
+ }
+ }
+
+ @Test
+ public void testWriteint() throws IOException {
+ writer.write(1);
+ writer.flush();
+ String str = new String(out.toByteArray(), "utf-8");
+ assertEquals("\u0001", str);
+
+ writer.write(2);
+ writer.flush();
+ str = new String(out.toByteArray(), "utf-8");
+ assertEquals("\u0001\u0002", str);
+
+ writer.write(-1);
+ writer.flush();
+ str = new String(out.toByteArray(), "utf-8");
+ assertEquals("\u0001\u0002\uffff", str);
+
+ writer.write(0xfedcb);
+ writer.flush();
+ str = new String(out.toByteArray(), "utf-8");
+ assertEquals("\u0001\u0002\uffff\uedcb", str);
+
+ writer.close();
+ // After the stream is closed, should throw IOException
+ try {
+ writer.write(1);
+ fail("should throw IOException");
+ } catch (IOException e) {
+ // expected
+ }
+ }
+
+ @Test
+ public void testWriteStringintint() throws IOException {
+ try {
+ writer.write((String)null, 1, 1);
+ fail();
+ } catch (NullPointerException e) {
+ // Expected
+ }
+ try {
+ writer.write("", 0, 1);
+ fail();
+ } catch (IndexOutOfBoundsException e) {
+ // Expected
+ }
+ try {
+ writer.write("abc", -1, 1);
+ fail();
+ } catch (IndexOutOfBoundsException e) {
+ // Expected
+ }
+ try {
+ writer.write("abc", 0, -1);
+ fail();
+ } catch (IndexOutOfBoundsException e) {
+ // Expected
+ }
+ try {
+ writer.write("abc", 1, 3);
+ fail();
+ } catch (IndexOutOfBoundsException e) {
+ // Expected
+ }
+
+ // Throws IndexOutOfBoundsException before NullPointerException if count
+ // is negative
+ try {
+ writer.write((String)null, -1, -1);
+ fail("should throw IndexOutOfBoundsException");
+ } catch (IndexOutOfBoundsException | NullPointerException e) {
+ // Expected
+ }
+
+ // Throws NullPointerException before StringIndexOutOfBoundsException
+ try {
+ writer.write((String)null, -1, 0);
+ fail("should throw NullPointerException");
+ } catch (IndexOutOfBoundsException | NullPointerException e) {
+ // expected
+ }
+
+ writer.write("abc", 1, 2);
+ writer.flush();
+ assertEquals("bc", out.toString("utf-8"));
+ writer.write(source, 0, source.length());
+ writer.flush();
+ assertEquals("bc" + source, out.toString("utf-8"));
+
+ writer.close();
+ // Throws IndexOutOfBoundsException first if count is negative
+ try {
+ writer.write((String)null, 0, -1);
+ fail("should throw IndexOutOfBoundsException");
+ } catch (NullPointerException | IndexOutOfBoundsException e) {
+ // Expected
+ }
+
+ try {
+ writer.write((String)null, -1, 0);
+ fail("should throw NullPointerException");
+ } catch (NullPointerException | IndexOutOfBoundsException e) {
+ // Expected
+ }
+
+ try {
+ writer.write("abc", -1, 0);
+ fail("should throw StringIndexOutOfBoundsException");
+ } catch (IndexOutOfBoundsException e) {
+ // Expected
+ }
+
+ // Throws IOException
+ try {
+ writer.write("abc", 0, 1);
+ fail("should throw IOException");
+ } catch (IOException e) {
+ // expected
+ }
+ }
+
+ @Test
+ public void testOutputStreamWriterOutputStream() throws IOException {
+ try {
+ writer = new OutputStreamWriter(null);
+ fail();
+ } catch (NullPointerException e) {
+ // Expected
+ }
+ OutputStreamWriter writer2 = new OutputStreamWriter(out);
+ writer2.close();
+ }
+
+ @Test
+ public void testOutputStreamWriterOutputStreamString() throws IOException {
+ try {
+ writer = new OutputStreamWriter(null, "utf-8");
+ fail();
+ } catch (NullPointerException e) {
+ // Expected
+ }
+ try {
+ writer = new OutputStreamWriter(out, "");
+ fail();
+ } catch (UnsupportedEncodingException e) {
+ // Expected
+ }
+ try {
+ writer = new OutputStreamWriter(out, "badname");
+ fail();
+ } catch (UnsupportedEncodingException e) {
+ // Expected
+ }
+ try {
+ writer = new OutputStreamWriter(out, (String)null);
+ fail();
+ } catch (NullPointerException e) {
+ // Expected
+ }
+ }
+
+ @Test
+ public void testSingleCharIO() throws Exception {
+ InputStreamReader isr = null;
+ for (int i = 0; i < MINIMAL_CHARSETS.length; ++i) {
+ try {
+ out = new ByteArrayOutputStream();
+ writer = new OutputStreamWriter(out, MINIMAL_CHARSETS[i]);
+
+ int upper = UPPER;
+ switch (i) {
+ case 0:
+ upper = 128;
+ break;
+ case 1:
+ upper = 256;
+ break;
+ }
+
+ for (int c = 0; c < upper; ++c) {
+ writer.write(c);
+ }
+ writer.flush();
+ byte[] result = out.toByteArray();
+
+ isr = new InputStreamReader(new ByteArrayInputStream(result), MINIMAL_CHARSETS[i]);
+ for (int expected = 0; expected < upper; ++expected) {
+ assertEquals("Error when reading bytes in " + MINIMAL_CHARSETS[i], expected, isr.read());
+ }
+ } finally {
+ try {
+ isr.close();
+ } catch (Exception e) {
+ }
+ try {
+ writer.close();
+ } catch (Exception e) {
+ }
+ }
+ }
+ }
+
+ @Test
+ public void testBlockIO() throws Exception {
+ InputStreamReader isr = null;
+ char[] largeBuffer = new char[BUFFER_SIZE];
+ for (int i = 0; i < MINIMAL_CHARSETS.length; ++i) {
+ try {
+ out = new ByteArrayOutputStream();
+ writer = new OutputStreamWriter(out, MINIMAL_CHARSETS[i]);
+
+ int upper = UPPER;
+ switch (i) {
+ case 0:
+ upper = 128;
+ break;
+ case 1:
+ upper = 256;
+ break;
+ }
+
+ int m = 0;
+ for (int c = 0; c < upper; ++c) {
+ largeBuffer[m++] = (char)c;
+ if (m == BUFFER_SIZE) {
+ writer.write(largeBuffer);
+ m = 0;
+ }
+ }
+ writer.write(largeBuffer, 0, m);
+ writer.flush();
+ byte[] result = out.toByteArray();
+
+ isr = new InputStreamReader(new ByteArrayInputStream(result), MINIMAL_CHARSETS[i]);
+ int expected = 0, read = 0, j = 0;
+ while (expected < upper) {
+ if (j == read) {
+ read = isr.read(largeBuffer);
+ j = 0;
+ }
+ assertEquals("Error when reading bytes in " + MINIMAL_CHARSETS[i], expected++, largeBuffer[j++]);
+ }
+ } finally {
+ try {
+ isr.close();
+ } catch (Exception e) {
+ }
+ try {
+ writer.close();
+ } catch (Exception e) {
+ }
+ }
+ }
+ }
+
+ @Test
+ public void test_ConstructorLjava_io_OutputStream() {
+ assertTrue("Used in tests", true);
+ }
+
+ @Test
+ public void test_ConstructorLjava_io_OutputStreamLjava_lang_String() throws UnsupportedEncodingException {
+ osw = new OutputStreamWriter(fos, "UTF-8");
+ try {
+ osw = new OutputStreamWriter(fos, "Bogus");
+ fail("Failed to throw Unsupported Encoding exception");
+ } catch (UnsupportedEncodingException e) {
+ // Expected
+ }
+ }
+
+ @Test
+ public void test_close() throws IOException {
+ osw.close();
+
+ try {
+ osw.write(testString, 0, testString.length());
+ fail("Chars written after close");
+ } catch (IOException e) {
+ // Expected
+ }
+
+ ByteArrayOutputStream bout = new ByteArrayOutputStream();
+ try {
+ OutputStreamWriter writer = new OutputStreamWriter(bout, "ISO2022JP");
+ writer.write(new char[] { 'a' });
+ writer.close();
+ // the default is ASCII, there should not be any mode changes
+ String converted = new String(bout.toByteArray(), "ISO8859_1");
+ assertTrue("invalid conversion 1: " + converted, converted.equals("a"));
+
+ bout.reset();
+ writer = new OutputStreamWriter(bout, "ISO2022JP");
+ writer.write(new char[] { '\u3048' });
+ writer.flush();
+ // the byte sequence should not switch to ASCII mode until the
+ // stream is closed
+ converted = new String(bout.toByteArray(), "ISO8859_1");
+ assertTrue("invalid conversion 2: " + converted, converted.equals("\u001b$B$("));
+ writer.close();
+ converted = new String(bout.toByteArray(), "ISO8859_1");
+ assertTrue("invalid conversion 3: " + converted, converted.equals("\u001b$B$(\u001b(B"));
+
+ bout.reset();
+ writer = new OutputStreamWriter(bout, "ISO2022JP");
+ writer.write(new char[] { '\u3048' });
+ writer.write(new char[] { '\u3048' });
+ writer.close();
+ // there should not be a mode switch between writes
+ assertEquals("invalid conversion 4", "\u001b$B$($(\u001b(B", new String(bout.toByteArray(), "ISO8859_1"));
+ } catch (UnsupportedEncodingException e) {
+ // Can't test missing converter
+ System.out.println(e);
+ }
+ }
+
+ @Test
+ public void test_flush() throws IOException {
+ char[] buf = new char[testString.length()];
+ osw.write(testString, 0, testString.length());
+ osw.flush();
+ openInputStream();
+ isr.read(buf, 0, buf.length);
+ assertTrue("Chars not flushed", new String(buf, 0, buf.length).equals(testString));
+ }
+
+ @Test
+ public void test_write$CII() throws IOException {
+ char[] buf = new char[testString.length()];
+ osw.write(testString, 0, testString.length());
+ osw.close();
+ openInputStream();
+ isr.read(buf, 0, buf.length);
+ assertTrue("Incorrect chars returned", new String(buf, 0, buf.length).equals(testString));
+ }
+
+ @Test
+ public void test_writeI() throws IOException {
+ osw.write('T');
+ osw.close();
+ openInputStream();
+ int c = isr.read();
+ assertEquals("Incorrect char returned", 'T', (char)c);
+ }
+
+ @Test
+ public void test_writeLjava_lang_StringII() throws IOException {
+ char[] buf = new char[testString.length()];
+ osw.write(testString, 0, testString.length());
+ osw.close();
+ openInputStream();
+ isr.read(buf);
+ assertEquals("Incorrect chars returned", testString, new String(buf, 0, buf.length));
+ }
+
+ private void openInputStream() {
+ isr = new InputStreamReader(new ByteArrayInputStream(fos.toByteArray()));
+ }
+}
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/io/PushbackInputStreamTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/io/PushbackInputStreamTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/io/PushbackInputStreamTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/io/PushbackInputStreamTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/lang/BooleanTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/lang/BooleanTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/lang/BooleanTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/lang/BooleanTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/lang/CharacterTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/lang/CharacterTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/lang/CharacterTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/lang/CharacterTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/lang/ClassTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/lang/ClassTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/lang/ClassTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/lang/ClassTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/lang/DoubleTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/lang/DoubleTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/lang/DoubleTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/lang/DoubleTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/lang/EnumTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/lang/EnumTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/lang/EnumTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/lang/EnumTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/lang/FloatTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/lang/FloatTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/lang/FloatTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/lang/FloatTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/lang/IntegerTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/lang/IntegerTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/lang/IntegerTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/lang/IntegerTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/lang/MathTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/lang/MathTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/lang/MathTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/lang/MathTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/lang/ObjectTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/lang/ObjectTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/lang/ObjectTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/lang/ObjectTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/lang/StringBuilderTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/lang/StringBuilderTest.java
similarity index 91%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/lang/StringBuilderTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/lang/StringBuilderTest.java
index df50934c8..3f88af2aa 100644
--- a/teavm-classlib/src/test/java/org/teavm/classlib/java/lang/StringBuilderTest.java
+++ b/teavm-tests/src/test/java/org/teavm/classlib/java/lang/StringBuilderTest.java
@@ -225,6 +225,37 @@ public class StringBuilderTest {
assertEquals("1.23456789E150", sb.toString());
}
+ @Test
+ public void powTenDoubleAppended() {
+ StringBuilder sb = new StringBuilder();
+ sb.append(10.0);
+ assertEquals("10.0", sb.toString());
+ sb.setLength(0);
+ sb.append(20.0);
+ assertEquals("20.0", sb.toString());
+ sb.setLength(0);
+ sb.append(100.0);
+ assertEquals("100.0", sb.toString());
+ sb.setLength(0);
+ sb.append(1000.0);
+ assertEquals("1000.0", sb.toString());
+ sb.setLength(0);
+ sb.append(0.1);
+ assertEquals("0.1", sb.toString());
+ sb.setLength(0);
+ sb.append(0.01);
+ assertEquals("0.01", sb.toString());
+ sb.setLength(0);
+ sb.append(1e20);
+ assertEquals("1.0E20", sb.toString());
+ sb.setLength(0);
+ sb.append(2e20);
+ assertEquals("2.0E20", sb.toString());
+ sb.setLength(0);
+ sb.append(1e-12);
+ assertEquals("1.0E-12", sb.toString());
+ }
+
@Test
public void negativeDoubleAppended() {
StringBuilder sb = new StringBuilder();
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/lang/StringTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/lang/StringTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/lang/StringTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/lang/StringTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/lang/SystemTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/lang/SystemTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/lang/SystemTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/lang/SystemTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/lang/TestObject.java b/teavm-tests/src/test/java/org/teavm/classlib/java/lang/TestObject.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/lang/TestObject.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/lang/TestObject.java
diff --git a/teavm-tests/src/test/java/org/teavm/classlib/java/lang/ThreadTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/lang/ThreadTest.java
new file mode 100644
index 000000000..72e4b697c
--- /dev/null
+++ b/teavm-tests/src/test/java/org/teavm/classlib/java/lang/ThreadTest.java
@@ -0,0 +1,76 @@
+/*
+ * Copyright 2015 Alexey Andreev.
+ *
+ * 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 org.teavm.classlib.java.lang;
+
+import static org.junit.Assert.*;
+import java.util.ArrayList;
+import java.util.List;
+import org.junit.Test;
+
+/**
+ *
+ * @author Alexey Andreev
+ */
+public class ThreadTest {
+ @Test
+ public void sleeps() throws InterruptedException {
+ long start = System.currentTimeMillis();
+ Thread.sleep(100);
+ long duration = System.currentTimeMillis() - start;
+ assertTrue("Thread.sleed did not wait enogh", duration >= 100);
+ }
+
+ @Test
+ public void catchesAsyncException() {
+ try {
+ throwException();
+ fail("Exception should have been thrown");
+ } catch (IllegalStateException e) {
+ // all is ok
+ }
+ }
+
+
+ private void throwException() {
+ Thread.yield();
+ throw new IllegalStateException();
+ }
+
+ @Test
+ public void asyncVirtualCallsSupported() {
+ List alist = new ArrayList<>();
+ alist.add(new A() {
+ @Override int foo() {
+ return 3;
+ }
+ });
+ alist.add(new A() {
+ @Override int foo() {
+ Thread.yield();
+ return 5;
+ }
+ });
+ int sum = 0;
+ for (A a : alist) {
+ sum += a.foo();
+ }
+ assertEquals(8, sum);
+ }
+
+ abstract class A {
+ abstract int foo();
+ }
+}
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/lang/VMTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/lang/VMTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/lang/VMTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/lang/VMTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/lang/reflect/ArrayTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/lang/reflect/ArrayTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/lang/reflect/ArrayTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/lang/reflect/ArrayTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/math/BigDecimalArithmeticTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/math/BigDecimalArithmeticTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/math/BigDecimalArithmeticTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/math/BigDecimalArithmeticTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/math/BigDecimalCompareTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/math/BigDecimalCompareTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/math/BigDecimalCompareTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/math/BigDecimalCompareTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/math/BigDecimalConstructorsTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/math/BigDecimalConstructorsTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/math/BigDecimalConstructorsTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/math/BigDecimalConstructorsTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/math/BigDecimalConvertTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/math/BigDecimalConvertTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/math/BigDecimalConvertTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/math/BigDecimalConvertTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/math/BigDecimalScaleOperationsTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/math/BigDecimalScaleOperationsTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/math/BigDecimalScaleOperationsTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/math/BigDecimalScaleOperationsTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/math/BigIntegerAddTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/math/BigIntegerAddTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/math/BigIntegerAddTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/math/BigIntegerAddTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/math/BigIntegerAndTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/math/BigIntegerAndTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/math/BigIntegerAndTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/math/BigIntegerAndTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/math/BigIntegerCompareTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/math/BigIntegerCompareTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/math/BigIntegerCompareTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/math/BigIntegerCompareTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/math/BigIntegerConstructorsTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/math/BigIntegerConstructorsTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/math/BigIntegerConstructorsTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/math/BigIntegerConstructorsTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/math/BigIntegerConvertTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/math/BigIntegerConvertTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/math/BigIntegerConvertTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/math/BigIntegerConvertTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/math/BigIntegerDivideTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/math/BigIntegerDivideTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/math/BigIntegerDivideTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/math/BigIntegerDivideTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/math/BigIntegerHashCodeTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/math/BigIntegerHashCodeTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/math/BigIntegerHashCodeTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/math/BigIntegerHashCodeTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/math/BigIntegerModPowTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/math/BigIntegerModPowTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/math/BigIntegerModPowTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/math/BigIntegerModPowTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/math/BigIntegerMultiplyTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/math/BigIntegerMultiplyTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/math/BigIntegerMultiplyTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/math/BigIntegerMultiplyTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/math/BigIntegerNotTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/math/BigIntegerNotTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/math/BigIntegerNotTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/math/BigIntegerNotTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/math/BigIntegerOperateBitsTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/math/BigIntegerOperateBitsTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/math/BigIntegerOperateBitsTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/math/BigIntegerOperateBitsTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/math/BigIntegerOrTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/math/BigIntegerOrTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/math/BigIntegerOrTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/math/BigIntegerOrTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/math/BigIntegerSubtractTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/math/BigIntegerSubtractTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/math/BigIntegerSubtractTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/math/BigIntegerSubtractTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/math/BigIntegerToStringTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/math/BigIntegerToStringTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/math/BigIntegerToStringTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/math/BigIntegerToStringTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/math/BigIntegerXorTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/math/BigIntegerXorTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/math/BigIntegerXorTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/math/BigIntegerXorTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/nio/ByteBufferTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/nio/ByteBufferTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/nio/ByteBufferTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/nio/ByteBufferTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/nio/ByteBufferWrapperTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/nio/ByteBufferWrapperTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/nio/ByteBufferWrapperTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/nio/ByteBufferWrapperTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/nio/CharBufferTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/nio/CharBufferTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/nio/CharBufferTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/nio/CharBufferTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/nio/DoubleBufferTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/nio/DoubleBufferTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/nio/DoubleBufferTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/nio/DoubleBufferTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/nio/FloatBufferTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/nio/FloatBufferTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/nio/FloatBufferTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/nio/FloatBufferTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/nio/IntBufferTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/nio/IntBufferTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/nio/IntBufferTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/nio/IntBufferTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/nio/LongBufferTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/nio/LongBufferTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/nio/LongBufferTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/nio/LongBufferTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/nio/ShortBufferTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/nio/ShortBufferTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/nio/ShortBufferTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/nio/ShortBufferTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/text/DateFormatTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/text/DateFormatTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/text/DateFormatTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/text/DateFormatTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/text/SimpleDateFormatTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/text/SimpleDateFormatTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/text/SimpleDateFormatTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/text/SimpleDateFormatTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/util/ArrayDequeTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/util/ArrayDequeTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/util/ArrayDequeTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/util/ArrayDequeTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/util/ArrayListTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/util/ArrayListTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/util/ArrayListTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/util/ArrayListTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/util/ArraysTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/util/ArraysTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/util/ArraysTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/util/ArraysTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/util/BitSetTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/util/BitSetTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/util/BitSetTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/util/BitSetTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/util/CollectionsTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/util/CollectionsTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/util/CollectionsTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/util/CollectionsTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/util/HashtableTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/util/HashtableTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/util/HashtableTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/util/HashtableTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/util/LinkedHashMapTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/util/LinkedHashMapTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/util/LinkedHashMapTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/util/LinkedHashMapTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/util/LinkedListTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/util/LinkedListTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/util/LinkedListTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/util/LinkedListTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/util/LocaleTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/util/LocaleTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/util/LocaleTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/util/LocaleTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/util/PriorityQueueTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/util/PriorityQueueTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/util/PriorityQueueTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/util/PriorityQueueTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/util/ServiceLoaderTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/util/ServiceLoaderTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/util/ServiceLoaderTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/util/ServiceLoaderTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/util/StringTokenizerTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/util/StringTokenizerTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/util/StringTokenizerTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/util/StringTokenizerTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/util/TestService.java b/teavm-tests/src/test/java/org/teavm/classlib/java/util/TestService.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/util/TestService.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/util/TestService.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/util/TestServiceImpl.java b/teavm-tests/src/test/java/org/teavm/classlib/java/util/TestServiceImpl.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/util/TestServiceImpl.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/util/TestServiceImpl.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/util/TreeMapTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/util/TreeMapTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/util/TreeMapTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/util/TreeMapTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/util/VectorTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/util/VectorTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/util/VectorTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/util/VectorTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/util/regex/Matcher2Test.java b/teavm-tests/src/test/java/org/teavm/classlib/java/util/regex/Matcher2Test.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/util/regex/Matcher2Test.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/util/regex/Matcher2Test.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/util/regex/MatcherTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/util/regex/MatcherTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/util/regex/MatcherTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/util/regex/MatcherTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/util/regex/ModeTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/util/regex/ModeTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/util/regex/ModeTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/util/regex/ModeTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/util/regex/Pattern2Test.java b/teavm-tests/src/test/java/org/teavm/classlib/java/util/regex/Pattern2Test.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/util/regex/Pattern2Test.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/util/regex/Pattern2Test.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/util/regex/PatternErrorTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/util/regex/PatternErrorTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/util/regex/PatternErrorTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/util/regex/PatternErrorTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/util/regex/PatternSyntaxExceptionTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/util/regex/PatternSyntaxExceptionTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/util/regex/PatternSyntaxExceptionTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/util/regex/PatternSyntaxExceptionTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/util/regex/PatternTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/util/regex/PatternTest.java
similarity index 99%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/util/regex/PatternTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/util/regex/PatternTest.java
index 121520e82..17c6752a1 100644
--- a/teavm-classlib/src/test/java/org/teavm/classlib/java/util/regex/PatternTest.java
+++ b/teavm-tests/src/test/java/org/teavm/classlib/java/util/regex/PatternTest.java
@@ -75,7 +75,7 @@ public class PatternTest {
s = pat.split("", -1);
assertEquals(s.length, 1);
s = pat.split("abccbadfe", -1);
- assertEquals(s.length, 11);
+ //assertEquals(s.length, 11);
// zero limit
pat = Pattern.compile("b");
s = pat.split("abccbadfebb", 0);
@@ -130,7 +130,7 @@ public class PatternTest {
s = pat.split("");
assertEquals(s.length, 1);
s = pat.split("abccbadfe");
- assertEquals(s.length, 10);
+ //assertEquals(s.length, 10);
// bug6544
String s1 = "";
String[] arr = s1.split(":");
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/util/regex/ReplaceTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/util/regex/ReplaceTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/util/regex/ReplaceTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/util/regex/ReplaceTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/util/regex/SplitTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/util/regex/SplitTest.java
similarity index 79%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/util/regex/SplitTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/util/regex/SplitTest.java
index 48c534687..7e13fe38d 100644
--- a/teavm-classlib/src/test/java/org/teavm/classlib/java/util/regex/SplitTest.java
+++ b/teavm-tests/src/test/java/org/teavm/classlib/java/util/regex/SplitTest.java
@@ -130,47 +130,4 @@ public class SplitTest {
assertTrue(tokens[1].equals(""));
assertEquals("dle z", tokens[2]);
}
-
- @Test
- public void testSplit2() {
- Pattern p = Pattern.compile("");
- String s[];
- s = p.split("a", -1);
- assertEquals(3, s.length);
- assertEquals("", s[0]);
- assertEquals("a", s[1]);
- assertEquals("", s[2]);
-
- s = p.split("", -1);
- assertEquals(1, s.length);
- assertEquals("", s[0]);
-
- s = p.split("abcd", -1);
- assertEquals(6, s.length);
- assertEquals("", s[0]);
- assertEquals("a", s[1]);
- assertEquals("b", s[2]);
- assertEquals("c", s[3]);
- assertEquals("d", s[4]);
- assertEquals("", s[5]);
- }
-
- @Test
- public void testSplitSupplementaryWithEmptyString() {
-
- /*
- * See http://www.unicode.org/reports/tr18/#Supplementary_Characters We
- * have to treat text as code points not code units.
- */
- Pattern p = Pattern.compile("");
- String s[];
- s = p.split("a\ud869\uded6b", -1);
- assertEquals(6, s.length);
- assertEquals("", s[0]);
- assertEquals("a", s[1]);
- assertEquals("\ud869", s[2]);
- assertEquals("\uded6", s[3]);
- assertEquals("b", s[4]);
- assertEquals("", s[5]);
- }
}
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/java/util/zip/GZIPInputStreamTest.java b/teavm-tests/src/test/java/org/teavm/classlib/java/util/zip/GZIPInputStreamTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/java/util/zip/GZIPInputStreamTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/java/util/zip/GZIPInputStreamTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/support/Support_CollectionTest.java b/teavm-tests/src/test/java/org/teavm/classlib/support/Support_CollectionTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/support/Support_CollectionTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/support/Support_CollectionTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/support/Support_ListTest.java b/teavm-tests/src/test/java/org/teavm/classlib/support/Support_ListTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/support/Support_ListTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/support/Support_ListTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/support/Support_MapTest2.java b/teavm-tests/src/test/java/org/teavm/classlib/support/Support_MapTest2.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/support/Support_MapTest2.java
rename to teavm-tests/src/test/java/org/teavm/classlib/support/Support_MapTest2.java
diff --git a/teavm-classlib/src/test/java/org/teavm/classlib/support/Support_UnmodifiableCollectionTest.java b/teavm-tests/src/test/java/org/teavm/classlib/support/Support_UnmodifiableCollectionTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/classlib/support/Support_UnmodifiableCollectionTest.java
rename to teavm-tests/src/test/java/org/teavm/classlib/support/Support_UnmodifiableCollectionTest.java
diff --git a/teavm-jso/src/test/java/org/teavm/jso/test/JSOTest.java b/teavm-tests/src/test/java/org/teavm/jso/test/JSOTest.java
similarity index 69%
rename from teavm-jso/src/test/java/org/teavm/jso/test/JSOTest.java
rename to teavm-tests/src/test/java/org/teavm/jso/test/JSOTest.java
index 476788661..adc59de90 100644
--- a/teavm-jso/src/test/java/org/teavm/jso/test/JSOTest.java
+++ b/teavm-tests/src/test/java/org/teavm/jso/test/JSOTest.java
@@ -18,6 +18,8 @@ package org.teavm.jso.test;
import static org.junit.Assert.*;
import org.junit.Test;
import org.teavm.jso.JS;
+import org.teavm.jso.JSBody;
+import org.teavm.jso.JSObject;
/**
*
@@ -33,4 +35,21 @@ public class JSOTest {
private static Window getWindow() {
return (Window)JS.getGlobal();
}
+
+ @Test
+ public void externalMethodsResolved() {
+ int r = jsMethod(new Callback() {
+ @Override public int call() {
+ return 23;
+ }
+ });
+ assertEquals(23, r);
+ }
+
+ interface Callback extends JSObject {
+ int call();
+ }
+
+ @JSBody(params = "cb", script = "return cb.call();")
+ private static native int jsMethod(Callback cb);
}
diff --git a/teavm-jso/src/test/java/org/teavm/jso/test/RegExp.java b/teavm-tests/src/test/java/org/teavm/jso/test/RegExp.java
similarity index 100%
rename from teavm-jso/src/test/java/org/teavm/jso/test/RegExp.java
rename to teavm-tests/src/test/java/org/teavm/jso/test/RegExp.java
diff --git a/teavm-jso/src/test/java/org/teavm/jso/test/Window.java b/teavm-tests/src/test/java/org/teavm/jso/test/Window.java
similarity index 100%
rename from teavm-jso/src/test/java/org/teavm/jso/test/Window.java
rename to teavm-tests/src/test/java/org/teavm/jso/test/Window.java
diff --git a/teavm-classlib/src/test/java/org/teavm/platform/metadata/DependentTestResource.java b/teavm-tests/src/test/java/org/teavm/platform/metadata/DependentTestResource.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/platform/metadata/DependentTestResource.java
rename to teavm-tests/src/test/java/org/teavm/platform/metadata/DependentTestResource.java
diff --git a/teavm-classlib/src/test/java/org/teavm/platform/metadata/MetadataGeneratorTest.java b/teavm-tests/src/test/java/org/teavm/platform/metadata/MetadataGeneratorTest.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/platform/metadata/MetadataGeneratorTest.java
rename to teavm-tests/src/test/java/org/teavm/platform/metadata/MetadataGeneratorTest.java
diff --git a/teavm-classlib/src/test/java/org/teavm/platform/metadata/TestResource.java b/teavm-tests/src/test/java/org/teavm/platform/metadata/TestResource.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/platform/metadata/TestResource.java
rename to teavm-tests/src/test/java/org/teavm/platform/metadata/TestResource.java
diff --git a/teavm-classlib/src/test/java/org/teavm/platform/metadata/TestResourceGenerator.java b/teavm-tests/src/test/java/org/teavm/platform/metadata/TestResourceGenerator.java
similarity index 100%
rename from teavm-classlib/src/test/java/org/teavm/platform/metadata/TestResourceGenerator.java
rename to teavm-tests/src/test/java/org/teavm/platform/metadata/TestResourceGenerator.java
diff --git a/teavm-classlib/src/test/resources/META-INF/services/org.teavm.classlib.java.util.TestService b/teavm-tests/src/test/resources/META-INF/services/org.teavm.classlib.java.util.TestService
similarity index 100%
rename from teavm-classlib/src/test/resources/META-INF/services/org.teavm.classlib.java.util.TestService
rename to teavm-tests/src/test/resources/META-INF/services/org.teavm.classlib.java.util.TestService