diff --git a/compiler/noirc_frontend/src/ast/expression.rs b/compiler/noirc_frontend/src/ast/expression.rs index d1d71112de0..f8c67781a49 100644 --- a/compiler/noirc_frontend/src/ast/expression.rs +++ b/compiler/noirc_frontend/src/ast/expression.rs @@ -26,6 +26,7 @@ pub enum ExpressionKind { Variable(Path), Tuple(Vec), Lambda(Box), + Parenthesized(Box), Error, } @@ -479,6 +480,7 @@ impl Display for ExpressionKind { write!(f, "({})", elements.join(", ")) } Lambda(lambda) => lambda.fmt(f), + Parenthesized(subexpr) => write!(f, "({subexpr})"), Error => write!(f, "Error"), } } diff --git a/compiler/noirc_frontend/src/hir/resolution/resolver.rs b/compiler/noirc_frontend/src/hir/resolution/resolver.rs index 60a12ad601d..97c71eedf4e 100644 --- a/compiler/noirc_frontend/src/hir/resolution/resolver.rs +++ b/compiler/noirc_frontend/src/hir/resolution/resolver.rs @@ -1260,6 +1260,7 @@ impl<'a> Resolver<'a> { captures: lambda_context.captures, }) }), + ExpressionKind::Parenthesized(subexpr) => return self.resolve_expression(*subexpr), }; let expr_id = self.interner.push_expr(hir_expr); diff --git a/compiler/noirc_frontend/src/lexer/lexer.rs b/compiler/noirc_frontend/src/lexer/lexer.rs index 0b8922efce6..593c5bdfb05 100644 --- a/compiler/noirc_frontend/src/lexer/lexer.rs +++ b/compiler/noirc_frontend/src/lexer/lexer.rs @@ -19,6 +19,7 @@ pub struct Lexer<'a> { char_iter: Peekable, RangeFrom>>, position: Position, done: bool, + skip_comments: bool, } pub type SpannedTokenResult = Result; @@ -39,15 +40,21 @@ impl<'a> Lexer<'a> { (Tokens(tokens), errors) } - fn new(source: &'a str) -> Self { + pub fn new(source: &'a str) -> Self { Lexer { // We zip with the character index here to ensure the first char has index 0 char_iter: source.chars().zip(0..).peekable(), position: 0, done: false, + skip_comments: true, } } + pub fn skip_comments(mut self, flag: bool) -> Self { + self.skip_comments = flag; + self + } + /// Iterates the cursor and returns the char at the new cursor position fn next_char(&mut self) -> Option { let (c, index) = self.char_iter.next()?; @@ -176,13 +183,16 @@ impl<'a> Lexer<'a> { Token::Minus => self.single_double_peek_token('>', prev_token, Token::Arrow), Token::Colon => self.single_double_peek_token(':', prev_token, Token::DoubleColon), Token::Slash => { + let start = self.position; + if self.peek_char_is('/') { self.next_char(); - return self.parse_comment(); + return self.parse_comment(start); } else if self.peek_char_is('*') { self.next_char(); - return self.parse_block_comment(); + return self.parse_block_comment(start); } + Ok(spanned_prev_token) } _ => Err(LexerErrorKind::NotADoubleChar { @@ -377,15 +387,18 @@ impl<'a> Lexer<'a> { } } - fn parse_comment(&mut self) -> SpannedTokenResult { - let _ = self.eat_while(None, |ch| ch != '\n'); - self.next_token() + fn parse_comment(&mut self, start: u32) -> SpannedTokenResult { + let comment = self.eat_while(None, |ch| ch != '\n'); + if self.skip_comments { + return self.next_token(); + } + Ok(Token::LineComment(comment).into_span(start, self.position)) } - fn parse_block_comment(&mut self) -> SpannedTokenResult { - let start = self.position; + fn parse_block_comment(&mut self, start: u32) -> SpannedTokenResult { let mut depth = 1usize; + let mut content = String::new(); while let Some(ch) = self.next_char() { match ch { '/' if self.peek_char_is('*') => { @@ -403,12 +416,15 @@ impl<'a> Lexer<'a> { break; } } - _ => {} + ch => content.push(ch), } } if depth == 0 { - self.next_token() + if self.skip_comments { + return self.next_token(); + } + Ok(Token::BlockComment(content).into_span(start, self.position)) } else { let span = Span::inclusive(start, self.position); Err(LexerErrorKind::UnterminatedBlockComment { span }) @@ -431,7 +447,6 @@ impl<'a> Iterator for Lexer<'a> { } } } - #[cfg(test)] mod tests { use super::*; @@ -497,7 +512,7 @@ mod tests { let input = r#"#[deprecated]"#; let mut lexer = Lexer::new(input); - let token = lexer.next().unwrap().unwrap(); + let token = lexer.next_token().unwrap(); assert_eq!( token.token(), &Token::Attribute(Attribute::Secondary(SecondaryAttribute::Deprecated(None))) @@ -509,7 +524,7 @@ mod tests { let input = r#"#[deprecated("hello")]"#; let mut lexer = Lexer::new(input); - let token = lexer.next().unwrap().unwrap(); + let token = lexer.next_token().unwrap(); assert_eq!( token.token(), &Token::Attribute(Attribute::Secondary(crate::token::SecondaryAttribute::Deprecated( @@ -542,7 +557,7 @@ mod tests { let input = r#"#[custom(hello)]"#; let mut lexer = Lexer::new(input); - let token = lexer.next().unwrap().unwrap(); + let token = lexer.next_token().unwrap(); assert_eq!( token.token(), &Token::Attribute(Attribute::Secondary(SecondaryAttribute::Custom( @@ -556,7 +571,7 @@ mod tests { let input = r#"#[test]"#; let mut lexer = Lexer::new(input); - let token = lexer.next().unwrap().unwrap(); + let token = lexer.next_token().unwrap(); assert_eq!( token.token(), &Token::Attribute(Attribute::Function(FunctionAttribute::Test(TestScope::None))) @@ -568,7 +583,7 @@ mod tests { let input = r#"#[contract_library_method]"#; let mut lexer = Lexer::new(input); - let token = lexer.next().unwrap().unwrap(); + let token = lexer.next_token().unwrap(); assert_eq!( token.token(), &Token::Attribute(Attribute::Secondary(SecondaryAttribute::ContractLibraryMethod)) @@ -580,7 +595,7 @@ mod tests { let input = r#"#[test(should_fail)]"#; let mut lexer = Lexer::new(input); - let token = lexer.next().unwrap().unwrap(); + let token = lexer.next_token().unwrap(); assert_eq!( token.token(), &Token::Attribute(Attribute::Function(FunctionAttribute::Test( @@ -594,7 +609,7 @@ mod tests { let input = r#"#[test(should_fail_with = "hello")]"#; let mut lexer = Lexer::new(input); - let token = lexer.next().unwrap().unwrap(); + let token = lexer.next_token().unwrap(); assert_eq!( token.token(), &Token::Attribute(Attribute::Function(FunctionAttribute::Test( diff --git a/compiler/noirc_frontend/src/lexer/token.rs b/compiler/noirc_frontend/src/lexer/token.rs index 2ad2f3902b1..8b2acb5b28e 100644 --- a/compiler/noirc_frontend/src/lexer/token.rs +++ b/compiler/noirc_frontend/src/lexer/token.rs @@ -19,6 +19,8 @@ pub enum Token { Keyword(Keyword), IntType(IntType), Attribute(Attribute), + LineComment(String), + BlockComment(String), /// < Less, /// <= @@ -149,6 +151,8 @@ impl fmt::Display for Token { Token::FmtStr(ref b) => write!(f, "f{b}"), Token::Keyword(k) => write!(f, "{k}"), Token::Attribute(ref a) => write!(f, "{a}"), + Token::LineComment(ref s) => write!(f, "//{s}"), + Token::BlockComment(ref s) => write!(f, "/*{s}*/"), Token::IntType(ref i) => write!(f, "{i}"), Token::Less => write!(f, "<"), Token::LessEqual => write!(f, "<="), diff --git a/compiler/noirc_frontend/src/parser/parser.rs b/compiler/noirc_frontend/src/parser/parser.rs index 2da17c997c1..083646ceab0 100644 --- a/compiler/noirc_frontend/src/parser/parser.rs +++ b/compiler/noirc_frontend/src/parser/parser.rs @@ -1554,7 +1554,9 @@ where literal(), )) .map_with_span(Expression::new) - .or(parenthesized(expr_parser.clone())) + .or(parenthesized(expr_parser.clone()).map_with_span(|subexpr, span| { + Expression::new(ExpressionKind::Parenthesized(subexpr.into()), span) + })) .or(tuple(expr_parser)) .labelled(ParsingRuleLabel::Atom) } diff --git a/tooling/nargo_fmt/src/visitor/expr.rs b/tooling/nargo_fmt/src/visitor/expr.rs index 7677a6284ec..d8bea6c7530 100644 --- a/tooling/nargo_fmt/src/visitor/expr.rs +++ b/tooling/nargo_fmt/src/visitor/expr.rs @@ -1,6 +1,6 @@ use noirc_frontend::{ - hir::resolution::errors::Span, ArrayLiteral, BlockExpression, Expression, ExpressionKind, - Literal, Statement, + hir::resolution::errors::Span, lexer::Lexer, token::Token, ArrayLiteral, BlockExpression, + Expression, ExpressionKind, Literal, Statement, }; use super::FmtVisitor; @@ -10,6 +10,7 @@ impl FmtVisitor<'_> { let span = expr.span; let rewrite = self.format_expr(expr); + let rewrite = recover_comment_removed(slice!(self, span.start(), span.end()), rewrite); self.push_rewrite(rewrite, span); self.last_position = span.end(); @@ -94,6 +95,7 @@ impl FmtVisitor<'_> { literal.to_string() } }, + ExpressionKind::Parenthesized(subexpr) => format!("({})", self.format_expr(*subexpr)), // TODO: _expr => slice!(self, span.start(), span.end()).to_string(), } @@ -159,3 +161,30 @@ impl FmtVisitor<'_> { self.push_str(&block_str); } } + +fn recover_comment_removed(original: &str, new: String) -> String { + if changed_comment_content(original, &new) { + original.to_string() + } else { + new + } +} + +fn changed_comment_content(original: &str, new: &str) -> bool { + comments(original) != comments(new) +} + +fn comments(source: &str) -> Vec { + Lexer::new(source) + .skip_comments(false) + .flatten() + .filter_map(|spanned| { + if let Token::LineComment(content) | Token::BlockComment(content) = spanned.into_token() + { + Some(content) + } else { + None + } + }) + .collect() +} diff --git a/tooling/nargo_fmt/tests/expected/expr.nr b/tooling/nargo_fmt/tests/expected/expr.nr index 01bdb9d1399..6d4fedba4b3 100644 --- a/tooling/nargo_fmt/tests/expected/expr.nr +++ b/tooling/nargo_fmt/tests/expected/expr.nr @@ -71,3 +71,11 @@ fn test() { // 324 34 } + +fn parenthesized() { + value + (x as Field) +} + +fn parenthesized() { + (i as u8) + (j as u8) + (k as u8) + x + y + z +} diff --git a/tooling/nargo_fmt/tests/expected/infix.nr b/tooling/nargo_fmt/tests/expected/infix.nr index 13b53170da6..f930f79ebcb 100644 --- a/tooling/nargo_fmt/tests/expected/infix.nr +++ b/tooling/nargo_fmt/tests/expected/infix.nr @@ -2,4 +2,7 @@ fn foo() { 40 + 2; !40 + 2; 40 + 2 == 42; + + 40/*test*/ + 2 == 42; + 40 + 2/*test*/ == 42; } diff --git a/tooling/nargo_fmt/tests/expected/unary_operators.nr b/tooling/nargo_fmt/tests/expected/unary_operators.nr index 88140ac9a6e..e65690cf482 100644 --- a/tooling/nargo_fmt/tests/expected/unary_operators.nr +++ b/tooling/nargo_fmt/tests/expected/unary_operators.nr @@ -1,3 +1,5 @@ fn main() { - -1 + -1; + -/*test*/1; + -/*test*/1; } diff --git a/tooling/nargo_fmt/tests/input/expr.nr b/tooling/nargo_fmt/tests/input/expr.nr index 601e946cb0c..25602c640d3 100644 --- a/tooling/nargo_fmt/tests/input/expr.nr +++ b/tooling/nargo_fmt/tests/input/expr.nr @@ -80,3 +80,11 @@ fn test() { // 324 34 } + +fn parenthesized() { + value + ( x as Field ) +} + +fn parenthesized() { + ( i as u8 ) + ( j as u8 ) + ( k as u8 ) + x + y + z +} diff --git a/tooling/nargo_fmt/tests/input/infix.nr b/tooling/nargo_fmt/tests/input/infix.nr index c01969f0502..df57f956097 100644 --- a/tooling/nargo_fmt/tests/input/infix.nr +++ b/tooling/nargo_fmt/tests/input/infix.nr @@ -2,4 +2,7 @@ fn foo() { 40 + 2; !40+2; 40 + 2 == 42; + +40/*test*/ + 2 == 42; + 40 + 2/*test*/ == 42; } diff --git a/tooling/nargo_fmt/tests/input/unary_operators.nr b/tooling/nargo_fmt/tests/input/unary_operators.nr index 5805c7081f5..62e4a1c4d27 100644 --- a/tooling/nargo_fmt/tests/input/unary_operators.nr +++ b/tooling/nargo_fmt/tests/input/unary_operators.nr @@ -1,3 +1,5 @@ fn main() { - -1 + -1; +-/*test*/1; + -/*test*/1; } \ No newline at end of file