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

org.teavm.jso.impl.JavaInvocationProcessor Maven / Gradle / Ivy

There is a newer version: 0.10.2
Show newest version
/*
 *  Copyright 2015 Alexey Andreev.
 *
 *  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.teavm.jso.impl;

import java.util.HashSet;
import org.mozilla.javascript.Token;
import org.mozilla.javascript.ast.AstNode;
import org.mozilla.javascript.ast.FunctionCall;
import org.mozilla.javascript.ast.InfixExpression;
import org.mozilla.javascript.ast.Name;
import org.mozilla.javascript.ast.NodeVisitor;
import org.mozilla.javascript.ast.PropertyGet;
import org.mozilla.javascript.ast.StringLiteral;
import org.teavm.diagnostics.Diagnostics;
import org.teavm.jso.JSObject;
import org.teavm.model.CallLocation;
import org.teavm.model.ClassReaderSource;
import org.teavm.model.ElementModifier;
import org.teavm.model.MethodReader;
import org.teavm.model.MethodReference;
import org.teavm.model.ValueType;

/**
 *
 * @author Alexey Andreev
 */
class JavaInvocationProcessor implements NodeVisitor {
    private ClassReaderSource classSource;
    private JSTypeHelper typeHelper;
    private JSBodyRepository repository;
    private Diagnostics diagnostics;
    private CallLocation location;
    private int idGenerator;

    public JavaInvocationProcessor(JSTypeHelper typeHelper, JSBodyRepository repository,
            ClassReaderSource classSource, Diagnostics diagnostics) {
        this.typeHelper = typeHelper;
        this.repository = repository;
        this.classSource = classSource;
        this.diagnostics = diagnostics;
    }

    public void process(CallLocation location, AstNode root) {
        this.location = location;
        root.visit(this);
    }

    @Override
    public boolean visit(AstNode node) {
        if (node instanceof FunctionCall) {
            return visit((FunctionCall) node);
        }
        return true;
    }

    private boolean visit(FunctionCall call) {
        if (!(call.getTarget() instanceof PropertyGet)) {
            return true;
        }

        PropertyGet propertyGet = (PropertyGet) call.getTarget();
        MethodReference methodRef = getJavaMethodSelector(propertyGet.getTarget());
        if (methodRef == null || !propertyGet.getProperty().getIdentifier().equals("invoke")) {
            return true;
        }

        for (AstNode arg : call.getArguments()) {
            arg.visit(this);
        }

        MethodReader method = classSource.resolve(methodRef);
        if (method == null) {
            diagnostics.error(location, "Java method not found: {{m0}}", methodRef);
            return false;
        }

        int requiredParams = methodRef.parameterCount();
        if (!method.hasModifier(ElementModifier.STATIC)) {
            ++requiredParams;
        }
        if (call.getArguments().size() != requiredParams) {
            diagnostics.error(location, "Invalid number of arguments for method {{m0}}. Expected: " + requiredParams
                    + ", encountered: " + call.getArguments().size(), methodRef);
        }

        MethodReference caller = createCallbackMethod(method);
        MethodReference delegate = repository.methodMap.get(location.getMethod());
        repository.callbackCallees.put(caller, methodRef);
        repository.callbackMethods.computeIfAbsent(delegate, key -> new HashSet<>()).add(caller);
        validateSignature(method);

        StringLiteral newTarget = new StringLiteral();
        newTarget.setValue("$$JSO$$_" + caller);
        propertyGet.setTarget(newTarget);

        return false;
    }

    private void validateSignature(MethodReader method) {
        if (!method.hasModifier(ElementModifier.STATIC)) {
            if (!typeHelper.isJavaScriptClass(method.getOwnerName())) {
                diagnostics.error(location, "Can't call method {{m0}} of non-JS class", method.getReference());
            }
        }
        for (int i = 0; i < method.parameterCount(); ++i) {
            if (!typeHelper.isSupportedType(method.parameterType(i))) {
                diagnostics.error(location, "Invalid type {{t0}} of parameter " + (i + 1) + " of method {{m1}}",
                        method.parameterType(i), method.getReference());
            }
        }
        if (method.getResultType() != ValueType.VOID && !typeHelper.isSupportedType(method.getResultType())) {
            diagnostics.error(location, "Invalid type {{t0}} of return value of method {{m1}}", method.getResultType(),
                    method.getReference());
        }
    }

    private MethodReference createCallbackMethod(MethodReader method) {
        int paramCount = method.parameterCount();
        if (!method.hasModifier(ElementModifier.STATIC)) {
            paramCount++;
        }
        ValueType[] signature = new ValueType[paramCount + 1];
        for (int i = 0; i < paramCount; ++i) {
            signature[i] = ValueType.object(JSObject.class.getName());
        }
        signature[paramCount] = method.getResultType() == ValueType.VOID ? ValueType.VOID
                : ValueType.object(JSObject.class.getName());
        return new MethodReference(location.getMethod().getClassName(),
                method.getName() + "$jsocb$_" + idGenerator++, signature);
    }

    private MethodReference getJavaMethodSelector(AstNode node) {
        if (!(node instanceof FunctionCall)) {
            return null;
        }
        FunctionCall call = (FunctionCall) node;
        if (!isJavaMethodRepository(call.getTarget())) {
            return null;
        }
        if (call.getArguments().size() != 1) {
            diagnostics.error(location, "javaMethods.get method should take exactly one argument");
            return null;
        }
        StringBuilder nameBuilder = new StringBuilder();
        if (!extractMethodName(call.getArguments().get(0), nameBuilder)) {
            diagnostics.error(location, "javaMethods.get method should take string constant");
            return null;
        }

        MethodReference method = MethodReference.parseIfPossible(nameBuilder.toString());
        if (method == null) {
            diagnostics.error(location, "Wrong method reference: " + nameBuilder);
        }
        return method;
    }

    private boolean extractMethodName(AstNode node, StringBuilder sb) {
        if (node.getType() == Token.ADD) {
            InfixExpression infix = (InfixExpression) node;
            return extractMethodName(infix.getLeft(), sb) && extractMethodName(infix.getRight(), sb);
        } else if (node.getType() == Token.STRING) {
            sb.append(((StringLiteral) node).getValue());
            return true;
        } else {
            return false;
        }
    }

    private boolean isJavaMethodRepository(AstNode node) {
        if (!(node instanceof PropertyGet)) {
            return false;
        }
        PropertyGet propertyGet = (PropertyGet) node;

        if (!(propertyGet.getLeft() instanceof Name)) {
            return false;
        }
        if (!((Name) propertyGet.getTarget()).getIdentifier().equals("javaMethods")) {
            return false;
        }
        if (!propertyGet.getProperty().getIdentifier().equals("get")) {
            return false;
        }

        return true;
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy