#instrumentation (1)

💻 Programming

[ASM 5.0] 1. BCI를 이용한 클래스 파일 읽기

BCI ( Byte Code Instrumentation )을 위한 ASM 5.0 프레임 워크에 대해서 공부해볼 예정입니다.

ASM에 대한 사용자메뉴얼 및 개발자가이드는 http://asm.ow2.org/index.html 이곳에서 쉽게 찾을 수 있습니다.

 

오늘은 그 첫 시간. 사용자 메뉴얼을 읽어가면서 조금씩 조금씩 정리해서 블로그에 올릴 예정입니다.

오늘은 클래스 파일 읽어서 정보 얻어오는 부분만 알아보도록 하겠습니다.

 

우선 이클립스에 ASM 5.0 프레임워크를 설치합니다.  

 

http://forge.ow2.org/project/showfiles.php?group_id=23

 

위 링크에 가시면 버전별로 zip 파일을 다운로드 받을 수 있습니다.( 링크가 깨질 것을 염려하여 최신 버전 5.0.3을 첨부하였습니다 )

 

다운로드 받은 zip파일을 압축해제를 합니다.

 

이제 이클립스를 실행시키고 프로젝트에 라이브러리를 추가해주기만하면 됩니다.

 

configure build path하셔서 라이브러리 추가하면 설치 완료!

 

자 이제 프레임워크를 설치했으니 사용해보기로 하죠.

 

BCITest패키지를 만들고 그 밑에 세 개의 클래스 파일을 만듭니다.

 

BCITest.java
HelloWorld.java
MyClassVisitor.java

 

// BCITest.java 소스 ( JUnit Test를 이용했습니다 ) 


package BCItest;

import java.io.IOException;
import org.junit.Test;
import org.objectweb.asm.ClassReader;

public class BCITest {
    @Test
    public void test() throws IOException {
        MyClassVisitor cv = new MyClassVisitor(327680);    // ASM5 = 327680
        ClassReader cr = new ClassReader("BCITest.HelloWorld");       // 대소문자 구분 X ( bcitest.helloworld 라고해도 실행됨 )
        cr.accept(cv, 2);
    }

 

테스트 해보실 때에는 BCITest.HelloWorld 대신에 다른 임의의 클래스를 넣어보시면 됩니다. 예를 들어 java.lang.String이라든가 java.lang.Exception이라든가 하는 클래스들 말이죠.

 

// HelloWorld.java 소스

 

package BCItest;

public class HelloWorld {
    public static void main(String[] args) {
        System.out.println("Hello World!!");
    }
}

 

// MyClassVisitor.java 소스

package BCItest;

import org.objectweb.asm.AnnotationVisitor;
import org.objectweb.asm.Attribute;
import org.objectweb.asm.ClassVisitor;
import org.objectweb.asm.FieldVisitor;
import org.objectweb.asm.MethodVisitor;
import org.objectweb.asm.TypePath;

public class MyClassVisitor extends ClassVisitor {

    public MyClassVisitor(int arg0) {
        super(arg0);
    }
    @Override
    public void visit(int version, int access, String name,
            String signature, String superName, String[] interfaces) {
        super.visit(version, access, name, signature, superName, interfaces);
        System.out.println(name + " extends " + superName + " {");
    }
    @Override
    public void visitEnd() {
        super.visitEnd();
        System.out.println("}");
    }
    @Override
    public FieldVisitor visitField(int access, String name, String desc,
            String signature, Object value) {
        System.out.println(" " + desc + " " + name);
        return super.visitField(access, name, desc, signature, value);
    }
    @Override
    public MethodVisitor visitMethod(int access, String name,
            String desc, String signature, String[] exceptions) {
        System.out.println(" " + name + desc);
        return super.visitMethod(access, name, desc, signature, exceptions);
    }
}
 

위에서 MyClassVisitor는 ClassVisitor를 상속한 뒤 몇몇 메소드만 오버라이드 했습니다.

 

우선 제가 제공해드린 소스를 기준으로 테스트를 진행하면 아래와 같은 결과가 나옵니다.

 

BCItest/HelloWorld extends java/lang/Object {
 <init>()V
 main([Ljava/lang/String;)V

 

여기서 <init>은 생성자를 말하고 마지막의 V는 void를 의미합니다.  

 

어떤가요? 어렵지 않죠?   

 

HelloWorld 클래스에 메소드를 더 넣어보고 ( 리턴 타입이나 파라미터 개수, 종류를 다르게 해서 ) 테스트해보세요~ 


참고로 아래는 Type Descriptor 와 Method descriptor 입니다.  

 

뭐가 뭘 의미하는지를 나타내주는 거니까 이거는 외우시는게 좋을거에요~







참고로 클래스를 읽어들이는 방법은 BCITest클래스에서 ClassReader생성할 때 이름을 주는 방법 외에 다른 방법도 있습니다.

사 용자 메뉴얼에는 아래와 같이 설명이 되어있습니다. 이름, 값, byte array, input stream을 이용해서 읽어들일 수 있는데 input stream을 이용해서 클래스를 읽어들일 때는 클래스 로더의 getResourceAsStream 메소드를 이용할 수 있다고 합니다. 사용 예제가 바로 밑에 나와있죠?


The class that must be read can be specified by name, as above, or by value, as a byte array or as an InputStream. An input stream to read the content of a class can be obtained with the ClassLoader’s getResourceAsStream method with:

 

cl.getResourceAsStream(classname.replace(’.’, ’/’) + ".class");

 

자, 클래스 파일 읽기는 이쯤에서 끝내겠습니다. 설명이 길면 머리만 아프죠.

 

다음 시간에는 기존의 클래스를 읽는것이 아니라 클래스를 생성해보도록 해보겠습니다~

 

그럼 오늘은 여기서 이만~ 

 

 출처 : ASM 4 사용자 가이드 ( 테스트는 ASM 5 로 했습니다. )