201 lines
7.4 KiB
Java
201 lines
7.4 KiB
Java
/*
|
|
* Copyright 2010-2013 JetBrains s.r.o.
|
|
*
|
|
* 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.jetbrains.jet.codegen;
|
|
|
|
import com.google.common.io.Files;
|
|
import com.intellij.openapi.util.Disposer;
|
|
import com.intellij.openapi.util.io.FileUtil;
|
|
import org.jetbrains.annotations.NotNull;
|
|
import org.jetbrains.asm4.*;
|
|
import org.jetbrains.jet.JetTestUtils;
|
|
import org.jetbrains.jet.cli.jvm.compiler.JetCoreEnvironment;
|
|
import org.jetbrains.jet.lang.psi.JetFile;
|
|
import org.jetbrains.jet.test.TestCaseWithTmpdir;
|
|
|
|
import java.io.File;
|
|
import java.io.IOException;
|
|
import java.nio.charset.Charset;
|
|
import java.util.ArrayList;
|
|
import java.util.List;
|
|
import java.util.regex.Matcher;
|
|
import java.util.regex.Pattern;
|
|
|
|
/**
|
|
* Test correctness of written local variables in class file for specified method
|
|
*/
|
|
|
|
public abstract class AbstractCheckLocalVariablesTableTest extends TestCaseWithTmpdir {
|
|
|
|
private File ktFile;
|
|
private JetCoreEnvironment jetCoreEnvironment;
|
|
|
|
public AbstractCheckLocalVariablesTableTest() {
|
|
}
|
|
|
|
@Override
|
|
protected void setUp() throws Exception {
|
|
super.setUp();
|
|
jetCoreEnvironment = JetTestUtils.createEnvironmentWithMockJdkAndIdeaAnnotations(myTestRootDisposable);
|
|
}
|
|
|
|
@Override
|
|
protected void tearDown() throws Exception {
|
|
jetCoreEnvironment = null;
|
|
super.tearDown();
|
|
}
|
|
|
|
protected void doTest(@NotNull String ktFileName) throws Exception {
|
|
ktFile = new File(ktFileName);
|
|
String text = FileUtil.loadFile(ktFile);
|
|
|
|
JetFile psiFile = JetTestUtils.createFile(ktFile.getName(), text, jetCoreEnvironment.getProject());
|
|
assert psiFile != null;
|
|
|
|
final ClassFileFactory factory = GenerationUtils.compileFileGetClassFileFactoryForTest(psiFile);
|
|
|
|
String modifiedTestName = ktFile.getName().replace(".kt", ".class");
|
|
boolean isClassFound = false;
|
|
for (String filename : factory.files()) {
|
|
if (filename.equals(modifiedTestName)) {
|
|
isClassFound = true;
|
|
ClassReader cr = new ClassReader(factory.asBytes(filename));
|
|
final List<LocalVariable> expectedLocalVariables = parseExpectations();
|
|
final List<LocalVariable> actualLocalVariables = readLocalVariable(cr, parseMethodName());
|
|
|
|
assertEquals("Count of variables are different", expectedLocalVariables.size(), actualLocalVariables.size());
|
|
|
|
int index = 0;
|
|
for (LocalVariable expectedVariable : expectedLocalVariables) {
|
|
LocalVariable actualVariable = actualLocalVariables.get(index);
|
|
assertEquals("Names are different", expectedVariable.name, actualVariable.name);
|
|
assertEquals("Types are different", expectedVariable.type, actualVariable.type);
|
|
assertEquals("Indexes are different", expectedVariable.index, actualVariable.index);
|
|
index++;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!isClassFound) {
|
|
throw new AssertionError("file name should be the same as class name. File name is " + modifiedTestName);
|
|
}
|
|
|
|
|
|
Disposer.dispose(myTestRootDisposable);
|
|
}
|
|
|
|
|
|
private static class LocalVariable {
|
|
private final String name;
|
|
private final String type;
|
|
private final int index;
|
|
|
|
private LocalVariable(
|
|
@NotNull String name,
|
|
@NotNull String type,
|
|
int index
|
|
) {
|
|
this.name = name;
|
|
this.type = type;
|
|
this.index = index;
|
|
}
|
|
}
|
|
|
|
@NotNull
|
|
private static final Pattern methodPattern = Pattern.compile("^// METHOD : *(.*)");
|
|
|
|
private static final Pattern namePattern = Pattern.compile("^// VARIABLE : NAME=*(.*) TYPE=.* INDEX=.*");
|
|
private static final Pattern typePattern = Pattern.compile("^// VARIABLE : NAME=.* TYPE=*(.*) INDEX=.*");
|
|
private static final Pattern indexPattern = Pattern.compile("^// VARIABLE : NAME=.* TYPE=.* INDEX=*(.*)");
|
|
|
|
|
|
private List<LocalVariable> parseExpectations() throws IOException {
|
|
List<String> lines = Files.readLines(ktFile, Charset.forName("utf-8"));
|
|
List<LocalVariable> expectedLocalVariables = new ArrayList<LocalVariable>();
|
|
for (int i = lines.size() - 3; i < lines.size(); ++i) {
|
|
Matcher nameMatcher = namePattern.matcher(lines.get(i));
|
|
if (nameMatcher.matches()) {
|
|
Matcher typeMatcher = typePattern.matcher(lines.get(i));
|
|
Matcher indexMatcher = indexPattern.matcher(lines.get(i));
|
|
|
|
if (!typeMatcher.matches()
|
|
|| !indexMatcher.matches()) {
|
|
throw new AssertionError("Incorrect test instructions format");
|
|
}
|
|
|
|
|
|
expectedLocalVariables.add(new LocalVariable(nameMatcher.group(1),
|
|
typeMatcher.group(1),
|
|
Integer.parseInt(indexMatcher.group(1))));
|
|
}
|
|
}
|
|
if (expectedLocalVariables.size() == 0) {
|
|
throw new AssertionError("test instructions not found in " + ktFile);
|
|
}
|
|
else {
|
|
return expectedLocalVariables;
|
|
}
|
|
}
|
|
|
|
private String parseMethodName() throws IOException {
|
|
List<String> lines = Files.readLines(ktFile, Charset.forName("utf-8"));
|
|
for (int i = 0; i < lines.size(); ++i) {
|
|
Matcher methodMatcher = methodPattern.matcher(lines.get(i));
|
|
if (methodMatcher.matches()) {
|
|
return methodMatcher.group(1);
|
|
}
|
|
}
|
|
|
|
assertTrue("method instructions not found", false);
|
|
return null;
|
|
}
|
|
|
|
@NotNull
|
|
private List<LocalVariable> readLocalVariable(ClassReader cr, final String methodName) throws Exception {
|
|
class Visitor extends ClassVisitor {
|
|
List<LocalVariable> readVariables = new ArrayList<LocalVariable>();
|
|
|
|
public Visitor() {
|
|
super(Opcodes.ASM4);
|
|
}
|
|
|
|
@Override
|
|
public MethodVisitor visitMethod(int access, String name, final String desc, final String signature, String[] exceptions) {
|
|
if (methodName.equals(name + desc)) {
|
|
return new MethodVisitor(Opcodes.ASM4) {
|
|
@Override
|
|
public void visitLocalVariable(String name, String desc, String signature, Label start, Label end, int index) {
|
|
readVariables.add(new LocalVariable(name, desc, index));
|
|
}
|
|
|
|
};
|
|
}
|
|
else {
|
|
return super.visitMethod(access, name, desc, signature, exceptions);
|
|
}
|
|
}
|
|
}
|
|
Visitor visitor = new Visitor();
|
|
|
|
cr.accept(visitor, ClassReader.SKIP_FRAMES);
|
|
|
|
assertFalse("method not found: " + methodName, visitor.readVariables.size() == 0);
|
|
|
|
return visitor.readVariables;
|
|
}
|
|
}
|
|
|