blob: 024e32f2ddbce55614766e2db0555fc99f8a5075 [file] [log] [blame]
/*
* Copyright (C) 2013 The Android Open Source Project
*
* 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 com.android.tools.layoutlib.create;
import java.util.Arrays;
import java.util.HashMap;
import org.objectweb.asm.ClassVisitor;
import org.objectweb.asm.MethodVisitor;
public class RefactorClassAdapter extends AbstractClassAdapter {
private final HashMap<String, String> mRefactorClasses;
RefactorClassAdapter(ClassVisitor cv, HashMap<String, String> refactorClasses) {
super(cv);
mRefactorClasses = refactorClasses;
}
@Override
public MethodVisitor visitMethod(int access, String name, String desc, String signature,
String[] exceptions) {
MethodVisitor mw = super.visitMethod(access, name, desc, signature, exceptions);
return new RefactorStackMapAdapter(mw);
}
@Override
protected String renameInternalType(String oldClassName) {
if (oldClassName != null) {
String newName = mRefactorClasses.get(oldClassName);
if (newName != null) {
return newName;
}
int pos = oldClassName.indexOf('$');
if (pos > 0) {
newName = mRefactorClasses.get(oldClassName.substring(0, pos));
if (newName != null) {
return newName + oldClassName.substring(pos);
}
}
}
return oldClassName;
}
/**
* A method visitor that renames all references from an old class name to a new class name in
* the stackmap of the method.
*/
private class RefactorStackMapAdapter extends MethodVisitor {
private RefactorStackMapAdapter(MethodVisitor mv) {
super(Main.ASM_VERSION, mv);
}
private Object[] renameFrame(Object[] elements) {
if (elements == null) {
return null;
}
// The input array cannot be modified. We only copy the source array on write
boolean copied = false;
for (int i = 0; i < elements.length; i++) {
if (!(elements[i] instanceof String)) {
continue;
}
if (!copied) {
elements = Arrays.copyOf(elements, elements.length);
copied = true;
}
String type = (String)elements[i];
if (type.indexOf(';') > 0) {
elements[i] = renameTypeDesc(type);
} else {
elements[i] = renameInternalType(type);
}
}
return elements;
}
@Override
public void visitFrame(int type, int nLocal, Object[] local, int nStack, Object[] stack) {
super.visitFrame(type, nLocal, renameFrame(local), nStack, renameFrame(stack));
}
}
}