/******************************************************************************* * Copyright (c) 2014 Jeff Martin. * All rights reserved. This program and the accompanying materials * are made available under the terms of the GNU Public License v3.0 * which accompanies this distribution, and is available at * http://www.gnu.org/licenses/gpl.html * * Contributors: * Jeff Martin - initial API and implementation ******************************************************************************/ package cuchaz.enigma.bytecode; import javassist.CtBehavior; import javassist.CtClass; import javassist.CtField; import javassist.CtMethod; import javassist.bytecode.ConstPool; import javassist.bytecode.Descriptor; import javassist.bytecode.SourceFileAttribute; import cuchaz.enigma.mapping.BehaviorEntry; import cuchaz.enigma.mapping.ClassEntry; import cuchaz.enigma.mapping.EntryFactory; import cuchaz.enigma.mapping.FieldEntry; import cuchaz.enigma.mapping.Signature; import cuchaz.enigma.mapping.Translator; import cuchaz.enigma.mapping.Type; public class ClassTranslator { private Translator m_translator; public ClassTranslator(Translator translator) { m_translator = translator; } public void translate(CtClass c) { // NOTE: the order of these translations is very important // translate all the field and method references in the code by editing the constant pool ConstPool constants = c.getClassFile().getConstPool(); ConstPoolEditor editor = new ConstPoolEditor(constants); for (int i = 1; i < constants.getSize(); i++) { switch (constants.getTag(i)) { case ConstPool.CONST_Fieldref: { // translate the name and type FieldEntry entry = EntryFactory.getFieldEntry( Descriptor.toJvmName(constants.getFieldrefClassName(i)), constants.getFieldrefName(i), constants.getFieldrefType(i) ); FieldEntry translatedEntry = m_translator.translateEntry(entry); if (!entry.equals(translatedEntry)) { editor.changeMemberrefNameAndType(i, translatedEntry.getName(), translatedEntry.getType().toString()); } } break; case ConstPool.CONST_Methodref: case ConstPool.CONST_InterfaceMethodref: { // translate the name and type (ie signature) BehaviorEntry entry = EntryFactory.getBehaviorEntry( Descriptor.toJvmName(editor.getMemberrefClassname(i)), editor.getMemberrefName(i), editor.getMemberrefType(i) ); BehaviorEntry translatedEntry = m_translator.translateEntry(entry); if (!entry.equals(translatedEntry)) { editor.changeMemberrefNameAndType(i, translatedEntry.getName(), translatedEntry.getSignature().toString()); } } break; } } ClassEntry classEntry = new ClassEntry(Descriptor.toJvmName(c.getName())); // translate all the fields for (CtField field : c.getDeclaredFields()) { // translate the name FieldEntry entry = EntryFactory.getFieldEntry(field); String translatedName = m_translator.translate(entry); if (translatedName != null) { field.setName(translatedName); } // translate the type Type translatedType = m_translator.translateType(entry.getType()); field.getFieldInfo().setDescriptor(translatedType.toString()); } // translate all the methods and constructors for (CtBehavior behavior : c.getDeclaredBehaviors()) { BehaviorEntry entry = EntryFactory.getBehaviorEntry(behavior); if (behavior instanceof CtMethod) { CtMethod method = (CtMethod)behavior; // translate the name String translatedName = m_translator.translate(entry); if (translatedName != null) { method.setName(translatedName); } } if (entry.getSignature() != null) { // translate the signature Signature translatedSignature = m_translator.translateSignature(entry.getSignature()); behavior.getMethodInfo().setDescriptor(translatedSignature.toString()); } } // translate all the class names referenced in the code // the above code only changed method/field/reference names and types, but not the rest of the class references ClassRenamer.renameClasses(c, m_translator); // translate the source file attribute too ClassEntry deobfClassEntry = m_translator.translateEntry(classEntry); if (deobfClassEntry != null) { String sourceFile = Descriptor.toJvmName(deobfClassEntry.getOutermostClassName()) + ".java"; c.getClassFile().addAttribute(new SourceFileAttribute(constants, sourceFile)); } } }