blob: 4467fb2dd6bc83d9076751a20fe0c3f204c9f238 [file] [log] [blame]
/***
* ASM: a very small and fast Java bytecode manipulation framework
* Copyright (c) 2000-2011 INRIA, France Telecom
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
* 3. Neither the name of the copyright holders nor the names of its
* contributors may be used to endorse or promote products derived from
* this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
* LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
* THE POSSIBILITY OF SUCH DAMAGE.
*/
package org.objectweb.asm.commons;
import java.util.Arrays;
import java.util.Collections;
import java.util.HashMap;
import java.util.ListIterator;
import java.util.Map;
import junit.framework.TestCase;
import org.objectweb.asm.ClassVisitor;
import org.objectweb.asm.Label;
import org.objectweb.asm.MethodVisitor;
import org.objectweb.asm.Opcodes;
import org.objectweb.asm.Type;
import org.objectweb.asm.tree.AbstractInsnNode;
import org.objectweb.asm.tree.ClassNode;
import org.objectweb.asm.tree.FieldInsnNode;
import org.objectweb.asm.tree.FieldNode;
import org.objectweb.asm.tree.FrameNode;
import org.objectweb.asm.tree.InnerClassNode;
import org.objectweb.asm.tree.LdcInsnNode;
import org.objectweb.asm.tree.MethodInsnNode;
import org.objectweb.asm.tree.MethodNode;
import org.objectweb.asm.tree.MultiANewArrayInsnNode;
import org.objectweb.asm.tree.TryCatchBlockNode;
import org.objectweb.asm.tree.TypeInsnNode;
public class RemappingClassAdapterTest extends TestCase implements Opcodes {
public void testRemappingClassAdapter() throws Exception {
Map<String, String> map = new HashMap<String, String>();
map.put("Boo", "B1");
map.put("Coo", "C1");
map.put("Doo", "D1");
Remapper remapper = new SimpleRemapper(map);
ClassNode cn = new ClassNode();
dump(new RemappingClassAdapter(cn, remapper));
assertEquals("D1", cn.name);
assertEquals("B1", cn.superName);
assertEquals(Arrays.asList(new String[] { "I", "I", "C1", "J", "B1" }),
cn.interfaces);
assertEquals("LB1;", field(cn, 0).desc);
assertEquals("[LB1;", field(cn, 1).desc);
assertEquals("D1", innerClass(cn, 0).name);
assertEquals("B1", innerClass(cn, 0).outerName);
// assertEquals("Doo", innerClass(cn, 0).innerName);
assertEquals("B1", cn.outerClass);
assertEquals("([[LB1;LC1;LD1;)LC1;", cn.outerMethodDesc);
MethodNode mn0 = cn.methods.get(0);
ListIterator<AbstractInsnNode> it = mn0.instructions.iterator();
FieldInsnNode n0 = (FieldInsnNode) it.next();
assertEquals("D1", n0.owner);
assertEquals("LB1;", n0.desc);
assertEquals(Type.getType("LB1;"), ((LdcInsnNode) it.next()).cst);
assertEquals(Type.getType("[LD1;"), ((LdcInsnNode) it.next()).cst);
assertEquals(Type.getType("[I"), ((LdcInsnNode) it.next()).cst);
assertEquals(Type.getType("J"), ((LdcInsnNode) it.next()).cst);
assertEquals("B1", ((TypeInsnNode) it.next()).desc);
assertEquals("[LD1;", ((TypeInsnNode) it.next()).desc);
assertEquals("[I", ((TypeInsnNode) it.next()).desc);
assertEquals("J", ((TypeInsnNode) it.next()).desc);
MultiANewArrayInsnNode n3 = (MultiANewArrayInsnNode) it.next();
assertEquals("[[LB1;", n3.desc);
MethodInsnNode n4 = (MethodInsnNode) it.next();
assertEquals("D1", n4.owner);
assertEquals("([[LB1;LC1;LD1;)LC1;", n4.desc);
FrameNode fn0 = (FrameNode) it.next();
assertEquals(Collections.EMPTY_LIST, fn0.local);
assertEquals(Collections.EMPTY_LIST, fn0.stack);
assertEquals(Arrays.asList(new Object[] { "B1", "C1", "D1" }),
((FrameNode) it.next()).local);
assertEquals(
Arrays.asList(new Object[] { Opcodes.INTEGER, "C1",
Opcodes.INTEGER, "D1" }), ((FrameNode) it.next()).local);
assertEquals(Arrays.asList(new Object[] { Opcodes.INTEGER,
Opcodes.INTEGER }), ((FrameNode) it.next()).local);
// assertEquals(Collections.EMPTY_LIST, fn0.stack);
TryCatchBlockNode tryCatchBlockNode = mn0.tryCatchBlocks.get(0);
assertEquals("C1", tryCatchBlockNode.type);
MethodNode mn1 = cn.methods.get(1);
assertEquals("([[LB1;LC1;LD1;)V", mn1.desc);
assertEquals(Arrays.asList(new String[] { "I", "J" }), mn1.exceptions);
}
private FieldNode field(ClassNode cn, int n) {
return cn.fields.get(n);
}
private InnerClassNode innerClass(ClassNode cn, int n) {
return cn.innerClasses.get(n);
}
public static void dump(ClassVisitor cv) throws Exception {
cv.visit(V1_5, 0, "Doo", null, "Boo", new String[] { "I", "I", "Coo",
"J", "Boo" });
cv.visitInnerClass("Doo", "Boo", "Doo", 0);
cv.visitOuterClass("Boo", "foo", "([[LBoo;LCoo;LDoo;)LCoo;");
cv.visitField(0, "boo", "LBoo;", null, null).visitEnd();
cv.visitField(0, "boo1", "[LBoo;", null, null).visitEnd();
cv.visitField(0, "s", "Ljava/lang/String;", null, null).visitEnd();
cv.visitField(0, "i", "I", null, null).visitEnd();
MethodVisitor mv;
mv = cv.visitMethod(0, "foo", "()V", null, null);
mv.visitCode();
mv.visitFieldInsn(GETFIELD, "Doo", "boo", "LBoo;");
mv.visitLdcInsn(Type.getType("LBoo;"));
mv.visitLdcInsn(Type.getType("[LDoo;"));
mv.visitLdcInsn(Type.getType("[I"));
mv.visitLdcInsn(Type.getType("J"));
mv.visitTypeInsn(ANEWARRAY, "Boo");
mv.visitTypeInsn(ANEWARRAY, "[LDoo;");
mv.visitTypeInsn(ANEWARRAY, "[I");
mv.visitTypeInsn(ANEWARRAY, "J");
mv.visitMultiANewArrayInsn("[[LBoo;", 2);
mv.visitMethodInsn(INVOKEVIRTUAL, "Doo", "goo",
"([[LBoo;LCoo;LDoo;)LCoo;");
mv.visitFrame(Opcodes.F_NEW, 0, new Object[5], 0, new Object[10]);
mv.visitFrame(Opcodes.F_NEW, 3, new Object[] { "Boo", "Coo", "Doo" },
0, new Object[0]);
mv.visitFrame(Opcodes.F_NEW, 4, new Object[] { Opcodes.INTEGER, "Coo",
Opcodes.INTEGER, "Doo" }, 0, new Object[0]);
mv.visitFrame(Opcodes.F_NEW, 2, new Object[] { Opcodes.INTEGER,
Opcodes.INTEGER }, 0, new Object[0]);
Label l = new Label();
mv.visitLocalVariable("boo", "LBoo;", null, l, l, 1);
mv.visitLocalVariable("boo1", "[LBoo;", null, l, l, 3);
mv.visitLocalVariable("boo2", "[[LBoo;", null, l, l, 4);
mv.visitMaxs(0, 0);
mv.visitTryCatchBlock(l, l, l, "Coo");
mv.visitEnd();
mv = cv.visitMethod(0, "goo", "([[LBoo;LCoo;LDoo;)V", null,
new String[] { "I", "J" });
mv.visitEnd();
cv.visitEnd();
}
// /*
public static class Boo {
}
public static interface Coo {
}
public static class Doo extends Boo implements Coo {
Boo boo = new Boo();
Boo[] boo1 = new Boo[2];
String s = "";
int i = 5;
static final Class<?> c1 = Boo.class;
static final Class<?> c2 = Boo[].class;
public Doo() {
}
public Doo(int i, Coo coo, Boo boo) {
}
void foo() {
class Eoo {
String s;
}
Eoo e = new Eoo();
e.s = "aaa";
// visitFieldInsn(int, String, String, String)
// visitLocalVariable(String, String, String, Label, Label, int)
Boo boo = this.boo;
// visitLdcInsn(Object)
Class<?> cc = Boo.class;
// visitTypeInsn(int, String)
Boo[] boo1 = new Boo[2];
// visitMultiANewArrayInsn(String, int)
Boo[][] boo2 = new Boo[2][2];
// visitMethodInsn(int, String, String, String)
goo(boo2, this, this);
}
Coo goo(Boo[][] boo2, Coo coo, Doo doo) {
return null;
}
}
// */
}