#![feature(rustc_private)] extern crate rustc_ast; extern crate rustc_driver; extern crate rustc_errors; extern crate rustc_parse; extern crate rustc_session; extern crate rustc_span; use rustc_ast::ast::{AttrKind, Attribute, DUMMY_NODE_ID, Expr}; use rustc_ast::mut_visit::{self, MutVisitor}; use rustc_ast::node_id::NodeId; use rustc_ast::ptr::P; use rustc_ast::token::{self, Token}; use rustc_ast::tokenstream::{AttrTokenStream, AttrTokenTree, LazyAttrTokenStream}; use rustc_errors::Diag; use rustc_parse::parser::Recovery; use rustc_session::parse::ParseSess; use rustc_span::{AttrId, DUMMY_SP, FileName, Span}; use std::sync::Arc; pub fn parse_expr(psess: &ParseSess, source_code: &str) -> Option> { let parser = rustc_parse::unwrap_or_emit_fatal(rustc_parse::new_parser_from_source_str( psess, FileName::anon_source_code(source_code), source_code.to_owned(), )); let mut parser = parser.recovery(Recovery::Forbidden); let mut expr = parser.parse_expr().map_err(Diag::cancel).ok()?; if parser.token != token::Eof { return None; } Normalize.visit_expr(&mut expr); Some(expr) } // Erase Span information that could distinguish between identical expressions // parsed from different source strings. struct Normalize; impl MutVisitor for Normalize { fn visit_id(&mut self, id: &mut NodeId) { *id = DUMMY_NODE_ID; } fn visit_span(&mut self, span: &mut Span) { *span = DUMMY_SP; } fn visit_attribute(&mut self, attr: &mut Attribute) { attr.id = AttrId::from_u32(0); if let AttrKind::Normal(normal_attr) = &mut attr.kind { if let Some(tokens) = &mut normal_attr.tokens { let mut stream = tokens.to_attr_token_stream(); normalize_attr_token_stream(&mut stream); *tokens = LazyAttrTokenStream::new_direct(stream); } } mut_visit::walk_attribute(self, attr); } } fn normalize_attr_token_stream(stream: &mut AttrTokenStream) { Arc::make_mut(&mut stream.0) .iter_mut() .for_each(normalize_attr_token_tree); } fn normalize_attr_token_tree(token: &mut AttrTokenTree) { match token { AttrTokenTree::Token(token, _spacing) => { Normalize.visit_span(&mut token.span); } AttrTokenTree::Delimited(dspan, _spacing, _delim, stream) => { normalize_attr_token_stream(stream); Normalize.visit_span(&mut dspan.open); Normalize.visit_span(&mut dspan.close); } AttrTokenTree::AttrsTarget(_) => unimplemented!("AttrTokenTree::AttrsTarget"), } }