com.toolazydogs.aunit.internal.ParserFactory.java Source code

Java tutorial

Introduction

Here is the source code for com.toolazydogs.aunit.internal.ParserFactory.java

Source

/**
 *
 * Copyright 2010-2011 (C) The original author or authors
 *
 * 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.toolazydogs.aunit.internal;

import com.toolazydogs.aunit.ParserSetup;
import org.antlr.runtime.CharStream;
import org.antlr.runtime.Parser;
import org.antlr.runtime.RecognizerSharedState;
import org.antlr.runtime.TokenStream;
import org.objectweb.asm.ClassWriter;
import org.objectweb.asm.FieldVisitor;
import org.objectweb.asm.MethodVisitor;
import org.objectweb.asm.Opcodes;

import java.lang.reflect.Constructor;

/**
 *
 */
public class ParserFactory<P extends Parser> implements Opcodes {
    private final Class<P> parserClass;
    private final ParserSetup<P> parserSetup;
    private final boolean failOnError;
    private Class<P> wrapperClass;

    public ParserFactory(Class parserClass, boolean failOnError, ParserSetup<P> parserSetup) {
        assert parserClass != null;

        this.parserClass = parserClass;
        this.parserSetup = parserSetup;
        this.failOnError = failOnError;
    }

    public Class getParserClass() {
        return parserClass;
    }

    public Parser generate(TokenStream input) throws Exception {
        Class<P> wc = getWrapperClass();
        Constructor c = wc.getConstructor(TokenStream.class);
        ParserWrapper wrapper = (ParserWrapper) c.newInstance(input);

        if (parserSetup != null) {
            parserSetup.config((P) wrapper);
        }

        wrapper.setFailOnError(failOnError);

        return (Parser) wrapper;
    }

    public Parser generate(TokenStream input, RecognizerSharedState state) throws Exception {
        Class<P> wc = getWrapperClass();
        Constructor c = wc.getConstructor(CharStream.class, RecognizerSharedState.class);
        ParserWrapper wrapper = (ParserWrapper) c.newInstance(input, state);

        if (parserSetup != null) {
            parserSetup.config((P) wrapper);
        }

        wrapper.setFailOnError(failOnError);

        return (Parser) wrapper;
    }

    private Class<P> getWrapperClass() {
        if (wrapperClass == null)
            wrapperClass = loadClass();
        return wrapperClass;
    }

    private Class<P> loadClass() {
        final String parent = parserClass.getName().replace('.', '/');
        final String name = "com/toolazydogs/aunit/asm/" + parent;

        ClassWriter cw = new ClassWriter(0);
        FieldVisitor fv;
        MethodVisitor mv;

        cw.visit(V1_5, ACC_PUBLIC + ACC_SUPER, name, null, parent,
                new String[] { "com/toolazydogs/aunit/internal/ParserWrapper" });

        {
            fv = cw.visitField(ACC_PRIVATE + ACC_FINAL, "errors", "Ljava/util/List;",
                    "Ljava/util/List<Ljava/lang/String;>;", null);
            fv.visitEnd();
        }
        {
            fv = cw.visitField(ACC_PRIVATE, "failOnError", "Z", null, null);
            fv.visitEnd();
        }
        {
            mv = cw.visitMethod(ACC_PUBLIC, "<init>", "(Lorg/antlr/runtime/TokenStream;)V", null, null);
            mv.visitCode();
            mv.visitVarInsn(ALOAD, 0);
            mv.visitVarInsn(ALOAD, 1);
            mv.visitMethodInsn(INVOKESPECIAL, parent, "<init>", "(Lorg/antlr/runtime/TokenStream;)V");
            mv.visitVarInsn(ALOAD, 0);
            mv.visitTypeInsn(NEW, "java/util/ArrayList");
            mv.visitInsn(DUP);
            mv.visitMethodInsn(INVOKESPECIAL, "java/util/ArrayList", "<init>", "()V");
            mv.visitFieldInsn(PUTFIELD, name, "errors", "Ljava/util/List;");
            mv.visitVarInsn(ALOAD, 0);
            mv.visitInsn(ICONST_1);
            mv.visitFieldInsn(PUTFIELD, name, "failOnError", "Z");
            mv.visitInsn(RETURN);
            mv.visitMaxs(3, 2);
            mv.visitEnd();
        }
        {
            mv = cw.visitMethod(ACC_PUBLIC, "<init>",
                    "(Lorg/antlr/runtime/TokenStream;Lorg/antlr/runtime/RecognizerSharedState;)V", null, null);
            mv.visitCode();
            mv.visitVarInsn(ALOAD, 0);
            mv.visitVarInsn(ALOAD, 1);
            mv.visitVarInsn(ALOAD, 2);
            mv.visitMethodInsn(INVOKESPECIAL, parent, "<init>",
                    "(Lorg/antlr/runtime/TokenStream;Lorg/antlr/runtime/RecognizerSharedState;)V");
            mv.visitVarInsn(ALOAD, 0);
            mv.visitTypeInsn(NEW, "java/util/ArrayList");
            mv.visitInsn(DUP);
            mv.visitMethodInsn(INVOKESPECIAL, "java/util/ArrayList", "<init>", "()V");
            mv.visitFieldInsn(PUTFIELD, name, "errors", "Ljava/util/List;");
            mv.visitVarInsn(ALOAD, 0);
            mv.visitInsn(ICONST_1);
            mv.visitFieldInsn(PUTFIELD, name, "failOnError", "Z");
            mv.visitInsn(RETURN);
            mv.visitMaxs(3, 3);
            mv.visitEnd();
        }
        {
            mv = cw.visitMethod(ACC_PUBLIC, "emitErrorMessage", "(Ljava/lang/String;)V", null, null);
            mv.visitCode();
            mv.visitVarInsn(ALOAD, 0);
            mv.visitFieldInsn(GETFIELD, name, "errors", "Ljava/util/List;");
            mv.visitVarInsn(ALOAD, 1);
            mv.visitMethodInsn(INVOKEINTERFACE, "java/util/List", "add", "(Ljava/lang/Object;)Z");
            mv.visitInsn(POP);
            mv.visitInsn(RETURN);
            mv.visitMaxs(2, 2);
            mv.visitEnd();
        }
        {
            mv = cw.visitMethod(ACC_PUBLIC, "getErrors", "()Ljava/util/List;",
                    "()Ljava/util/List<Ljava/lang/String;>;", null);
            mv.visitCode();
            mv.visitVarInsn(ALOAD, 0);
            mv.visitFieldInsn(GETFIELD, name, "errors", "Ljava/util/List;");
            mv.visitInsn(ARETURN);
            mv.visitMaxs(1, 1);
            mv.visitEnd();
        }
        {
            mv = cw.visitMethod(ACC_PUBLIC, "isFailOnError", "()Z", null, null);
            mv.visitCode();
            mv.visitVarInsn(ALOAD, 0);
            mv.visitFieldInsn(GETFIELD, name, "failOnError", "Z");
            mv.visitInsn(IRETURN);
            mv.visitMaxs(1, 1);
            mv.visitEnd();
        }
        {
            mv = cw.visitMethod(ACC_PUBLIC, "setFailOnError", "(Z)V", null, null);
            mv.visitCode();
            mv.visitVarInsn(ALOAD, 0);
            mv.visitVarInsn(ILOAD, 1);
            mv.visitFieldInsn(PUTFIELD, name, "failOnError", "Z");
            mv.visitInsn(RETURN);
            mv.visitMaxs(2, 2);
            mv.visitEnd();
        }
        cw.visitEnd();

        byte[] b = cw.toByteArray();

        AunitClassloader ac = new AunitClassloader(parserClass.getClassLoader());
        return ac.defineClass(name.replace('/', '.'), b);
    }
}