Disk cached classes and programs

This commit is contained in:
konsoletyper 2014-09-09 13:35:43 +04:00
parent 13b64fd4c1
commit 7de36a6d33
8 changed files with 68 additions and 68 deletions

View File

@ -1,57 +0,0 @@
/*
* Copyright 2014 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.cache;
import java.io.File;
import java.net.URISyntaxException;
import java.net.URL;
import java.util.Date;
/**
*
* @author Alexey Andreev
*/
public class ClassPathClassDateProvider implements ClassDateProvider {
private ClassLoader classLoader;
public ClassPathClassDateProvider(ClassLoader classLoader) {
this.classLoader = classLoader;
}
@Override
public Date getModificationDate(String className) {
URL url = classLoader.getResource(className.replace('.', '/') + ".class");
if (url == null) {
return null;
}
if (url.getProtocol().equals("file")) {
try {
File file = new File(url.toURI());
return file.exists() ? new Date(file.lastModified()) : null;
} catch (URISyntaxException e) {
// If URI is invalid, we just report that class should be reparsed
return null;
}
} else if (url.getProtocol().equals("jar") && url.getPath().startsWith("file:")) {
int exclIndex = url.getPath().indexOf('!');
String jarFileName = exclIndex >= 0 ? url.getPath().substring(0, exclIndex) : url.getPath();
File file = new File(jarFileName.substring("file:".length()));
return file.exists() ? new Date(file.lastModified()) : null;
} else {
return null;
}
}
}

View File

@ -18,6 +18,7 @@ package org.teavm.cache;
import java.io.*; import java.io.*;
import java.util.*; import java.util.*;
import org.teavm.model.*; import org.teavm.model.*;
import org.teavm.parsing.ClassDateProvider;
/** /**
* *
@ -53,7 +54,7 @@ public class DiskCachedClassHolderSource implements ClassHolderSource {
if (classFile.exists()) { if (classFile.exists()) {
Date classDate = classDateProvider.getModificationDate(name); Date classDate = classDateProvider.getModificationDate(name);
if (classDate != null && classDate.before(new Date(classFile.lastModified()))) { if (classDate != null && classDate.before(new Date(classFile.lastModified()))) {
try (InputStream input = new FileInputStream(classFile)) { try (InputStream input = new BufferedInputStream(new FileInputStream(classFile))) {
item.cls = readClass(input, name); item.cls = readClass(input, name);
} catch (IOException e) { } catch (IOException e) {
// We could not access cache file, so let's parse class file // We could not access cache file, so let's parse class file

View File

@ -19,6 +19,7 @@ import java.io.*;
import java.util.*; import java.util.*;
import org.teavm.model.*; import org.teavm.model.*;
import org.teavm.model.instructions.*; import org.teavm.model.instructions.*;
import org.teavm.parsing.ClassDateProvider;
/** /**
* *
@ -46,7 +47,7 @@ public class DiskProgramCache implements ProgramCache {
cache.put(method, item); cache.put(method, item);
File file = getMethodFile(method); File file = getMethodFile(method);
if (file.exists()) { if (file.exists()) {
try (InputStream stream = new FileInputStream(file)) { try (InputStream stream = new BufferedInputStream(new FileInputStream(file))) {
DataInput input = new DataInputStream(stream); DataInput input = new DataInputStream(stream);
int depCount = input.readShort(); int depCount = input.readShort();
boolean dependenciesChanged = false; boolean dependenciesChanged = false;
@ -66,9 +67,6 @@ public class DiskProgramCache implements ProgramCache {
} }
} }
} }
if (item.program == null) {
System.out.println(method);
}
return item.program; return item.program;
} }

View File

@ -39,7 +39,7 @@ public class FileSymbolTable implements SymbolTable {
symbols.clear(); symbols.clear();
symbolMap.clear(); symbolMap.clear();
firstUnstoredIndex = 0; firstUnstoredIndex = 0;
try (DataInputStream input = new DataInputStream(new FileInputStream(file))) { try (DataInputStream input = new DataInputStream(new BufferedInputStream(new FileInputStream(file)))) {
while (true) { while (true) {
int length = input.read(); int length = input.read();
if (length == -1) { if (length == -1) {

View File

@ -130,7 +130,7 @@ public class ProgramIO {
TryCatchBlock tryCatch = new TryCatchBlock(); TryCatchBlock tryCatch = new TryCatchBlock();
int typeIndex = data.readInt(); int typeIndex = data.readInt();
if (typeIndex >= 0) { if (typeIndex >= 0) {
tryCatch.setExceptionType(symbolTable.at(data.readShort())); tryCatch.setExceptionType(symbolTable.at(typeIndex));
} }
short varIndex = data.readShort(); short varIndex = data.readShort();
if (varIndex >= 0) { if (varIndex >= 0) {

View File

@ -13,7 +13,7 @@
* See the License for the specific language governing permissions and * See the License for the specific language governing permissions and
* limitations under the License. * limitations under the License.
*/ */
package org.teavm.cache; package org.teavm.parsing;
import java.util.Date; import java.util.Date;

View File

@ -15,6 +15,7 @@
*/ */
package org.teavm.parsing; package org.teavm.parsing;
import java.util.Date;
import org.teavm.model.ClassHolder; import org.teavm.model.ClassHolder;
import org.teavm.model.ClassHolderSource; import org.teavm.model.ClassHolderSource;
import org.teavm.resource.ClasspathResourceReader; import org.teavm.resource.ClasspathResourceReader;
@ -25,13 +26,14 @@ import org.teavm.resource.ResourceClassHolderMapper;
* *
* @author Alexey Andreev <konsoletyper@gmail.com> * @author Alexey Andreev <konsoletyper@gmail.com>
*/ */
public class ClasspathClassHolderSource implements ClassHolderSource { public class ClasspathClassHolderSource implements ClassHolderSource, ClassDateProvider {
private MapperClassHolderSource innerClassSource; private MapperClassHolderSource innerClassSource;
private ClasspathResourceMapper classPathMapper;
public ClasspathClassHolderSource(ClassLoader classLoader) { public ClasspathClassHolderSource(ClassLoader classLoader) {
ClasspathResourceReader reader = new ClasspathResourceReader(classLoader); ClasspathResourceReader reader = new ClasspathResourceReader(classLoader);
ResourceClassHolderMapper rawMapper = new ResourceClassHolderMapper(reader); ResourceClassHolderMapper rawMapper = new ResourceClassHolderMapper(reader);
ClasspathResourceMapper classPathMapper = new ClasspathResourceMapper(classLoader, rawMapper); classPathMapper = new ClasspathResourceMapper(classLoader, rawMapper);
innerClassSource = new MapperClassHolderSource(classPathMapper); innerClassSource = new MapperClassHolderSource(classPathMapper);
} }
@ -43,4 +45,9 @@ public class ClasspathClassHolderSource implements ClassHolderSource {
public ClassHolder get(String name) { public ClassHolder get(String name) {
return innerClassSource.get(name); return innerClassSource.get(name);
} }
@Override
public Date getModificationDate(String className) {
return classPathMapper.getModificationDate(className);
}
} }

View File

@ -15,8 +15,10 @@
*/ */
package org.teavm.parsing; package org.teavm.parsing;
import java.io.File;
import java.io.IOException; import java.io.IOException;
import java.io.InputStream; import java.io.InputStream;
import java.net.URISyntaxException;
import java.net.URL; import java.net.URL;
import java.util.*; import java.util.*;
import org.teavm.common.CachedMapper; import org.teavm.common.CachedMapper;
@ -27,12 +29,13 @@ import org.teavm.model.ClassHolder;
* *
* @author Alexey Andreev <konsoletyper@gmail.com> * @author Alexey Andreev <konsoletyper@gmail.com>
*/ */
public class ClasspathResourceMapper implements Mapper<String, ClassHolder> { public class ClasspathResourceMapper implements Mapper<String, ClassHolder>, ClassDateProvider {
private static final String PACKAGE_PREFIX = "packagePrefix."; private static final String PACKAGE_PREFIX = "packagePrefix.";
private static final String CLASS_PREFIX = "classPrefix."; private static final String CLASS_PREFIX = "classPrefix.";
private Mapper<String, ClassHolder> innerMapper; private Mapper<String, ClassHolder> innerMapper;
private List<Transformation> transformations = new ArrayList<>(); private List<Transformation> transformations = new ArrayList<>();
private ClassRefsRenamer renamer; private ClassRefsRenamer renamer;
private ClassLoader classLoader;
private static class Transformation { private static class Transformation {
String packageName; String packageName;
@ -59,6 +62,7 @@ public class ClasspathResourceMapper implements Mapper<String, ClassHolder> {
throw new RuntimeException("Error reading resources", e); throw new RuntimeException("Error reading resources", e);
} }
renamer = new ClassRefsRenamer(new CachedMapper<>(classNameMapper)); renamer = new ClassRefsRenamer(new CachedMapper<>(classNameMapper));
this.classLoader = classLoader;
} }
private void loadProperties(Properties properties, Map<String, Transformation> cache) { private void loadProperties(Properties properties, Map<String, Transformation> cache) {
@ -126,4 +130,51 @@ public class ClasspathResourceMapper implements Mapper<String, ClassHolder> {
return renameClass(preimage); return renameClass(preimage);
} }
}; };
@Override
public Date getModificationDate(String className) {
int dotIndex = className.lastIndexOf('.');
String packageName;
String simpleName;
if (dotIndex > 0) {
packageName = className.substring(0, dotIndex + 1);
simpleName = className.substring(dotIndex + 1);
} else {
packageName = "";
simpleName = className;
}
for (Transformation transformation : transformations) {
if (packageName.startsWith(transformation.packageName)) {
String fullName = transformation.packagePrefix + packageName + transformation.classPrefix + simpleName;
Date date = getOriginalModificationDate(fullName);
if (date != null) {
return date;
}
}
}
return getOriginalModificationDate(className);
}
private Date getOriginalModificationDate(String className) {
URL url = classLoader.getResource(className.replace('.', '/') + ".class");
if (url == null) {
return null;
}
if (url.getProtocol().equals("file")) {
try {
File file = new File(url.toURI());
return file.exists() ? new Date(file.lastModified()) : null;
} catch (URISyntaxException e) {
// If URI is invalid, we just report that class should be reparsed
return null;
}
} else if (url.getProtocol().equals("jar") && url.getPath().startsWith("file:")) {
int exclIndex = url.getPath().indexOf('!');
String jarFileName = exclIndex >= 0 ? url.getPath().substring(0, exclIndex) : url.getPath();
File file = new File(jarFileName.substring("file:".length()));
return file.exists() ? new Date(file.lastModified()) : null;
} else {
return null;
}
}
} }