Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add Callgraph #185

Merged
merged 58 commits into from
Nov 6, 2024
Merged
Show file tree
Hide file tree
Changes from 14 commits
Commits
Show all changes
58 commits
Select commit Hold shift + click to select a range
5d8727d
fix: Draft Pr for check CallGraph
Esorat Oct 17, 2024
57bc931
chore: spell for pass ci
Esorat Oct 17, 2024
27ffbba
fix: remove unused export from class
Esorat Oct 17, 2024
b2c2cf0
fix: make properties public
Esorat Oct 18, 2024
26d582f
fix: add callGraph to ir CompilationUnit. When building the IR and c…
Esorat Oct 18, 2024
3c07cec
fix: add switch case, add forEachExpression for processExpressions, u…
Esorat Oct 18, 2024
b5b93e3
fix: add case statement_augmentedassign
Esorat Oct 18, 2024
0b8cb2b
fix: del test/good folder
Esorat Oct 18, 2024
a8f31b1
fix: enhance CallGraph with efficient edge tracking and lookup
Esorat Oct 18, 2024
12a0817
chore: remove extra newlines
Esorat Oct 18, 2024
fd02604
fix: back areConnected func
Esorat Oct 18, 2024
2ce8095
fix: add BFS to areConnected function
Esorat Oct 18, 2024
4695954
fix: add dumpCallgraph
Esorat Oct 19, 2024
8e1b8eb
fix: remake CallGraph add type Aliases.
Esorat Oct 19, 2024
c127dc5
fix: Callgraph export type.
Esorat Oct 20, 2024
5ee3309
chore: enabledByDefault: false
Esorat Oct 20, 2024
608a0d2
chore: remove export { TactASTStore };
Esorat Oct 20, 2024
2ff1b58
Merge remote-tracking branch 'upstream/master' into 91-add-callgraph
Esorat Oct 20, 2024
135e42b
fix: add correct export and import
Esorat Oct 20, 2024
2b47fb0
chore: fix spell
Esorat Oct 20, 2024
879708d
chore: update CHANGELOG
Esorat Oct 20, 2024
7ed87d8
chore: add expected files for test
Esorat Oct 20, 2024
277fc3a
Update CHANGELOG.md
jubnzv Oct 20, 2024
d2a8663
Update src/tools/dumpCallgraph.ts
jubnzv Oct 20, 2024
b7a7c6c
Update src/tools/dumpCallgraph.ts
jubnzv Oct 20, 2024
8c23973
fix: refactor
Esorat Oct 20, 2024
a70024d
Merge branch '91-add-callgraph' of https://github.com/Esorat/misti in…
Esorat Oct 20, 2024
b3411fe
fix: pass ci test
Esorat Oct 20, 2024
7a795db
Merge remote-tracking branch 'upstream' into 91-add-callgraph
Esorat Oct 25, 2024
eea2ec4
fix: remove dublicate and add needed expected output
Esorat Oct 25, 2024
ddba926
Merge remote-tracking branch 'upstream' into 91-add-callgraph
Esorat Oct 27, 2024
997056d
WIP: Refactoring
Esorat Oct 28, 2024
f2db6b9
fix: refactor dumpCallgraph
Esorat Oct 31, 2024
364460e
fix: dumpCallgraph
Esorat Oct 31, 2024
30ff0ec
fix: Update Callgraph and dumpCallgraph
Esorat Nov 2, 2024
7a7c5d3
fix: Refactor logic. Add branded types. Docstrings.
Esorat Nov 3, 2024
e393e1e
Merge remote-tracking branch 'upstream' into 91-add-callgraph
Esorat Nov 4, 2024
e18aa54
Fix: Refactor
Esorat Nov 4, 2024
82eb520
Fix: add 'make' for consistency with other builders.
Esorat Nov 6, 2024
f1e7a23
Chore: Add a proper docstring for the areConnected method
Esorat Nov 6, 2024
e4a9fec
Fix: Remove return undefined because unreachable throws.
Esorat Nov 6, 2024
d3d0397
Fix: Documented undefined value. Add logger.debug
Esorat Nov 6, 2024
ec66e96
Chore: Add needed docstring for parts of the public API
Esorat Nov 6, 2024
dc0a489
Chore: Update CHANGELOG
Esorat Nov 6, 2024
818b826
Update CHANGELOG.md
jubnzv Nov 6, 2024
0f59933
Update src/internals/ir/callGraph.ts
jubnzv Nov 6, 2024
76b8753
Update src/internals/ir/callGraph.ts
jubnzv Nov 6, 2024
5932983
Update src/internals/ir/callGraph.ts
Esorat Nov 6, 2024
53ddeb2
Update src/internals/ir/callGraph.ts
Esorat Nov 6, 2024
89294c1
Update src/internals/ir/callGraph.ts
Esorat Nov 6, 2024
8fe7500
Update src/internals/ir/callGraph.ts
Esorat Nov 6, 2024
33cae65
Update src/internals/ir/callGraph.ts
Esorat Nov 6, 2024
94da560
Update src/internals/ir/callGraph.ts
Esorat Nov 6, 2024
3145037
Update src/internals/ir/callGraph.ts
Esorat Nov 6, 2024
08b21c0
Chore: yarn fix-all
Esorat Nov 6, 2024
d780e4a
Update src/internals/ir/callGraph.ts
jubnzv Nov 6, 2024
e65a098
Update src/internals/ir/callGraph.ts
jubnzv Nov 6, 2024
95359b8
Update src/internals/ir/callGraph.ts
jubnzv Nov 6, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
3 changes: 2 additions & 1 deletion cspell.json
Original file line number Diff line number Diff line change
Expand Up @@ -101,7 +101,8 @@
"Dont",
"consteval",
"Georgiy",
"Komarov"
"Komarov",
"callgraph"
],
"flagWords": [],
"ignorePaths": [
Expand Down
2 changes: 1 addition & 1 deletion src/detectors/detector.ts
Original file line number Diff line number Diff line change
Expand Up @@ -355,7 +355,7 @@ export const BuiltInDetectors: Record<string, DetectorEntry> = {
import("./builtin/sendInLoop").then(
(module) => new module.SendInLoop(ctx),
),
enabledByDefault: false,
enabledByDefault: true,
Esorat marked this conversation as resolved.
Show resolved Hide resolved
},
CellOverflow: {
loader: (ctx: MistiContext) =>
Expand Down
1 change: 1 addition & 0 deletions src/internals/ir/builders/astStore.ts
Original file line number Diff line number Diff line change
Expand Up @@ -225,3 +225,4 @@ export class TactASTStoreBuilder {
}
}
}
export { TactASTStore };
Esorat marked this conversation as resolved.
Show resolved Hide resolved
20 changes: 18 additions & 2 deletions src/internals/ir/builders/ir.ts
Original file line number Diff line number Diff line change
Expand Up @@ -13,13 +13,15 @@ import {
FunctionName,
ProjectName,
} from "..";
import { TactASTStoreBuilder } from "./astStore";
import { ImportGraphBuilder } from "./imports";
import { MistiContext } from "../../context";
import { InternalException } from "../../exceptions";
import { formatPosition } from "../../tact";
import { TactConfigManager } from "../../tact/config";
import { unreachable } from "../../util";
import { CallGraph } from "../callGraph";
import { TactASTStoreBuilder } from "./astStore";
import { TactASTStore } from "./astStore";
import {
AstContractDeclaration,
AstExpression,
Expand Down Expand Up @@ -54,6 +56,17 @@ function generateReceiveName(receive: AstReceiver): string {
}
}

export class TactCallGraphBuilder {
/**
* Builds a CallGraph for the provided AST Store.
* @param astStore The TactASTStore containing AST nodes and function definitions.
* @returns A CallGraph instance built from the AST nodes.
*/
static buildCallGraph(astStore: TactASTStore): CallGraph {
const callGraph = new CallGraph();
return callGraph.build(astStore);
}
}
/**
* Represents a stateful object which is responsible for constructing the IR of a Tact project.
*
Expand Down Expand Up @@ -97,12 +110,15 @@ export class TactIRBuilder {
build(): CompilationUnit {
const functions = this.createFunctions();
const contracts = this.createContracts();
const tactASTStore = TactASTStoreBuilder.make(this.ctx, this.ast).build();
const callGraph = TactCallGraphBuilder.buildCallGraph(tactASTStore);
return new CompilationUnit(
this.projectName,
TactASTStoreBuilder.make(this.ctx, this.ast).build(),
tactASTStore,
ImportGraphBuilder.make(this.ctx).build(),
functions,
contracts,
callGraph,
);
}

Expand Down
237 changes: 237 additions & 0 deletions src/internals/ir/callGraph.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,237 @@
import { unreachable } from "../util";
import { TactASTStore } from "./astStore";
import { IdxGenerator } from "./indices";
import {
AstFunctionDef,
AstReceiver,
AstContractInit,
AstStatement,
AstStaticCall,
AstMethodCall,
AstExpression,
AstStructFieldInitializer,
} from "@tact-lang/compiler/dist/grammar/ast";

type NodeId = number;
Esorat marked this conversation as resolved.
Show resolved Hide resolved
type EdgeId = number;

export class CGEdge {
public idx: EdgeId;
constructor(
public src: NodeId,
public dst: NodeId,
) {
this.idx = IdxGenerator.next("cg_edge");
}
}

export class CGNode {
public idx: NodeId;
public inEdges: Set<EdgeId> = new Set();
public outEdges: Set<EdgeId> = new Set();
constructor(
public astId: number,
Esorat marked this conversation as resolved.
Show resolved Hide resolved
public name: string,
) {
this.idx = IdxGenerator.next("cg_node");
}
}

export class CallGraph {
Esorat marked this conversation as resolved.
Show resolved Hide resolved
private nodeMap: Map<NodeId, CGNode> = new Map();
private edgesMap: Map<EdgeId, CGEdge> = new Map();

public getNodes(): Map<NodeId, CGNode> {
return this.nodeMap;
}

public getEdges(): Map<EdgeId, CGEdge> {
return this.edgesMap;
}

build(astStore: TactASTStore): CallGraph {
this.addFunctionsToNodes(astStore);
this.analyzeFunctionCalls(astStore);
return this;
}

private addFunctionsToNodes(astStore: TactASTStore) {
for (const func of astStore.getFunctions()) {
const funcName = this.getFunctionName(func);
if (funcName) {
const node = new CGNode(func.id, funcName);
this.nodeMap.set(func.id, node);
}
}
}

private analyzeFunctionCalls(astStore: TactASTStore) {
for (const func of astStore.getFunctions()) {
const funcName = this.getFunctionName(func);
if (funcName) {
this.processStatements(func.statements, func.id);
}
}
}

/**
* Get function name based on its kind.
* @param func - The function definition or initializer.
* @returns The name of the function.
*/
private getFunctionName(
func: AstFunctionDef | AstReceiver | AstContractInit,
): string {
switch (func.kind) {
case "function_def":
return func.name.text;
case "contract_init":
jubnzv marked this conversation as resolved.
Show resolved Hide resolved
Esorat marked this conversation as resolved.
Show resolved Hide resolved
return `contract_init_${func.id}`;
case "receiver":
return `receiver_${func.id}`;
default:
unreachable(func);
}
}

/**
* Process the statements of a function.
* @param statements - The AST statements.
* @param callerId - The ID of the calling function.
*
* Note: We don't use `foldStatements` or `forEachStatement` here because
* the logic is specific to call graph construction, and we want to
* maintain control over the traversal and processing.
*/
private processStatements(statements: AstStatement[], callerId: number) {
Esorat marked this conversation as resolved.
Show resolved Hide resolved
Esorat marked this conversation as resolved.
Show resolved Hide resolved
for (const stmt of statements) {
switch (stmt.kind) {
case "statement_expression":
this.processExpression(stmt.expression, callerId);
break;
case "statement_condition":
this.processStatements(stmt.trueStatements, callerId);
if (stmt.falseStatements) {
this.processStatements(stmt.falseStatements, callerId);
}
break;
case "statement_while":
case "statement_until":
case "statement_repeat":
case "statement_foreach":
this.processStatements(stmt.statements, callerId);
break;
case "statement_try":
this.processStatements(stmt.statements, callerId);
break;
case "statement_try_catch":
this.processStatements(stmt.statements, callerId);
this.processStatements(stmt.catchStatements, callerId);
break;
case "statement_let":
case "statement_return":
case "statement_assign":
case "statement_augmentedassign":
if (stmt.expression) {
this.processExpression(stmt.expression, callerId);
}
break;
default:
console.warn(`Unhandled statement type: ${(stmt as any).kind}`, stmt);
}
}
}

private processExpression(
Esorat marked this conversation as resolved.
Show resolved Hide resolved
expr: AstExpression | AstStructFieldInitializer,
callerId: NodeId,
) {
this.forEachExpression(expr, (nestedExpr) => {
let calleeId: NodeId | undefined;
if (nestedExpr.kind === "static_call") {
const staticCall = nestedExpr as AstStaticCall;
calleeId = this.findOrAddFunction(staticCall.function.text);
} else if (nestedExpr.kind === "method_call") {
const methodCall = nestedExpr as AstMethodCall;
calleeId = this.findOrAddFunction(methodCall.method.text);
}
if (calleeId !== undefined) {
this.addEdge(callerId, calleeId);
}
});
}

private forEachExpression(
expr: AstExpression | AstStructFieldInitializer,
callback: (expr: AstExpression) => void,
) {
if (expr.kind !== "struct_field_initializer") {
callback(expr as AstExpression);
}

switch (expr.kind) {
case "static_call":
case "method_call": {
const callExpr = expr as AstStaticCall | AstMethodCall;
for (const arg of callExpr.args) {
this.forEachExpression(arg, callback);
}
break;
}
default:
break;
}
}

private findOrAddFunction(name: string): NodeId {
const existingNode = Array.from(this.nodeMap.values()).find(
(node: CGNode) => node.name === name,
);
if (existingNode) {
return existingNode.astId;
}
const newNode = new CGNode(IdxGenerator.next("cg_node"), name);
this.nodeMap.set(newNode.astId, newNode);
return newNode.astId;
}

private addEdge(src: NodeId, dst: NodeId) {
const srcNode = this.nodeMap.get(src);
const dstNode = this.nodeMap.get(dst);
if (srcNode && dstNode) {
const edge = new CGEdge(src, dst);
this.edgesMap.set(edge.idx, edge);
srcNode.outEdges.add(edge.idx);
dstNode.inEdges.add(edge.idx);
}
}

areConnected(src: NodeId, dst: NodeId): boolean {
const srcNode = this.nodeMap.get(src);
const dstNode = this.nodeMap.get(dst);
if (!srcNode || !dstNode) {
return false;
}
const queue: NodeId[] = [src];
const visited = new Set<NodeId>([src]);

while (queue.length > 0) {
const current = queue.shift()!;

if (current === dst) {
return true;
}
const currentNode = this.nodeMap.get(current);
if (currentNode) {
for (const edgeId of currentNode.outEdges) {
const edge = this.edgesMap.get(edgeId);
if (edge && !visited.has(edge.dst)) {
visited.add(edge.dst);
queue.push(edge.dst);
}
}
}
}
return false;
}
}
1 change: 1 addition & 0 deletions src/internals/ir/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,7 @@ export {
EntryOrigin,
} from "./cfg";
export { TactASTStore } from "./astStore";
export { CGEdge, CGNode } from "./callGraph";
export * from "./imports";
export { CompilationUnit, Contract } from "./ir";
export * from "./types";
Expand Down
2 changes: 2 additions & 0 deletions src/internals/ir/ir.ts
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@
* @packageDocumentation
*/
import { TactASTStore } from "./astStore";
import { CallGraph } from "./callGraph";
import { BasicBlock, CFG } from "./cfg";
import { ImportGraph } from "./imports";
import { IdxGenerator } from "./indices";
Expand Down Expand Up @@ -36,6 +37,7 @@ export class CompilationUnit {
public imports: ImportGraph,
public functions: Map<CFGIdx, CFG>,
public contracts: Map<ContractIdx, Contract>,
public callGraph: CallGraph,
) {}

/**
Expand Down
Loading