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

org.teavm.model.ClassReaderSourceHelper Maven / Gradle / Ivy

There is a newer version: 0.10.2
Show newest version
/*
 *  Copyright 2017 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.model;

import java.util.ArrayList;
import java.util.List;
import java.util.Optional;
import java.util.Set;

final class ClassReaderSourceHelper {
    private ClassReaderSourceHelper() {
    }

    static MethodReader resolveMethodImplementation(ClassReaderSource classSource, String className,
            MethodDescriptor methodDescriptor, Set visited) {
        if (!visited.add(className)) {
            return null;
        }

        ClassReader cls = classSource.get(className);
        if (cls == null) {
            return null;
        }

        MethodReader method = cls.getMethod(methodDescriptor);
        if (method != null && !method.hasModifier(ElementModifier.ABSTRACT)) {
            return method;
        }

        MethodReader mostSpecificMethod = null;
        List superClasses = new ArrayList<>();
        if (cls.getParent() != null) {
            superClasses.add(cls.getParent());
        }
        superClasses.addAll(cls.getInterfaces());

        for (String superClass : superClasses) {
            MethodReader resultFromSuperClass = resolveMethodImplementation(classSource, superClass,
                    methodDescriptor, visited);
            if (resultFromSuperClass != null) {
                if (mostSpecificMethod == null || classSource.isSuperType(mostSpecificMethod.getOwnerName(),
                        resultFromSuperClass.getOwnerName()).orElse(false)) {
                    mostSpecificMethod = resultFromSuperClass;
                }
            }
        }

        return mostSpecificMethod;
    }

    static Optional isSuperType(ClassReaderSource classSource, String superType, String subType) {
        if (superType.equals("java.lang.Object")) {
            return Optional.of(true);
        }

        ClassReader cls = classSource.get(superType);
        if (cls != null && !cls.hasModifier(ElementModifier.INTERFACE)) {
            return isSuperTypeSimple(classSource, superType, subType);
        }

        return isSuperTypeInterface(classSource, superType, subType);
    }

    private static Optional isSuperTypeSimple(ClassReaderSource classSource,
            String superType, String subType) {
        while (!superType.equals(subType)) {
            ClassReader cls = classSource.get(subType);
            if (cls == null) {
                return Optional.empty();
            }

            subType = cls.getParent();
            if (subType == null) {
                return Optional.of(false);
            }
        }

        return Optional.of(true);
    }

    private static Optional isSuperTypeInterface(ClassReaderSource classSource,
            String superType, String subType) {
        if (superType.equals(subType)) {
            return Optional.of(true);
        }
        ClassReader cls = classSource.get(subType);
        if (cls == null) {
            return Optional.empty();
        }
        if (cls.getParent() != null) {
            if (isSuperTypeInterface(classSource, superType, cls.getParent()).orElse(false)) {
                return Optional.of(true);
            }
        }
        for (String iface : cls.getInterfaces()) {
            if (isSuperTypeInterface(classSource, superType, iface).orElse(false)) {
                return Optional.of(true);
            }
        }
        return Optional.of(false);
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy