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

pascal.taie.analysis.pta.plugin.reflection.LogBasedModel Maven / Gradle / Ivy

The newest version!
/*
 * Tai-e: A Static Analysis Framework for Java
 *
 * Copyright (C) 2022 Tian Tan 
 * Copyright (C) 2022 Yue Li 
 *
 * This file is part of Tai-e.
 *
 * Tai-e is free software: you can redistribute it and/or modify
 * it under the terms of the GNU Lesser General Public License
 * as published by the Free Software Foundation, either version 3
 * of the License, or (at your option) any later version.
 *
 * Tai-e is distributed in the hope that it will be useful,but WITHOUT
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
 * or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General
 * Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public
 * License along with Tai-e. If not, see .
 */

package pascal.taie.analysis.pta.plugin.reflection;

import org.apache.logging.log4j.LogManager;
import org.apache.logging.log4j.Logger;
import pascal.taie.analysis.pta.core.cs.context.Context;
import pascal.taie.analysis.pta.core.cs.element.CSMethod;
import pascal.taie.analysis.pta.core.solver.Solver;
import pascal.taie.analysis.pta.plugin.Plugin;
import pascal.taie.analysis.pta.plugin.util.InvokeUtils;
import pascal.taie.analysis.pta.plugin.util.SolverHolder;
import pascal.taie.ir.exp.Var;
import pascal.taie.ir.stmt.Invoke;
import pascal.taie.language.classes.ClassMember;
import pascal.taie.language.classes.ClassNames;
import pascal.taie.language.classes.JClass;
import pascal.taie.language.classes.JField;
import pascal.taie.language.classes.JMethod;
import pascal.taie.language.type.ArrayType;
import pascal.taie.language.type.ClassType;
import pascal.taie.language.type.PrimitiveType;
import pascal.taie.util.collection.Maps;
import pascal.taie.util.collection.MultiMap;
import pascal.taie.util.collection.Sets;

import javax.annotation.Nullable;
import java.nio.file.Path;
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
import java.util.Map;
import java.util.Set;

import static pascal.taie.analysis.pta.plugin.util.InvokeUtils.BASE;

class LogBasedModel extends SolverHolder implements Plugin {

    private static final Logger logger = LogManager.getLogger(LogBasedModel.class);

    private final Set supportedApis = Set.of(
            "Class.forName",
            "Class.newInstance",
            "Constructor.newInstance",
            "Method.invoke",
            "Field.get",
            "Field.set",
            "Array.newInstance"
    );

    private final Map fullNames = Map.of(
            "Class", ClassNames.CLASS,
            "Constructor", ClassNames.CONSTRUCTOR,
            "Method", ClassNames.METHOD,
            "Field", ClassNames.FIELD,
            "Array", ClassNames.ARRAY
    );

    private final MetaObjHelper helper;

    private final Set loggedInvokes = Sets.newSet();

    private final Set relevantMethods = Sets.newSet();

    /**
     * Targets for Class.forName(...).
     */
    private final MultiMap forNameTargets = Maps.newMultiMap();

    /**
     * Targets for Class.newInstance().
     */
    private final MultiMap classTargets = Maps.newMultiMap();

    /**
     * Targets for Constructor.newInstance(...), Method.invoke(...), and Field.get/set(...).
     */
    private final MultiMap memberTargets = Maps.newMultiMap();

    /**
     * Targets for Array.newInstance(...).
     */
    private final MultiMap arrayTypeTargets = Maps.newMultiMap();

    /**
     * Callers or targets that are absent in the closed world.
     */
    private final Set missingItems = Sets.newSet();

    LogBasedModel(Solver solver, MetaObjHelper helper, @Nullable String logPath) {
        super(solver);
        this.helper = helper;
        if (logPath != null) {
            logger.info("Using reflection log from {}",
                    Path.of(logPath).toAbsolutePath());
            LogItem.load(logPath).forEach(this::addItem);
        }
    }

    private void addItem(LogItem item) {
        if (!supportedApis.contains(item.api)) {
            return;
        }
        // obtain reflective target
        Object target = switch (item.api) {
            case "Class.forName", "Class.newInstance" -> hierarchy.getClass(item.target);
            case "Constructor.newInstance", "Method.invoke" -> hierarchy.getMethod(item.target);
            case "Field.get", "Field.set" -> hierarchy.getField(item.target);
            case "Array.newInstance" -> typeSystem.getType(item.target);
            default -> null;
        };
        // ignore get/set of fields of primitive types
        if (target instanceof JField field &&
                field.getType() instanceof PrimitiveType) {
            return;
        }
        // add target specified in the item
        if (target != null) {
            List invokes = getMatchedInvokes(item);
            if (target instanceof JClass jclass) {
                if (item.api.equals("Class.forName")) {
                    for (Invoke invoke : invokes) {
                        forNameTargets.put(invoke, jclass);
                    }
                } else {
                    for (Invoke invoke : invokes) {
                        classTargets.put(invoke, jclass);
                    }
                }
            } else if (target instanceof ClassMember member) {
                for (Invoke invoke : invokes) {
                    memberTargets.put(invoke, member);
                }
            } else if (target instanceof ArrayType arrayType) {
                // Note that currently we only support Array.newInstance(Class,int),
                // and ignore primitive arrays.
                if (arrayType.baseType() instanceof ClassType baseClass) {
                    for (Invoke invoke : invokes) {
                        arrayTypeTargets.put(invoke, baseClass);
                    }
                }
            }
            invokes.forEach(invoke -> {
                loggedInvokes.add(invoke);
                relevantMethods.add(invoke.getContainer());
            });
        } else if (missingItems.add(item.target)) {
            logger.warn("Reflective target '{}' for {} is not found", item.target, item.api);
        }
    }

    private List getMatchedInvokes(LogItem item) {
        int lastDot = item.caller.lastIndexOf('.');
        String callerClass = item.caller.substring(0, lastDot);
        String callerMethod = item.caller.substring(lastDot + 1);
        JClass clazz = hierarchy.getClass(callerClass);
        if (clazz == null) {
            if (missingItems.add(callerClass)) {
                logger.warn("Reflective caller class '{}' is absent", callerClass);
            }
            return List.of();
        }
        List invokes = new ArrayList<>();
        clazz.getDeclaredMethods()
                .stream()
                .filter(m -> m.getName().equals(callerMethod) && !m.isAbstract())
                .forEach(caller ->
                        caller.getIR()
                                .invokes(false)
                                .filter(invoke -> isMatched(item, invoke))
                                .forEach(invokes::add));
        if (invokes.isEmpty()) {
            logger.warn("No matched invokes found for {}/{}",
                    item.caller, item.lineNumber);
        }
        return invokes;
    }

    private boolean isMatched(LogItem item, Invoke invoke) {
        int lastDot = item.api.lastIndexOf('.');
        String apiClass = fullNames.get(item.api.substring(0, lastDot));
        String apiMethod = item.api.substring(lastDot + 1);
        JMethod callee = invoke.getMethodRef().resolve();
        return callee.getDeclaringClass().getName().equals(apiClass) &&
                callee.getName().equals(apiMethod) &&
                (item.lineNumber == LogItem.UNKNOWN ||
                        item.lineNumber == invoke.getLineNumber());
    }

    /**
     * @return set of reflective invocations that are annotated by the log.
     */
    Set getInvokesWithLog() {
        return Collections.unmodifiableSet(loggedInvokes);
    }

    MultiMap getForNameTargets() {
        return forNameTargets;
    }

    @Override
    public void onNewCSMethod(CSMethod csMethod) {
        JMethod method = csMethod.getMethod();
        if (relevantMethods.contains(method)) {
            method.getIR()
                    .invokes(false)
                    .forEach(invoke -> {
                        handleForName(csMethod, invoke);
                        passTarget(classTargets, csMethod, invoke, BASE);
                        passTarget(memberTargets, csMethod, invoke, BASE);
                        passTarget(arrayTypeTargets, csMethod, invoke, 0);
                    });
        }
    }

    private void handleForName(CSMethod csMethod, Invoke invoke) {
        if (forNameTargets.containsKey(invoke)) {
            Context context = csMethod.getContext();
            Var result = invoke.getResult();
            forNameTargets.get(invoke).forEach(target -> {
                solver.initializeClass(target);
                if (result != null) {
                    solver.addVarPointsTo(context, result, helper.getLogMetaObj(target));
                }
            });
        }
    }

    private  void passTarget(MultiMap targetMap,
                                CSMethod csMethod, Invoke invoke, int index) {
        if (targetMap.containsKey(invoke)) {
            Context context = csMethod.getContext();
            Var var = InvokeUtils.getVar(invoke, index);
            targetMap.get(invoke).forEach(target ->
                    solver.addVarPointsTo(context, var, helper.getLogMetaObj(target)));
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy