
org.parboiled.transform.ParserTransformer Maven / Gradle / Ivy
/*
* Copyright (C) 2009-2011 Mathias Doenitz
*
* 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 org.parboiled.transform;
import static org.parboiled.common.Preconditions.*;
import org.objectweb.asm.ClassWriter;
import org.parboiled.common.ImmutableList;
import java.util.List;
import static org.parboiled.transform.AsmUtils.*;
public class ParserTransformer {
private ParserTransformer() {}
@SuppressWarnings({"unchecked"})
public static synchronized Class extends T> transformParser(Class parserClass) throws Exception {
checkArgNotNull(parserClass, "parserClass");
// first check whether we did not already create and load the extension of the given parser class
Class> extendedClass = AsmUtils.loadClass(
getExtendedParserClassName(parserClass.getName()), parserClass
);
return (Class extends T>)
(extendedClass != null ? extendedClass : extendParserClass(parserClass).getExtendedClass());
}
static ParserClassNode extendParserClass(Class> parserClass) throws Exception {
ParserClassNode classNode = new ParserClassNode(parserClass);
new ClassNodeInitializer().process(classNode);
runMethodTransformers(classNode);
new ConstructorGenerator().process(classNode);
defineExtendedParserClass(classNode);
return classNode;
}
@SuppressWarnings({"unchecked"})
private static void runMethodTransformers(ParserClassNode classNode) throws Exception {
List methodProcessors = createRuleMethodProcessors();
// iterate through all rule methods
// since the ruleMethods map on the classnode is a treemap we get the methods sorted by name which puts
// all super methods first (since they are prefixed with one or more '$')
for (RuleMethod ruleMethod : classNode.getRuleMethods().values()) {
if (!ruleMethod.hasDontExtend()) {
for (RuleMethodProcessor methodProcessor : methodProcessors) {
if (methodProcessor.appliesTo(classNode, ruleMethod)) {
methodProcessor.process(classNode, ruleMethod);
}
}
}
}
for (RuleMethod ruleMethod : classNode.getRuleMethods().values()) {
if (!ruleMethod.isGenerationSkipped()) {
classNode.methods.add(ruleMethod);
}
}
}
static List createRuleMethodProcessors() {
return ImmutableList.of(
new UnusedLabelsRemover(),
new ReturnInstructionUnifier(),
new InstructionGraphCreator(),
new ImplicitActionsConverter(),
new InstructionGroupCreator(),
new InstructionGroupPreparer(),
new ActionClassGenerator(false),
new VarInitClassGenerator(false),
new RuleMethodRewriter(),
new SuperCallRewriter(),
new BodyWithSuperCallReplacer(),
new VarFramingGenerator(),
new LabellingGenerator(),
new FlagMarkingGenerator(),
new CachingGenerator()
);
}
private static void defineExtendedParserClass(final ParserClassNode classNode) {
ClassWriter classWriter = new ClassWriter(ASMSettings.FRAMES) {
@Override
protected ClassLoader getClassLoader() {
return classNode.getParentClass().getClassLoader();
}
};
classNode.accept(classWriter);
classNode.setClassCode(classWriter.toByteArray());
classNode.setExtendedClass(defineClass(
classNode.name.replace('/', '.'),
classNode.getClassCode(),
classNode.getParentClass()
));
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy