From d3c90ded2b9a4f75e101fa3abc60cd3aebc439c9 Mon Sep 17 00:00:00 2001 From: Aleksey Kladov Date: Fri, 17 Aug 2018 22:00:13 +0300 Subject: [PATCH] Borrowed AST --- crates/cli/src/main.rs | 4 +- crates/libanalysis/src/lib.rs | 26 +- crates/libanalysis/src/symbol_index.rs | 4 +- crates/libeditor/src/code_actions.rs | 15 +- crates/libeditor/src/extend_selection.rs | 5 +- crates/libeditor/src/lib.rs | 26 +- crates/libeditor/src/symbols.rs | 50 +-- crates/libeditor/tests/test.rs | 8 +- crates/libsyntax2/src/algo/visit.rs | 6 +- crates/libsyntax2/src/ast/generated.rs | 432 ++++++++++---------- crates/libsyntax2/src/ast/generated.rs.tera | 31 +- crates/libsyntax2/src/ast/mod.rs | 61 +-- crates/libsyntax2/src/lib.rs | 2 +- crates/libsyntax2/src/utils.rs | 5 +- crates/libsyntax2/src/yellow/syntax.rs | 16 +- crates/libsyntax2/tests/test/main.rs | 2 +- 16 files changed, 350 insertions(+), 343 deletions(-) diff --git a/crates/cli/src/main.rs b/crates/cli/src/main.rs index fc28691fc9..e8e2ce7eed 100644 --- a/crates/cli/src/main.rs +++ b/crates/cli/src/main.rs @@ -10,7 +10,7 @@ use std::{ }; use clap::{App, Arg, SubCommand}; use tools::collect_tests; -use libeditor::{File, syntax_tree, file_structure}; +use libeditor::{ParsedFile, syntax_tree, file_structure}; type Result = ::std::result::Result; @@ -68,7 +68,7 @@ fn main() -> Result<()> { Ok(()) } -fn file() -> Result { +fn file() -> Result { let text = read_stdin()?; Ok(libeditor::parse(&text)) } diff --git a/crates/libanalysis/src/lib.rs b/crates/libanalysis/src/lib.rs index ae96a0d62d..15a433afc9 100644 --- a/crates/libanalysis/src/lib.rs +++ b/crates/libanalysis/src/lib.rs @@ -26,8 +26,8 @@ use std::{ }; use libsyntax2::{ - TextUnit, TextRange, RefRoot, - ast::{self, AstNode, NameOwner}, + TextUnit, TextRange, + ast::{self, AstNode, NameOwner, ParsedFile}, SyntaxKind::*, }; use libeditor::{LineIndex, FileSymbol, find_node}; @@ -109,7 +109,7 @@ impl WorldState { impl World { - pub fn file_syntax(&self, file_id: FileId) -> Result { + pub fn file_syntax(&self, file_id: FileId) -> Result { let data = self.file_data(file_id)?; Ok(data.syntax().clone()) } @@ -137,11 +137,11 @@ impl World { offset: TextUnit, ) -> Result> { let file = self.file_syntax(id)?; - let syntax = file.syntax_ref(); - if let Some(name_ref) = find_node::>(syntax, offset) { + let syntax = file.syntax(); + if let Some(name_ref) = find_node::(syntax, offset) { return Ok(self.index_resolve(name_ref)); } - if let Some(name) = find_node::>(syntax, offset) { + if let Some(name) = find_node::(syntax, offset) { if let Some(module) = name.syntax().parent().and_then(ast::Module::cast) { if module.has_semi() { return Ok(self.resolve_module(id, module)); @@ -151,7 +151,7 @@ impl World { Ok(vec![]) } - fn index_resolve(&self, name_ref: ast::NameRef) -> Vec<(FileId, FileSymbol)> { + fn index_resolve(&self, name_ref: ast::NameRef) -> Vec<(FileId, FileSymbol)> { let name = name_ref.text(); let mut query = Query::new(name.to_string()); query.exact(); @@ -159,7 +159,7 @@ impl World { self.world_symbols(query) } - fn resolve_module(&self, id: FileId, module: ast::Module) -> Vec<(FileId, FileSymbol)> { + fn resolve_module(&self, id: FileId, module: ast::Module) -> Vec<(FileId, FileSymbol)> { let name = match module.name() { Some(name) => name.text(), None => return Vec::new(), @@ -220,7 +220,7 @@ struct WorldData { struct FileData { text: String, symbols: OnceCell, - syntax: OnceCell, + syntax: OnceCell, lines: OnceCell, } @@ -234,14 +234,14 @@ impl FileData { } } - fn syntax(&self) -> &ast::File { + fn syntax(&self) -> &ParsedFile { self.syntax - .get_or_init(|| ast::File::parse(&self.text)) + .get_or_init(|| ParsedFile::parse(&self.text)) } - fn syntax_transient(&self) -> ast::File { + fn syntax_transient(&self) -> ParsedFile { self.syntax.get().map(|s| s.clone()) - .unwrap_or_else(|| ast::File::parse(&self.text)) + .unwrap_or_else(|| ParsedFile::parse(&self.text)) } fn symbols(&self) -> &FileSymbols { diff --git a/crates/libanalysis/src/symbol_index.rs b/crates/libanalysis/src/symbol_index.rs index 3c3252956b..426de4c760 100644 --- a/crates/libanalysis/src/symbol_index.rs +++ b/crates/libanalysis/src/symbol_index.rs @@ -1,6 +1,6 @@ use libeditor::{FileSymbol, file_symbols}; use libsyntax2::{ - ast, + ParsedFile, SyntaxKind::{self, *}, }; use fst::{self, IntoStreamer, Streamer}; @@ -12,7 +12,7 @@ pub(crate) struct FileSymbols { } impl FileSymbols { - pub(crate) fn new(file: &ast::File) -> FileSymbols { + pub(crate) fn new(file: &ParsedFile) -> FileSymbols { let mut symbols = file_symbols(file) .into_iter() .map(|s| (s.name.as_str().to_lowercase(), s)) diff --git a/crates/libeditor/src/code_actions.rs b/crates/libeditor/src/code_actions.rs index bb6eb0d618..80c3963378 100644 --- a/crates/libeditor/src/code_actions.rs +++ b/crates/libeditor/src/code_actions.rs @@ -1,8 +1,8 @@ -use {TextUnit, File, EditBuilder, Edit}; +use {TextUnit, EditBuilder, Edit}; use libsyntax2::{ - ast::{self, AstNode, AttrsOwner}, + ast::{self, AstNode, AttrsOwner, ParsedFile}, SyntaxKind::COMMA, - SyntaxNodeRef, RefRoot, + SyntaxNodeRef, algo::{ Direction, siblings, find_leaf_at_offset, ancestors, @@ -19,9 +19,8 @@ pub enum CursorPosition { Offset(TextUnit), } -pub fn flip_comma<'a>(file: &'a File, offset: TextUnit) -> Option ActionResult + 'a> { +pub fn flip_comma<'a>(file: &'a ParsedFile, offset: TextUnit) -> Option ActionResult + 'a> { let syntax = file.syntax(); - let syntax = syntax.as_ref(); let comma = find_leaf_at_offset(syntax, offset).find(|leaf| leaf.kind() == COMMA)?; let left = non_trivia_sibling(comma, Direction::Backward)?; @@ -37,8 +36,8 @@ pub fn flip_comma<'a>(file: &'a File, offset: TextUnit) -> Option(file: &'a File, offset: TextUnit) -> Option ActionResult + 'a> { - let nominal = find_node::>(file.syntax_ref(), offset)?; +pub fn add_derive<'a>(file: &'a ParsedFile, offset: TextUnit) -> Option ActionResult + 'a> { + let nominal = find_node::(file.syntax(), offset)?; Some(move || { let derive_attr = nominal .attrs() @@ -70,7 +69,7 @@ fn non_trivia_sibling(node: SyntaxNodeRef, direction: Direction) -> Option>>(syntax: SyntaxNodeRef<'a>, offset: TextUnit) -> Option { +pub fn find_node<'a, N: AstNode<'a>>(syntax: SyntaxNodeRef<'a>, offset: TextUnit) -> Option { let leaves = find_leaf_at_offset(syntax, offset); let leaf = leaves.clone() .find(|leaf| !leaf.kind().is_trivia()) diff --git a/crates/libeditor/src/extend_selection.rs b/crates/libeditor/src/extend_selection.rs index ed7d9b3f77..cb6edb5769 100644 --- a/crates/libeditor/src/extend_selection.rs +++ b/crates/libeditor/src/extend_selection.rs @@ -1,11 +1,10 @@ use libsyntax2::{ - ast, AstNode, - TextRange, SyntaxNodeRef, + ParsedFile, TextRange, SyntaxNodeRef, SyntaxKind::WHITESPACE, algo::{find_leaf_at_offset, find_covering_node, ancestors}, }; -pub fn extend_selection(file: &ast::File, range: TextRange) -> Option { +pub fn extend_selection(file: &ParsedFile, range: TextRange) -> Option { let syntax = file.syntax(); extend(syntax.as_ref(), range) } diff --git a/crates/libeditor/src/lib.rs b/crates/libeditor/src/lib.rs index 76cb4d0285..d9d0369f6e 100644 --- a/crates/libeditor/src/lib.rs +++ b/crates/libeditor/src/lib.rs @@ -15,7 +15,7 @@ use libsyntax2::{ algo::{walk, find_leaf_at_offset}, SyntaxKind::{self, *}, }; -pub use libsyntax2::{File, TextRange, TextUnit}; +pub use libsyntax2::{ParsedFile, TextRange, TextUnit}; pub use self::{ line_index::{LineIndex, LineCol}, extend_selection::extend_selection, @@ -51,18 +51,18 @@ pub enum RunnableKind { Bin, } -pub fn parse(text: &str) -> ast::File { - ast::File::parse(text) +pub fn parse(text: &str) -> ast::ParsedFile { + ast::ParsedFile::parse(text) } -pub fn matching_brace(file: &ast::File, offset: TextUnit) -> Option { +pub fn matching_brace(file: &ast::ParsedFile, offset: TextUnit) -> Option { const BRACES: &[SyntaxKind] = &[ L_CURLY, R_CURLY, L_BRACK, R_BRACK, L_PAREN, R_PAREN, L_ANGLE, R_ANGLE, ]; - let (brace_node, brace_idx) = find_leaf_at_offset(file.syntax_ref(), offset) + let (brace_node, brace_idx) = find_leaf_at_offset(file.syntax(), offset) .filter_map(|node| { let idx = BRACES.iter().position(|&brace| brace == node.kind())?; Some((node, idx)) @@ -75,9 +75,9 @@ pub fn matching_brace(file: &ast::File, offset: TextUnit) -> Option { Some(matching_node.range().start()) } -pub fn highlight(file: &ast::File) -> Vec { +pub fn highlight(file: &ast::ParsedFile) -> Vec { let mut res = Vec::new(); - for node in walk::preorder(file.syntax_ref()) { + for node in walk::preorder(file.syntax()) { let tag = match node.kind() { ERROR => "error", COMMENT | DOC_COMMENT => "comment", @@ -98,10 +98,10 @@ pub fn highlight(file: &ast::File) -> Vec { res } -pub fn diagnostics(file: &ast::File) -> Vec { +pub fn diagnostics(file: &ast::ParsedFile) -> Vec { let mut res = Vec::new(); - for node in walk::preorder(file.syntax_ref()) { + for node in walk::preorder(file.syntax()) { if node.kind() == ERROR { res.push(Diagnostic { range: node.range(), @@ -116,12 +116,12 @@ pub fn diagnostics(file: &ast::File) -> Vec { res } -pub fn syntax_tree(file: &ast::File) -> String { - ::libsyntax2::utils::dump_tree(&file.syntax()) +pub fn syntax_tree(file: &ast::ParsedFile) -> String { + ::libsyntax2::utils::dump_tree(file.syntax()) } -pub fn runnables(file: &ast::File) -> Vec { - file +pub fn runnables(file: &ast::ParsedFile) -> Vec { + file.ast() .functions() .filter_map(|f| { let name = f.name()?.text(); diff --git a/crates/libeditor/src/symbols.rs b/crates/libeditor/src/symbols.rs index cf5bd2a412..d7bd111e6e 100644 --- a/crates/libeditor/src/symbols.rs +++ b/crates/libeditor/src/symbols.rs @@ -1,6 +1,6 @@ use smol_str::SmolStr; use libsyntax2::{ - SyntaxKind, SyntaxNodeRef, AstNode, RefRoot, + SyntaxKind, SyntaxNodeRef, AstNode, ParsedFile, ast::{self, NameOwner}, algo::{ visit::{visitor, Visitor}, @@ -25,14 +25,14 @@ pub struct FileSymbol { pub kind: SyntaxKind, } -pub fn file_symbols(file: &ast::File) -> Vec { - preorder(file.syntax_ref()) +pub fn file_symbols(file: &ParsedFile) -> Vec { + preorder(file.syntax()) .filter_map(to_symbol) .collect() } fn to_symbol(node: SyntaxNodeRef) -> Option { - fn decl<'a, N: NameOwner>>(node: N) -> Option { + fn decl<'a, N: NameOwner<'a>>(node: N) -> Option { let name = node.name()?; Some(FileSymbol { name: name.text(), @@ -41,23 +41,23 @@ fn to_symbol(node: SyntaxNodeRef) -> Option { }) } visitor() - .visit(decl::>) - .visit(decl::>) - .visit(decl::>) - .visit(decl::>) - .visit(decl::>) - .visit(decl::>) - .visit(decl::>) - .visit(decl::>) + .visit(decl::) + .visit(decl::) + .visit(decl::) + .visit(decl::) + .visit(decl::) + .visit(decl::) + .visit(decl::) + .visit(decl::) .accept(node)? } -pub fn file_structure(file: &ast::File) -> Vec { +pub fn file_structure(file: &ParsedFile) -> Vec { let mut res = Vec::new(); let mut stack = Vec::new(); - for event in walk(file.syntax_ref()) { + for event in walk(file.syntax()) { match event { WalkEvent::Enter(node) => { match structure_node(node) { @@ -80,7 +80,7 @@ pub fn file_structure(file: &ast::File) -> Vec { } fn structure_node(node: SyntaxNodeRef) -> Option { - fn decl<'a, N: NameOwner>>(node: N) -> Option { + fn decl<'a, N: NameOwner<'a>>(node: N) -> Option { let name = node.name()?; Some(StructureNode { parent: None, @@ -92,16 +92,16 @@ fn structure_node(node: SyntaxNodeRef) -> Option { } visitor() - .visit(decl::>) - .visit(decl::>) - .visit(decl::>) - .visit(decl::>) - .visit(decl::>) - .visit(decl::>) - .visit(decl::>) - .visit(decl::>) - .visit(decl::>) - .visit(|im: ast::ImplItem<_>| { + .visit(decl::) + .visit(decl::) + .visit(decl::) + .visit(decl::) + .visit(decl::) + .visit(decl::) + .visit(decl::) + .visit(decl::) + .visit(decl::) + .visit(|im: ast::ImplItem| { let target_type = im.target_type()?; let target_trait = im.target_trait(); let label = match target_trait { diff --git a/crates/libeditor/tests/test.rs b/crates/libeditor/tests/test.rs index cc42267104..a2e26003ac 100644 --- a/crates/libeditor/tests/test.rs +++ b/crates/libeditor/tests/test.rs @@ -5,7 +5,7 @@ extern crate assert_eq_text; use assert_eq_text::{assert_eq_dbg}; use libeditor::{ - File, TextUnit, TextRange, ActionResult, CursorPosition, + ParsedFile, TextUnit, TextRange, ActionResult, CursorPosition, highlight, runnables, extend_selection, file_structure, flip_comma, add_derive, matching_brace, }; @@ -146,11 +146,11 @@ fn test_matching_brace() { ); } -fn file(text: &str) -> File { - File::parse(text) +fn file(text: &str) -> ParsedFile { + ParsedFile::parse(text) } -fn check_action Option>( +fn check_action Option>( before: &str, after: &str, f: F, diff --git a/crates/libsyntax2/src/algo/visit.rs b/crates/libsyntax2/src/algo/visit.rs index 55eb72f595..a36c8f45e1 100644 --- a/crates/libsyntax2/src/algo/visit.rs +++ b/crates/libsyntax2/src/algo/visit.rs @@ -1,5 +1,5 @@ use std::marker::PhantomData; -use {SyntaxNodeRef, AstNode, RefRoot}; +use {SyntaxNodeRef, AstNode}; pub fn visitor<'a, T>() -> impl Visitor<'a, Output=T> { @@ -10,7 +10,7 @@ pub trait Visitor<'a>: Sized { type Output; fn accept(self, node: SyntaxNodeRef<'a>) -> Option; fn visit(self, f: F) -> Vis - where N: AstNode>, + where N: AstNode<'a>, F: FnOnce(N) -> Self::Output, { Vis { inner: self, f, ph: PhantomData } @@ -40,7 +40,7 @@ pub struct Vis { impl<'a, V, N, F> Visitor<'a> for Vis where V: Visitor<'a>, - N: AstNode>, + N: AstNode<'a>, F: FnOnce(N) -> >::Output, { type Output = >::Output; diff --git a/crates/libsyntax2/src/ast/generated.rs b/crates/libsyntax2/src/ast/generated.rs index 4fc405c8af..e8a87eba58 100644 --- a/crates/libsyntax2/src/ast/generated.rs +++ b/crates/libsyntax2/src/ast/generated.rs @@ -1,45 +1,45 @@ use { ast, - SyntaxNode, OwnedRoot, TreeRoot, AstNode, + SyntaxNodeRef, AstNode, SyntaxKind::*, }; // ArrayType #[derive(Debug, Clone, Copy)] -pub struct ArrayType { - syntax: SyntaxNode, +pub struct ArrayType<'a> { + syntax: SyntaxNodeRef<'a>, } -impl AstNode for ArrayType { - fn cast(syntax: SyntaxNode) -> Option { +impl<'a> AstNode<'a> for ArrayType<'a> { + fn cast(syntax: SyntaxNodeRef<'a>) -> Option { match syntax.kind() { ARRAY_TYPE => Some(ArrayType { syntax }), _ => None, } } - fn syntax(&self) -> &SyntaxNode { &self.syntax } + fn syntax(self) -> SyntaxNodeRef<'a> { self.syntax } } -impl ArrayType {} +impl<'a> ArrayType<'a> {} // Attr #[derive(Debug, Clone, Copy)] -pub struct Attr { - syntax: SyntaxNode, +pub struct Attr<'a> { + syntax: SyntaxNodeRef<'a>, } -impl AstNode for Attr { - fn cast(syntax: SyntaxNode) -> Option { +impl<'a> AstNode<'a> for Attr<'a> { + fn cast(syntax: SyntaxNodeRef<'a>) -> Option { match syntax.kind() { ATTR => Some(Attr { syntax }), _ => None, } } - fn syntax(&self) -> &SyntaxNode { &self.syntax } + fn syntax(self) -> SyntaxNodeRef<'a> { self.syntax } } -impl Attr { - pub fn value(&self) -> Option> { +impl<'a> Attr<'a> { + pub fn value(self) -> Option> { self.syntax() .children() .filter_map(TokenTree::cast) @@ -49,80 +49,80 @@ impl Attr { // ConstDef #[derive(Debug, Clone, Copy)] -pub struct ConstDef { - syntax: SyntaxNode, +pub struct ConstDef<'a> { + syntax: SyntaxNodeRef<'a>, } -impl AstNode for ConstDef { - fn cast(syntax: SyntaxNode) -> Option { +impl<'a> AstNode<'a> for ConstDef<'a> { + fn cast(syntax: SyntaxNodeRef<'a>) -> Option { match syntax.kind() { CONST_DEF => Some(ConstDef { syntax }), _ => None, } } - fn syntax(&self) -> &SyntaxNode { &self.syntax } + fn syntax(self) -> SyntaxNodeRef<'a> { self.syntax } } -impl ast::NameOwner for ConstDef {} -impl ast::AttrsOwner for ConstDef {} -impl ConstDef {} +impl<'a> ast::NameOwner<'a> for ConstDef<'a> {} +impl<'a> ast::AttrsOwner<'a> for ConstDef<'a> {} +impl<'a> ConstDef<'a> {} // DynTraitType #[derive(Debug, Clone, Copy)] -pub struct DynTraitType { - syntax: SyntaxNode, +pub struct DynTraitType<'a> { + syntax: SyntaxNodeRef<'a>, } -impl AstNode for DynTraitType { - fn cast(syntax: SyntaxNode) -> Option { +impl<'a> AstNode<'a> for DynTraitType<'a> { + fn cast(syntax: SyntaxNodeRef<'a>) -> Option { match syntax.kind() { DYN_TRAIT_TYPE => Some(DynTraitType { syntax }), _ => None, } } - fn syntax(&self) -> &SyntaxNode { &self.syntax } + fn syntax(self) -> SyntaxNodeRef<'a> { self.syntax } } -impl DynTraitType {} +impl<'a> DynTraitType<'a> {} // EnumDef #[derive(Debug, Clone, Copy)] -pub struct EnumDef { - syntax: SyntaxNode, +pub struct EnumDef<'a> { + syntax: SyntaxNodeRef<'a>, } -impl AstNode for EnumDef { - fn cast(syntax: SyntaxNode) -> Option { +impl<'a> AstNode<'a> for EnumDef<'a> { + fn cast(syntax: SyntaxNodeRef<'a>) -> Option { match syntax.kind() { ENUM_DEF => Some(EnumDef { syntax }), _ => None, } } - fn syntax(&self) -> &SyntaxNode { &self.syntax } + fn syntax(self) -> SyntaxNodeRef<'a> { self.syntax } } -impl ast::NameOwner for EnumDef {} -impl ast::AttrsOwner for EnumDef {} -impl EnumDef {} +impl<'a> ast::NameOwner<'a> for EnumDef<'a> {} +impl<'a> ast::AttrsOwner<'a> for EnumDef<'a> {} +impl<'a> EnumDef<'a> {} // File #[derive(Debug, Clone, Copy)] -pub struct File { - syntax: SyntaxNode, +pub struct File<'a> { + syntax: SyntaxNodeRef<'a>, } -impl AstNode for File { - fn cast(syntax: SyntaxNode) -> Option { +impl<'a> AstNode<'a> for File<'a> { + fn cast(syntax: SyntaxNodeRef<'a>) -> Option { match syntax.kind() { FILE => Some(File { syntax }), _ => None, } } - fn syntax(&self) -> &SyntaxNode { &self.syntax } + fn syntax(self) -> SyntaxNodeRef<'a> { self.syntax } } -impl File { - pub fn functions<'a>(&'a self) -> impl Iterator> + 'a { +impl<'a> File<'a> { + pub fn functions(self) -> impl Iterator> + 'a { self.syntax() .children() .filter_map(FnDef::cast) @@ -131,206 +131,206 @@ impl File { // FnDef #[derive(Debug, Clone, Copy)] -pub struct FnDef { - syntax: SyntaxNode, +pub struct FnDef<'a> { + syntax: SyntaxNodeRef<'a>, } -impl AstNode for FnDef { - fn cast(syntax: SyntaxNode) -> Option { +impl<'a> AstNode<'a> for FnDef<'a> { + fn cast(syntax: SyntaxNodeRef<'a>) -> Option { match syntax.kind() { FN_DEF => Some(FnDef { syntax }), _ => None, } } - fn syntax(&self) -> &SyntaxNode { &self.syntax } + fn syntax(self) -> SyntaxNodeRef<'a> { self.syntax } } -impl ast::NameOwner for FnDef {} -impl ast::AttrsOwner for FnDef {} -impl FnDef {} +impl<'a> ast::NameOwner<'a> for FnDef<'a> {} +impl<'a> ast::AttrsOwner<'a> for FnDef<'a> {} +impl<'a> FnDef<'a> {} // FnPointerType #[derive(Debug, Clone, Copy)] -pub struct FnPointerType { - syntax: SyntaxNode, +pub struct FnPointerType<'a> { + syntax: SyntaxNodeRef<'a>, } -impl AstNode for FnPointerType { - fn cast(syntax: SyntaxNode) -> Option { +impl<'a> AstNode<'a> for FnPointerType<'a> { + fn cast(syntax: SyntaxNodeRef<'a>) -> Option { match syntax.kind() { FN_POINTER_TYPE => Some(FnPointerType { syntax }), _ => None, } } - fn syntax(&self) -> &SyntaxNode { &self.syntax } + fn syntax(self) -> SyntaxNodeRef<'a> { self.syntax } } -impl FnPointerType {} +impl<'a> FnPointerType<'a> {} // ForType #[derive(Debug, Clone, Copy)] -pub struct ForType { - syntax: SyntaxNode, +pub struct ForType<'a> { + syntax: SyntaxNodeRef<'a>, } -impl AstNode for ForType { - fn cast(syntax: SyntaxNode) -> Option { +impl<'a> AstNode<'a> for ForType<'a> { + fn cast(syntax: SyntaxNodeRef<'a>) -> Option { match syntax.kind() { FOR_TYPE => Some(ForType { syntax }), _ => None, } } - fn syntax(&self) -> &SyntaxNode { &self.syntax } + fn syntax(self) -> SyntaxNodeRef<'a> { self.syntax } } -impl ForType {} +impl<'a> ForType<'a> {} // ImplItem #[derive(Debug, Clone, Copy)] -pub struct ImplItem { - syntax: SyntaxNode, +pub struct ImplItem<'a> { + syntax: SyntaxNodeRef<'a>, } -impl AstNode for ImplItem { - fn cast(syntax: SyntaxNode) -> Option { +impl<'a> AstNode<'a> for ImplItem<'a> { + fn cast(syntax: SyntaxNodeRef<'a>) -> Option { match syntax.kind() { IMPL_ITEM => Some(ImplItem { syntax }), _ => None, } } - fn syntax(&self) -> &SyntaxNode { &self.syntax } + fn syntax(self) -> SyntaxNodeRef<'a> { self.syntax } } -impl ImplItem {} +impl<'a> ImplItem<'a> {} // ImplTraitType #[derive(Debug, Clone, Copy)] -pub struct ImplTraitType { - syntax: SyntaxNode, +pub struct ImplTraitType<'a> { + syntax: SyntaxNodeRef<'a>, } -impl AstNode for ImplTraitType { - fn cast(syntax: SyntaxNode) -> Option { +impl<'a> AstNode<'a> for ImplTraitType<'a> { + fn cast(syntax: SyntaxNodeRef<'a>) -> Option { match syntax.kind() { IMPL_TRAIT_TYPE => Some(ImplTraitType { syntax }), _ => None, } } - fn syntax(&self) -> &SyntaxNode { &self.syntax } + fn syntax(self) -> SyntaxNodeRef<'a> { self.syntax } } -impl ImplTraitType {} +impl<'a> ImplTraitType<'a> {} // Module #[derive(Debug, Clone, Copy)] -pub struct Module { - syntax: SyntaxNode, +pub struct Module<'a> { + syntax: SyntaxNodeRef<'a>, } -impl AstNode for Module { - fn cast(syntax: SyntaxNode) -> Option { +impl<'a> AstNode<'a> for Module<'a> { + fn cast(syntax: SyntaxNodeRef<'a>) -> Option { match syntax.kind() { MODULE => Some(Module { syntax }), _ => None, } } - fn syntax(&self) -> &SyntaxNode { &self.syntax } + fn syntax(self) -> SyntaxNodeRef<'a> { self.syntax } } -impl ast::NameOwner for Module {} -impl ast::AttrsOwner for Module {} -impl Module {} +impl<'a> ast::NameOwner<'a> for Module<'a> {} +impl<'a> ast::AttrsOwner<'a> for Module<'a> {} +impl<'a> Module<'a> {} // Name #[derive(Debug, Clone, Copy)] -pub struct Name { - syntax: SyntaxNode, +pub struct Name<'a> { + syntax: SyntaxNodeRef<'a>, } -impl AstNode for Name { - fn cast(syntax: SyntaxNode) -> Option { +impl<'a> AstNode<'a> for Name<'a> { + fn cast(syntax: SyntaxNodeRef<'a>) -> Option { match syntax.kind() { NAME => Some(Name { syntax }), _ => None, } } - fn syntax(&self) -> &SyntaxNode { &self.syntax } + fn syntax(self) -> SyntaxNodeRef<'a> { self.syntax } } -impl Name {} +impl<'a> Name<'a> {} // NameRef #[derive(Debug, Clone, Copy)] -pub struct NameRef { - syntax: SyntaxNode, +pub struct NameRef<'a> { + syntax: SyntaxNodeRef<'a>, } -impl AstNode for NameRef { - fn cast(syntax: SyntaxNode) -> Option { +impl<'a> AstNode<'a> for NameRef<'a> { + fn cast(syntax: SyntaxNodeRef<'a>) -> Option { match syntax.kind() { NAME_REF => Some(NameRef { syntax }), _ => None, } } - fn syntax(&self) -> &SyntaxNode { &self.syntax } + fn syntax(self) -> SyntaxNodeRef<'a> { self.syntax } } -impl NameRef {} +impl<'a> NameRef<'a> {} // NamedField #[derive(Debug, Clone, Copy)] -pub struct NamedField { - syntax: SyntaxNode, +pub struct NamedField<'a> { + syntax: SyntaxNodeRef<'a>, } -impl AstNode for NamedField { - fn cast(syntax: SyntaxNode) -> Option { +impl<'a> AstNode<'a> for NamedField<'a> { + fn cast(syntax: SyntaxNodeRef<'a>) -> Option { match syntax.kind() { NAMED_FIELD => Some(NamedField { syntax }), _ => None, } } - fn syntax(&self) -> &SyntaxNode { &self.syntax } + fn syntax(self) -> SyntaxNodeRef<'a> { self.syntax } } -impl ast::NameOwner for NamedField {} -impl ast::AttrsOwner for NamedField {} -impl NamedField {} +impl<'a> ast::NameOwner<'a> for NamedField<'a> {} +impl<'a> ast::AttrsOwner<'a> for NamedField<'a> {} +impl<'a> NamedField<'a> {} // NeverType #[derive(Debug, Clone, Copy)] -pub struct NeverType { - syntax: SyntaxNode, +pub struct NeverType<'a> { + syntax: SyntaxNodeRef<'a>, } -impl AstNode for NeverType { - fn cast(syntax: SyntaxNode) -> Option { +impl<'a> AstNode<'a> for NeverType<'a> { + fn cast(syntax: SyntaxNodeRef<'a>) -> Option { match syntax.kind() { NEVER_TYPE => Some(NeverType { syntax }), _ => None, } } - fn syntax(&self) -> &SyntaxNode { &self.syntax } + fn syntax(self) -> SyntaxNodeRef<'a> { self.syntax } } -impl NeverType {} +impl<'a> NeverType<'a> {} // NominalDef #[derive(Debug, Clone, Copy)] -pub enum NominalDef { - StructDef(StructDef), - EnumDef(EnumDef), +pub enum NominalDef<'a> { + StructDef(StructDef<'a>), + EnumDef(EnumDef<'a>), } -impl AstNode for NominalDef { - fn cast(syntax: SyntaxNode) -> Option { +impl<'a> AstNode<'a> for NominalDef<'a> { + fn cast(syntax: SyntaxNodeRef<'a>) -> Option { match syntax.kind() { STRUCT_DEF => Some(NominalDef::StructDef(StructDef { syntax })), ENUM_DEF => Some(NominalDef::EnumDef(EnumDef { syntax })), _ => None, } } - fn syntax(&self) -> &SyntaxNode { + fn syntax(self) -> SyntaxNodeRef<'a> { match self { NominalDef::StructDef(inner) => inner.syntax(), NominalDef::EnumDef(inner) => inner.syntax(), @@ -338,157 +338,157 @@ impl AstNode for NominalDef { } } -impl ast::AttrsOwner for NominalDef {} -impl NominalDef {} +impl<'a> ast::AttrsOwner<'a> for NominalDef<'a> {} +impl<'a> NominalDef<'a> {} // ParenType #[derive(Debug, Clone, Copy)] -pub struct ParenType { - syntax: SyntaxNode, +pub struct ParenType<'a> { + syntax: SyntaxNodeRef<'a>, } -impl AstNode for ParenType { - fn cast(syntax: SyntaxNode) -> Option { +impl<'a> AstNode<'a> for ParenType<'a> { + fn cast(syntax: SyntaxNodeRef<'a>) -> Option { match syntax.kind() { PAREN_TYPE => Some(ParenType { syntax }), _ => None, } } - fn syntax(&self) -> &SyntaxNode { &self.syntax } + fn syntax(self) -> SyntaxNodeRef<'a> { self.syntax } } -impl ParenType {} +impl<'a> ParenType<'a> {} // PathType #[derive(Debug, Clone, Copy)] -pub struct PathType { - syntax: SyntaxNode, +pub struct PathType<'a> { + syntax: SyntaxNodeRef<'a>, } -impl AstNode for PathType { - fn cast(syntax: SyntaxNode) -> Option { +impl<'a> AstNode<'a> for PathType<'a> { + fn cast(syntax: SyntaxNodeRef<'a>) -> Option { match syntax.kind() { PATH_TYPE => Some(PathType { syntax }), _ => None, } } - fn syntax(&self) -> &SyntaxNode { &self.syntax } + fn syntax(self) -> SyntaxNodeRef<'a> { self.syntax } } -impl PathType {} +impl<'a> PathType<'a> {} // PlaceholderType #[derive(Debug, Clone, Copy)] -pub struct PlaceholderType { - syntax: SyntaxNode, +pub struct PlaceholderType<'a> { + syntax: SyntaxNodeRef<'a>, } -impl AstNode for PlaceholderType { - fn cast(syntax: SyntaxNode) -> Option { +impl<'a> AstNode<'a> for PlaceholderType<'a> { + fn cast(syntax: SyntaxNodeRef<'a>) -> Option { match syntax.kind() { PLACEHOLDER_TYPE => Some(PlaceholderType { syntax }), _ => None, } } - fn syntax(&self) -> &SyntaxNode { &self.syntax } + fn syntax(self) -> SyntaxNodeRef<'a> { self.syntax } } -impl PlaceholderType {} +impl<'a> PlaceholderType<'a> {} // PointerType #[derive(Debug, Clone, Copy)] -pub struct PointerType { - syntax: SyntaxNode, +pub struct PointerType<'a> { + syntax: SyntaxNodeRef<'a>, } -impl AstNode for PointerType { - fn cast(syntax: SyntaxNode) -> Option { +impl<'a> AstNode<'a> for PointerType<'a> { + fn cast(syntax: SyntaxNodeRef<'a>) -> Option { match syntax.kind() { POINTER_TYPE => Some(PointerType { syntax }), _ => None, } } - fn syntax(&self) -> &SyntaxNode { &self.syntax } + fn syntax(self) -> SyntaxNodeRef<'a> { self.syntax } } -impl PointerType {} +impl<'a> PointerType<'a> {} // ReferenceType #[derive(Debug, Clone, Copy)] -pub struct ReferenceType { - syntax: SyntaxNode, +pub struct ReferenceType<'a> { + syntax: SyntaxNodeRef<'a>, } -impl AstNode for ReferenceType { - fn cast(syntax: SyntaxNode) -> Option { +impl<'a> AstNode<'a> for ReferenceType<'a> { + fn cast(syntax: SyntaxNodeRef<'a>) -> Option { match syntax.kind() { REFERENCE_TYPE => Some(ReferenceType { syntax }), _ => None, } } - fn syntax(&self) -> &SyntaxNode { &self.syntax } + fn syntax(self) -> SyntaxNodeRef<'a> { self.syntax } } -impl ReferenceType {} +impl<'a> ReferenceType<'a> {} // SliceType #[derive(Debug, Clone, Copy)] -pub struct SliceType { - syntax: SyntaxNode, +pub struct SliceType<'a> { + syntax: SyntaxNodeRef<'a>, } -impl AstNode for SliceType { - fn cast(syntax: SyntaxNode) -> Option { +impl<'a> AstNode<'a> for SliceType<'a> { + fn cast(syntax: SyntaxNodeRef<'a>) -> Option { match syntax.kind() { SLICE_TYPE => Some(SliceType { syntax }), _ => None, } } - fn syntax(&self) -> &SyntaxNode { &self.syntax } + fn syntax(self) -> SyntaxNodeRef<'a> { self.syntax } } -impl SliceType {} +impl<'a> SliceType<'a> {} // StaticDef #[derive(Debug, Clone, Copy)] -pub struct StaticDef { - syntax: SyntaxNode, +pub struct StaticDef<'a> { + syntax: SyntaxNodeRef<'a>, } -impl AstNode for StaticDef { - fn cast(syntax: SyntaxNode) -> Option { +impl<'a> AstNode<'a> for StaticDef<'a> { + fn cast(syntax: SyntaxNodeRef<'a>) -> Option { match syntax.kind() { STATIC_DEF => Some(StaticDef { syntax }), _ => None, } } - fn syntax(&self) -> &SyntaxNode { &self.syntax } + fn syntax(self) -> SyntaxNodeRef<'a> { self.syntax } } -impl ast::NameOwner for StaticDef {} -impl ast::AttrsOwner for StaticDef {} -impl StaticDef {} +impl<'a> ast::NameOwner<'a> for StaticDef<'a> {} +impl<'a> ast::AttrsOwner<'a> for StaticDef<'a> {} +impl<'a> StaticDef<'a> {} // StructDef #[derive(Debug, Clone, Copy)] -pub struct StructDef { - syntax: SyntaxNode, +pub struct StructDef<'a> { + syntax: SyntaxNodeRef<'a>, } -impl AstNode for StructDef { - fn cast(syntax: SyntaxNode) -> Option { +impl<'a> AstNode<'a> for StructDef<'a> { + fn cast(syntax: SyntaxNodeRef<'a>) -> Option { match syntax.kind() { STRUCT_DEF => Some(StructDef { syntax }), _ => None, } } - fn syntax(&self) -> &SyntaxNode { &self.syntax } + fn syntax(self) -> SyntaxNodeRef<'a> { self.syntax } } -impl ast::NameOwner for StructDef {} -impl ast::AttrsOwner for StructDef {} -impl StructDef { - pub fn fields<'a>(&'a self) -> impl Iterator> + 'a { +impl<'a> ast::NameOwner<'a> for StructDef<'a> {} +impl<'a> ast::AttrsOwner<'a> for StructDef<'a> {} +impl<'a> StructDef<'a> { + pub fn fields(self) -> impl Iterator> + 'a { self.syntax() .children() .filter_map(NamedField::cast) @@ -497,100 +497,100 @@ impl StructDef { // TokenTree #[derive(Debug, Clone, Copy)] -pub struct TokenTree { - syntax: SyntaxNode, +pub struct TokenTree<'a> { + syntax: SyntaxNodeRef<'a>, } -impl AstNode for TokenTree { - fn cast(syntax: SyntaxNode) -> Option { +impl<'a> AstNode<'a> for TokenTree<'a> { + fn cast(syntax: SyntaxNodeRef<'a>) -> Option { match syntax.kind() { TOKEN_TREE => Some(TokenTree { syntax }), _ => None, } } - fn syntax(&self) -> &SyntaxNode { &self.syntax } + fn syntax(self) -> SyntaxNodeRef<'a> { self.syntax } } -impl TokenTree {} +impl<'a> TokenTree<'a> {} // TraitDef #[derive(Debug, Clone, Copy)] -pub struct TraitDef { - syntax: SyntaxNode, +pub struct TraitDef<'a> { + syntax: SyntaxNodeRef<'a>, } -impl AstNode for TraitDef { - fn cast(syntax: SyntaxNode) -> Option { +impl<'a> AstNode<'a> for TraitDef<'a> { + fn cast(syntax: SyntaxNodeRef<'a>) -> Option { match syntax.kind() { TRAIT_DEF => Some(TraitDef { syntax }), _ => None, } } - fn syntax(&self) -> &SyntaxNode { &self.syntax } + fn syntax(self) -> SyntaxNodeRef<'a> { self.syntax } } -impl ast::NameOwner for TraitDef {} -impl ast::AttrsOwner for TraitDef {} -impl TraitDef {} +impl<'a> ast::NameOwner<'a> for TraitDef<'a> {} +impl<'a> ast::AttrsOwner<'a> for TraitDef<'a> {} +impl<'a> TraitDef<'a> {} // TupleType #[derive(Debug, Clone, Copy)] -pub struct TupleType { - syntax: SyntaxNode, +pub struct TupleType<'a> { + syntax: SyntaxNodeRef<'a>, } -impl AstNode for TupleType { - fn cast(syntax: SyntaxNode) -> Option { +impl<'a> AstNode<'a> for TupleType<'a> { + fn cast(syntax: SyntaxNodeRef<'a>) -> Option { match syntax.kind() { TUPLE_TYPE => Some(TupleType { syntax }), _ => None, } } - fn syntax(&self) -> &SyntaxNode { &self.syntax } + fn syntax(self) -> SyntaxNodeRef<'a> { self.syntax } } -impl TupleType {} +impl<'a> TupleType<'a> {} // TypeDef #[derive(Debug, Clone, Copy)] -pub struct TypeDef { - syntax: SyntaxNode, +pub struct TypeDef<'a> { + syntax: SyntaxNodeRef<'a>, } -impl AstNode for TypeDef { - fn cast(syntax: SyntaxNode) -> Option { +impl<'a> AstNode<'a> for TypeDef<'a> { + fn cast(syntax: SyntaxNodeRef<'a>) -> Option { match syntax.kind() { TYPE_DEF => Some(TypeDef { syntax }), _ => None, } } - fn syntax(&self) -> &SyntaxNode { &self.syntax } + fn syntax(self) -> SyntaxNodeRef<'a> { self.syntax } } -impl ast::NameOwner for TypeDef {} -impl ast::AttrsOwner for TypeDef {} -impl TypeDef {} +impl<'a> ast::NameOwner<'a> for TypeDef<'a> {} +impl<'a> ast::AttrsOwner<'a> for TypeDef<'a> {} +impl<'a> TypeDef<'a> {} // TypeRef #[derive(Debug, Clone, Copy)] -pub enum TypeRef { - ParenType(ParenType), - TupleType(TupleType), - NeverType(NeverType), - PathType(PathType), - PointerType(PointerType), - ArrayType(ArrayType), - SliceType(SliceType), - ReferenceType(ReferenceType), - PlaceholderType(PlaceholderType), - FnPointerType(FnPointerType), - ForType(ForType), - ImplTraitType(ImplTraitType), - DynTraitType(DynTraitType), +pub enum TypeRef<'a> { + ParenType(ParenType<'a>), + TupleType(TupleType<'a>), + NeverType(NeverType<'a>), + PathType(PathType<'a>), + PointerType(PointerType<'a>), + ArrayType(ArrayType<'a>), + SliceType(SliceType<'a>), + ReferenceType(ReferenceType<'a>), + PlaceholderType(PlaceholderType<'a>), + FnPointerType(FnPointerType<'a>), + ForType(ForType<'a>), + ImplTraitType(ImplTraitType<'a>), + DynTraitType(DynTraitType<'a>), } -impl AstNode for TypeRef { - fn cast(syntax: SyntaxNode) -> Option { +impl<'a> AstNode<'a> for TypeRef<'a> { + fn cast(syntax: SyntaxNodeRef<'a>) -> Option { match syntax.kind() { PAREN_TYPE => Some(TypeRef::ParenType(ParenType { syntax })), TUPLE_TYPE => Some(TypeRef::TupleType(TupleType { syntax })), @@ -608,7 +608,7 @@ impl AstNode for TypeRef { _ => None, } } - fn syntax(&self) -> &SyntaxNode { + fn syntax(self) -> SyntaxNodeRef<'a> { match self { TypeRef::ParenType(inner) => inner.syntax(), TypeRef::TupleType(inner) => inner.syntax(), @@ -627,5 +627,5 @@ impl AstNode for TypeRef { } } -impl TypeRef {} +impl<'a> TypeRef<'a> {} diff --git a/crates/libsyntax2/src/ast/generated.rs.tera b/crates/libsyntax2/src/ast/generated.rs.tera index f83da03268..0572cceaab 100644 --- a/crates/libsyntax2/src/ast/generated.rs.tera +++ b/crates/libsyntax2/src/ast/generated.rs.tera @@ -1,21 +1,20 @@ -use std::sync::Arc; use { ast, - SyntaxNode, SyntaxRoot, TreeRoot, AstNode, + SyntaxNodeRef, AstNode, SyntaxKind::*, }; {% for node, methods in ast %} // {{ node }} {%- if methods.enum %} #[derive(Debug, Clone, Copy)] -pub enum {{ node }}> { +pub enum {{ node }}<'a> { {%- for kind in methods.enum %} - {{ kind }}({{ kind }}), + {{ kind }}({{ kind }}<'a>), {%- endfor %} } -impl AstNode for {{ node }} { - fn cast(syntax: SyntaxNode) -> Option { +impl<'a> AstNode<'a> for {{ node }}<'a> { + fn cast(syntax: SyntaxNodeRef<'a>) -> Option { match syntax.kind() { {%- for kind in methods.enum %} {{ kind | SCREAM }} => Some({{ node }}::{{ kind }}({{ kind }} { syntax })), @@ -23,7 +22,7 @@ impl AstNode for {{ node }} { _ => None, } } - fn syntax(&self) -> &SyntaxNode { + fn syntax(self) -> SyntaxNodeRef<'a> { match self { {%- for kind in methods.enum %} {{ node }}::{{ kind }}(inner) => inner.syntax(), @@ -33,32 +32,32 @@ impl AstNode for {{ node }} { } {% else %} #[derive(Debug, Clone, Copy)] -pub struct {{ node }}> { - syntax: SyntaxNode, +pub struct {{ node }}<'a> { + syntax: SyntaxNodeRef<'a>, } -impl AstNode for {{ node }} { - fn cast(syntax: SyntaxNode) -> Option { +impl<'a> AstNode<'a> for {{ node }}<'a> { + fn cast(syntax: SyntaxNodeRef<'a>) -> Option { match syntax.kind() { {{ node | SCREAM }} => Some({{ node }} { syntax }), _ => None, } } - fn syntax(&self) -> &SyntaxNode { &self.syntax } + fn syntax(self) -> SyntaxNodeRef<'a> { self.syntax } } {% endif %} {% if methods.traits -%} {%- for t in methods.traits -%} -impl ast::{{ t }} for {{ node }} {} +impl<'a> ast::{{ t }}<'a> for {{ node }}<'a> {} {% endfor -%} {%- endif -%} -impl {{ node }} { +impl<'a> {{ node }}<'a> { {%- if methods.collections -%} {%- for m in methods.collections -%} {%- set method_name = m.0 -%} {%- set ChildName = m.1 %} - pub fn {{ method_name }}<'a>(&'a self) -> impl Iterator> + 'a { + pub fn {{ method_name }}(self) -> impl Iterator> + 'a { self.syntax() .children() .filter_map({{ ChildName }}::cast) @@ -70,7 +69,7 @@ impl {{ node }} { {%- for m in methods.options -%} {%- set method_name = m.0 -%} {%- set ChildName = m.1 %} - pub fn {{ method_name }}(&self) -> Option<{{ ChildName }}> { + pub fn {{ method_name }}(self) -> Option<{{ ChildName }}<'a>> { self.syntax() .children() .filter_map({{ ChildName }}::cast) diff --git a/crates/libsyntax2/src/ast/mod.rs b/crates/libsyntax2/src/ast/mod.rs index b52230e9cb..46509b5ec6 100644 --- a/crates/libsyntax2/src/ast/mod.rs +++ b/crates/libsyntax2/src/ast/mod.rs @@ -4,22 +4,19 @@ use itertools::Itertools; use smol_str::SmolStr; use { - SyntaxNode, SyntaxNodeRef, OwnedRoot, TreeRoot, SyntaxError, + SyntaxNode, SyntaxNodeRef, TreeRoot, SyntaxError, SyntaxKind::*, }; pub use self::generated::*; -pub trait AstNode { - fn cast(syntax: SyntaxNode) -> Option +pub trait AstNode<'a>: Clone + Copy { + fn cast(syntax: SyntaxNodeRef<'a>) -> Option where Self: Sized; - fn syntax(&self) -> &SyntaxNode; - fn syntax_ref<'a>(&'a self) -> SyntaxNodeRef<'a> where R: 'a { - self.syntax().as_ref() - } + fn syntax(self) -> SyntaxNodeRef<'a>; } -pub trait NameOwner: AstNode { - fn name(&self) -> Option> { +pub trait NameOwner<'a>: AstNode<'a> { + fn name(self) -> Option> { self.syntax() .children() .filter_map(Name::cast) @@ -27,27 +24,37 @@ pub trait NameOwner: AstNode { } } -pub trait AttrsOwner: AstNode { - fn attrs<'a>(&'a self) -> Box> + 'a> where R: 'a { +pub trait AttrsOwner<'a>: AstNode<'a> { + fn attrs(&self) -> Box> + 'a> { let it = self.syntax().children() .filter_map(Attr::cast); Box::new(it) } } -impl File { - pub fn parse(text: &str) -> Self { - File::cast(::parse(text)).unwrap() - } +#[derive(Clone, Debug)] +pub struct ParsedFile { + root: SyntaxNode } -impl File { +impl ParsedFile { + pub fn parse(text: &str) -> Self { + let root = ::parse(text); + ParsedFile { root } + } + pub fn ast(&self) -> File { + File::cast(self.syntax()).unwrap() + } + pub fn syntax(&self) -> SyntaxNodeRef { + self.root.as_ref() + } pub fn errors(&self) -> Vec { self.syntax().root.syntax_root().errors.clone() } + } -impl FnDef { +impl<'a> FnDef<'a> { pub fn has_atom_attr(&self, atom: &str) -> bool { self.attrs() .filter_map(|x| x.as_atom()) @@ -55,7 +62,7 @@ impl FnDef { } } -impl Attr { +impl<'a> Attr<'a> { pub fn as_atom(&self) -> Option { let tt = self.value()?; let (_bra, attr, _ket) = tt.syntax().children().collect_tuple()?; @@ -66,7 +73,7 @@ impl Attr { } } - pub fn as_call(&self) -> Option<(SmolStr, TokenTree)> { + pub fn as_call(&self) -> Option<(SmolStr, TokenTree<'a>)> { let tt = self.value()?; let (_bra, attr, args, _ket) = tt.syntax().children().collect_tuple()?; let args = TokenTree::cast(args)?; @@ -78,7 +85,7 @@ impl Attr { } } -impl Name { +impl<'a> Name<'a> { pub fn text(&self) -> SmolStr { let ident = self.syntax().first_child() .unwrap(); @@ -86,7 +93,7 @@ impl Name { } } -impl NameRef { +impl<'a> NameRef<'a> { pub fn text(&self) -> SmolStr { let ident = self.syntax().first_child() .unwrap(); @@ -94,22 +101,22 @@ impl NameRef { } } -impl ImplItem { - pub fn target_type(&self) -> Option> { +impl<'a> ImplItem<'a> { + pub fn target_type(&self) -> Option> { match self.target() { (Some(t), None) | (_, Some(t)) => Some(t), _ => None, } } - pub fn target_trait(&self) -> Option> { + pub fn target_trait(&self) -> Option> { match self.target() { (Some(t), Some(_)) => Some(t), _ => None, } } - fn target(&self) -> (Option>, Option>) { + fn target(&self) -> (Option>, Option>) { let mut types = self.syntax().children().filter_map(TypeRef::cast); let first = types.next(); let second = types.next(); @@ -117,9 +124,9 @@ impl ImplItem { } } -impl Module { +impl<'a> Module<'a> { pub fn has_semi(&self) -> bool { - match self.syntax_ref().last_child() { + match self.syntax().last_child() { None => false, Some(node) => node.kind() == SEMI, } diff --git a/crates/libsyntax2/src/lib.rs b/crates/libsyntax2/src/lib.rs index 0e40fb65fb..e837a8d2db 100644 --- a/crates/libsyntax2/src/lib.rs +++ b/crates/libsyntax2/src/lib.rs @@ -41,7 +41,7 @@ mod yellow; pub mod utils; pub use { - ast::{AstNode, File}, + ast::{AstNode, ParsedFile}, lexer::{tokenize, Token}, syntax_kinds::SyntaxKind, text_unit::{TextRange, TextUnit}, diff --git a/crates/libsyntax2/src/utils.rs b/crates/libsyntax2/src/utils.rs index 4f60aa2a89..fbe48dd71c 100644 --- a/crates/libsyntax2/src/utils.rs +++ b/crates/libsyntax2/src/utils.rs @@ -1,12 +1,11 @@ use std::fmt::Write; use { algo::walk::{walk, WalkEvent}, - SyntaxNode, TreeRoot, + SyntaxNodeRef, TreeRoot, }; /// Parse a file and create a string representation of the resulting parse tree. -pub fn dump_tree(syntax: &SyntaxNode) -> String { - let syntax = syntax.as_ref(); +pub fn dump_tree(syntax: SyntaxNodeRef) -> String { let mut errors: Vec<_> = syntax.root.syntax_root().errors.iter().cloned().collect(); errors.sort_by_key(|e| e.offset); let mut err_pos = 0; diff --git a/crates/libsyntax2/src/yellow/syntax.rs b/crates/libsyntax2/src/yellow/syntax.rs index ef82ba408f..87e4a159de 100644 --- a/crates/libsyntax2/src/yellow/syntax.rs +++ b/crates/libsyntax2/src/yellow/syntax.rs @@ -71,12 +71,16 @@ impl SyntaxNode { self.red().green().text() } - pub fn children<'a>(&'a self) -> impl Iterator> + 'a { - let red = self.red(); - let n_children = red.n_children(); - (0..n_children).map(move |i| SyntaxNode { - root: self.root.clone(), - red: red.get_child(i).unwrap(), + pub fn children(&self) -> impl Iterator> { + let red = self.red; + let n_children = self.red().n_children(); + let root = self.root.clone(); + (0..n_children).map(move |i| { + let red = unsafe { red.get(&root) }; + SyntaxNode { + root: root.clone(), + red: red.get_child(i).unwrap(), + } }) } diff --git a/crates/libsyntax2/tests/test/main.rs b/crates/libsyntax2/tests/test/main.rs index 802dba0e9a..d35935c646 100644 --- a/crates/libsyntax2/tests/test/main.rs +++ b/crates/libsyntax2/tests/test/main.rs @@ -21,7 +21,7 @@ fn lexer_tests() { fn parser_tests() { dir_tests(&["parser/inline", "parser/ok", "parser/err"], |text| { let file = libsyntax2::parse(text); - libsyntax2::utils::dump_tree(&file) + libsyntax2::utils::dump_tree(file.as_ref()) }) }