1#![cfg_attr(feature = "in-rust-tree", feature(rustc_private))]
23
24#[cfg(feature = "in-rust-tree")]
25extern crate rustc_driver as _;
26
27mod parsing;
28mod ptr;
29mod syntax_error;
30mod syntax_node;
31#[cfg(test)]
32mod tests;
33mod token_text;
34mod validation;
35
36pub mod algo;
37pub mod ast;
38#[doc(hidden)]
39pub mod fuzz;
40pub mod hacks;
41pub mod syntax_editor;
42pub mod ted;
43pub mod utils;
44
45use std::{marker::PhantomData, ops::Range};
46
47use stdx::format_to;
48use triomphe::Arc;
49
50pub use crate::{
51 ast::{AstNode, AstToken},
52 ptr::{AstPtr, SyntaxNodePtr},
53 syntax_error::SyntaxError,
54 syntax_node::{
55 PreorderWithTokens, RustLanguage, SyntaxElement, SyntaxElementChildren, SyntaxNode,
56 SyntaxNodeChildren, SyntaxToken, SyntaxTreeBuilder,
57 },
58 token_text::TokenText,
59};
60pub use parser::{Edition, SyntaxKind, T};
61pub use rowan::{
62 Direction, GreenNode, NodeOrToken, SyntaxText, TextRange, TextSize, TokenAtOffset, WalkEvent,
63 api::Preorder,
64};
65pub use rustc_literal_escaper as unescape;
66pub use smol_str::{SmolStr, SmolStrBuilder, ToSmolStr, format_smolstr};
67
68#[derive(Debug, PartialEq, Eq)]
74pub struct Parse<T> {
75 green: Option<GreenNode>,
76 errors: Option<Arc<[SyntaxError]>>,
77 _ty: PhantomData<fn() -> T>,
78}
79
80impl<T> Clone for Parse<T> {
81 fn clone(&self) -> Parse<T> {
82 Parse { green: self.green.clone(), errors: self.errors.clone(), _ty: PhantomData }
83 }
84}
85
86impl<T> Parse<T> {
87 fn new(green: GreenNode, errors: Vec<SyntaxError>) -> Parse<T> {
88 Parse {
89 green: Some(green),
90 errors: if errors.is_empty() { None } else { Some(errors.into()) },
91 _ty: PhantomData,
92 }
93 }
94
95 pub fn syntax_node(&self) -> SyntaxNode {
96 SyntaxNode::new_root(self.green.as_ref().unwrap().clone())
97 }
98
99 pub fn errors(&self) -> Vec<SyntaxError> {
100 let mut errors = if let Some(e) = self.errors.as_deref() { e.to_vec() } else { vec![] };
101 validation::validate(&self.syntax_node(), &mut errors);
102 errors
103 }
104}
105
106impl<T: AstNode> Parse<T> {
107 pub fn to_syntax(mut self) -> Parse<SyntaxNode> {
109 let green = self.green.take();
110 let errors = self.errors.take();
111 Parse { green, errors, _ty: PhantomData }
112 }
113
114 pub fn tree(&self) -> T {
121 T::cast(self.syntax_node()).unwrap()
122 }
123
124 pub fn ok(self) -> Result<T, Vec<SyntaxError>> {
126 match self.errors() {
127 errors if !errors.is_empty() => Err(errors),
128 _ => Ok(self.tree()),
129 }
130 }
131}
132
133impl Parse<SyntaxNode> {
134 pub fn cast<N: AstNode>(mut self) -> Option<Parse<N>> {
135 if N::cast(self.syntax_node()).is_some() {
136 Some(Parse { green: self.green.take(), errors: self.errors.take(), _ty: PhantomData })
137 } else {
138 None
139 }
140 }
141}
142
143impl Parse<SourceFile> {
144 pub fn debug_dump(&self) -> String {
145 let mut buf = format!("{:#?}", self.tree().syntax());
146 for err in self.errors() {
147 format_to!(buf, "error {:?}: {}\n", err.range(), err);
148 }
149 buf
150 }
151
152 pub fn reparse(&self, delete: TextRange, insert: &str, edition: Edition) -> Parse<SourceFile> {
153 self.incremental_reparse(delete, insert, edition)
154 .unwrap_or_else(|| self.full_reparse(delete, insert, edition))
155 }
156
157 fn incremental_reparse(
158 &self,
159 delete: TextRange,
160 insert: &str,
161 edition: Edition,
162 ) -> Option<Parse<SourceFile>> {
163 parsing::incremental_reparse(
165 self.tree().syntax(),
166 delete,
167 insert,
168 self.errors.as_deref().unwrap_or_default().iter().cloned(),
169 edition,
170 )
171 .map(|(green_node, errors, _reparsed_range)| Parse {
172 green: Some(green_node),
173 errors: if errors.is_empty() { None } else { Some(errors.into()) },
174 _ty: PhantomData,
175 })
176 }
177
178 fn full_reparse(&self, delete: TextRange, insert: &str, edition: Edition) -> Parse<SourceFile> {
179 let mut text = self.tree().syntax().text().to_string();
180 text.replace_range(Range::<usize>::from(delete), insert);
181 SourceFile::parse(&text, edition)
182 }
183}
184
185impl ast::Expr {
186 pub fn parse(text: &str, edition: Edition) -> Parse<ast::Expr> {
195 let _p = tracing::info_span!("Expr::parse").entered();
196 let (green, errors) = parsing::parse_text_at(text, parser::TopEntryPoint::Expr, edition);
197 let root = SyntaxNode::new_root(green.clone());
198
199 assert!(
200 ast::Expr::can_cast(root.kind()) || root.kind() == SyntaxKind::ERROR,
201 "{:?} isn't an expression",
202 root.kind()
203 );
204 Parse::new(green, errors)
205 }
206}
207
208#[cfg(not(no_salsa_async_drops))]
209impl<T> Drop for Parse<T> {
210 fn drop(&mut self) {
211 let Some(green) = self.green.take() else {
212 return;
213 };
214 static PARSE_DROP_THREAD: std::sync::OnceLock<std::sync::mpsc::Sender<GreenNode>> =
215 std::sync::OnceLock::new();
216 PARSE_DROP_THREAD
217 .get_or_init(|| {
218 let (sender, receiver) = std::sync::mpsc::channel::<GreenNode>();
219 std::thread::Builder::new()
220 .name("ParseNodeDropper".to_owned())
221 .spawn(move || receiver.iter().for_each(drop))
222 .unwrap();
223 sender
224 })
225 .send(green)
226 .unwrap();
227 }
228}
229
230pub use crate::ast::SourceFile;
232
233impl SourceFile {
234 pub fn parse(text: &str, edition: Edition) -> Parse<SourceFile> {
235 let _p = tracing::info_span!("SourceFile::parse").entered();
236 let (green, errors) = parsing::parse_text(text, edition);
237 let root = SyntaxNode::new_root(green.clone());
238
239 assert_eq!(root.kind(), SyntaxKind::SOURCE_FILE);
240 Parse::new(green, errors)
241 }
242}
243
244#[macro_export]
259macro_rules! match_ast {
260 (match $node:ident { $($tt:tt)* }) => { $crate::match_ast!(match ($node) { $($tt)* }) };
261
262 (match ($node:expr) {
263 $( $( $path:ident )::+ ($it:pat) => $res:expr, )*
264 _ => $catch_all:expr $(,)?
265 }) => {{
266 $( if let Some($it) = $($path::)+cast($node.clone()) { $res } else )*
267 { $catch_all }
268 }};
269}
270
271#[test]
274fn api_walkthrough() {
275 use ast::{HasModuleItem, HasName};
276
277 let source_code = "
278 fn foo() {
279 1 + 1
280 }
281 ";
282 let parse = SourceFile::parse(source_code, parser::Edition::CURRENT);
287 assert!(parse.errors().is_empty());
288
289 let file: SourceFile = parse.tree();
292
293 let mut func = None;
296 for item in file.items() {
297 match item {
298 ast::Item::Fn(f) => func = Some(f),
299 _ => unreachable!(),
300 }
301 }
302 let func: ast::Fn = func.unwrap();
303
304 let name: Option<ast::Name> = func.name();
310 let name = name.unwrap();
311 assert_eq!(name.text(), "foo");
312
313 let body: ast::BlockExpr = func.body().unwrap();
315 let stmt_list: ast::StmtList = body.stmt_list().unwrap();
316 let expr: ast::Expr = stmt_list.tail_expr().unwrap();
317
318 let bin_expr: &ast::BinExpr = match &expr {
323 ast::Expr::BinExpr(e) => e,
324 _ => unreachable!(),
325 };
326
327 let expr_syntax: &SyntaxNode = expr.syntax();
330
331 assert!(expr_syntax == bin_expr.syntax());
333
334 let _expr: ast::Expr = match ast::Expr::cast(expr_syntax.clone()) {
336 Some(e) => e,
337 None => unreachable!(),
338 };
339
340 assert_eq!(expr_syntax.kind(), SyntaxKind::BIN_EXPR);
342
343 assert_eq!(expr_syntax.text_range(), TextRange::new(32.into(), 37.into()));
345
346 let text: SyntaxText = expr_syntax.text();
349 assert_eq!(text.to_string(), "1 + 1");
350
351 assert_eq!(expr_syntax.parent().as_ref(), Some(stmt_list.syntax()));
353 assert_eq!(stmt_list.syntax().first_child_or_token().map(|it| it.kind()), Some(T!['{']));
354 assert_eq!(
355 expr_syntax.next_sibling_or_token().map(|it| it.kind()),
356 Some(SyntaxKind::WHITESPACE)
357 );
358
359 let f = expr_syntax.ancestors().find_map(ast::Fn::cast);
361 assert_eq!(f, Some(func));
362 assert!(expr_syntax.siblings_with_tokens(Direction::Next).any(|it| it.kind() == T!['}']));
363 assert_eq!(
364 expr_syntax.descendants_with_tokens().count(),
365 8, );
369
370 let mut buf = String::new();
372 let mut indent = 0;
373 for event in expr_syntax.preorder_with_tokens() {
374 match event {
375 WalkEvent::Enter(node) => {
376 let text = match &node {
377 NodeOrToken::Node(it) => it.text().to_string(),
378 NodeOrToken::Token(it) => it.text().to_owned(),
379 };
380 format_to!(buf, "{:indent$}{:?} {:?}\n", " ", text, node.kind(), indent = indent);
381 indent += 2;
382 }
383 WalkEvent::Leave(_) => indent -= 2,
384 }
385 }
386 assert_eq!(indent, 0);
387 assert_eq!(
388 buf.trim(),
389 r#"
390"1 + 1" BIN_EXPR
391 "1" LITERAL
392 "1" INT_NUMBER
393 " " WHITESPACE
394 "+" PLUS
395 " " WHITESPACE
396 "1" LITERAL
397 "1" INT_NUMBER
398"#
399 .trim()
400 );
401
402 let exprs_cast: Vec<String> = file
409 .syntax()
410 .descendants()
411 .filter_map(ast::Expr::cast)
412 .map(|expr| expr.syntax().text().to_string())
413 .collect();
414
415 let mut exprs_visit = Vec::new();
417 for node in file.syntax().descendants() {
418 match_ast! {
419 match node {
420 ast::Expr(it) => {
421 let res = it.syntax().text().to_string();
422 exprs_visit.push(res);
423 },
424 _ => (),
425 }
426 }
427 }
428 assert_eq!(exprs_cast, exprs_visit);
429}