mirror of
https://github.com/Eaglercraft-TeaVM-Fork/eagler-teavm.git
synced 2025-01-06 15:14:11 -08:00
Fix html4j tests
This commit is contained in:
parent
7d2c76f711
commit
95a3a30a6e
classlib/src/main/java/org/teavm/classlib/java/lang/reflect
core/src/main/java/org/teavm/dependency
AbstractInstructionAnalyzer.javaDependencyGraphBuilder.javaDependencyNode.javaFastInstructionAnalyzer.java
platform/src/main/java/org/teavm/platform/plugin
|
@ -39,10 +39,15 @@ import org.teavm.model.MethodReference;
|
||||||
import org.teavm.model.ValueType;
|
import org.teavm.model.ValueType;
|
||||||
import org.teavm.model.emit.ProgramEmitter;
|
import org.teavm.model.emit.ProgramEmitter;
|
||||||
import org.teavm.model.emit.ValueEmitter;
|
import org.teavm.model.emit.ValueEmitter;
|
||||||
|
import org.teavm.platform.Platform;
|
||||||
import org.teavm.platform.PlatformAnnotationProvider;
|
import org.teavm.platform.PlatformAnnotationProvider;
|
||||||
|
import org.teavm.platform.PlatformClass;
|
||||||
|
|
||||||
public class AnnotationDependencyListener extends AbstractDependencyListener {
|
public class AnnotationDependencyListener extends AbstractDependencyListener {
|
||||||
private Set<MethodReference> reachedMethods = new HashSet<>();
|
private Set<MethodReference> reachedMethods = new HashSet<>();
|
||||||
|
private static final MethodReference GET_ANNOTATIONS_METHOD = new MethodReference(
|
||||||
|
Platform.class, "getAnnotations", PlatformClass.class, Annotation[].class);
|
||||||
|
private static final String ANNOTATIONS_READER_SUFFIX = "$$__annotations__$$";
|
||||||
|
|
||||||
private String getAnnotationImplementor(DependencyAgent agent, String annotationType) {
|
private String getAnnotationImplementor(DependencyAgent agent, String annotationType) {
|
||||||
String implementorName = annotationType + "$$_impl";
|
String implementorName = annotationType + "$$_impl";
|
||||||
|
@ -150,6 +155,9 @@ public class AnnotationDependencyListener extends AbstractDependencyListener {
|
||||||
private void reachGetAnnotations(DependencyAgent agent, DependencyNode node) {
|
private void reachGetAnnotations(DependencyAgent agent, DependencyNode node) {
|
||||||
node.getClassValueNode().addConsumer(type -> {
|
node.getClassValueNode().addConsumer(type -> {
|
||||||
String className = type.getName();
|
String className = type.getName();
|
||||||
|
if (className.endsWith(ANNOTATIONS_READER_SUFFIX)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
ClassReader cls = agent.getClassSource().get(className);
|
ClassReader cls = agent.getClassSource().get(className);
|
||||||
if (cls == null) {
|
if (cls == null) {
|
||||||
|
@ -165,12 +173,12 @@ public class AnnotationDependencyListener extends AbstractDependencyListener {
|
||||||
}
|
}
|
||||||
|
|
||||||
private void createAnnotationClass(DependencyAgent agent, String className) {
|
private void createAnnotationClass(DependencyAgent agent, String className) {
|
||||||
String readerClassName = className + "$$__annotations__$$";
|
String readerClassName = className + ANNOTATIONS_READER_SUFFIX;
|
||||||
if (agent.getClassSource().get(readerClassName) != null) {
|
if (agent.getClassSource().get(readerClassName) != null) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
ClassHolder cls = new ClassHolder(className + "$$__annotations__$$");
|
ClassHolder cls = new ClassHolder(readerClassName);
|
||||||
cls.setLevel(AccessLevel.PUBLIC);
|
cls.setLevel(AccessLevel.PUBLIC);
|
||||||
cls.setOwnerName("java.lang.Object");
|
cls.setOwnerName("java.lang.Object");
|
||||||
cls.getInterfaces().add(PlatformAnnotationProvider.class.getName());
|
cls.getInterfaces().add(PlatformAnnotationProvider.class.getName());
|
||||||
|
@ -183,9 +191,20 @@ public class AnnotationDependencyListener extends AbstractDependencyListener {
|
||||||
|
|
||||||
ClassReader annotatedClass = agent.getClassSource().get(className);
|
ClassReader annotatedClass = agent.getClassSource().get(className);
|
||||||
cls.addMethod(ctor);
|
cls.addMethod(ctor);
|
||||||
cls.addMethod(addReader(agent, annotatedClass));
|
MethodHolder reader = addReader(agent, annotatedClass);
|
||||||
|
cls.addMethod(reader);
|
||||||
|
|
||||||
agent.submitClass(cls);
|
agent.submitClass(cls);
|
||||||
|
|
||||||
|
MethodDependency ctorDep = agent.linkMethod(ctor.getReference());
|
||||||
|
ctorDep.getVariable(0).propagate(agent.getType(readerClassName));
|
||||||
|
ctorDep.use();
|
||||||
|
|
||||||
|
MethodDependency annotationsDep = agent.linkMethod(GET_ANNOTATIONS_METHOD);
|
||||||
|
MethodDependency readerDep = agent.linkMethod(reader.getReference());
|
||||||
|
readerDep.getVariable(0).propagate(agent.getType(readerClassName));
|
||||||
|
readerDep.getResult().getArrayItem().connect(annotationsDep.getResult().getArrayItem());
|
||||||
|
readerDep.use();
|
||||||
}
|
}
|
||||||
|
|
||||||
private MethodHolder addReader(DependencyAgent agent, ClassReader cls) {
|
private MethodHolder addReader(DependencyAgent agent, ClassReader cls) {
|
||||||
|
|
|
@ -191,10 +191,6 @@ abstract class AbstractInstructionAnalyzer extends AbstractInstructionReader {
|
||||||
invokeVirtual(receiver, instance, method, arguments);
|
invokeVirtual(receiver, instance, method, arguments);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
if (method.getName().equals("getClass") && method.parameterCount() == 0
|
|
||||||
&& method.getReturnType().isObject(Class.class) && receiver != null) {
|
|
||||||
getNode(instance).connect(getNode(receiver).getClassValueNode());
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -30,6 +30,7 @@ import org.teavm.model.ClassReader;
|
||||||
import org.teavm.model.ClassReaderSource;
|
import org.teavm.model.ClassReaderSource;
|
||||||
import org.teavm.model.ElementModifier;
|
import org.teavm.model.ElementModifier;
|
||||||
import org.teavm.model.IncomingReader;
|
import org.teavm.model.IncomingReader;
|
||||||
|
import org.teavm.model.MethodDescriptor;
|
||||||
import org.teavm.model.MethodHolder;
|
import org.teavm.model.MethodHolder;
|
||||||
import org.teavm.model.MethodReference;
|
import org.teavm.model.MethodReference;
|
||||||
import org.teavm.model.PhiReader;
|
import org.teavm.model.PhiReader;
|
||||||
|
@ -41,6 +42,7 @@ import org.teavm.model.instructions.ArrayElementType;
|
||||||
import org.teavm.model.text.ListingBuilder;
|
import org.teavm.model.text.ListingBuilder;
|
||||||
|
|
||||||
class DependencyGraphBuilder {
|
class DependencyGraphBuilder {
|
||||||
|
private static final MethodDescriptor GET_CLASS = new MethodDescriptor("getClass", Class.class);
|
||||||
private DependencyAnalyzer dependencyAnalyzer;
|
private DependencyAnalyzer dependencyAnalyzer;
|
||||||
private DependencyNode[] nodes;
|
private DependencyNode[] nodes;
|
||||||
private DependencyNode resultNode;
|
private DependencyNode resultNode;
|
||||||
|
@ -323,6 +325,10 @@ class DependencyGraphBuilder {
|
||||||
@Override
|
@Override
|
||||||
protected void invokeSpecial(VariableReader receiver, VariableReader instance, MethodReference method,
|
protected void invokeSpecial(VariableReader receiver, VariableReader instance, MethodReference method,
|
||||||
List<? extends VariableReader> arguments) {
|
List<? extends VariableReader> arguments) {
|
||||||
|
if (method.getDescriptor().equals(GET_CLASS)) {
|
||||||
|
invokeGetClass(receiver, instance);
|
||||||
|
return;
|
||||||
|
}
|
||||||
CallLocation callLocation = getCallLocation();
|
CallLocation callLocation = getCallLocation();
|
||||||
if (instance == null) {
|
if (instance == null) {
|
||||||
dependencyAnalyzer.linkClass(method.getClassName()).initClass(callLocation);
|
dependencyAnalyzer.linkClass(method.getClassName()).initClass(callLocation);
|
||||||
|
@ -359,6 +365,11 @@ class DependencyGraphBuilder {
|
||||||
@Override
|
@Override
|
||||||
protected void invokeVirtual(VariableReader receiver, VariableReader instance, MethodReference method,
|
protected void invokeVirtual(VariableReader receiver, VariableReader instance, MethodReference method,
|
||||||
List<? extends VariableReader> arguments) {
|
List<? extends VariableReader> arguments) {
|
||||||
|
if (method.getDescriptor().equals(GET_CLASS)) {
|
||||||
|
invokeGetClass(receiver, instance);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
DependencyNode[] actualArgs = new DependencyNode[arguments.size() + 1];
|
DependencyNode[] actualArgs = new DependencyNode[arguments.size() + 1];
|
||||||
for (int i = 0; i < arguments.size(); ++i) {
|
for (int i = 0; i < arguments.size(); ++i) {
|
||||||
actualArgs[i + 1] = nodes[arguments.get(i).getIndex()];
|
actualArgs[i + 1] = nodes[arguments.get(i).getIndex()];
|
||||||
|
@ -375,6 +386,21 @@ class DependencyGraphBuilder {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void invokeGetClass(VariableReader receiver, VariableReader instance) {
|
||||||
|
MethodDependency getClassDep = dependencyAnalyzer.linkMethod("java.lang.Object", GET_CLASS);
|
||||||
|
getClassDep.addLocation(getCallLocation());
|
||||||
|
getNode(instance).addConsumer(t -> {
|
||||||
|
getClassDep.getVariable(0).propagate(t);
|
||||||
|
if (receiver != null) {
|
||||||
|
getNode(receiver).getClassValueNode().propagate(t);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
if (receiver != null) {
|
||||||
|
getNode(receiver).propagate(dependencyAnalyzer.getType("java.lang.Class"));
|
||||||
|
}
|
||||||
|
getClassDep.use();
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void nullCheck(VariableReader receiver, VariableReader value) {
|
public void nullCheck(VariableReader receiver, VariableReader value) {
|
||||||
super.nullCheck(receiver, value);
|
super.nullCheck(receiver, value);
|
||||||
|
|
|
@ -32,7 +32,6 @@ import org.teavm.model.MethodReference;
|
||||||
import org.teavm.model.ValueType;
|
import org.teavm.model.ValueType;
|
||||||
|
|
||||||
public class DependencyNode implements ValueDependencyInfo {
|
public class DependencyNode implements ValueDependencyInfo {
|
||||||
private static final int SMALL_TYPES_THRESHOLD = 3;
|
|
||||||
private static final int DEGREE_THRESHOLD = 2;
|
private static final int DEGREE_THRESHOLD = 2;
|
||||||
DependencyAnalyzer dependencyAnalyzer;
|
DependencyAnalyzer dependencyAnalyzer;
|
||||||
List<DependencyConsumer> followers;
|
List<DependencyConsumer> followers;
|
||||||
|
@ -385,12 +384,6 @@ public class DependencyNode implements ValueDependencyInfo {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void propagateTypes(Transition transition) {
|
|
||||||
if (typeSet != null) {
|
|
||||||
dependencyAnalyzer.schedulePropagation(transition, getTypesInternal());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public void connect(DependencyNode node) {
|
public void connect(DependencyNode node) {
|
||||||
connect(node, null);
|
connect(node, null);
|
||||||
}
|
}
|
||||||
|
|
|
@ -31,6 +31,9 @@ class FastInstructionAnalyzer extends AbstractInstructionAnalyzer {
|
||||||
@Override
|
@Override
|
||||||
protected void invokeSpecial(VariableReader receiver, VariableReader instance, MethodReference method,
|
protected void invokeSpecial(VariableReader receiver, VariableReader instance, MethodReference method,
|
||||||
List<? extends VariableReader> arguments) {
|
List<? extends VariableReader> arguments) {
|
||||||
|
if (instance != null) {
|
||||||
|
invokeGetClass(method);
|
||||||
|
}
|
||||||
CallLocation callLocation = impreciseLocation;
|
CallLocation callLocation = impreciseLocation;
|
||||||
if (instance == null) {
|
if (instance == null) {
|
||||||
dependencyAnalyzer.linkClass(method.getClassName()).initClass(callLocation);
|
dependencyAnalyzer.linkClass(method.getClassName()).initClass(callLocation);
|
||||||
|
@ -43,12 +46,19 @@ class FastInstructionAnalyzer extends AbstractInstructionAnalyzer {
|
||||||
@Override
|
@Override
|
||||||
protected void invokeVirtual(VariableReader receiver, VariableReader instance, MethodReference method,
|
protected void invokeVirtual(VariableReader receiver, VariableReader instance, MethodReference method,
|
||||||
List<? extends VariableReader> arguments) {
|
List<? extends VariableReader> arguments) {
|
||||||
|
invokeGetClass(method);
|
||||||
dependencyAnalyzer.getVirtualCallConsumer(method).addLocation(impreciseLocation);
|
dependencyAnalyzer.getVirtualCallConsumer(method).addLocation(impreciseLocation);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void invokeGetClass(MethodReference method) {
|
||||||
|
if (method.getName().equals("getClass") && method.parameterCount() == 0
|
||||||
|
&& method.getReturnType().isObject(Class.class)) {
|
||||||
|
dependencyAnalyzer.instancesNode.connect(dependencyAnalyzer.classesNode);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void cloneArray(VariableReader receiver, VariableReader array) {
|
public void cloneArray(VariableReader receiver, VariableReader array) {
|
||||||
DependencyNode arrayNode = getNode(array);
|
|
||||||
MethodDependency cloneDep = getAnalyzer().linkMethod(CLONE_METHOD);
|
MethodDependency cloneDep = getAnalyzer().linkMethod(CLONE_METHOD);
|
||||||
cloneDep.addLocation(impreciseLocation);
|
cloneDep.addLocation(impreciseLocation);
|
||||||
cloneDep.use();
|
cloneDep.use();
|
||||||
|
|
|
@ -1,74 +0,0 @@
|
||||||
/*
|
|
||||||
* 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.lang.annotation.Annotation;
|
|
||||||
import org.teavm.dependency.AbstractDependencyListener;
|
|
||||||
import org.teavm.dependency.DependencyAgent;
|
|
||||||
import org.teavm.dependency.DependencyNode;
|
|
||||||
import org.teavm.dependency.MethodDependency;
|
|
||||||
import org.teavm.model.MethodReference;
|
|
||||||
import org.teavm.model.ValueType;
|
|
||||||
import org.teavm.platform.Platform;
|
|
||||||
import org.teavm.platform.PlatformAnnotationProvider;
|
|
||||||
|
|
||||||
public class AnnotationDependencySupport extends AbstractDependencyListener {
|
|
||||||
private DependencyNode allClasses;
|
|
||||||
private MethodDependency getAnnotationsDep;
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void started(DependencyAgent agent) {
|
|
||||||
allClasses = agent.createNode();
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void classReached(DependencyAgent agent, String className) {
|
|
||||||
allClasses.propagate(agent.getType(className));
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void methodReached(DependencyAgent agent, MethodDependency method) {
|
|
||||||
if (method.getReference().getClassName().equals(Platform.class.getName())
|
|
||||||
&& method.getReference().getName().equals("getAnnotations")) {
|
|
||||||
method.getResult().propagate(agent.getType("[" + ValueType.parse(Annotation.class).toString()));
|
|
||||||
if (getAnnotationsDep == null) {
|
|
||||||
getAnnotationsDep = agent.linkMethod(new MethodReference(PlatformAnnotationProvider.class,
|
|
||||||
"getAnnotations", Annotation[].class));
|
|
||||||
}
|
|
||||||
method.addLocationListener(getAnnotationsDep::addLocation);
|
|
||||||
|
|
||||||
allClasses.addConsumer(type -> {
|
|
||||||
if (type.getName().endsWith("$$__annotations__$$")) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
String className = type.getName() + "$$__annotations__$$";
|
|
||||||
MethodDependency initMethod = agent.linkMethod(new MethodReference(className, "<init>",
|
|
||||||
ValueType.VOID));
|
|
||||||
initMethod.propagate(0, className);
|
|
||||||
initMethod.use();
|
|
||||||
MethodDependency readMethod = agent.linkMethod(new MethodReference(className,
|
|
||||||
"getAnnotations", ValueType.parse(Annotation[].class)));
|
|
||||||
readMethod.getResult().getArrayItem().connect(method.getResult().getArrayItem());
|
|
||||||
readMethod.use();
|
|
||||||
|
|
||||||
method.addLocationListener(location -> {
|
|
||||||
initMethod.addLocation(location);
|
|
||||||
readMethod.addLocation(location);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -105,7 +105,6 @@ public class PlatformPlugin implements TeaVMPlugin {
|
||||||
host.add(new NewInstanceDependencySupport());
|
host.add(new NewInstanceDependencySupport());
|
||||||
host.add(new ClassLookupDependencySupport());
|
host.add(new ClassLookupDependencySupport());
|
||||||
host.add(new EnumDependencySupport());
|
host.add(new EnumDependencySupport());
|
||||||
host.add(new AnnotationDependencySupport());
|
|
||||||
host.add(new PlatformDependencyListener());
|
host.add(new PlatformDependencyListener());
|
||||||
host.add(new AsyncDependencyListener());
|
host.add(new AsyncDependencyListener());
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue
Block a user