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

pascal.taie.analysis.pta.plugin.invokedynamic.Java9StringConcatHandler 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.invokedynamic;

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.ir.exp.InvokeDynamic;
import pascal.taie.ir.exp.InvokeVirtual;
import pascal.taie.ir.exp.NewInstance;
import pascal.taie.ir.exp.Var;
import pascal.taie.ir.proginfo.MethodRef;
import pascal.taie.ir.stmt.Invoke;
import pascal.taie.ir.stmt.New;
import pascal.taie.ir.stmt.Stmt;
import pascal.taie.language.classes.ClassNames;
import pascal.taie.language.classes.JClass;
import pascal.taie.language.classes.JMethod;
import pascal.taie.language.classes.Signatures;
import pascal.taie.language.classes.Subsignature;
import pascal.taie.language.type.ClassType;
import pascal.taie.language.type.TypeSystem;
import pascal.taie.util.collection.Maps;

import java.util.ArrayList;
import java.util.Collection;
import java.util.List;
import java.util.Map;
import java.util.Objects;

/**
 * Since Java 9, String concatenation is implemented via invokedynamic.
 * This class handles string concatenation, and
 * see the documentation
 * for details.
 */
public class Java9StringConcatHandler implements Plugin {

    private Solver solver;

    private ClassType stringBuilder;

    private ClassType string;

    private MethodRef appendString;

    private MethodRef appendObject;

    private MethodRef toString;

    /**
     * Counter for naming temporary variables.
     */
    private int counter = 0;

    private final Map> method2GenStmts = Maps.newHybridMap();

    @Override
    public void setSolver(Solver solver) {
        this.solver = solver;
        TypeSystem typeSystem = solver.getTypeSystem();
        stringBuilder = typeSystem.getClassType(ClassNames.STRING_BUILDER);
        string = typeSystem.getClassType(ClassNames.STRING);
        JClass sb = stringBuilder.getJClass();
        appendString = Objects.requireNonNull(sb.getDeclaredMethod(Subsignature.get(
                        "java.lang.StringBuilder append(java.lang.String)")))
                .getRef();
        appendObject = Objects.requireNonNull(sb.getDeclaredMethod(Subsignature.get(
                        "java.lang.StringBuilder append(java.lang.Object)")))
                .getRef();
        toString = Objects.requireNonNull(sb.getDeclaredMethod(Subsignature.get(
                        "java.lang.String toString()")))
                .getRef();
    }

    /**
     * @return {@code true} if given {@code invoke} is an invokedynamic to
     * StringConcatFactory.makeConcatWithConstants(...).
     */
    static boolean isStringConcatFactoryMake(Invoke invoke) {
        // Declaring class of bootstrap method reference may be phantom
        // (looks like an issue of current (soot-based) front end),
        // thus we use resolveNullable() to avoid exception.
        // TODO: change to resolve() after using new front end
        JMethod bsm = ((InvokeDynamic) invoke.getInvokeExp())
                .getBootstrapMethodRef()
                .resolveNullable();
        if (bsm != null) {
            String bsmSig = bsm.getSignature();
            return bsmSig.equals(Signatures.STRING_CONCAT_FACTORY_MAKE);
        } else {
            return false;
        }
    }

    @Override
    public void onNewStmt(Stmt stmt, JMethod container) {
        if (stmt instanceof Invoke invoke &&
                invoke.isDynamic() &&
                isStringConcatFactoryMake(invoke) &&
                invoke.getResult() != null) {
            List stmts = generate(invoke, container);
            method2GenStmts.computeIfAbsent(container, __ -> new ArrayList<>())
                    .addAll(stmts);
        }
    }

    /**
     * Generates modelling statements for given invocation site of
     * StringConcatFactory.makeConcatWithConstants(...).
     * Currently, we only consider concatenation of reference types,
     * and ignore string constants and primitive types.
     */
    private List generate(Invoke stringConcatMake, JMethod container) {
        // generate invocations to model string concatenation
        List stmts = new ArrayList<>();
        // generate sb = new StringBuilder;
        Var sbVar = getTempVar(container);
        stmts.add(new New(container, sbVar, new NewInstance(stringBuilder)));
        // generate sb.append(...);
        InvokeDynamic indy = (InvokeDynamic) stringConcatMake.getInvokeExp();
        indy.getArgs().forEach(arg -> {
            if (arg.getType() instanceof ClassType argType) {
                MethodRef append = argType.equals(string) ? appendString : appendObject;
                stmts.add(new Invoke(container,
                        new InvokeVirtual(append, sbVar, List.of(arg))));
            }
        });
        // generate string = sb.toString();
        Var string = stringConcatMake.getResult();
        stmts.add(new Invoke(container,
                new InvokeVirtual(toString, sbVar, List.of()),
                string));
        return stmts;
    }

    private Var getTempVar(JMethod container) {
        String varName = "%stringconcat-" + counter++;
        return new Var(container, varName, stringBuilder, -1);
    }

    @Override
    public void onNewCSMethod(CSMethod csMethod) {
        Collection stmts = method2GenStmts.get(csMethod.getMethod());
        if (stmts != null) {
            solver.addStmts(csMethod, stmts);
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy