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

feat: variable declaration statements are not allowed as the body of loop #158

Merged
merged 22 commits into from
Dec 10, 2024
Merged
Show file tree
Hide file tree
Changes from 19 commits
Commits
Show all changes
22 commits
Select commit Hold shift + click to select a range
fdbfbf2
feat: variable declaration statements are not allowed as the body of …
TilakMaddy Dec 8, 2024
7d89225
fix: recurse on unchecked block
TilakMaddy Dec 8, 2024
6991f9c
Merge remote-tracking branch 'upstream/main' into feat/ast-validation…
TilakMaddy Dec 8, 2024
c91e5b0
chore: bringing back changes on loops.sol
TilakMaddy Dec 8, 2024
053b8ec
chore: cargo uibless
TilakMaddy Dec 8, 2024
213120d
fix: added error annotation for var decl as body of loop
TilakMaddy Dec 8, 2024
1893ba0
Update crates/sema/src/ast_passes.rs
TilakMaddy Dec 9, 2024
ea00e73
fix: reverting to original fix for loops.sol and adjust the message
TilakMaddy Dec 9, 2024
de03c7f
Merge branch 'main' into feat/ast-validations-p1
TilakMaddy Dec 9, 2024
8c1dff5
chore: moved to separate function
TilakMaddy Dec 9, 2024
8f52d50
fix: remove trails of x_depth
TilakMaddy Dec 9, 2024
5bbf944
fix: cargo clippy recommendations
TilakMaddy Dec 9, 2024
6031788
fix
TilakMaddy Dec 10, 2024
1b4d1f7
Merge branch 'main' into feat/ast-validations-p1
TilakMaddy Dec 10, 2024
276ad0c
feat: made it simple like solidity
TilakMaddy Dec 10, 2024
162d40b
feat: cleanup + added check on if statement
TilakMaddy Dec 10, 2024
a007222
fix: change error message
TilakMaddy Dec 10, 2024
a2ba5ed
Merge branch 'main' of github.com:paradigmxyz/solar into feat/ast-val…
TilakMaddy Dec 10, 2024
f650db9
fix: cleanup
TilakMaddy Dec 10, 2024
24b9195
fix: i missed the space it's annoying haha
TilakMaddy Dec 10, 2024
a3c3446
chore: clean up visitor
DaniPopes Dec 10, 2024
10d8b17
chore: clippy
DaniPopes Dec 10, 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
66 changes: 50 additions & 16 deletions crates/sema/src/ast_passes.rs
Original file line number Diff line number Diff line change
Expand Up @@ -28,7 +28,7 @@ struct AstValidator<'sess, 'ast> {
placeholder_count: u64,
}

impl<'sess> AstValidator<'sess, '_> {
impl<'sess, 'ast> AstValidator<'sess, 'ast> {
fn new(sess: &'sess Session) -> Self {
Self {
span: Span::DUMMY,
Expand All @@ -50,6 +50,16 @@ impl<'sess> AstValidator<'sess, '_> {
fn in_loop(&self) -> bool {
self.in_loop_depth != 0
}

fn check_single_statement_variable_declaration(&self, stmt: &'ast &'ast mut ast::Stmt<'ast>) {
if matches!(stmt.kind, ast::StmtKind::DeclSingle(..) | ast::StmtKind::DeclMulti(..)) {
self.dcx()
.err("variable declarations can only be used inside blocks")
.span(stmt.span)
.help("wrap the statement in a block (`{ ... }`)")
.emit();
}
}
}

impl<'ast> Visit<'ast> for AstValidator<'_, 'ast> {
Expand Down Expand Up @@ -121,13 +131,46 @@ impl<'ast> Visit<'ast> for AstValidator<'_, 'ast> {

fn visit_stmt(&mut self, stmt: &'ast ast::Stmt<'ast>) -> ControlFlow<Self::BreakValue> {
match &stmt.kind {
ast::StmtKind::While(_, body, ..)
| ast::StmtKind::DoWhile(body, ..)
| ast::StmtKind::For { body, .. } => {
ast::StmtKind::While(cond, body) => {
self.visit_expr(cond)?;
self.in_loop_depth += 1;
let r = self.walk_stmt(body);
self.visit_stmt(body);
self.check_single_statement_variable_declaration(body);
self.in_loop_depth -= 1;
return r;
return ControlFlow::Continue(());
}
ast::StmtKind::DoWhile(body, ..) => {
self.in_loop_depth += 1;
self.visit_stmt(body)?;
self.check_single_statement_variable_declaration(body);
self.in_loop_depth -= 1;
return ControlFlow::Continue(());
}
ast::StmtKind::For { init, cond, next, body } => {
if let Some(init) = init {
self.visit_stmt(init)?;
}
if let Some(cond) = cond {
self.visit_expr(cond)?;
}
if let Some(next) = next {
self.visit_expr(next)?;
}
self.in_loop_depth += 1;
self.visit_stmt(body)?;
self.check_single_statement_variable_declaration(body);
self.in_loop_depth -= 1;
return ControlFlow::Continue(());
}
ast::StmtKind::If(cond, then, else_) => {
self.visit_expr(cond)?;
self.visit_stmt(then)?;
self.check_single_statement_variable_declaration(then);
if let Some(else_) = else_ {
self.visit_stmt(else_)?;
self.check_single_statement_variable_declaration(then);
}
return ControlFlow::Continue(());
}
ast::StmtKind::Break | ast::StmtKind::Continue => {
if !self.in_loop() {
Expand All @@ -147,7 +190,7 @@ impl<'ast> Visit<'ast> for AstValidator<'_, 'ast> {

let prev = self.in_unchecked_block;
self.in_unchecked_block = true;
let r = self.walk_block(block);
let r = self.visit_block(block);
self.in_unchecked_block = prev;
return r;
}
Expand Down Expand Up @@ -263,13 +306,4 @@ impl<'ast> Visit<'ast> for AstValidator<'_, 'ast> {
}
self.walk_using_directive(using)
}

// Intentionally override unused default implementations to reduce bloat.
fn visit_expr(&mut self, _expr: &'ast ast::Expr<'ast>) -> ControlFlow<Self::BreakValue> {
ControlFlow::Continue(())
}

fn visit_ty(&mut self, _ty: &'ast ast::Type<'ast>) -> ControlFlow<Self::BreakValue> {
ControlFlow::Continue(())
}
}
14 changes: 6 additions & 8 deletions tests/ui/resolve/loops.sol
Original file line number Diff line number Diff line change
Expand Up @@ -8,27 +8,25 @@ function funky() {

for (;;) break;
for (; i < 40; i++) continue;
for (; i++ < 50;) continue;
for (; i++ < 50; ) continue;

// ---
// TODO: `Error: Variable declarations can only be used inside blocks.`

while (a == 0) uint a = 0; //~ ERROR: unresolved symbol
while (a == 0) { uint a = 0; } //~ ERROR: unresolved symbol
a; //~ ERROR: unresolved symbol
while (b == 0) { uint b = 0; } //~ ERROR: unresolved symbol
b; //~ ERROR: unresolved symbol

do uint c; while (c == 0); //~ ERROR: unresolved symbol
do { uint c; } while (c == 0); //~ ERROR: unresolved symbol
c; //~ ERROR: unresolved symbol
do { uint d; } while (d == 0); //~ ERROR: unresolved symbol
d; //~ ERROR: unresolved symbol

for (; false; e++) uint e; //~ ERROR: unresolved symbol
for (; false; e++) { uint e; } //~ ERROR: unresolved symbol
e; //~ ERROR: unresolved symbol
for (; false; f++) { uint f; } //~ ERROR: unresolved symbol
for (; false; f++) { uint f; }//~ ERROR: unresolved symbol
f; //~ ERROR: unresolved symbol
for (uint g; false; g++) {
g;
}
g; //~ ERROR: unresolved symbol
}

8 changes: 4 additions & 4 deletions tests/ui/resolve/loops.stderr
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
error: unresolved symbol `a`
--> ROOT/tests/ui/resolve/loops.sol:LL:CC
|
LL | while (a == 0) uint a = 0;
LL | while (a == 0) { uint a = 0; }
| ^
|

Expand Down Expand Up @@ -29,8 +29,8 @@ LL | b;
error: unresolved symbol `c`
--> ROOT/tests/ui/resolve/loops.sol:LL:CC
|
LL | do uint c; while (c == 0);
| ^
LL | do { uint c; } while (c == 0);
| ^
|

error: unresolved symbol `c`
Expand All @@ -57,7 +57,7 @@ LL | d;
error: unresolved symbol `e`
--> ROOT/tests/ui/resolve/loops.sol:LL:CC
|
LL | for (; false; e++) uint e;
LL | for (; false; e++) { uint e; }
| ^
|

Expand Down
28 changes: 28 additions & 0 deletions tests/ui/typeck/var_decl_as_loop_body.sol
Original file line number Diff line number Diff line change
@@ -0,0 +1,28 @@
contract C {
function var_decl_inside_loops() external {
for (uint256 i = 0; i < 100; ++i) {
uint256 m_count = i + 1 * 2;
}

for (uint256 i = 0; i < 100; ++i) uint256 m_count = i + 1 * 2; //~ ERROR: variable declarations can only be used inside blocks
for (uint256 i = 0; i < 100; ++i)
for (uint256 j = 0; i < 100; ++j) {
uint256 k = i + j;
}

for (uint256 i = 0; i < 100; ++i)
for (uint256 j = 0; i < 100; ++j) uint256 k = i + j; //~ ERROR: variable declarations can only be used inside blocks

while (true) uint256 x = 4; //~ ERROR: variable declarations can only be used inside blocks

do uint256 x = 4; while (true); //~ ERROR: variable declarations can only be used inside blocks

unchecked {
{
{
for (uint256 i = 0; i < 10; ++i) uint256 y = 0; //~ ERROR: variable declarations can only be used inside blocks
}
}
}
}
}
42 changes: 42 additions & 0 deletions tests/ui/typeck/var_decl_as_loop_body.stderr
Original file line number Diff line number Diff line change
@@ -0,0 +1,42 @@
error: variable declarations can only be used inside blocks
--> ROOT/tests/ui/typeck/var_decl_as_loop_body.sol:LL:CC
|
LL | for (uint256 i = 0; i < 100; ++i) uint256 m_count = i + 1 * 2;
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^
|
= help: wrap the statement in a block (`{ ... }`)

error: variable declarations can only be used inside blocks
--> ROOT/tests/ui/typeck/var_decl_as_loop_body.sol:LL:CC
|
LL | for (uint256 j = 0; i < 100; ++j) uint256 k = i + j;
| ^^^^^^^^^^^^^^^^^^
|
= help: wrap the statement in a block (`{ ... }`)

error: variable declarations can only be used inside blocks
--> ROOT/tests/ui/typeck/var_decl_as_loop_body.sol:LL:CC
|
LL | while (true) uint256 x = 4;
| ^^^^^^^^^^^^^^
|
= help: wrap the statement in a block (`{ ... }`)

error: variable declarations can only be used inside blocks
--> ROOT/tests/ui/typeck/var_decl_as_loop_body.sol:LL:CC
|
LL | do uint256 x = 4; while (true);
| ^^^^^^^^^^^^^^
|
= help: wrap the statement in a block (`{ ... }`)

error: variable declarations can only be used inside blocks
--> ROOT/tests/ui/typeck/var_decl_as_loop_body.sol:LL:CC
|
LL | ... for (uint256 i = 0; i < 10; ++i) uint256 y = 0;
| ^^^^^^^^^^^^^^
|
= help: wrap the statement in a block (`{ ... }`)

error: aborting due to 5 previous errors

Loading