All Downloads are FREE. Search and download functionalities are using the official Maven repository.

net.wpm.codegen.ExpressionCall Maven / Gradle / Ivy

The newest version!
/*
 * Copyright (C) 2015 SoftIndex LLC.
 *
 * 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 net.wpm.codegen;

import org.objectweb.asm.Type;
import org.objectweb.asm.commons.GeneratorAdapter;

import net.wpm.codegen.Context;
import net.wpm.codegen.Expression;
import net.wpm.codegen.ExpressionCall;

import java.lang.reflect.Method;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;

import static java.lang.String.format;
import static net.wpm.codegen.Utils.*;
import static org.objectweb.asm.Type.getType;

/**
 * Defines methods for using static methods from other classes
 */
public final class ExpressionCall implements Expression {
	private final Expression owner;
	private final String methodName;
	private final List arguments = new ArrayList<>();

	ExpressionCall(Expression owner, String methodName, Expression... arguments) {
		this.owner = owner;
		this.methodName = methodName;
		this.arguments.addAll(Arrays.asList(arguments));
	}

	@Override
	public Type type(Context ctx) {
		List> argumentClasses = new ArrayList<>();
		List argumentTypes = new ArrayList<>();
		for (Expression argument : arguments) {
			argumentTypes.add(argument.type(ctx));
			if (argument.type(ctx).equals(getType(Object[].class))) {
				argumentClasses.add(Object[].class);
			} else {
				argumentClasses.add(getJavaType(ctx.getClassLoader(), argument.type(ctx)));
			}
		}
		Type returnType;
		try {
			if (ctx.getThisType().equals(owner.type(ctx))) {
				for (org.objectweb.asm.commons.Method method : ctx.getMethods()) {
					if (method.getName().equals(methodName)) {
						if (method.getArgumentTypes().length == arguments.size()) {
							Type[] methodTypes = method.getArgumentTypes();
							boolean isSame = true;
							for (int i = 0; i < arguments.size(); i++) {
								if (!methodTypes[i].equals(argumentTypes.get(i))) {
									isSame = false;
									break;
								}
							}
							if (isSame) return method.getReturnType();
						}
					}
				}
				throw new NoSuchMethodException();
			} else {
				Class ownerJavaType = getJavaType(ctx.getClassLoader(), owner.type(ctx));
				Method method = ownerJavaType.getMethod(methodName, argumentClasses.toArray(new Class[]{}));
				Class returnClass = method.getReturnType();
				returnType = getType(returnClass);
			}

		} catch (NoSuchMethodException e) {
			throw new RuntimeException(format("No method %s.%s(%s). %s",
					owner.type(ctx).getClassName(),
					methodName,
					(!argumentClasses.isEmpty() ? argsToString(argumentClasses) : ""),
					exceptionInGeneratedClass(ctx)
			));
		}

		return returnType;
	}

	@Override
	public Type load(Context ctx) {
		GeneratorAdapter g = ctx.getGeneratorAdapter();

		owner.load(ctx);

		List> argumentClasses = new ArrayList<>();
		List argumentTypes = new ArrayList<>();
		for (Expression argument : arguments) {
			argument.load(ctx);
			argumentTypes.add(argument.type(ctx));
			argumentClasses.add(getJavaType(ctx.getClassLoader(), argument.type(ctx)));
		}

		Type returnType;
		try {
			if (ctx.getThisType().equals(owner.type(ctx))) {
				org.objectweb.asm.commons.Method method = null;
				for (org.objectweb.asm.commons.Method m : ctx.getMethods()) {
					if (m.getName().equals(methodName)) {
						if (m.getArgumentTypes().length == arguments.size()) {
							Type[] methodTypes = m.getArgumentTypes();
							boolean isSame = true;
							for (int i = 0; i < arguments.size(); i++) {
								if (!methodTypes[i].equals(argumentTypes.get(i))) {
									isSame = false;
									break;
								}
							}
							if (isSame) {
								method = m;
								break;
							}
						}
					}
				}
				if (method == null) throw new NoSuchMethodException();
				g.invokeVirtual(owner.type(ctx), method);
				return method.getReturnType();
			}
			Class ownerJavaType = getJavaType(ctx.getClassLoader(), owner.type(ctx));
			Method method = ownerJavaType.getMethod(methodName, argumentClasses.toArray(new Class[]{}));
			Class returnClass = method.getReturnType();
			returnType = getType(returnClass);

			invokeVirtualOrInterface(g, ownerJavaType, new org.objectweb.asm.commons.Method(methodName,
					returnType, argumentTypes.toArray(new Type[]{})));

		} catch (NoSuchMethodException e) {
			throw new RuntimeException(format("No method %s.%s(%s). %s",
					owner.type(ctx).getClassName(),
					methodName,
					(!argumentClasses.isEmpty() ? argsToString(argumentClasses) : ""),
					exceptionInGeneratedClass(ctx)));
		}
		return returnType;
	}

	@Override
	public boolean equals(Object o) {
		if (this == o) return true;
		if (o == null || getClass() != o.getClass()) return false;

		ExpressionCall that = (ExpressionCall) o;

		if (arguments != null ? !arguments.equals(that.arguments) : that.arguments != null) return false;
		if (methodName != null ? !methodName.equals(that.methodName) : that.methodName != null) return false;
		if (owner != null ? !owner.equals(that.owner) : that.owner != null) return false;

		return true;
	}

	@Override
	public int hashCode() {
		int result = owner != null ? owner.hashCode() : 0;
		result = 31 * result + (methodName != null ? methodName.hashCode() : 0);
		result = 31 * result + (arguments != null ? arguments.hashCode() : 0);
		return result;
	}
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy