aboutsummaryrefslogtreecommitdiff
path: root/crates/ra_syntax
diff options
context:
space:
mode:
Diffstat (limited to 'crates/ra_syntax')
-rw-r--r--crates/ra_syntax/Cargo.toml1
-rw-r--r--crates/ra_syntax/src/ast.rs46
-rw-r--r--crates/ra_syntax/src/ast/extensions.rs10
-rw-r--r--crates/ra_syntax/src/ast/generated.rs2
-rw-r--r--crates/ra_syntax/src/ast/traits.rs12
-rw-r--r--crates/ra_syntax/src/grammar.ron2
-rw-r--r--crates/ra_syntax/src/parsing/lexer.rs35
-rw-r--r--crates/ra_syntax/src/validation.rs25
-rw-r--r--crates/ra_syntax/src/validation/field_expr.rs13
-rw-r--r--crates/ra_syntax/test_data/parser/inline/err/0010_bad_tuple_index_expr.txt7
-rw-r--r--crates/ra_syntax/test_data/parser/inline/ok/0011_field_expr.txt6
-rw-r--r--crates/ra_syntax/test_data/parser/inline/ok/0061_struct_lit.rs1
-rw-r--r--crates/ra_syntax/test_data/parser/inline/ok/0061_struct_lit.txt33
-rw-r--r--crates/ra_syntax/test_data/parser/inline/ok/0137_await_expr.txt6
-rw-r--r--crates/ra_syntax/test_data/parser/inline/ok/0138_associated_type_bounds.rs1
-rw-r--r--crates/ra_syntax/test_data/parser/inline/ok/0138_associated_type_bounds.txt55
-rw-r--r--crates/ra_syntax/test_data/parser/inline/ok/0138_self_param_outer_attr.rs1
-rw-r--r--crates/ra_syntax/test_data/parser/inline/ok/0138_self_param_outer_attr.txt23
-rw-r--r--crates/ra_syntax/test_data/parser/inline/ok/0139_param_outer_arg.rs1
-rw-r--r--crates/ra_syntax/test_data/parser/inline/ok/0139_param_outer_arg.txt32
-rw-r--r--crates/ra_syntax/test_data/parser/ok/0051_parameter_attrs.rs21
-rw-r--r--crates/ra_syntax/test_data/parser/ok/0051_parameter_attrs.txt477
22 files changed, 755 insertions, 55 deletions
diff --git a/crates/ra_syntax/Cargo.toml b/crates/ra_syntax/Cargo.toml
index bc1c88070..5f8585878 100644
--- a/crates/ra_syntax/Cargo.toml
+++ b/crates/ra_syntax/Cargo.toml
@@ -8,7 +8,6 @@ description = "Comment and whitespace preserving parser for the Rust langauge"
8repository = "https://github.com/rust-analyzer/rust-analyzer" 8repository = "https://github.com/rust-analyzer/rust-analyzer"
9 9
10[dependencies] 10[dependencies]
11unicode-xid = "0.1.0"
12itertools = "0.8.0" 11itertools = "0.8.0"
13rowan = "0.6.1" 12rowan = "0.6.1"
14ra_rustc_lexer = { version = "0.1.0-pre.2" } 13ra_rustc_lexer = { version = "0.1.0-pre.2" }
diff --git a/crates/ra_syntax/src/ast.rs b/crates/ra_syntax/src/ast.rs
index c5746d98d..6f0489617 100644
--- a/crates/ra_syntax/src/ast.rs
+++ b/crates/ra_syntax/src/ast.rs
@@ -139,6 +139,52 @@ fn test_doc_comment_preserves_newlines() {
139} 139}
140 140
141#[test] 141#[test]
142fn test_doc_comment_single_line_block_strips_suffix() {
143 let file = SourceFile::parse(
144 r#"
145 /** this is mod foo*/
146 mod foo {}
147 "#,
148 )
149 .ok()
150 .unwrap();
151 let module = file.syntax().descendants().find_map(Module::cast).unwrap();
152 assert_eq!("this is mod foo", module.doc_comment_text().unwrap());
153}
154
155#[test]
156fn test_doc_comment_single_line_block_strips_suffix_whitespace() {
157 let file = SourceFile::parse(
158 r#"
159 /** this is mod foo */
160 mod foo {}
161 "#,
162 )
163 .ok()
164 .unwrap();
165 let module = file.syntax().descendants().find_map(Module::cast).unwrap();
166 assert_eq!("this is mod foo", module.doc_comment_text().unwrap());
167}
168
169#[test]
170fn test_doc_comment_multi_line_block_strips_suffix() {
171 let file = SourceFile::parse(
172 r#"
173 /**
174 this
175 is
176 mod foo
177 */
178 mod foo {}
179 "#,
180 )
181 .ok()
182 .unwrap();
183 let module = file.syntax().descendants().find_map(Module::cast).unwrap();
184 assert_eq!(" this\n is\n mod foo", module.doc_comment_text().unwrap());
185}
186
187#[test]
142fn test_where_predicates() { 188fn test_where_predicates() {
143 fn assert_bound(text: &str, bound: Option<TypeBound>) { 189 fn assert_bound(text: &str, bound: Option<TypeBound>) {
144 assert_eq!(text, bound.unwrap().syntax().text().to_string()); 190 assert_eq!(text, bound.unwrap().syntax().text().to_string());
diff --git a/crates/ra_syntax/src/ast/extensions.rs b/crates/ra_syntax/src/ast/extensions.rs
index d4873b39a..2a59cf653 100644
--- a/crates/ra_syntax/src/ast/extensions.rs
+++ b/crates/ra_syntax/src/ast/extensions.rs
@@ -91,6 +91,7 @@ impl ast::Attr {
91#[derive(Debug, Clone, PartialEq, Eq)] 91#[derive(Debug, Clone, PartialEq, Eq)]
92pub enum PathSegmentKind { 92pub enum PathSegmentKind {
93 Name(ast::NameRef), 93 Name(ast::NameRef),
94 Type { type_ref: Option<ast::TypeRef>, trait_ref: Option<ast::PathType> },
94 SelfKw, 95 SelfKw,
95 SuperKw, 96 SuperKw,
96 CrateKw, 97 CrateKw,
@@ -112,6 +113,15 @@ impl ast::PathSegment {
112 T![self] => PathSegmentKind::SelfKw, 113 T![self] => PathSegmentKind::SelfKw,
113 T![super] => PathSegmentKind::SuperKw, 114 T![super] => PathSegmentKind::SuperKw,
114 T![crate] => PathSegmentKind::CrateKw, 115 T![crate] => PathSegmentKind::CrateKw,
116 T![<] => {
117 // <T> or <T as Trait>
118 // T is any TypeRef, Trait has to be a PathType
119 let mut type_refs =
120 self.syntax().children().filter(|node| ast::TypeRef::can_cast(node.kind()));
121 let type_ref = type_refs.next().and_then(ast::TypeRef::cast);
122 let trait_ref = type_refs.next().and_then(ast::PathType::cast);
123 PathSegmentKind::Type { type_ref, trait_ref }
124 }
115 _ => return None, 125 _ => return None,
116 } 126 }
117 }; 127 };
diff --git a/crates/ra_syntax/src/ast/generated.rs b/crates/ra_syntax/src/ast/generated.rs
index da8cf4ae8..f322e1d84 100644
--- a/crates/ra_syntax/src/ast/generated.rs
+++ b/crates/ra_syntax/src/ast/generated.rs
@@ -2013,6 +2013,7 @@ impl AstNode for Param {
2013 2013
2014 2014
2015impl ast::TypeAscriptionOwner for Param {} 2015impl ast::TypeAscriptionOwner for Param {}
2016impl ast::AttrsOwner for Param {}
2016impl Param { 2017impl Param {
2017 pub fn pat(&self) -> Option<Pat> { 2018 pub fn pat(&self) -> Option<Pat> {
2018 super::child_opt(self) 2019 super::child_opt(self)
@@ -2667,6 +2668,7 @@ impl AstNode for SelfParam {
2667 2668
2668 2669
2669impl ast::TypeAscriptionOwner for SelfParam {} 2670impl ast::TypeAscriptionOwner for SelfParam {}
2671impl ast::AttrsOwner for SelfParam {}
2670impl SelfParam {} 2672impl SelfParam {}
2671 2673
2672// SlicePat 2674// SlicePat
diff --git a/crates/ra_syntax/src/ast/traits.rs b/crates/ra_syntax/src/ast/traits.rs
index 6ed1b5213..1b9a2b20c 100644
--- a/crates/ra_syntax/src/ast/traits.rs
+++ b/crates/ra_syntax/src/ast/traits.rs
@@ -115,8 +115,8 @@ pub trait DocCommentsOwner: AstNode {
115 } 115 }
116 116
117 /// Returns the textual content of a doc comment block as a single string. 117 /// Returns the textual content of a doc comment block as a single string.
118 /// That is, strips leading `///` (+ optional 1 character of whitespace) 118 /// That is, strips leading `///` (+ optional 1 character of whitespace),
119 /// and joins lines. 119 /// trailing `*/`, trailing whitespace and then joins the lines.
120 fn doc_comment_text(&self) -> Option<String> { 120 fn doc_comment_text(&self) -> Option<String> {
121 let mut has_comments = false; 121 let mut has_comments = false;
122 let docs = self 122 let docs = self
@@ -136,7 +136,13 @@ pub trait DocCommentsOwner: AstNode {
136 prefix_len 136 prefix_len
137 }; 137 };
138 138
139 line[pos..].to_owned() 139 let end = if comment.kind().shape.is_block() && line.ends_with("*/") {
140 line.len() - 2
141 } else {
142 line.len()
143 };
144
145 line[pos..end].trim_end().to_owned()
140 }) 146 })
141 .join("\n"); 147 .join("\n");
142 148
diff --git a/crates/ra_syntax/src/grammar.ron b/crates/ra_syntax/src/grammar.ron
index 817dedfbf..f2c20573e 100644
--- a/crates/ra_syntax/src/grammar.ron
+++ b/crates/ra_syntax/src/grammar.ron
@@ -642,12 +642,14 @@ Grammar(
642 "SelfParam": ( 642 "SelfParam": (
643 traits: [ 643 traits: [
644 "TypeAscriptionOwner", 644 "TypeAscriptionOwner",
645 "AttrsOwner",
645 ] 646 ]
646 ), 647 ),
647 "Param": ( 648 "Param": (
648 options: [ "Pat" ], 649 options: [ "Pat" ],
649 traits: [ 650 traits: [
650 "TypeAscriptionOwner", 651 "TypeAscriptionOwner",
652 "AttrsOwner",
651 ] 653 ]
652 ), 654 ),
653 "UseItem": ( 655 "UseItem": (
diff --git a/crates/ra_syntax/src/parsing/lexer.rs b/crates/ra_syntax/src/parsing/lexer.rs
index 2a4343b0a..06822ea91 100644
--- a/crates/ra_syntax/src/parsing/lexer.rs
+++ b/crates/ra_syntax/src/parsing/lexer.rs
@@ -12,6 +12,19 @@ pub struct Token {
12 pub len: TextUnit, 12 pub len: TextUnit,
13} 13}
14 14
15fn match_literal_kind(kind: ra_rustc_lexer::LiteralKind) -> SyntaxKind {
16 match kind {
17 ra_rustc_lexer::LiteralKind::Int { .. } => INT_NUMBER,
18 ra_rustc_lexer::LiteralKind::Float { .. } => FLOAT_NUMBER,
19 ra_rustc_lexer::LiteralKind::Char { .. } => CHAR,
20 ra_rustc_lexer::LiteralKind::Byte { .. } => BYTE,
21 ra_rustc_lexer::LiteralKind::Str { .. } => STRING,
22 ra_rustc_lexer::LiteralKind::ByteStr { .. } => BYTE_STRING,
23 ra_rustc_lexer::LiteralKind::RawStr { .. } => RAW_STRING,
24 ra_rustc_lexer::LiteralKind::RawByteStr { .. } => RAW_BYTE_STRING,
25 }
26}
27
15/// Break a string up into its component tokens 28/// Break a string up into its component tokens
16pub fn tokenize(text: &str) -> Vec<Token> { 29pub fn tokenize(text: &str) -> Vec<Token> {
17 if text.is_empty() { 30 if text.is_empty() {
@@ -53,16 +66,7 @@ pub fn tokenize(text: &str) -> Vec<Token> {
53 } 66 }
54 } 67 }
55 ra_rustc_lexer::TokenKind::RawIdent => IDENT, 68 ra_rustc_lexer::TokenKind::RawIdent => IDENT,
56 ra_rustc_lexer::TokenKind::Literal { kind, .. } => match kind { 69 ra_rustc_lexer::TokenKind::Literal { kind, .. } => match_literal_kind(kind),
57 ra_rustc_lexer::LiteralKind::Int { .. } => INT_NUMBER,
58 ra_rustc_lexer::LiteralKind::Float { .. } => FLOAT_NUMBER,
59 ra_rustc_lexer::LiteralKind::Char { .. } => CHAR,
60 ra_rustc_lexer::LiteralKind::Byte { .. } => BYTE,
61 ra_rustc_lexer::LiteralKind::Str { .. } => STRING,
62 ra_rustc_lexer::LiteralKind::ByteStr { .. } => BYTE_STRING,
63 ra_rustc_lexer::LiteralKind::RawStr { .. } => RAW_STRING,
64 ra_rustc_lexer::LiteralKind::RawByteStr { .. } => RAW_BYTE_STRING,
65 },
66 ra_rustc_lexer::TokenKind::Lifetime { .. } => LIFETIME, 70 ra_rustc_lexer::TokenKind::Lifetime { .. } => LIFETIME,
67 ra_rustc_lexer::TokenKind::Semi => SEMI, 71 ra_rustc_lexer::TokenKind::Semi => SEMI,
68 ra_rustc_lexer::TokenKind::Comma => COMMA, 72 ra_rustc_lexer::TokenKind::Comma => COMMA,
@@ -131,16 +135,7 @@ pub fn classify_literal(text: &str) -> Option<Token> {
131 return None; 135 return None;
132 } 136 }
133 let kind = match t.kind { 137 let kind = match t.kind {
134 ra_rustc_lexer::TokenKind::Literal { kind, .. } => match kind { 138 ra_rustc_lexer::TokenKind::Literal { kind, .. } => match_literal_kind(kind),
135 ra_rustc_lexer::LiteralKind::Int { .. } => INT_NUMBER,
136 ra_rustc_lexer::LiteralKind::Float { .. } => FLOAT_NUMBER,
137 ra_rustc_lexer::LiteralKind::Char { .. } => CHAR,
138 ra_rustc_lexer::LiteralKind::Byte { .. } => BYTE,
139 ra_rustc_lexer::LiteralKind::Str { .. } => STRING,
140 ra_rustc_lexer::LiteralKind::ByteStr { .. } => BYTE_STRING,
141 ra_rustc_lexer::LiteralKind::RawStr { .. } => RAW_STRING,
142 ra_rustc_lexer::LiteralKind::RawByteStr { .. } => RAW_BYTE_STRING,
143 },
144 _ => return None, 139 _ => return None,
145 }; 140 };
146 Some(Token { kind, len: TextUnit::from_usize(t.len) }) 141 Some(Token { kind, len: TextUnit::from_usize(t.len) })
diff --git a/crates/ra_syntax/src/validation.rs b/crates/ra_syntax/src/validation.rs
index 1f904434e..2bb3c0a03 100644
--- a/crates/ra_syntax/src/validation.rs
+++ b/crates/ra_syntax/src/validation.rs
@@ -1,13 +1,12 @@
1mod block; 1mod block;
2mod field_expr;
3 2
4use ra_rustc_lexer::unescape; 3use ra_rustc_lexer::unescape;
5 4
6use crate::{ 5use crate::{
7 algo::visit::{visitor_ctx, VisitorCtx}, 6 algo::visit::{visitor_ctx, VisitorCtx},
8 ast, SyntaxError, SyntaxErrorKind, 7 ast, AstNode, SyntaxError, SyntaxErrorKind,
9 SyntaxKind::{BYTE, BYTE_STRING, CHAR, STRING}, 8 SyntaxKind::{BYTE, BYTE_STRING, CHAR, INT_NUMBER, STRING},
10 SyntaxNode, TextUnit, T, 9 SyntaxNode, SyntaxToken, TextUnit, T,
11}; 10};
12 11
13#[derive(Clone, Copy, Debug, PartialEq, Eq, PartialOrd, Ord, Hash)] 12#[derive(Clone, Copy, Debug, PartialEq, Eq, PartialOrd, Ord, Hash)]
@@ -101,7 +100,8 @@ pub(crate) fn validate(root: &SyntaxNode) -> Vec<SyntaxError> {
101 let _ = visitor_ctx(&mut errors) 100 let _ = visitor_ctx(&mut errors)
102 .visit::<ast::Literal, _>(validate_literal) 101 .visit::<ast::Literal, _>(validate_literal)
103 .visit::<ast::Block, _>(block::validate_block_node) 102 .visit::<ast::Block, _>(block::validate_block_node)
104 .visit::<ast::FieldExpr, _>(field_expr::validate_field_expr_node) 103 .visit::<ast::FieldExpr, _>(|it, errors| validate_numeric_name(it.name_ref(), errors))
104 .visit::<ast::NamedField, _>(|it, errors| validate_numeric_name(it.name_ref(), errors))
105 .accept(&node); 105 .accept(&node);
106 } 106 }
107 errors 107 errors
@@ -189,3 +189,18 @@ pub(crate) fn validate_block_structure(root: &SyntaxNode) {
189 } 189 }
190 } 190 }
191} 191}
192
193fn validate_numeric_name(name_ref: Option<ast::NameRef>, errors: &mut Vec<SyntaxError>) {
194 if let Some(int_token) = int_token(name_ref) {
195 if int_token.text().chars().any(|c| !c.is_digit(10)) {
196 errors.push(SyntaxError::new(
197 SyntaxErrorKind::InvalidTupleIndexFormat,
198 int_token.text_range(),
199 ));
200 }
201 }
202
203 fn int_token(name_ref: Option<ast::NameRef>) -> Option<SyntaxToken> {
204 name_ref?.syntax().first_child_or_token()?.into_token().filter(|it| it.kind() == INT_NUMBER)
205 }
206}
diff --git a/crates/ra_syntax/src/validation/field_expr.rs b/crates/ra_syntax/src/validation/field_expr.rs
deleted file mode 100644
index 004f199fd..000000000
--- a/crates/ra_syntax/src/validation/field_expr.rs
+++ /dev/null
@@ -1,13 +0,0 @@
1use crate::{
2 ast::{self, FieldKind},
3 SyntaxError,
4 SyntaxErrorKind::*,
5};
6
7pub(crate) fn validate_field_expr_node(node: ast::FieldExpr, errors: &mut Vec<SyntaxError>) {
8 if let Some(FieldKind::Index(idx)) = node.field_access() {
9 if idx.text().chars().any(|c| c < '0' || c > '9') {
10 errors.push(SyntaxError::new(InvalidTupleIndexFormat, idx.text_range()));
11 }
12 }
13}
diff --git a/crates/ra_syntax/test_data/parser/inline/err/0010_bad_tuple_index_expr.txt b/crates/ra_syntax/test_data/parser/inline/err/0010_bad_tuple_index_expr.txt
index a21b29c80..465e79e7b 100644
--- a/crates/ra_syntax/test_data/parser/inline/err/0010_bad_tuple_index_expr.txt
+++ b/crates/ra_syntax/test_data/parser/inline/err/0010_bad_tuple_index_expr.txt
@@ -30,7 +30,8 @@ SOURCE_FILE@[0; 47)
30 NAME_REF@[25; 26) 30 NAME_REF@[25; 26)
31 IDENT@[25; 26) "x" 31 IDENT@[25; 26) "x"
32 DOT@[26; 27) "." 32 DOT@[26; 27) "."
33 INT_NUMBER@[27; 31) "1i32" 33 NAME_REF@[27; 31)
34 INT_NUMBER@[27; 31) "1i32"
34 SEMI@[31; 32) ";" 35 SEMI@[31; 32) ";"
35 WHITESPACE@[32; 37) "\n " 36 WHITESPACE@[32; 37) "\n "
36 EXPR_STMT@[37; 44) 37 EXPR_STMT@[37; 44)
@@ -41,11 +42,11 @@ SOURCE_FILE@[0; 47)
41 NAME_REF@[37; 38) 42 NAME_REF@[37; 38)
42 IDENT@[37; 38) "x" 43 IDENT@[37; 38) "x"
43 DOT@[38; 39) "." 44 DOT@[38; 39) "."
44 INT_NUMBER@[39; 43) "0x01" 45 NAME_REF@[39; 43)
46 INT_NUMBER@[39; 43) "0x01"
45 SEMI@[43; 44) ";" 47 SEMI@[43; 44) ";"
46 WHITESPACE@[44; 45) "\n" 48 WHITESPACE@[44; 45) "\n"
47 R_CURLY@[45; 46) "}" 49 R_CURLY@[45; 46) "}"
48 WHITESPACE@[46; 47) "\n" 50 WHITESPACE@[46; 47) "\n"
49error [17; 19): Tuple (struct) field access is only allowed through decimal integers with no underscores or suffix
50error [27; 31): Tuple (struct) field access is only allowed through decimal integers with no underscores or suffix 51error [27; 31): Tuple (struct) field access is only allowed through decimal integers with no underscores or suffix
51error [39; 43): Tuple (struct) field access is only allowed through decimal integers with no underscores or suffix 52error [39; 43): Tuple (struct) field access is only allowed through decimal integers with no underscores or suffix
diff --git a/crates/ra_syntax/test_data/parser/inline/ok/0011_field_expr.txt b/crates/ra_syntax/test_data/parser/inline/ok/0011_field_expr.txt
index 78054ec5a..1d2cf2761 100644
--- a/crates/ra_syntax/test_data/parser/inline/ok/0011_field_expr.txt
+++ b/crates/ra_syntax/test_data/parser/inline/ok/0011_field_expr.txt
@@ -32,7 +32,8 @@ SOURCE_FILE@[0; 48)
32 NAME_REF@[26; 27) 32 NAME_REF@[26; 27)
33 IDENT@[26; 27) "x" 33 IDENT@[26; 27) "x"
34 DOT@[27; 28) "." 34 DOT@[27; 28) "."
35 INT_NUMBER@[28; 29) "0" 35 NAME_REF@[28; 29)
36 INT_NUMBER@[28; 29) "0"
36 DOT@[29; 30) "." 37 DOT@[29; 30) "."
37 NAME_REF@[30; 33) 38 NAME_REF@[30; 33)
38 IDENT@[30; 33) "bar" 39 IDENT@[30; 33) "bar"
@@ -47,7 +48,8 @@ SOURCE_FILE@[0; 48)
47 NAME_REF@[39; 40) 48 NAME_REF@[39; 40)
48 IDENT@[39; 40) "x" 49 IDENT@[39; 40) "x"
49 DOT@[40; 41) "." 50 DOT@[40; 41) "."
50 INT_NUMBER@[41; 42) "0" 51 NAME_REF@[41; 42)
52 INT_NUMBER@[41; 42) "0"
51 ARG_LIST@[42; 44) 53 ARG_LIST@[42; 44)
52 L_PAREN@[42; 43) "(" 54 L_PAREN@[42; 43) "("
53 R_PAREN@[43; 44) ")" 55 R_PAREN@[43; 44) ")"
diff --git a/crates/ra_syntax/test_data/parser/inline/ok/0061_struct_lit.rs b/crates/ra_syntax/test_data/parser/inline/ok/0061_struct_lit.rs
index eb711f68a..6285e5549 100644
--- a/crates/ra_syntax/test_data/parser/inline/ok/0061_struct_lit.rs
+++ b/crates/ra_syntax/test_data/parser/inline/ok/0061_struct_lit.rs
@@ -2,4 +2,5 @@ fn foo() {
2 S {}; 2 S {};
3 S { x, y: 32, }; 3 S { x, y: 32, };
4 S { x, y: 32, ..Default::default() }; 4 S { x, y: 32, ..Default::default() };
5 TupleStruct { 0: 1 };
5} 6}
diff --git a/crates/ra_syntax/test_data/parser/inline/ok/0061_struct_lit.txt b/crates/ra_syntax/test_data/parser/inline/ok/0061_struct_lit.txt
index 94d1bfe2e..d06594cae 100644
--- a/crates/ra_syntax/test_data/parser/inline/ok/0061_struct_lit.txt
+++ b/crates/ra_syntax/test_data/parser/inline/ok/0061_struct_lit.txt
@@ -1,5 +1,5 @@
1SOURCE_FILE@[0; 86) 1SOURCE_FILE@[0; 112)
2 FN_DEF@[0; 85) 2 FN_DEF@[0; 111)
3 FN_KW@[0; 2) "fn" 3 FN_KW@[0; 2) "fn"
4 WHITESPACE@[2; 3) " " 4 WHITESPACE@[2; 3) " "
5 NAME@[3; 6) 5 NAME@[3; 6)
@@ -8,7 +8,7 @@ SOURCE_FILE@[0; 86)
8 L_PAREN@[6; 7) "(" 8 L_PAREN@[6; 7) "("
9 R_PAREN@[7; 8) ")" 9 R_PAREN@[7; 8) ")"
10 WHITESPACE@[8; 9) " " 10 WHITESPACE@[8; 9) " "
11 BLOCK@[9; 85) 11 BLOCK@[9; 111)
12 L_CURLY@[9; 10) "{" 12 L_CURLY@[9; 10) "{"
13 WHITESPACE@[10; 15) "\n " 13 WHITESPACE@[10; 15) "\n "
14 EXPR_STMT@[15; 20) 14 EXPR_STMT@[15; 20)
@@ -92,6 +92,27 @@ SOURCE_FILE@[0; 86)
92 WHITESPACE@[80; 81) " " 92 WHITESPACE@[80; 81) " "
93 R_CURLY@[81; 82) "}" 93 R_CURLY@[81; 82) "}"
94 SEMI@[82; 83) ";" 94 SEMI@[82; 83) ";"
95 WHITESPACE@[83; 84) "\n" 95 WHITESPACE@[83; 88) "\n "
96 R_CURLY@[84; 85) "}" 96 EXPR_STMT@[88; 109)
97 WHITESPACE@[85; 86) "\n" 97 STRUCT_LIT@[88; 108)
98 PATH@[88; 99)
99 PATH_SEGMENT@[88; 99)
100 NAME_REF@[88; 99)
101 IDENT@[88; 99) "TupleStruct"
102 WHITESPACE@[99; 100) " "
103 NAMED_FIELD_LIST@[100; 108)
104 L_CURLY@[100; 101) "{"
105 WHITESPACE@[101; 102) " "
106 NAMED_FIELD@[102; 106)
107 NAME_REF@[102; 103)
108 INT_NUMBER@[102; 103) "0"
109 COLON@[103; 104) ":"
110 WHITESPACE@[104; 105) " "
111 LITERAL@[105; 106)
112 INT_NUMBER@[105; 106) "1"
113 WHITESPACE@[106; 107) " "
114 R_CURLY@[107; 108) "}"
115 SEMI@[108; 109) ";"
116 WHITESPACE@[109; 110) "\n"
117 R_CURLY@[110; 111) "}"
118 WHITESPACE@[111; 112) "\n"
diff --git a/crates/ra_syntax/test_data/parser/inline/ok/0137_await_expr.txt b/crates/ra_syntax/test_data/parser/inline/ok/0137_await_expr.txt
index 99bd76ace..7adb662de 100644
--- a/crates/ra_syntax/test_data/parser/inline/ok/0137_await_expr.txt
+++ b/crates/ra_syntax/test_data/parser/inline/ok/0137_await_expr.txt
@@ -31,7 +31,8 @@ SOURCE_FILE@[0; 67)
31 NAME_REF@[28; 29) 31 NAME_REF@[28; 29)
32 IDENT@[28; 29) "x" 32 IDENT@[28; 29) "x"
33 DOT@[29; 30) "." 33 DOT@[29; 30) "."
34 INT_NUMBER@[30; 31) "0" 34 NAME_REF@[30; 31)
35 INT_NUMBER@[30; 31) "0"
35 DOT@[31; 32) "." 36 DOT@[31; 32) "."
36 AWAIT_KW@[32; 37) "await" 37 AWAIT_KW@[32; 37) "await"
37 SEMI@[37; 38) ";" 38 SEMI@[37; 38) ";"
@@ -48,7 +49,8 @@ SOURCE_FILE@[0; 67)
48 NAME_REF@[43; 44) 49 NAME_REF@[43; 44)
49 IDENT@[43; 44) "x" 50 IDENT@[43; 44) "x"
50 DOT@[44; 45) "." 51 DOT@[44; 45) "."
51 INT_NUMBER@[45; 46) "0" 52 NAME_REF@[45; 46)
53 INT_NUMBER@[45; 46) "0"
52 ARG_LIST@[46; 48) 54 ARG_LIST@[46; 48)
53 L_PAREN@[46; 47) "(" 55 L_PAREN@[46; 47) "("
54 R_PAREN@[47; 48) ")" 56 R_PAREN@[47; 48) ")"
diff --git a/crates/ra_syntax/test_data/parser/inline/ok/0138_associated_type_bounds.rs b/crates/ra_syntax/test_data/parser/inline/ok/0138_associated_type_bounds.rs
new file mode 100644
index 000000000..eb21a657b
--- /dev/null
+++ b/crates/ra_syntax/test_data/parser/inline/ok/0138_associated_type_bounds.rs
@@ -0,0 +1 @@
fn print_all<T: Iterator<Item: Display>>(printables: T) {}
diff --git a/crates/ra_syntax/test_data/parser/inline/ok/0138_associated_type_bounds.txt b/crates/ra_syntax/test_data/parser/inline/ok/0138_associated_type_bounds.txt
new file mode 100644
index 000000000..33e75510d
--- /dev/null
+++ b/crates/ra_syntax/test_data/parser/inline/ok/0138_associated_type_bounds.txt
@@ -0,0 +1,55 @@
1SOURCE_FILE@[0; 59)
2 FN_DEF@[0; 58)
3 FN_KW@[0; 2) "fn"
4 WHITESPACE@[2; 3) " "
5 NAME@[3; 12)
6 IDENT@[3; 12) "print_all"
7 TYPE_PARAM_LIST@[12; 40)
8 L_ANGLE@[12; 13) "<"
9 TYPE_PARAM@[13; 39)
10 NAME@[13; 14)
11 IDENT@[13; 14) "T"
12 COLON@[14; 15) ":"
13 WHITESPACE@[15; 16) " "
14 TYPE_BOUND_LIST@[16; 39)
15 TYPE_BOUND@[16; 39)
16 PATH_TYPE@[16; 39)
17 PATH@[16; 39)
18 PATH_SEGMENT@[16; 39)
19 NAME_REF@[16; 24)
20 IDENT@[16; 24) "Iterator"
21 TYPE_ARG_LIST@[24; 39)
22 L_ANGLE@[24; 25) "<"
23 ASSOC_TYPE_ARG@[25; 38)
24 NAME_REF@[25; 29)
25 IDENT@[25; 29) "Item"
26 COLON@[29; 30) ":"
27 WHITESPACE@[30; 31) " "
28 TYPE_BOUND_LIST@[31; 38)
29 TYPE_BOUND@[31; 38)
30 PATH_TYPE@[31; 38)
31 PATH@[31; 38)
32 PATH_SEGMENT@[31; 38)
33 NAME_REF@[31; 38)
34 IDENT@[31; 38) "Display"
35 R_ANGLE@[38; 39) ">"
36 R_ANGLE@[39; 40) ">"
37 PARAM_LIST@[40; 55)
38 L_PAREN@[40; 41) "("
39 PARAM@[41; 54)
40 BIND_PAT@[41; 51)
41 NAME@[41; 51)
42 IDENT@[41; 51) "printables"
43 COLON@[51; 52) ":"
44 WHITESPACE@[52; 53) " "
45 PATH_TYPE@[53; 54)
46 PATH@[53; 54)
47 PATH_SEGMENT@[53; 54)
48 NAME_REF@[53; 54)
49 IDENT@[53; 54) "T"
50 R_PAREN@[54; 55) ")"
51 WHITESPACE@[55; 56) " "
52 BLOCK@[56; 58)
53 L_CURLY@[56; 57) "{"
54 R_CURLY@[57; 58) "}"
55 WHITESPACE@[58; 59) "\n"
diff --git a/crates/ra_syntax/test_data/parser/inline/ok/0138_self_param_outer_attr.rs b/crates/ra_syntax/test_data/parser/inline/ok/0138_self_param_outer_attr.rs
new file mode 100644
index 000000000..35155057a
--- /dev/null
+++ b/crates/ra_syntax/test_data/parser/inline/ok/0138_self_param_outer_attr.rs
@@ -0,0 +1 @@
fn f(#[must_use] self) {}
diff --git a/crates/ra_syntax/test_data/parser/inline/ok/0138_self_param_outer_attr.txt b/crates/ra_syntax/test_data/parser/inline/ok/0138_self_param_outer_attr.txt
new file mode 100644
index 000000000..49b14e632
--- /dev/null
+++ b/crates/ra_syntax/test_data/parser/inline/ok/0138_self_param_outer_attr.txt
@@ -0,0 +1,23 @@
1SOURCE_FILE@[0; 26)
2 FN_DEF@[0; 25)
3 FN_KW@[0; 2) "fn"
4 WHITESPACE@[2; 3) " "
5 NAME@[3; 4)
6 IDENT@[3; 4) "f"
7 PARAM_LIST@[4; 22)
8 L_PAREN@[4; 5) "("
9 ATTR@[5; 16)
10 POUND@[5; 6) "#"
11 TOKEN_TREE@[6; 16)
12 L_BRACK@[6; 7) "["
13 IDENT@[7; 15) "must_use"
14 R_BRACK@[15; 16) "]"
15 WHITESPACE@[16; 17) " "
16 SELF_PARAM@[17; 21)
17 SELF_KW@[17; 21) "self"
18 R_PAREN@[21; 22) ")"
19 WHITESPACE@[22; 23) " "
20 BLOCK@[23; 25)
21 L_CURLY@[23; 24) "{"
22 R_CURLY@[24; 25) "}"
23 WHITESPACE@[25; 26) "\n"
diff --git a/crates/ra_syntax/test_data/parser/inline/ok/0139_param_outer_arg.rs b/crates/ra_syntax/test_data/parser/inline/ok/0139_param_outer_arg.rs
new file mode 100644
index 000000000..c238be791
--- /dev/null
+++ b/crates/ra_syntax/test_data/parser/inline/ok/0139_param_outer_arg.rs
@@ -0,0 +1 @@
fn f(#[attr1] pat: Type) {}
diff --git a/crates/ra_syntax/test_data/parser/inline/ok/0139_param_outer_arg.txt b/crates/ra_syntax/test_data/parser/inline/ok/0139_param_outer_arg.txt
new file mode 100644
index 000000000..91c5e5f9a
--- /dev/null
+++ b/crates/ra_syntax/test_data/parser/inline/ok/0139_param_outer_arg.txt
@@ -0,0 +1,32 @@
1SOURCE_FILE@[0; 28)
2 FN_DEF@[0; 27)
3 FN_KW@[0; 2) "fn"
4 WHITESPACE@[2; 3) " "
5 NAME@[3; 4)
6 IDENT@[3; 4) "f"
7 PARAM_LIST@[4; 24)
8 L_PAREN@[4; 5) "("
9 ATTR@[5; 13)
10 POUND@[5; 6) "#"
11 TOKEN_TREE@[6; 13)
12 L_BRACK@[6; 7) "["
13 IDENT@[7; 12) "attr1"
14 R_BRACK@[12; 13) "]"
15 WHITESPACE@[13; 14) " "
16 PARAM@[14; 23)
17 BIND_PAT@[14; 17)
18 NAME@[14; 17)
19 IDENT@[14; 17) "pat"
20 COLON@[17; 18) ":"
21 WHITESPACE@[18; 19) " "
22 PATH_TYPE@[19; 23)
23 PATH@[19; 23)
24 PATH_SEGMENT@[19; 23)
25 NAME_REF@[19; 23)
26 IDENT@[19; 23) "Type"
27 R_PAREN@[23; 24) ")"
28 WHITESPACE@[24; 25) " "
29 BLOCK@[25; 27)
30 L_CURLY@[25; 26) "{"
31 R_CURLY@[26; 27) "}"
32 WHITESPACE@[27; 28) "\n"
diff --git a/crates/ra_syntax/test_data/parser/ok/0051_parameter_attrs.rs b/crates/ra_syntax/test_data/parser/ok/0051_parameter_attrs.rs
new file mode 100644
index 000000000..de350d858
--- /dev/null
+++ b/crates/ra_syntax/test_data/parser/ok/0051_parameter_attrs.rs
@@ -0,0 +1,21 @@
1fn g1(#[attr1] #[attr2] pat: Type) {}
2fn g2(#[attr1] x: u8) {}
3
4extern "C" { fn printf(format: *const i8, #[attr] ...) -> i32; }
5
6fn foo<F: FnMut(#[attr] &mut Foo<'a>)>(){}
7
8trait Foo {
9 fn bar(#[attr] _: u64, # [attr] mut x: i32);
10}
11
12impl S {
13 fn f(#[must_use] self) {}
14 fn g1(#[attr] self) {}
15 fn g2(#[attr] &self) {}
16 fn g3<'a>(#[attr] &mut self) {}
17 fn g4<'a>(#[attr] &'a self) {}
18 fn g5<'a>(#[attr] &'a mut self) {}
19 fn c(#[attr] self: Self) {}
20 fn d(#[attr] self: Rc<Self>) {}
21} \ No newline at end of file
diff --git a/crates/ra_syntax/test_data/parser/ok/0051_parameter_attrs.txt b/crates/ra_syntax/test_data/parser/ok/0051_parameter_attrs.txt
new file mode 100644
index 000000000..b360f29f4
--- /dev/null
+++ b/crates/ra_syntax/test_data/parser/ok/0051_parameter_attrs.txt
@@ -0,0 +1,477 @@
1SOURCE_FILE@[0; 519)
2 FN_DEF@[0; 37)
3 FN_KW@[0; 2) "fn"
4 WHITESPACE@[2; 3) " "
5 NAME@[3; 5)
6 IDENT@[3; 5) "g1"
7 PARAM_LIST@[5; 34)
8 L_PAREN@[5; 6) "("
9 ATTR@[6; 14)
10 POUND@[6; 7) "#"
11 TOKEN_TREE@[7; 14)
12 L_BRACK@[7; 8) "["
13 IDENT@[8; 13) "attr1"
14 R_BRACK@[13; 14) "]"
15 WHITESPACE@[14; 15) " "
16 ATTR@[15; 23)
17 POUND@[15; 16) "#"
18 TOKEN_TREE@[16; 23)
19 L_BRACK@[16; 17) "["
20 IDENT@[17; 22) "attr2"
21 R_BRACK@[22; 23) "]"
22 WHITESPACE@[23; 24) " "
23 PARAM@[24; 33)
24 BIND_PAT@[24; 27)
25 NAME@[24; 27)
26 IDENT@[24; 27) "pat"
27 COLON@[27; 28) ":"
28 WHITESPACE@[28; 29) " "
29 PATH_TYPE@[29; 33)
30 PATH@[29; 33)
31 PATH_SEGMENT@[29; 33)
32 NAME_REF@[29; 33)
33 IDENT@[29; 33) "Type"
34 R_PAREN@[33; 34) ")"
35 WHITESPACE@[34; 35) " "
36 BLOCK@[35; 37)
37 L_CURLY@[35; 36) "{"
38 R_CURLY@[36; 37) "}"
39 WHITESPACE@[37; 38) "\n"
40 FN_DEF@[38; 62)
41 FN_KW@[38; 40) "fn"
42 WHITESPACE@[40; 41) " "
43 NAME@[41; 43)
44 IDENT@[41; 43) "g2"
45 PARAM_LIST@[43; 59)
46 L_PAREN@[43; 44) "("
47 ATTR@[44; 52)
48 POUND@[44; 45) "#"
49 TOKEN_TREE@[45; 52)
50 L_BRACK@[45; 46) "["
51 IDENT@[46; 51) "attr1"
52 R_BRACK@[51; 52) "]"
53 WHITESPACE@[52; 53) " "
54 PARAM@[53; 58)
55 BIND_PAT@[53; 54)
56 NAME@[53; 54)
57 IDENT@[53; 54) "x"
58 COLON@[54; 55) ":"
59 WHITESPACE@[55; 56) " "
60 PATH_TYPE@[56; 58)
61 PATH@[56; 58)
62 PATH_SEGMENT@[56; 58)
63 NAME_REF@[56; 58)
64 IDENT@[56; 58) "u8"
65 R_PAREN@[58; 59) ")"
66 WHITESPACE@[59; 60) " "
67 BLOCK@[60; 62)
68 L_CURLY@[60; 61) "{"
69 R_CURLY@[61; 62) "}"
70 WHITESPACE@[62; 64) "\n\n"
71 EXTERN_BLOCK@[64; 128)
72 ABI@[64; 74)
73 EXTERN_KW@[64; 70) "extern"
74 WHITESPACE@[70; 71) " "
75 STRING@[71; 74) "\"C\""
76 WHITESPACE@[74; 75) " "
77 EXTERN_ITEM_LIST@[75; 128)
78 L_CURLY@[75; 76) "{"
79 WHITESPACE@[76; 77) " "
80 FN_DEF@[77; 126)
81 FN_KW@[77; 79) "fn"
82 WHITESPACE@[79; 80) " "
83 NAME@[80; 86)
84 IDENT@[80; 86) "printf"
85 PARAM_LIST@[86; 118)
86 L_PAREN@[86; 87) "("
87 PARAM@[87; 104)
88 BIND_PAT@[87; 93)
89 NAME@[87; 93)
90 IDENT@[87; 93) "format"
91 COLON@[93; 94) ":"
92 WHITESPACE@[94; 95) " "
93 POINTER_TYPE@[95; 104)
94 STAR@[95; 96) "*"
95 CONST_KW@[96; 101) "const"
96 WHITESPACE@[101; 102) " "
97 PATH_TYPE@[102; 104)
98 PATH@[102; 104)
99 PATH_SEGMENT@[102; 104)
100 NAME_REF@[102; 104)
101 IDENT@[102; 104) "i8"
102 COMMA@[104; 105) ","
103 WHITESPACE@[105; 106) " "
104 ATTR@[106; 113)
105 POUND@[106; 107) "#"
106 TOKEN_TREE@[107; 113)
107 L_BRACK@[107; 108) "["
108 IDENT@[108; 112) "attr"
109 R_BRACK@[112; 113) "]"
110 WHITESPACE@[113; 114) " "
111 DOTDOTDOT@[114; 117) "..."
112 R_PAREN@[117; 118) ")"
113 WHITESPACE@[118; 119) " "
114 RET_TYPE@[119; 125)
115 THIN_ARROW@[119; 121) "->"
116 WHITESPACE@[121; 122) " "
117 PATH_TYPE@[122; 125)
118 PATH@[122; 125)
119 PATH_SEGMENT@[122; 125)
120 NAME_REF@[122; 125)
121 IDENT@[122; 125) "i32"
122 SEMI@[125; 126) ";"
123 WHITESPACE@[126; 127) " "
124 R_CURLY@[127; 128) "}"
125 WHITESPACE@[128; 130) "\n\n"
126 FN_DEF@[130; 172)
127 FN_KW@[130; 132) "fn"
128 WHITESPACE@[132; 133) " "
129 NAME@[133; 136)
130 IDENT@[133; 136) "foo"
131 TYPE_PARAM_LIST@[136; 168)
132 L_ANGLE@[136; 137) "<"
133 TYPE_PARAM@[137; 167)
134 NAME@[137; 138)
135 IDENT@[137; 138) "F"
136 COLON@[138; 139) ":"
137 WHITESPACE@[139; 140) " "
138 TYPE_BOUND_LIST@[140; 167)
139 TYPE_BOUND@[140; 167)
140 PATH_TYPE@[140; 167)
141 PATH@[140; 167)
142 PATH_SEGMENT@[140; 167)
143 NAME_REF@[140; 145)
144 IDENT@[140; 145) "FnMut"
145 PARAM_LIST@[145; 167)
146 L_PAREN@[145; 146) "("
147 ATTR@[146; 153)
148 POUND@[146; 147) "#"
149 TOKEN_TREE@[147; 153)
150 L_BRACK@[147; 148) "["
151 IDENT@[148; 152) "attr"
152 R_BRACK@[152; 153) "]"
153 WHITESPACE@[153; 154) " "
154 PARAM@[154; 166)
155 REFERENCE_TYPE@[154; 166)
156 AMP@[154; 155) "&"
157 MUT_KW@[155; 158) "mut"
158 WHITESPACE@[158; 159) " "
159 PATH_TYPE@[159; 166)
160 PATH@[159; 166)
161 PATH_SEGMENT@[159; 166)
162 NAME_REF@[159; 162)
163 IDENT@[159; 162) "Foo"
164 TYPE_ARG_LIST@[162; 166)
165 L_ANGLE@[162; 163) "<"
166 LIFETIME_ARG@[163; 165)
167 LIFETIME@[163; 165) "\'a"
168 R_ANGLE@[165; 166) ">"
169 R_PAREN@[166; 167) ")"
170 R_ANGLE@[167; 168) ">"
171 PARAM_LIST@[168; 170)
172 L_PAREN@[168; 169) "("
173 R_PAREN@[169; 170) ")"
174 BLOCK@[170; 172)
175 L_CURLY@[170; 171) "{"
176 R_CURLY@[171; 172) "}"
177 WHITESPACE@[172; 174) "\n\n"
178 TRAIT_DEF@[174; 236)
179 TRAIT_KW@[174; 179) "trait"
180 WHITESPACE@[179; 180) " "
181 NAME@[180; 183)
182 IDENT@[180; 183) "Foo"
183 WHITESPACE@[183; 184) " "
184 ITEM_LIST@[184; 236)
185 L_CURLY@[184; 185) "{"
186 WHITESPACE@[185; 190) "\n "
187 FN_DEF@[190; 234)
188 FN_KW@[190; 192) "fn"
189 WHITESPACE@[192; 193) " "
190 NAME@[193; 196)
191 IDENT@[193; 196) "bar"
192 PARAM_LIST@[196; 233)
193 L_PAREN@[196; 197) "("
194 ATTR@[197; 204)
195 POUND@[197; 198) "#"
196 TOKEN_TREE@[198; 204)
197 L_BRACK@[198; 199) "["
198 IDENT@[199; 203) "attr"
199 R_BRACK@[203; 204) "]"
200 WHITESPACE@[204; 205) " "
201 PARAM@[205; 211)
202 PLACEHOLDER_PAT@[205; 206)
203 UNDERSCORE@[205; 206) "_"
204 COLON@[206; 207) ":"
205 WHITESPACE@[207; 208) " "
206 PATH_TYPE@[208; 211)
207 PATH@[208; 211)
208 PATH_SEGMENT@[208; 211)
209 NAME_REF@[208; 211)
210 IDENT@[208; 211) "u64"
211 COMMA@[211; 212) ","
212 WHITESPACE@[212; 213) " "
213 ATTR@[213; 221)
214 POUND@[213; 214) "#"
215 WHITESPACE@[214; 215) " "
216 TOKEN_TREE@[215; 221)
217 L_BRACK@[215; 216) "["
218 IDENT@[216; 220) "attr"
219 R_BRACK@[220; 221) "]"
220 WHITESPACE@[221; 222) " "
221 PARAM@[222; 232)
222 BIND_PAT@[222; 227)
223 MUT_KW@[222; 225) "mut"
224 WHITESPACE@[225; 226) " "
225 NAME@[226; 227)
226 IDENT@[226; 227) "x"
227 COLON@[227; 228) ":"
228 WHITESPACE@[228; 229) " "
229 PATH_TYPE@[229; 232)
230 PATH@[229; 232)
231 PATH_SEGMENT@[229; 232)
232 NAME_REF@[229; 232)
233 IDENT@[229; 232) "i32"
234 R_PAREN@[232; 233) ")"
235 SEMI@[233; 234) ";"
236 WHITESPACE@[234; 235) "\n"
237 R_CURLY@[235; 236) "}"
238 WHITESPACE@[236; 238) "\n\n"
239 IMPL_BLOCK@[238; 519)
240 IMPL_KW@[238; 242) "impl"
241 WHITESPACE@[242; 243) " "
242 PATH_TYPE@[243; 244)
243 PATH@[243; 244)
244 PATH_SEGMENT@[243; 244)
245 NAME_REF@[243; 244)
246 IDENT@[243; 244) "S"
247 WHITESPACE@[244; 245) " "
248 ITEM_LIST@[245; 519)
249 L_CURLY@[245; 246) "{"
250 WHITESPACE@[246; 252) "\n "
251 FN_DEF@[252; 277)
252 FN_KW@[252; 254) "fn"
253 WHITESPACE@[254; 255) " "
254 NAME@[255; 256)
255 IDENT@[255; 256) "f"
256 PARAM_LIST@[256; 274)
257 L_PAREN@[256; 257) "("
258 ATTR@[257; 268)
259 POUND@[257; 258) "#"
260 TOKEN_TREE@[258; 268)
261 L_BRACK@[258; 259) "["
262 IDENT@[259; 267) "must_use"
263 R_BRACK@[267; 268) "]"
264 WHITESPACE@[268; 269) " "
265 SELF_PARAM@[269; 273)
266 SELF_KW@[269; 273) "self"
267 R_PAREN@[273; 274) ")"
268 WHITESPACE@[274; 275) " "
269 BLOCK@[275; 277)
270 L_CURLY@[275; 276) "{"
271 R_CURLY@[276; 277) "}"
272 WHITESPACE@[277; 283) "\n "
273 FN_DEF@[283; 305)
274 FN_KW@[283; 285) "fn"
275 WHITESPACE@[285; 286) " "
276 NAME@[286; 288)
277 IDENT@[286; 288) "g1"
278 PARAM_LIST@[288; 302)
279 L_PAREN@[288; 289) "("
280 ATTR@[289; 296)
281 POUND@[289; 290) "#"
282 TOKEN_TREE@[290; 296)
283 L_BRACK@[290; 291) "["
284 IDENT@[291; 295) "attr"
285 R_BRACK@[295; 296) "]"
286 WHITESPACE@[296; 297) " "
287 SELF_PARAM@[297; 301)
288 SELF_KW@[297; 301) "self"
289 R_PAREN@[301; 302) ")"
290 WHITESPACE@[302; 303) " "
291 BLOCK@[303; 305)
292 L_CURLY@[303; 304) "{"
293 R_CURLY@[304; 305) "}"
294 WHITESPACE@[305; 311) "\n "
295 FN_DEF@[311; 334)
296 FN_KW@[311; 313) "fn"
297 WHITESPACE@[313; 314) " "
298 NAME@[314; 316)
299 IDENT@[314; 316) "g2"
300 PARAM_LIST@[316; 331)
301 L_PAREN@[316; 317) "("
302 ATTR@[317; 324)
303 POUND@[317; 318) "#"
304 TOKEN_TREE@[318; 324)
305 L_BRACK@[318; 319) "["
306 IDENT@[319; 323) "attr"
307 R_BRACK@[323; 324) "]"
308 WHITESPACE@[324; 325) " "
309 SELF_PARAM@[325; 330)
310 AMP@[325; 326) "&"
311 SELF_KW@[326; 330) "self"
312 R_PAREN@[330; 331) ")"
313 WHITESPACE@[331; 332) " "
314 BLOCK@[332; 334)
315 L_CURLY@[332; 333) "{"
316 R_CURLY@[333; 334) "}"
317 WHITESPACE@[334; 340) "\n "
318 FN_DEF@[340; 371)
319 FN_KW@[340; 342) "fn"
320 WHITESPACE@[342; 343) " "
321 NAME@[343; 345)
322 IDENT@[343; 345) "g3"
323 TYPE_PARAM_LIST@[345; 349)
324 L_ANGLE@[345; 346) "<"
325 LIFETIME_PARAM@[346; 348)
326 LIFETIME@[346; 348) "\'a"
327 R_ANGLE@[348; 349) ">"
328 PARAM_LIST@[349; 368)
329 L_PAREN@[349; 350) "("
330 ATTR@[350; 357)
331 POUND@[350; 351) "#"
332 TOKEN_TREE@[351; 357)
333 L_BRACK@[351; 352) "["
334 IDENT@[352; 356) "attr"
335 R_BRACK@[356; 357) "]"
336 WHITESPACE@[357; 358) " "
337 SELF_PARAM@[358; 367)
338 AMP@[358; 359) "&"
339 MUT_KW@[359; 362) "mut"
340 WHITESPACE@[362; 363) " "
341 SELF_KW@[363; 367) "self"
342 R_PAREN@[367; 368) ")"
343 WHITESPACE@[368; 369) " "
344 BLOCK@[369; 371)
345 L_CURLY@[369; 370) "{"
346 R_CURLY@[370; 371) "}"
347 WHITESPACE@[371; 377) "\n "
348 FN_DEF@[377; 407)
349 FN_KW@[377; 379) "fn"
350 WHITESPACE@[379; 380) " "
351 NAME@[380; 382)
352 IDENT@[380; 382) "g4"
353 TYPE_PARAM_LIST@[382; 386)
354 L_ANGLE@[382; 383) "<"
355 LIFETIME_PARAM@[383; 385)
356 LIFETIME@[383; 385) "\'a"
357 R_ANGLE@[385; 386) ">"
358 PARAM_LIST@[386; 404)
359 L_PAREN@[386; 387) "("
360 ATTR@[387; 394)
361 POUND@[387; 388) "#"
362 TOKEN_TREE@[388; 394)
363 L_BRACK@[388; 389) "["
364 IDENT@[389; 393) "attr"
365 R_BRACK@[393; 394) "]"
366 WHITESPACE@[394; 395) " "
367 SELF_PARAM@[395; 403)
368 AMP@[395; 396) "&"
369 LIFETIME@[396; 398) "\'a"
370 WHITESPACE@[398; 399) " "
371 SELF_KW@[399; 403) "self"
372 R_PAREN@[403; 404) ")"
373 WHITESPACE@[404; 405) " "
374 BLOCK@[405; 407)
375 L_CURLY@[405; 406) "{"
376 R_CURLY@[406; 407) "}"
377 WHITESPACE@[407; 413) "\n "
378 FN_DEF@[413; 447)
379 FN_KW@[413; 415) "fn"
380 WHITESPACE@[415; 416) " "
381 NAME@[416; 418)
382 IDENT@[416; 418) "g5"
383 TYPE_PARAM_LIST@[418; 422)
384 L_ANGLE@[418; 419) "<"
385 LIFETIME_PARAM@[419; 421)
386 LIFETIME@[419; 421) "\'a"
387 R_ANGLE@[421; 422) ">"
388 PARAM_LIST@[422; 444)
389 L_PAREN@[422; 423) "("
390 ATTR@[423; 430)
391 POUND@[423; 424) "#"
392 TOKEN_TREE@[424; 430)
393 L_BRACK@[424; 425) "["
394 IDENT@[425; 429) "attr"
395 R_BRACK@[429; 430) "]"
396 WHITESPACE@[430; 431) " "
397 SELF_PARAM@[431; 443)
398 AMP@[431; 432) "&"
399 LIFETIME@[432; 434) "\'a"
400 WHITESPACE@[434; 435) " "
401 MUT_KW@[435; 438) "mut"
402 WHITESPACE@[438; 439) " "
403 SELF_KW@[439; 443) "self"
404 R_PAREN@[443; 444) ")"
405 WHITESPACE@[444; 445) " "
406 BLOCK@[445; 447)
407 L_CURLY@[445; 446) "{"
408 R_CURLY@[446; 447) "}"
409 WHITESPACE@[447; 453) "\n "
410 FN_DEF@[453; 480)
411 FN_KW@[453; 455) "fn"
412 WHITESPACE@[455; 456) " "
413 NAME@[456; 457)
414 IDENT@[456; 457) "c"
415 PARAM_LIST@[457; 477)
416 L_PAREN@[457; 458) "("
417 ATTR@[458; 465)
418 POUND@[458; 459) "#"
419 TOKEN_TREE@[459; 465)
420 L_BRACK@[459; 460) "["
421 IDENT@[460; 464) "attr"
422 R_BRACK@[464; 465) "]"
423 WHITESPACE@[465; 466) " "
424 SELF_PARAM@[466; 476)
425 SELF_KW@[466; 470) "self"
426 COLON@[470; 471) ":"
427 WHITESPACE@[471; 472) " "
428 PATH_TYPE@[472; 476)
429 PATH@[472; 476)
430 PATH_SEGMENT@[472; 476)
431 NAME_REF@[472; 476)
432 IDENT@[472; 476) "Self"
433 R_PAREN@[476; 477) ")"
434 WHITESPACE@[477; 478) " "
435 BLOCK@[478; 480)
436 L_CURLY@[478; 479) "{"
437 R_CURLY@[479; 480) "}"
438 WHITESPACE@[480; 486) "\n "
439 FN_DEF@[486; 517)
440 FN_KW@[486; 488) "fn"
441 WHITESPACE@[488; 489) " "
442 NAME@[489; 490)
443 IDENT@[489; 490) "d"
444 PARAM_LIST@[490; 514)
445 L_PAREN@[490; 491) "("
446 ATTR@[491; 498)
447 POUND@[491; 492) "#"
448 TOKEN_TREE@[492; 498)
449 L_BRACK@[492; 493) "["
450 IDENT@[493; 497) "attr"
451 R_BRACK@[497; 498) "]"
452 WHITESPACE@[498; 499) " "
453 SELF_PARAM@[499; 513)
454 SELF_KW@[499; 503) "self"
455 COLON@[503; 504) ":"
456 WHITESPACE@[504; 505) " "
457 PATH_TYPE@[505; 513)
458 PATH@[505; 513)
459 PATH_SEGMENT@[505; 513)
460 NAME_REF@[505; 507)
461 IDENT@[505; 507) "Rc"
462 TYPE_ARG_LIST@[507; 513)
463 L_ANGLE@[507; 508) "<"
464 TYPE_ARG@[508; 512)
465 PATH_TYPE@[508; 512)
466 PATH@[508; 512)
467 PATH_SEGMENT@[508; 512)
468 NAME_REF@[508; 512)
469 IDENT@[508; 512) "Self"
470 R_ANGLE@[512; 513) ">"
471 R_PAREN@[513; 514) ")"
472 WHITESPACE@[514; 515) " "
473 BLOCK@[515; 517)
474 L_CURLY@[515; 516) "{"
475 R_CURLY@[516; 517) "}"
476 WHITESPACE@[517; 518) "\n"
477 R_CURLY@[518; 519) "}"