]> git.lizzy.rs Git - rust.git/blob - src/libsyntax_ext/global_asm.rs
Remove `with_legacy_ctxt`
[rust.git] / src / libsyntax_ext / global_asm.rs
1 /// Module-level assembly support.
2 ///
3 /// The macro defined here allows you to specify "top-level",
4 /// "file-scoped", or "module-level" assembly. These synonyms
5 /// all correspond to LLVM's module-level inline assembly instruction.
6 ///
7 /// For example, `global_asm!("some assembly here")` codegens to
8 /// LLVM's `module asm "some assembly here"`. All of LLVM's caveats
9 /// therefore apply.
10
11 use errors::DiagnosticBuilder;
12
13 use syntax::ast;
14 use syntax::source_map::respan;
15 use syntax::ext::base::{self, *};
16 use syntax::parse::token;
17 use syntax::ptr::P;
18 use syntax_pos::Span;
19 use syntax::tokenstream::TokenStream;
20 use smallvec::smallvec;
21
22 pub fn expand_global_asm<'cx>(cx: &'cx mut ExtCtxt<'_>,
23                               sp: Span,
24                               tts: TokenStream) -> Box<dyn base::MacResult + 'cx> {
25     match parse_global_asm(cx, sp, tts) {
26         Ok(Some(global_asm)) => {
27             MacEager::items(smallvec![P(ast::Item {
28                 ident: ast::Ident::invalid(),
29                 attrs: Vec::new(),
30                 id: ast::DUMMY_NODE_ID,
31                 node: ast::ItemKind::GlobalAsm(P(global_asm)),
32                 vis: respan(sp.shrink_to_lo(), ast::VisibilityKind::Inherited),
33                 span: cx.with_def_site_ctxt(sp),
34                 tokens: None,
35             })])
36         }
37         Ok(None) => DummyResult::any(sp),
38         Err(mut err) => {
39             err.emit();
40             DummyResult::any(sp)
41         }
42     }
43 }
44
45 fn parse_global_asm<'a>(
46     cx: &mut ExtCtxt<'a>,
47     sp: Span,
48     tts: TokenStream
49 ) -> Result<Option<ast::GlobalAsm>, DiagnosticBuilder<'a>> {
50     let mut p = cx.new_parser_from_tts(tts);
51
52     if p.token == token::Eof {
53         let mut err = cx.struct_span_err(sp, "macro requires a string literal as an argument");
54         err.span_label(sp, "string literal required");
55         return Err(err);
56     }
57
58     let expr = p.parse_expr()?;
59     let (asm, _) = match expr_to_string(cx, expr, "inline assembly must be a string literal") {
60         Some((s, st)) => (s, st),
61         None => return Ok(None),
62     };
63
64     Ok(Some(ast::GlobalAsm { asm }))
65 }