org.qkit.core.asm.adapters.AddMethodAdapter.java Source code

Java tutorial

Introduction

Here is the source code for org.qkit.core.asm.adapters.AddMethodAdapter.java

Source

/**
 * This file is part of ParaBot.
 *
 * ParaBot is free software: you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation, either version 3 of the License, or
 * (at your option) any later version.
 *
 * ParaBot 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 for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with ParaBot.  If not, see <http://www.gnu.org/licenses/>.
 */
package org.qkit.core.asm.adapters;

import org.objectweb.asm.ClassVisitor;
import org.objectweb.asm.Opcodes;
import org.objectweb.asm.tree.*;

/**
 * @since 1.7
 * @author Queue
 */
public class AddMethodAdapter extends ClassVisitor implements Opcodes {

    private String fieldName = null;
    private String getterName = null;
    private String fieldDescriptor = null;
    private String signature = null;

    private boolean isFieldPresent = false;
    private boolean isMethodPresent = false;

    private ClassVisitor next;

    private int varInsn, retInsn;

    public AddMethodAdapter(final ClassVisitor cv, final String fieldName, final String fieldDescriptor,
            final String getterName, final int varInsn, final int retInsn) {
        super(ASM4, new ClassNode());
        next = cv;

        this.fieldName = fieldName;
        this.getterName = getterName;
        this.varInsn = varInsn;
        this.retInsn = retInsn;
        this.fieldDescriptor = fieldDescriptor;
    }

    @Override
    public void visitEnd() {
        ClassNode cn = (ClassNode) cv;

        for (FieldNode f : cn.fields) {
            if (fieldName.equals(f.name) && fieldDescriptor.equals(f.desc)) {
                isFieldPresent = true;
                signature = f.signature;
                break;
            }
        }

        for (MethodNode mv : cn.methods) {
            if (getterName.equals(cn.name) && fieldDescriptor.equals(mv.desc)) {
                isMethodPresent = true;
                break;
            }
        }

        if (isFieldPresent && !isMethodPresent) {
            MethodNode mn = new MethodNode(ACC_PUBLIC, getterName, "()" + fieldDescriptor, signature, null);

            mn.instructions.add(new VarInsnNode(varInsn, 0));
            mn.instructions.add(new FieldInsnNode(GETFIELD, cn.name, fieldName, fieldDescriptor));
            mn.instructions.add(new InsnNode(retInsn));

            mn.visitMaxs(0, 0);

            cn.methods.add(mn);
            System.out.println(
                    "[+Method Addition] " + fieldDescriptor + " " + getterName + "() returns " + fieldName);
        }

        try {
            cn.accept(next);
        } catch (Exception ez) {
            ez.printStackTrace();
        }

    }

}