pascal.taie.analysis.pta.plugin.invokedynamic.LookupModel Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of tai-e Show documentation
Show all versions of tai-e Show documentation
An easy-to-learn/use static analysis framework for Java
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.context.Context;
import pascal.taie.analysis.pta.core.solver.Solver;
import pascal.taie.analysis.pta.plugin.util.AnalysisModelPlugin;
import pascal.taie.analysis.pta.plugin.util.CSObjs;
import pascal.taie.analysis.pta.plugin.util.InvokeHandler;
import pascal.taie.analysis.pta.pts.PointsToSet;
import pascal.taie.ir.exp.MethodHandle;
import pascal.taie.ir.exp.Var;
import pascal.taie.ir.stmt.Invoke;
import pascal.taie.language.classes.JClass;
import pascal.taie.language.classes.JMethod;
import pascal.taie.language.classes.Reflections;
import java.util.function.BiFunction;
import java.util.function.Predicate;
import java.util.stream.Stream;
/**
* Models java.lang.invoke.MethodHandles.Lookup.find*(...).
* For details, please refer to
* MethodHandles.Lookup.html
* TODO: take Lookup.lookupClass's visibility into account
* TODO: take MethodType into account
*/
public class LookupModel extends AnalysisModelPlugin {
LookupModel(Solver solver) {
super(solver);
}
@InvokeHandler(signature = "", argIndexes = {0})
public void findConstructor(Context context, Invoke invoke, PointsToSet clsObjs) {
Var result = invoke.getResult();
if (result != null) {
clsObjs.forEach(clsObj -> {
JClass cls = CSObjs.toClass(clsObj);
if (cls != null) {
Reflections.getDeclaredConstructors(cls)
.map(ctor -> MethodHandle.get(
MethodHandle.Kind.REF_newInvokeSpecial,
ctor.getRef()))
.map(heapModel::getConstantObj)
.forEach(mhObj -> solver.addVarPointsTo(context, result, mhObj));
}
});
}
}
@InvokeHandler(signature = "", argIndexes = {0, 1})
public void findVirtual(Context context, Invoke invoke,
PointsToSet clsObjs, PointsToSet nameObjs) {
// TODO: find private methods in (direct/indirect) super class.
findMethod(context, invoke, clsObjs, nameObjs, (cls, name) ->
Reflections.getDeclaredMethods(cls, name)
.filter(Predicate.not(JMethod::isStatic)),
MethodHandle.Kind.REF_invokeVirtual);
}
@InvokeHandler(signature = "", argIndexes = {0, 1})
public void findStatic(Context context, Invoke invoke,
PointsToSet clsObjs, PointsToSet nameObjs) {
// TODO: find static methods in (direct/indirect) super class.
findMethod(context, invoke, clsObjs, nameObjs, (cls, name) ->
Reflections.getDeclaredMethods(cls, name)
.filter(JMethod::isStatic),
MethodHandle.Kind.REF_invokeStatic);
}
private void findMethod(Context context, Invoke invoke,
PointsToSet clsObjs, PointsToSet nameObjs,
BiFunction> getter,
MethodHandle.Kind kind) {
Var result = invoke.getResult();
if (result != null) {
clsObjs.forEach(clsObj -> {
JClass cls = CSObjs.toClass(clsObj);
if (cls != null) {
nameObjs.forEach(nameObj -> {
String name = CSObjs.toString(nameObj);
if (name != null) {
getter.apply(cls, name)
.map(mtd -> MethodHandle.get(kind, mtd.getRef()))
.map(heapModel::getConstantObj)
.forEach(mhObj ->
solver.addVarPointsTo(context, result, mhObj));
}
});
}
});
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy