com.facebook.presto.bytecode.MethodDefinition Maven / Gradle / Ivy
/*
* 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.facebook.presto.bytecode;
import com.google.common.base.Joiner;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.Lists;
import org.objectweb.asm.ClassVisitor;
import org.objectweb.asm.MethodVisitor;
import org.objectweb.asm.tree.InsnNode;
import javax.annotation.concurrent.NotThreadSafe;
import java.util.ArrayList;
import java.util.EnumSet;
import java.util.List;
import java.util.Optional;
import static com.facebook.presto.bytecode.Access.STATIC;
import static com.facebook.presto.bytecode.Access.toAccessModifier;
import static com.facebook.presto.bytecode.ParameterizedType.type;
import static com.google.common.collect.Iterables.transform;
import static org.objectweb.asm.Opcodes.RETURN;
@SuppressWarnings("UnusedDeclaration")
@NotThreadSafe
public class MethodDefinition
{
private final Scope scope;
private final ClassDefinition declaringClass;
private final EnumSet access;
private final String name;
private final List annotations = new ArrayList<>();
private final ParameterizedType returnType;
private final List parameters;
private final List parameterTypes;
private final List> parameterAnnotations;
private final List exceptions = new ArrayList<>();
private final BytecodeBlock body;
private String comment;
public MethodDefinition(
ClassDefinition declaringClass,
EnumSet access,
String name,
ParameterizedType returnType,
Parameter... parameters
)
{
this(declaringClass, access, name, returnType, ImmutableList.copyOf(parameters));
}
public MethodDefinition(
ClassDefinition declaringClass,
EnumSet access,
String name,
ParameterizedType returnType,
Iterable parameters
)
{
this.declaringClass = declaringClass;
body = new BytecodeBlock();
this.access = access;
this.name = name;
if (returnType != null) {
this.returnType = returnType;
}
else {
this.returnType = type(void.class);
}
this.parameters = ImmutableList.copyOf(parameters);
this.parameterTypes = Lists.transform(this.parameters, Parameter::getType);
this.parameterAnnotations = ImmutableList.copyOf(transform(parameters, input -> new ArrayList<>()));
Optional thisType = Optional.empty();
if (!access.contains(STATIC)) {
thisType = Optional.of(declaringClass.getType());
}
scope = new Scope(thisType, parameters);
}
public ClassDefinition getDeclaringClass()
{
return declaringClass;
}
public List getAnnotations()
{
return ImmutableList.copyOf(annotations);
}
public List getParameterAnnotations(int index)
{
return ImmutableList.copyOf(parameterAnnotations.get(index));
}
public EnumSet getAccess()
{
return access;
}
public String getName()
{
return name;
}
public ParameterizedType getReturnType()
{
return returnType;
}
public List getParameters()
{
return parameters;
}
public List getParameterTypes()
{
return parameterTypes;
}
public List getExceptions()
{
return exceptions;
}
public MethodDefinition addException(Class extends Throwable> exceptionClass)
{
exceptions.add(type(exceptionClass));
return this;
}
public MethodDefinition comment(String format, Object... args)
{
this.comment = String.format(format, args);
return this;
}
public String getComment()
{
return comment;
}
public Scope getScope()
{
return scope;
}
public Variable getThis()
{
return scope.getThis();
}
public String getMethodDescriptor()
{
return methodDescription(returnType, parameterTypes);
}
public BytecodeBlock getBody()
{
return body;
}
public AnnotationDefinition declareAnnotation(Class> type)
{
AnnotationDefinition annotationDefinition = new AnnotationDefinition(type);
annotations.add(annotationDefinition);
return annotationDefinition;
}
public AnnotationDefinition declareAnnotation(ParameterizedType type)
{
AnnotationDefinition annotationDefinition = new AnnotationDefinition(type);
annotations.add(annotationDefinition);
return annotationDefinition;
}
public AnnotationDefinition declareParameterAnnotation(Class> type, int parameterIndex)
{
AnnotationDefinition annotationDefinition = new AnnotationDefinition(type);
parameterAnnotations.get(parameterIndex).add(annotationDefinition);
return annotationDefinition;
}
public AnnotationDefinition declareParameterAnnotation(ParameterizedType type, int parameterIndex)
{
AnnotationDefinition annotationDefinition = new AnnotationDefinition(type);
parameterAnnotations.get(parameterIndex).add(annotationDefinition);
return annotationDefinition;
}
public void visit(ClassVisitor visitor)
{
visit(visitor, false);
}
public void visit(ClassVisitor visitor, boolean addReturn)
{
String[] exceptions = new String[this.exceptions.size()];
for (int i = 0; i < exceptions.length; i++) {
exceptions[i] = this.exceptions.get(i).getClassName();
}
MethodVisitor methodVisitor = visitor.visitMethod(toAccessModifier(access),
name,
getMethodDescriptor(),
genericMethodSignature(returnType, parameterTypes),
exceptions);
if (methodVisitor == null) {
return;
}
// visit method annotations
for (AnnotationDefinition annotation : annotations) {
annotation.visitMethodAnnotation(methodVisitor);
}
// visit parameter annotations
for (int parameterIndex = 0; parameterIndex < parameterAnnotations.size(); parameterIndex++) {
List parameterAnnotations1 = this.parameterAnnotations.get(parameterIndex);
for (AnnotationDefinition parameterAnnotation : parameterAnnotations1) {
parameterAnnotation.visitParameterAnnotation(parameterIndex, methodVisitor);
}
}
// visit code
methodVisitor.visitCode();
// visit instructions
MethodGenerationContext generationContext = new MethodGenerationContext(methodVisitor);
generationContext.enterScope(scope);
body.accept(methodVisitor, generationContext);
if (addReturn) {
new InsnNode(RETURN).accept(methodVisitor);
}
generationContext.exitScope(scope);
// done
methodVisitor.visitMaxs(-1, -1);
methodVisitor.visitEnd();
}
public String toSourceString()
{
StringBuilder sb = new StringBuilder();
Joiner.on(' ').appendTo(sb, access).append(' ');
sb.append(returnType.getJavaClassName()).append(' ');
sb.append(name).append('(');
Joiner.on(", ").appendTo(sb, transform(parameters, Parameter::getSourceString)).append(')');
return sb.toString();
}
@Override
public String toString()
{
return toSourceString();
}
public static String methodDescription(Class> returnType, Class>... parameterTypes)
{
return methodDescription(returnType, ImmutableList.copyOf(parameterTypes));
}
public static String methodDescription(Class> returnType, List> parameterTypes)
{
return methodDescription(
type(returnType),
Lists.transform(parameterTypes, ParameterizedType::type)
);
}
public static String methodDescription(
ParameterizedType returnType,
ParameterizedType... parameterTypes
)
{
return methodDescription(returnType, ImmutableList.copyOf(parameterTypes));
}
public static String methodDescription(
ParameterizedType returnType,
List parameterTypes
)
{
StringBuilder sb = new StringBuilder();
sb.append("(");
Joiner.on("").appendTo(sb, transform(parameterTypes, ParameterizedType::getType));
sb.append(")");
sb.append(returnType.getType());
return sb.toString();
}
public static String genericMethodSignature(
ParameterizedType returnType,
ParameterizedType... parameterTypes
)
{
return genericMethodSignature(returnType, ImmutableList.copyOf(parameterTypes));
}
public static String genericMethodSignature(
ParameterizedType returnType,
List parameterTypes
)
{
StringBuilder sb = new StringBuilder();
sb.append("(");
Joiner.on("").appendTo(sb, parameterTypes);
sb.append(")");
sb.append(returnType);
return sb.toString();
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy