Skip to content

Commit

Permalink
parent 9e567c8
Browse files Browse the repository at this point in the history
author Jessie Zijia Xu <[email protected]> 1720048025 -0700
committer Jessie Xu <[email protected]> 1724974035 -0700

early return bug & for loop fixes
  • Loading branch information
jxuatlas committed Aug 29, 2024
1 parent 9e567c8 commit d186ee1
Show file tree
Hide file tree
Showing 7 changed files with 792 additions and 66 deletions.
1 change: 1 addition & 0 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,7 @@ Arguments:
Options:
-d, --debug
--dump-ir <DUMP_IR> Dump the IR for the specified function.
-dt, --dump-dt <DUMP_DOM_TREE> Dump the Dominator Tree for the specified app
-f, --function <FUNCTION> A specific function to scan. Must be an entrypoint specified in `manifest.yml`
-h, --help Print help information
-V, --version Print version information
Expand Down
139 changes: 123 additions & 16 deletions crates/forge_analyzer/src/definitions.rs
Original file line number Diff line number Diff line change
Expand Up @@ -909,6 +909,11 @@ impl<'cx> FunctionAnalyzer<'cx> {
self.body.set_terminator(self.block, term);
}

#[inline]
fn get_curr_terminator(&mut self) -> Option<Terminator> {
self.body.get_terminator(self.block)
}

fn as_intrinsic(&self, callee: &[PropPath], first_arg: Option<&Expr>) -> Option<Intrinsic> {
fn is_storage_read(prop: &JsWord) -> bool {
*prop == *"get" || *prop == *"getSecret" || *prop == *"query"
Expand Down Expand Up @@ -1051,7 +1056,9 @@ impl<'cx> FunctionAnalyzer<'cx> {
/// Sets the current block to `block` and returns the previous block.
#[inline]
fn goto_block(&mut self, block: BasicBlockId) -> BasicBlockId {
self.set_curr_terminator(Terminator::Goto(block));
if self.get_curr_terminator().is_none() {
self.set_curr_terminator(Terminator::Goto(block));
}
mem::replace(&mut self.block, block)
}

Expand Down Expand Up @@ -1217,7 +1224,6 @@ impl<'cx> FunctionAnalyzer<'cx> {
}

fn lower_call(&mut self, callee: CalleeRef<'_>, args: &[ExprOrSpread]) -> Operand {
// debug!("in da lower call");
let props = normalize_callee_expr(callee, self.res, self.module);
if let Some(&PropPath::Def(id)) = props.first() {
if self.res.is_imported_from(id, "@forge/ui").map_or(
Expand All @@ -1243,6 +1249,7 @@ impl<'cx> FunctionAnalyzer<'cx> {
Expr::Fn(FnExpr { ident: _, function }) => {
if let Some(body) = &function.body {
self.lower_stmts(&body.stmts);

return Operand::UNDEF;
}
}
Expand Down Expand Up @@ -1530,6 +1537,7 @@ impl<'cx> FunctionAnalyzer<'cx> {
.push_tmp(self.block, Rvalue::Unary(op.into(), arg), None);
Operand::with_var(tmp)
}

Expr::Update(UpdateExpr {
op, prefix, arg, ..
}) => {
Expand Down Expand Up @@ -1626,17 +1634,20 @@ impl<'cx> FunctionAnalyzer<'cx> {
}) => {
let cond = self.lower_expr(test, None);
let curr = self.block;
let rest = self.body.new_block();
let cons_block = self.body.new_block();
let alt_block = self.body.new_block();
let [temp1, temp2, temp3] = self.body.new_blocks();
let rest = self.body.new_blockbuilder();
let cons_block = self.body.new_blockbuilder();
let alt_block = self.body.new_blockbuilder();
self.set_curr_terminator(Terminator::If {
// TODO: COME BACK TO?
cond,
cons: cons_block,
alt: alt_block,
});
self.block = cons_block;
let cons = self.lower_expr(cons, None);
let cons_phi = self.body.push_tmp(self.block, Rvalue::Read(cons), None);

self.set_curr_terminator(Terminator::Goto(rest));
self.block = alt_block;
let alt = self.lower_expr(alt, None);
Expand All @@ -1650,6 +1661,7 @@ impl<'cx> FunctionAnalyzer<'cx> {
);
Operand::with_var(phi)
}

Expr::Call(CallExpr { callee, args, .. }) => self.lower_call(callee.into(), args),
Expr::New(NewExpr { callee, args, .. }) => {
if let Expr::Ident(ident) = &**callee {
Expand Down Expand Up @@ -1750,12 +1762,28 @@ impl<'cx> FunctionAnalyzer<'cx> {
}
}

// Lowers all statements from the given `stmts`.
// If a return is encounted, return early to prevent
// unreachable statements that come afterwards from being lowered.
fn lower_stmts(&mut self, stmts: &[Stmt]) {
for stmt in stmts {
self.lower_stmt(stmt);
if let Stmt::Return(_) = stmt {
return;
}
}
}

// Lowers a single statement by pushing corresponding instruction(s)/expression(s)
// onto self.body.blockbuilders[block].
//
// Corresponding instruction(s)/expression(s) are initially placed in
// self.body.blockbuilders[block] and transferred to self.body.blocks[block]
// once the block's terminator gets set.
//
// B/c of this, an empty block is pushed to self.body.blocks whenever
// a new block is added to self.body.blockbuilders, to ensure space is allocated
// on self.body.blocks for all blocks when instructions are moved.
fn lower_stmt(&mut self, n: &Stmt) {
match n {
Stmt::Block(BlockStmt { stmts, .. }) => self.lower_stmts(stmts),
Expand All @@ -1764,6 +1792,7 @@ impl<'cx> FunctionAnalyzer<'cx> {
Stmt::With(WithStmt { obj, body, .. }) => {
let opnd = self.lower_expr(obj, None);
self.body.push_expr(self.block, Rvalue::Read(opnd));

self.lower_stmt(body);
}
Stmt::Return(ReturnStmt { arg, .. }) => {
Expand All @@ -1777,17 +1806,29 @@ impl<'cx> FunctionAnalyzer<'cx> {
Stmt::Labeled(LabeledStmt { label, body, .. }) => {
self.lower_stmt(body);
}
// TODO: Lower Break and Continue
Stmt::Break(BreakStmt { label, .. }) => {}
Stmt::Continue(ContinueStmt { label, .. }) => {}
Stmt::If(IfStmt {
test, cons, alt, ..
}) => {
let [cons_block, cont] = self.body.new_blocks();
// Adds two blocks to the body:
// - cons_block: block to store insts that run if the test condition of the Stmt::If is true
// - cont: block to store insts that run after the Stmt::If
let [temp1, temp2] = self.body.new_blocks();
let [cons_block, cont] = self.body.new_blockbuilders();

// If an alt block (`else` case) is present, add another block to the body
let alt_block = if let Some(alt) = alt {
let alt_block = self.body.new_block();
let temp3 = self.body.new_block();
let alt_block = self.body.new_blockbuilder();
let old_block = mem::replace(&mut self.block, alt_block);
self.lower_stmt(alt);
self.set_curr_terminator(Terminator::Goto(cont));

if self.get_curr_terminator().is_none() {
self.set_curr_terminator(Terminator::Goto(cont));
}

self.block = old_block;
alt_block
} else {
Expand All @@ -1801,6 +1842,7 @@ impl<'cx> FunctionAnalyzer<'cx> {
});
self.block = cons_block;
self.lower_stmt(cons);

self.goto_block(cont);
}
Stmt::Switch(SwitchStmt {
Expand Down Expand Up @@ -1829,7 +1871,8 @@ impl<'cx> FunctionAnalyzer<'cx> {
}
}
Stmt::While(WhileStmt { test, body, .. }) => {
let [check, cont, body_id] = self.body.new_blocks();
let [temp1, temp2, temp3] = self.body.new_blocks();
let [check, cont, body_id] = self.body.new_blockbuilders();
self.set_curr_terminator(Terminator::Goto(check));
self.block = check;
let cond = self.lower_expr(test, None);
Expand All @@ -1844,7 +1887,8 @@ impl<'cx> FunctionAnalyzer<'cx> {
self.block = cont;
}
Stmt::DoWhile(DoWhileStmt { test, body, .. }) => {
let [check, cont, body_id] = self.body.new_blocks();
let [temp1, temp2, temp3] = self.body.new_blocks();
let [check, cont, body_id] = self.body.new_blockbuilders();
self.set_curr_terminator(Terminator::Goto(body_id));
self.block = body_id;
self.lower_stmt(body);
Expand Down Expand Up @@ -1874,7 +1918,8 @@ impl<'cx> FunctionAnalyzer<'cx> {
}
None => {}
}
let [check, cont, body_id] = self.body.new_blocks();
let [temp1, temp2, temp3] = self.body.new_blocks();
let [check, cont, body_id] = self.body.new_blockbuilders();
self.goto_block(check);
if let Some(test) = test {
let cond = self.lower_expr(test, None);
Expand Down Expand Up @@ -2233,7 +2278,18 @@ impl Visit for FunctionCollector<'_> {
};
if let Some(BlockStmt { stmts, .. }) = &n.body {
analyzer.lower_stmts(stmts);
let body = analyzer.body;
let mut body = analyzer.body;

let mut blocks_to_update: Vec<BasicBlockId> = Vec::new();
for (id, block) in body.blocks.iter_enumerated() {
if !block.set_term_called {
blocks_to_update.push(id);
}
}
for id in blocks_to_update {
body.set_terminator(id, Terminator::Ret);
}

*self.res.def_mut(*owner).expect_body() = body;
}
}
Expand Down Expand Up @@ -2280,7 +2336,18 @@ impl Visit for FunctionCollector<'_> {
};
if let Some(BlockStmt { stmts, .. }) = &n.function.body {
analyzer.lower_stmts(stmts);
let body = analyzer.body;
let mut body = analyzer.body;

let mut blocks_to_update: Vec<BasicBlockId> = Vec::new();
for (id, block) in body.blocks.iter_enumerated() {
if !block.set_term_called {
blocks_to_update.push(id);
}
}
for id in blocks_to_update {
body.set_terminator(id, Terminator::Ret);
}

*self.res.def_mut(*owner).expect_body() = body;
}
}
Expand Down Expand Up @@ -2341,7 +2408,19 @@ impl Visit for FunctionCollector<'_> {
.push_inst(analyzer.block, Inst::Assign(RETURN_VAR, Rvalue::Read(opnd)));
}
}
*self.res.def_mut(owner).expect_body() = analyzer.body;
let mut body = analyzer.body;

let mut blocks_to_update: Vec<BasicBlockId> = Vec::new();
for (id, block) in body.blocks.iter_enumerated() {
if !block.set_term_called {
blocks_to_update.push(id);
}
}
for id in blocks_to_update {
body.set_terminator(id, Terminator::Ret);
}

*self.res.def_mut(owner).expect_body() = body;
self.parent = old_parent;
}

Expand Down Expand Up @@ -2449,6 +2528,10 @@ impl Visit for FunctionCollector<'_> {
analyzer.block,
Inst::Assign(RETURN_VAR, Rvalue::Read(opnd)),
);

analyzer
.body
.set_terminator(analyzer.block, Terminator::Ret);
*self.res.def_mut(owner).expect_body() = analyzer.body;
self.parent = old_parent;
}
Expand Down Expand Up @@ -2576,7 +2659,21 @@ impl FunctionCollector<'_> {
};
if let Some(BlockStmt { stmts, .. }) = &n.body {
analyzer.lower_stmts(stmts);
let body = analyzer.body;
let mut body = analyzer.body;

let mut blocks_to_update: Vec<BasicBlockId> = Vec::new();
for (id, block) in body.blocks.iter_enumerated() {
if !block.set_term_called {
blocks_to_update.push(id);
}
}

// Ensures that instructions of all blocks from body.blockbuilders
// are moved to body.blocks
for id in blocks_to_update {
body.set_terminator(id, Terminator::Ret);
}

*self.res.def_mut(owner).expect_body() = body;
}
}
Expand Down Expand Up @@ -3111,7 +3208,17 @@ impl Visit for GlobalCollector<'_> {
}
}
analyzer.lower_stmts(all_module_items.as_slice());
let body = analyzer.body;
let mut body = analyzer.body;

let mut blocks_to_update: Vec<BasicBlockId> = Vec::new();
for (id, block) in body.blocks.iter_enumerated() {
if !block.set_term_called {
blocks_to_update.push(id);
}
}
for id in blocks_to_update {
body.set_terminator(id, Terminator::Ret);
}

*self.res.def_mut(owner).expect_body() = body;
}
Expand Down
Loading

0 comments on commit d186ee1

Please sign in to comment.