cfdc64fcb4
Reviewed-by: liach, redestad, vromero
280 lines
12 KiB
Java
280 lines
12 KiB
Java
/*
|
|
* Copyright (c) 2014, Oracle and/or its affiliates. All rights reserved.
|
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
|
*
|
|
* This code is free software; you can redistribute it and/or modify it
|
|
* under the terms of the GNU General Public License version 2 only, as
|
|
* published by the Free Software Foundation.
|
|
*
|
|
* This code is distributed in the hope that it will be useful, but WITHOUT
|
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
|
* version 2 for more details (a copy is included in the LICENSE file that
|
|
* accompanied this code).
|
|
*
|
|
* You should have received a copy of the GNU General Public License version
|
|
* 2 along with this work; if not, write to the Free Software Foundation,
|
|
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
*
|
|
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
|
|
* or visit www.oracle.com if you need additional information or have any
|
|
* questions.
|
|
*/
|
|
|
|
import java.lang.classfile.*;
|
|
import java.lang.classfile.attribute.*;
|
|
|
|
import java.io.IOException;
|
|
import java.lang.annotation.Repeatable;
|
|
import java.lang.annotation.Retention;
|
|
import java.lang.annotation.RetentionPolicy;
|
|
import java.util.Collection;
|
|
import java.util.HashMap;
|
|
import java.util.List;
|
|
import java.util.Map;
|
|
import java.util.stream.Stream;
|
|
|
|
import static java.lang.String.format;
|
|
import static java.util.stream.Collectors.*;
|
|
|
|
/**
|
|
* Base class for LocalVariableTable and LocalVariableTypeTable attributes tests.
|
|
* To add tests cases you should extend this class.
|
|
* Then implement {@link #getVariableTables} to get LocalVariableTable or LocalVariableTypeTable attribute.
|
|
* Then add method with local variables.
|
|
* Finally, annotate method with information about expected variables and their types
|
|
* by several {@link LocalVariableTestBase.ExpectedLocals} annotations.
|
|
* To run test invoke {@link #test()} method.
|
|
* If there are variables with the same name, set different scopes for them.
|
|
*
|
|
* @see #test()
|
|
*/
|
|
public abstract class LocalVariableTestBase extends TestBase {
|
|
public static final int DEFAULT_SCOPE = 0;
|
|
private final ClassModel classFile;
|
|
private final Class<?> clazz;
|
|
|
|
/**
|
|
* @param clazz class to test. Must contains annotated methods with expected results.
|
|
*/
|
|
public LocalVariableTestBase(Class<?> clazz) {
|
|
this.clazz = clazz;
|
|
try {
|
|
this.classFile = ClassFile.of().parse(getClassFile(clazz).toPath());
|
|
} catch (IOException e) {
|
|
throw new IllegalArgumentException("Can't read classfile for specified class", e);
|
|
}
|
|
}
|
|
|
|
protected abstract List<VariableTable> getVariableTables(CodeAttribute codeAttribute);
|
|
|
|
/**
|
|
* Finds expected variables with their type in VariableTable.
|
|
* Also does consistency checks, like variables from the same scope must point to different indexes.
|
|
*/
|
|
public void test() throws IOException {
|
|
List<java.lang.reflect.Method> testMethods = Stream.of(clazz.getDeclaredMethods())
|
|
.filter(m -> m.getAnnotationsByType(ExpectedLocals.class).length > 0)
|
|
.toList();
|
|
int failed = 0;
|
|
for (java.lang.reflect.Method method : testMethods) {
|
|
try {
|
|
Map<String, String> expectedLocals2Types = new HashMap<>();
|
|
Map<String, Integer> sig2scope = new HashMap<>();
|
|
for (ExpectedLocals anno : method.getDeclaredAnnotationsByType(ExpectedLocals.class)) {
|
|
expectedLocals2Types.put(anno.name(), anno.type());
|
|
sig2scope.put(anno.name() + "&" + anno.type(), anno.scope());
|
|
}
|
|
|
|
test(method.getName(), expectedLocals2Types, sig2scope);
|
|
} catch (AssertionFailedException ex) {
|
|
System.err.printf("Test %s failed.%n", method.getName());
|
|
ex.printStackTrace();
|
|
failed++;
|
|
}
|
|
}
|
|
if (failed > 0)
|
|
throw new RuntimeException(format("Failed %d out of %d. See logs.", failed, testMethods.size()));
|
|
}
|
|
|
|
public void test(String methodName, Map<String, String> expectedLocals2Types, Map<String, Integer> sig2scope)
|
|
throws IOException {
|
|
|
|
for (MethodModel m : classFile.methods()) {
|
|
String mName = m.methodName().stringValue();
|
|
if (methodName.equals(mName)) {
|
|
System.out.println("Testing local variable table in method " + mName);
|
|
CodeAttribute code_attribute = m.findAttribute(Attributes.code()).orElse(null);
|
|
assert code_attribute != null;
|
|
List<? extends VariableTable> variableTables = getVariableTables(code_attribute);
|
|
generalLocalVariableTableCheck(variableTables);
|
|
|
|
List<VariableTable.Entry> entries = variableTables.stream()
|
|
.flatMap(table -> table.entries().stream())
|
|
.collect(toList());
|
|
|
|
generalEntriesCheck(entries, code_attribute);
|
|
assertIndexesAreUnique(entries, sig2scope);
|
|
checkNamesAndTypes(entries, expectedLocals2Types);
|
|
checkDoubleAndLongIndexes(entries, sig2scope, code_attribute.maxLocals());
|
|
}
|
|
}
|
|
}
|
|
|
|
private void generalLocalVariableTableCheck(List<? extends VariableTable> variableTables) {
|
|
for (VariableTable localTable : variableTables) {
|
|
//only one per variable.
|
|
assertEquals(localTable.localVariableTableLength(),
|
|
localTable.entries().size(), "Incorrect local variable table length");
|
|
//attribute length is offset(line_number_table_length) + element_size*element_count
|
|
assertEquals(localTable.attributeLength(),
|
|
2 + (5 * 2) * localTable.localVariableTableLength(), "Incorrect attribute length");
|
|
}
|
|
}
|
|
|
|
private void generalEntriesCheck(List<VariableTable.Entry> entries, CodeAttribute code_attribute) {
|
|
for (VariableTable.Entry e : entries) {
|
|
assertTrue(e.index() >= 0 && e.index() < code_attribute.maxLocals(),
|
|
"Index " + e.index() + " out of variable array. Size of array is " + code_attribute.maxLocals());
|
|
assertTrue(e.startPC() >= 0, "StartPC is less then 0. StartPC = " + e.startPC());
|
|
assertTrue(e.length() >= 0, "Length is less then 0. Length = " + e.length());
|
|
assertTrue(e.startPC() + e.length() <= code_attribute.codeLength(),
|
|
format("StartPC+Length > code length.%n" +
|
|
"%s%n" +
|
|
"code_length = %s"
|
|
, e, code_attribute.codeLength()));
|
|
}
|
|
}
|
|
|
|
private void checkNamesAndTypes(List<VariableTable.Entry> entries,
|
|
Map<String, String> expectedLocals2Types) {
|
|
Map<String, List<String>> actualNames2Types = entries.stream()
|
|
.collect(
|
|
groupingBy(VariableTable.Entry::name,
|
|
mapping(VariableTable.Entry::type, toList())));
|
|
for (Map.Entry<String, String> name2type : expectedLocals2Types.entrySet()) {
|
|
String name = name2type.getKey();
|
|
String type = name2type.getValue();
|
|
|
|
assertTrue(actualNames2Types.containsKey(name),
|
|
format("There is no record for local variable %s%nEntries: %s", name, entries));
|
|
|
|
assertTrue(actualNames2Types.get(name).contains(type),
|
|
format("Types are different for local variable %s%nExpected type: %s%nActual type: %s",
|
|
name, type, actualNames2Types.get(name)));
|
|
}
|
|
}
|
|
|
|
|
|
private void assertIndexesAreUnique(Collection<VariableTable.Entry> entries, Map<String, Integer> scopes) {
|
|
//check every scope separately
|
|
Map<Object, List<VariableTable.Entry>> entriesByScope = groupByScope(entries, scopes);
|
|
for (Map.Entry<Object, List<VariableTable.Entry>> mapEntry : entriesByScope.entrySet()) {
|
|
mapEntry.getValue().stream()
|
|
.collect(groupingBy(VariableTable.Entry::index))
|
|
.entrySet()
|
|
.forEach(e ->
|
|
assertTrue(e.getValue().size() == 1,
|
|
"Multiple variables point to the same index in common scope. " + e.getValue()));
|
|
}
|
|
|
|
}
|
|
|
|
private void checkDoubleAndLongIndexes(Collection<VariableTable.Entry> entries,
|
|
Map<String, Integer> scopes, int maxLocals) {
|
|
//check every scope separately
|
|
Map<Object, List<VariableTable.Entry>> entriesByScope = groupByScope(entries, scopes);
|
|
for (List<VariableTable.Entry> entryList : entriesByScope.values()) {
|
|
Map<Integer, VariableTable.Entry> index2Entry = entryList.stream()
|
|
.collect(toMap(VariableTable.Entry::index, e -> e));
|
|
|
|
entryList.stream()
|
|
.filter(e -> "J".equals(e.type()) || "D".equals(e.type()))
|
|
.forEach(e -> {
|
|
assertTrue(e.index() + 1 < maxLocals,
|
|
format("Index %s is out of variable array. Long and double occupy 2 cells." +
|
|
" Size of array is %d", e.index() + 1, maxLocals));
|
|
assertTrue(!index2Entry.containsKey(e.index() + 1),
|
|
format("An entry points to the second cell of long/double entry.%n%s%n%s", e,
|
|
index2Entry.get(e.index() + 1)));
|
|
});
|
|
}
|
|
}
|
|
|
|
private Map<Object, List<VariableTable.Entry>> groupByScope(
|
|
Collection<VariableTable.Entry> entries, Map<String, Integer> scopes) {
|
|
return entries.stream().collect(groupingBy(e -> scopes.getOrDefault(e.name() + "&" + e.type(), DEFAULT_SCOPE)));
|
|
}
|
|
|
|
// Don't need the getString method now
|
|
// protected String getString(int i) {
|
|
// try {
|
|
// return classFile.constant_pool.getUTF8Info(i).value;
|
|
// } catch (ConstantPool.InvalidIndex | ConstantPool.UnexpectedEntry ex) {
|
|
// ex.printStackTrace();
|
|
// throw new AssertionFailedException("Issue while reading constant pool");
|
|
// }
|
|
// }
|
|
|
|
/**
|
|
* LocalVariableTable and LocalVariableTypeTable are similar.
|
|
* VariableTable interface is introduced to test this attributes in the same way without code duplication.
|
|
*/
|
|
interface VariableTable {
|
|
|
|
int localVariableTableLength();
|
|
|
|
List<VariableTable.Entry> entries();
|
|
|
|
int attributeLength();
|
|
|
|
interface Entry {
|
|
|
|
int index();
|
|
|
|
int startPC();
|
|
|
|
int length();
|
|
|
|
String name();
|
|
|
|
String type();
|
|
|
|
default String dump() {
|
|
return format("Entry{" +
|
|
"%n name = %s" +
|
|
"%n type = %s" +
|
|
"%n index = %d" +
|
|
"%n startPC = %d" +
|
|
"%n length = %d" +
|
|
"%n}", name(), type(), index(), startPC(), length());
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Used to store expected results in sources
|
|
*/
|
|
@Retention(RetentionPolicy.RUNTIME)
|
|
@Repeatable(Container.class)
|
|
@interface ExpectedLocals {
|
|
/**
|
|
* @return name of a local variable
|
|
*/
|
|
String name();
|
|
|
|
/**
|
|
* @return type of local variable in the internal format.
|
|
*/
|
|
String type();
|
|
|
|
//variables from different scopes can share the local variable table index and/or name.
|
|
int scope() default DEFAULT_SCOPE;
|
|
}
|
|
|
|
@Retention(RetentionPolicy.RUNTIME)
|
|
@interface Container {
|
|
ExpectedLocals[] value();
|
|
}
|
|
}
|