| #![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::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<Box<Expr>> { | 
 |     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(), | 
 |         rustc_parse::lexer::StripTokens::Nothing, | 
 |     )); | 
 |  | 
 |     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"), | 
 |     } | 
 | } |