Eclipse builder is triggered by resource modification

This commit is contained in:
Alexey Andreev 2015-06-22 17:59:44 +04:00
parent 4641f9ca87
commit e3017828d8
4 changed files with 273 additions and 13 deletions

View File

@ -0,0 +1,205 @@
/*
* 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.tooling;
import java.util.List;
import java.util.Set;
import org.teavm.model.BasicBlockReader;
import org.teavm.model.FieldReference;
import org.teavm.model.InstructionLocation;
import org.teavm.model.MethodReference;
import org.teavm.model.ValueType;
import org.teavm.model.VariableReader;
import org.teavm.model.instructions.ArrayElementType;
import org.teavm.model.instructions.BinaryBranchingCondition;
import org.teavm.model.instructions.BinaryOperation;
import org.teavm.model.instructions.BranchingCondition;
import org.teavm.model.instructions.CastIntegerDirection;
import org.teavm.model.instructions.InstructionReader;
import org.teavm.model.instructions.IntegerSubtype;
import org.teavm.model.instructions.InvocationType;
import org.teavm.model.instructions.NumericOperandType;
import org.teavm.model.instructions.SwitchTableEntryReader;
/**
*
* @author Alexey Andreev
*/
class InstructionLocationReader implements InstructionReader {
private Set<String> resources;
public InstructionLocationReader(Set<String> resources) {
this.resources = resources;
}
@Override
public void location(InstructionLocation location) {
if (location != null && location.getFileName() != null) {
resources.add(location.getFileName());
}
}
@Override
public void nop() {
}
@Override
public void classConstant(VariableReader receiver, ValueType cst) {
}
@Override
public void nullConstant(VariableReader receiver) {
}
@Override
public void integerConstant(VariableReader receiver, int cst) {
}
@Override
public void longConstant(VariableReader receiver, long cst) {
}
@Override
public void floatConstant(VariableReader receiver, float cst) {
}
@Override
public void doubleConstant(VariableReader receiver, double cst) {
}
@Override
public void stringConstant(VariableReader receiver, String cst) {
}
@Override
public void binary(BinaryOperation op, VariableReader receiver, VariableReader first, VariableReader second,
NumericOperandType type) {
}
@Override
public void negate(VariableReader receiver, VariableReader operand, NumericOperandType type) {
}
@Override
public void assign(VariableReader receiver, VariableReader assignee) {
}
@Override
public void cast(VariableReader receiver, VariableReader value, ValueType targetType) {
}
@Override
public void cast(VariableReader receiver, VariableReader value, NumericOperandType sourceType,
NumericOperandType targetType) {
}
@Override
public void cast(VariableReader receiver, VariableReader value, IntegerSubtype type,
CastIntegerDirection targetType) {
}
@Override
public void jumpIf(BranchingCondition cond, VariableReader operand, BasicBlockReader consequent,
BasicBlockReader alternative) {
}
@Override
public void jumpIf(BinaryBranchingCondition cond, VariableReader first, VariableReader second,
BasicBlockReader consequent, BasicBlockReader alternative) {
}
@Override
public void jump(BasicBlockReader target) {
}
@Override
public void choose(VariableReader condition, List<? extends SwitchTableEntryReader> table,
BasicBlockReader defaultTarget) {
}
@Override
public void exit(VariableReader valueToReturn) {
}
@Override
public void raise(VariableReader exception) {
}
@Override
public void createArray(VariableReader receiver, ValueType itemType, VariableReader size) {
}
@Override
public void createArray(VariableReader receiver, ValueType itemType, List<? extends VariableReader> dimensions) {
}
@Override
public void create(VariableReader receiver, String type) {
}
@Override
public void getField(VariableReader receiver, VariableReader instance, FieldReference field, ValueType fieldType) {
}
@Override
public void putField(VariableReader instance, FieldReference field, VariableReader value, ValueType fieldType) {
}
@Override
public void arrayLength(VariableReader receiver, VariableReader array) {
}
@Override
public void cloneArray(VariableReader receiver, VariableReader array) {
}
@Override
public void unwrapArray(VariableReader receiver, VariableReader array, ArrayElementType elementType) {
}
@Override
public void getElement(VariableReader receiver, VariableReader array, VariableReader index) {
}
@Override
public void putElement(VariableReader array, VariableReader index, VariableReader value) {
}
@Override
public void invoke(VariableReader receiver, VariableReader instance, MethodReference method,
List<? extends VariableReader> arguments, InvocationType type) {
}
@Override
public void isInstance(VariableReader receiver, VariableReader value, ValueType type) {
}
@Override
public void initClass(String className) {
}
@Override
public void nullCheck(VariableReader receiver, VariableReader value) {
}
@Override
public void monitorEnter(VariableReader objectRef) {
}
@Override
public void monitorExit(VariableReader objectRef) {
}
}

View File

@ -213,6 +213,38 @@ public class TeaVMTool {
return vm != null ? vm.getClasses() : Collections.<String>emptyList(); return vm != null ? vm.getClasses() : Collections.<String>emptyList();
} }
public Collection<String> getUsedResources() {
if (vm == null) {
return Collections.<String>emptyList();
}
Set<String> resources = new HashSet<>();
ClassReaderSource classSource = vm.getDependencyClassSource();
InstructionLocationReader reader = new InstructionLocationReader(resources);
for (MethodReference methodRef : vm.getMethods()) {
ClassReader cls = classSource.get(methodRef.getClassName());
if (cls == null) {
continue;
}
MethodReader method = cls.getMethod(methodRef.getDescriptor());
if (method == null) {
continue;
}
ProgramReader program = method.getProgram();
if (program == null) {
continue;
}
for (int i = 0; i < program.basicBlockCount(); ++i) {
program.basicBlockAt(i).readAllInstructions(reader);
}
}
return resources;
}
public void addSourceFileProvider(SourceFileProvider sourceFileProvider) { public void addSourceFileProvider(SourceFileProvider sourceFileProvider) {
sourceFileProviders.add(sourceFileProvider); sourceFileProviders.add(sourceFileProvider);
} }

View File

@ -297,10 +297,23 @@ public class TeaVM implements TeaVMHost, ServiceRepository {
return classSource; return classSource;
} }
/**
* Gets a {@link ClassReaderSource} which is similar to that of {@link #getClassSource()},
* except that it also contains classes with applied transformations together with
* classes, generated via {@link DependencyChecker#submitClass(ClassHolder)}.
*/
public ClassReaderSource getDependencyClassSource() {
return dependencyChecker.getClassSource();
}
public Collection<String> getClasses() { public Collection<String> getClasses() {
return dependencyChecker.getAchievableClasses(); return dependencyChecker.getAchievableClasses();
} }
public Collection<MethodReference> getMethods() {
return dependencyChecker.getAchievableMethods();
}
public DependencyInfo getDependencyInfo() { public DependencyInfo getDependencyInfo() {
return dependencyChecker; return dependencyChecker;
} }

View File

@ -50,7 +50,7 @@ public class TeaVMProjectBuilder extends IncrementalProjectBuilder {
private SourceFileProvider[] sourceProviders; private SourceFileProvider[] sourceProviders;
private Set<IProject> usedProjects = new HashSet<>(); private Set<IProject> usedProjects = new HashSet<>();
private static Map<TeaVMProfile, Set<String>> profileClasses = new WeakHashMap<>(); private static Map<TeaVMProfile, Set<String>> profileClasses = new WeakHashMap<>();
private static Map<TeaVMProfile, Set<String>> profileClassNames = new WeakHashMap<>(); private static Map<TeaVMProfile, Set<IProject>> profileDependencies = new WeakHashMap<>();
private static Pattern newLinePattern = Pattern.compile("\\r|\\n|\\r\\n"); private static Pattern newLinePattern = Pattern.compile("\\r|\\n|\\r\\n");
@Override @Override
@ -93,9 +93,15 @@ public class TeaVMProjectBuilder extends IncrementalProjectBuilder {
private void buildProfile(int kind, IProgressMonitor monitor, TeaVMProfile profile, ClassLoader classLoader) private void buildProfile(int kind, IProgressMonitor monitor, TeaVMProfile profile, ClassLoader classLoader)
throws CoreException { throws CoreException {
if ((kind == AUTO_BUILD || kind == INCREMENTAL_BUILD) && !shouldBuild(profile)) { if ((kind == AUTO_BUILD || kind == INCREMENTAL_BUILD)) {
if (!shouldBuild(profile)) {
Set<IProject> dependencies = profileDependencies.get(profile);
if (dependencies != null) {
usedProjects.addAll(dependencies);
}
return; return;
} }
}
IStringVariableManager varManager = VariablesPlugin.getDefault().getStringVariableManager(); IStringVariableManager varManager = VariablesPlugin.getDefault().getStringVariableManager();
TeaVMTool tool = new TeaVMTool(); TeaVMTool tool = new TeaVMTool();
tool.setClassLoader(classLoader); tool.setClassLoader(classLoader);
@ -133,9 +139,8 @@ public class TeaVMProjectBuilder extends IncrementalProjectBuilder {
removeMarkers(profile); removeMarkers(profile);
putMarkers(tool.getDependencyInfo().getCallGraph(), tool.getProblemProvider().getProblems(), putMarkers(tool.getDependencyInfo().getCallGraph(), tool.getProblemProvider().getProblems(),
profile); profile);
Set<String> classNames = new HashSet<String>(tool.getClasses()); profileDependencies.put(profile, new HashSet<IProject>(usedProjects));
profileClassNames.put(profile, classNames); classesToResources(profile, tool);
setClasses(profile, classesToResources(classNames));
refreshTarget(tool.getTargetDirectory()); refreshTarget(tool.getTargetDirectory());
} }
if (!monitor.isCanceled()) { if (!monitor.isCanceled()) {
@ -179,9 +184,9 @@ public class TeaVMProjectBuilder extends IncrementalProjectBuilder {
} }
} }
private Set<String> classesToResources(Collection<String> classNames) { private void classesToResources(TeaVMProfile profile, TeaVMTool tool) {
Set<String> resourcePaths = new HashSet<>(); Set<String> resourcePaths = new HashSet<>();
for (String className : classNames) { for (String className : tool.getClasses()) {
for (IContainer clsContainer : classFileContainers) { for (IContainer clsContainer : classFileContainers) {
IResource res = clsContainer.findMember(className.replace('.', '/') + ".class"); IResource res = clsContainer.findMember(className.replace('.', '/') + ".class");
if (res != null) { if (res != null) {
@ -190,7 +195,16 @@ public class TeaVMProjectBuilder extends IncrementalProjectBuilder {
} }
} }
} }
return resourcePaths; for (String resourceName : tool.getUsedResources()) {
for (IContainer clsContainer : classFileContainers) {
IResource res = clsContainer.findMember(resourceName);
if (res != null) {
resourcePaths.add(res.getFullPath().toString());
usedProjects.add(res.getProject());
}
}
}
setClasses(profile, resourcePaths);
} }
private boolean shouldBuild(TeaVMProfile profile) throws CoreException { private boolean shouldBuild(TeaVMProfile profile) throws CoreException {
@ -204,10 +218,6 @@ public class TeaVMProjectBuilder extends IncrementalProjectBuilder {
return true; return true;
} }
} }
Set<String> classNames = profileClassNames.get(profile);
if (classNames != null) {
classesToResources(classNames);
}
return false; return false;
} }