com.facebook.presto.bytecode.ClassDefinition 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.ImmutableSet;
import org.objectweb.asm.ClassVisitor;
import javax.annotation.concurrent.NotThreadSafe;
import java.util.ArrayList;
import java.util.EnumSet;
import java.util.List;
import java.util.Set;
import static com.facebook.presto.bytecode.Access.STATIC;
import static com.facebook.presto.bytecode.Access.a;
import static com.facebook.presto.bytecode.Access.toAccessModifier;
import static com.google.common.collect.Iterables.any;
import static com.google.common.collect.Iterables.concat;
import static java.util.Objects.requireNonNull;
import static org.objectweb.asm.Opcodes.ACC_SUPER;
import static org.objectweb.asm.Opcodes.V1_7;
@NotThreadSafe
public class ClassDefinition
{
private final EnumSet access;
private final ParameterizedType type;
private final ParameterizedType superClass;
private final List interfaces = new ArrayList<>();
private final List annotations = new ArrayList<>();
private final List fields = new ArrayList<>();
private final List methods = new ArrayList<>();
private final MethodDefinition classInitializer;
private String source;
private String debug;
public ClassDefinition(
EnumSet access,
String name,
ParameterizedType superClass,
ParameterizedType... interfaces)
{
this(access, new ParameterizedType(name), superClass, interfaces);
}
public ClassDefinition(
EnumSet access,
ParameterizedType type,
ParameterizedType superClass,
ParameterizedType... interfaces)
{
requireNonNull(access, "access is null");
requireNonNull(access, "access is null");
requireNonNull(superClass, "superClass is null");
requireNonNull(interfaces, "interfaces is null");
this.access = access;
this.type = type;
this.superClass = superClass;
this.interfaces.addAll(ImmutableList.copyOf(interfaces));
classInitializer = new MethodDefinition(this, a(STATIC), "", ParameterizedType.type(void.class), ImmutableList.of());
}
public Set getAccess()
{
return ImmutableSet.copyOf(access);
}
public String getName()
{
return type.getClassName();
}
public ParameterizedType getType()
{
return type;
}
public ParameterizedType getSuperClass()
{
return superClass;
}
public String getSource()
{
return source;
}
public List getInterfaces()
{
return ImmutableList.copyOf(interfaces);
}
public List getAnnotations()
{
return ImmutableList.copyOf(annotations);
}
public List getFields()
{
return ImmutableList.copyOf(fields);
}
public List getMethods()
{
return ImmutableList.copyOf(methods);
}
public void visit(ClassVisitor visitor)
{
// Generic signature if super class or any interface is generic
String signature = null;
if (superClass.isGeneric() || any(interfaces, ParameterizedType::isGeneric)) {
signature = genericClassSignature(superClass, interfaces);
}
String[] interfaces = new String[this.interfaces.size()];
for (int i = 0; i < interfaces.length; i++) {
interfaces[i] = this.interfaces.get(i).getClassName();
}
visitor.visit(V1_7, toAccessModifier(access) | ACC_SUPER, type.getClassName(), signature, superClass.getClassName(), interfaces);
// visit source
if (source != null) {
visitor.visitSource(source, debug);
}
// visit annotations
for (AnnotationDefinition annotation : annotations) {
annotation.visitClassAnnotation(visitor);
}
// visit fields
for (FieldDefinition field : fields) {
field.visit(visitor);
}
// visit clinit method
classInitializer.visit(visitor, true);
// visit methods
for (MethodDefinition method : methods) {
method.visit(visitor);
}
// done
visitor.visitEnd();
}
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 FieldDefinition declareField(EnumSet access, String name, Class> type)
{
FieldDefinition fieldDefinition = new FieldDefinition(this, access, name, type);
fields.add(fieldDefinition);
return fieldDefinition;
}
public ClassDefinition addField(EnumSet access, String name, Class> type)
{
declareField(access, name, type);
return this;
}
public FieldDefinition declareField(EnumSet access, String name, ParameterizedType type)
{
FieldDefinition fieldDefinition = new FieldDefinition(this, access, name, type);
fields.add(fieldDefinition);
return fieldDefinition;
}
public ClassDefinition addField(EnumSet access, String name, ParameterizedType type)
{
declareField(access, name, type);
return this;
}
public ClassDefinition addField(FieldDefinition field)
{
fields.add(field);
return this;
}
public MethodDefinition getClassInitializer()
{
return classInitializer;
}
public MethodDefinition declareConstructor(
EnumSet access,
Parameter... parameters)
{
return declareMethod(access, "", ParameterizedType.type(void.class), ImmutableList.copyOf(parameters));
}
public MethodDefinition declareConstructor(
EnumSet access,
Iterable parameters)
{
return declareMethod(access, "", ParameterizedType.type(void.class), ImmutableList.copyOf(parameters));
}
public ClassDefinition declareDefaultConstructor(EnumSet access)
{
MethodDefinition constructor = declareConstructor(access);
constructor
.getBody()
.append(constructor.getThis())
.invokeConstructor(superClass)
.ret();
return this;
}
public ClassDefinition addMethod(MethodDefinition method)
{
methods.add(method);
return this;
}
public ClassDefinition visitSource(String source, String debug)
{
this.source = source;
this.debug = debug;
return this;
}
public MethodDefinition declareMethod(
EnumSet access,
String name,
ParameterizedType returnType,
Parameter... parameters)
{
return declareMethod(access, name, returnType, ImmutableList.copyOf(parameters));
}
public MethodDefinition declareMethod(
EnumSet access,
String name,
ParameterizedType returnType,
Iterable parameters)
{
MethodDefinition methodDefinition = new MethodDefinition(this, access, name, returnType, parameters);
for (MethodDefinition method : methods) {
if (name.equals(method.getName()) && method.getParameterTypes().equals(methodDefinition.getParameterTypes())) {
throw new IllegalArgumentException("Method with same name and signature already exists: " + name);
}
}
methods.add(methodDefinition);
return methodDefinition;
}
public static String genericClassSignature(
ParameterizedType classType,
ParameterizedType... interfaceTypes
)
{
return Joiner.on("").join(
concat(ImmutableList.of(classType), ImmutableList.copyOf(interfaceTypes))
);
}
public static String genericClassSignature(
ParameterizedType classType,
List interfaceTypes
)
{
return Joiner.on("").join(concat(ImmutableList.of(classType), interfaceTypes));
}
@Override
public String toString()
{
final StringBuilder sb = new StringBuilder();
sb.append("ClassDefinition");
sb.append("{access=").append(access);
sb.append(", type=").append(type);
sb.append('}');
return sb.toString();
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy