2013-08-07 00:50:23 -04:00
|
|
|
// Copyright 2012-2013 The Rust Project Developers. See the COPYRIGHT
|
2013-03-10 22:08:38 -07:00
|
|
|
// file at the top-level directory of this distribution and at
|
|
|
|
// http://rust-lang.org/COPYRIGHT.
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
|
|
|
|
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
|
|
|
|
// option. This file may not be copied, modified, or distributed
|
|
|
|
// except according to those terms.
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Inline assembly support.
|
|
|
|
*/
|
|
|
|
|
|
|
|
use ast;
|
2014-09-27 01:33:36 -07:00
|
|
|
use codemap;
|
2013-08-31 18:13:04 +02:00
|
|
|
use codemap::Span;
|
2013-03-10 22:08:38 -07:00
|
|
|
use ext::base;
|
|
|
|
use ext::base::*;
|
2014-01-10 14:02:36 -08:00
|
|
|
use parse::token::InternedString;
|
2013-03-12 00:01:09 -07:00
|
|
|
use parse::token;
|
2014-09-13 19:06:01 +03:00
|
|
|
use ptr::P;
|
2014-06-11 19:33:52 -07:00
|
|
|
|
2013-03-12 00:01:09 -07:00
|
|
|
enum State {
|
|
|
|
Asm,
|
|
|
|
Outputs,
|
|
|
|
Inputs,
|
|
|
|
Clobbers,
|
2014-03-09 23:41:18 +01:00
|
|
|
Options,
|
|
|
|
StateNone
|
2013-03-12 00:01:09 -07:00
|
|
|
}
|
|
|
|
|
2014-03-09 23:41:18 +01:00
|
|
|
impl State {
|
|
|
|
fn next(&self) -> State {
|
|
|
|
match *self {
|
|
|
|
Asm => Outputs,
|
|
|
|
Outputs => Inputs,
|
|
|
|
Inputs => Clobbers,
|
|
|
|
Clobbers => Options,
|
|
|
|
Options => StateNone,
|
|
|
|
StateNone => StateNone
|
|
|
|
}
|
2013-03-12 00:01:09 -07:00
|
|
|
}
|
|
|
|
}
|
2013-03-10 22:08:38 -07:00
|
|
|
|
2014-03-09 23:41:18 +01:00
|
|
|
static OPTIONS: &'static [&'static str] = &["volatile", "alignstack", "intel"];
|
|
|
|
|
2014-08-27 21:46:52 -04:00
|
|
|
pub fn expand_asm<'cx>(cx: &'cx mut ExtCtxt, sp: Span, tts: &[ast::TokenTree])
|
|
|
|
-> Box<base::MacResult+'cx> {
|
2014-07-03 11:42:24 +02:00
|
|
|
let mut p = cx.new_parser_from_tts(tts);
|
2014-01-10 14:02:36 -08:00
|
|
|
let mut asm = InternedString::new("");
|
2013-10-08 02:49:10 +02:00
|
|
|
let mut asm_str_style = None;
|
2014-02-28 13:09:09 -08:00
|
|
|
let mut outputs = Vec::new();
|
|
|
|
let mut inputs = Vec::new();
|
2014-05-25 03:10:11 -07:00
|
|
|
let mut cons = "".to_string();
|
2013-03-12 00:01:09 -07:00
|
|
|
let mut volatile = false;
|
2013-03-12 01:02:58 -07:00
|
|
|
let mut alignstack = false;
|
2014-01-09 15:05:33 +02:00
|
|
|
let mut dialect = ast::AsmAtt;
|
2013-03-12 00:01:09 -07:00
|
|
|
|
|
|
|
let mut state = Asm;
|
2013-04-26 16:19:26 -07:00
|
|
|
|
2014-03-09 23:41:18 +01:00
|
|
|
'statement: loop {
|
2013-03-12 00:01:09 -07:00
|
|
|
match state {
|
|
|
|
Asm => {
|
2014-06-21 03:39:03 -07:00
|
|
|
let (s, style) = match expr_to_string(cx, p.parse_expr(),
|
2014-01-18 01:53:10 +11:00
|
|
|
"inline assembly must be a string literal.") {
|
|
|
|
Some((s, st)) => (s, st),
|
|
|
|
// let compilation continue
|
2014-04-15 22:00:14 +10:00
|
|
|
None => return DummyResult::expr(sp),
|
2014-01-18 01:53:10 +11:00
|
|
|
};
|
2013-10-08 02:49:10 +02:00
|
|
|
asm = s;
|
|
|
|
asm_str_style = Some(style);
|
2013-03-12 00:01:09 -07:00
|
|
|
}
|
|
|
|
Outputs => {
|
2014-10-27 19:22:52 +11:00
|
|
|
while p.token != token::Eof &&
|
|
|
|
p.token != token::Colon &&
|
|
|
|
p.token != token::ModSep {
|
2013-03-12 00:09:53 -07:00
|
|
|
|
2013-03-12 00:01:09 -07:00
|
|
|
if outputs.len() != 0 {
|
2014-10-27 19:22:52 +11:00
|
|
|
p.eat(&token::Comma);
|
2013-03-12 00:01:09 -07:00
|
|
|
}
|
2013-03-12 00:09:53 -07:00
|
|
|
|
2013-10-08 02:49:10 +02:00
|
|
|
let (constraint, _str_style) = p.parse_str();
|
2013-10-17 21:24:41 +03:00
|
|
|
|
2014-03-09 23:41:18 +01:00
|
|
|
let span = p.last_span;
|
2013-10-17 21:24:41 +03:00
|
|
|
|
2014-10-27 19:22:52 +11:00
|
|
|
p.expect(&token::LParen);
|
2013-03-12 00:01:09 -07:00
|
|
|
let out = p.parse_expr();
|
2014-10-27 19:22:52 +11:00
|
|
|
p.expect(&token::RParen);
|
2013-03-12 00:01:09 -07:00
|
|
|
|
2014-03-09 23:41:18 +01:00
|
|
|
// Expands a read+write operand into two operands.
|
|
|
|
//
|
|
|
|
// Use '+' modifier when you want the same expression
|
|
|
|
// to be both an input and an output at the same time.
|
|
|
|
// It's the opposite of '=&' which means that the memory
|
|
|
|
// cannot be shared with any other operand (usually when
|
|
|
|
// a register is clobbered early.)
|
|
|
|
let output = match constraint.get().slice_shift_char() {
|
|
|
|
(Some('='), _) => None,
|
|
|
|
(Some('+'), operand) => {
|
2014-05-19 23:19:56 -07:00
|
|
|
Some(token::intern_and_get_ident(format!(
|
|
|
|
"={}",
|
|
|
|
operand).as_slice()))
|
2014-03-09 23:41:18 +01:00
|
|
|
}
|
|
|
|
_ => {
|
|
|
|
cx.span_err(span, "output operand constraint lacks '=' or '+'");
|
|
|
|
None
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2014-08-19 20:39:26 +01:00
|
|
|
let is_rw = output.is_some();
|
|
|
|
outputs.push((output.unwrap_or(constraint), out, is_rw));
|
2013-03-12 00:01:09 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
Inputs => {
|
2014-10-27 19:22:52 +11:00
|
|
|
while p.token != token::Eof &&
|
|
|
|
p.token != token::Colon &&
|
|
|
|
p.token != token::ModSep {
|
2013-03-12 00:09:53 -07:00
|
|
|
|
2013-03-12 00:01:09 -07:00
|
|
|
if inputs.len() != 0 {
|
2014-10-27 19:22:52 +11:00
|
|
|
p.eat(&token::Comma);
|
2013-03-12 00:01:09 -07:00
|
|
|
}
|
2013-03-12 00:09:53 -07:00
|
|
|
|
2013-10-08 02:49:10 +02:00
|
|
|
let (constraint, _str_style) = p.parse_str();
|
2013-10-17 21:24:41 +03:00
|
|
|
|
2014-01-15 18:30:40 -08:00
|
|
|
if constraint.get().starts_with("=") {
|
2013-12-30 15:30:14 -08:00
|
|
|
cx.span_err(p.last_span, "input operand constraint contains '='");
|
2014-01-15 18:30:40 -08:00
|
|
|
} else if constraint.get().starts_with("+") {
|
2013-12-30 15:30:14 -08:00
|
|
|
cx.span_err(p.last_span, "input operand constraint contains '+'");
|
2013-10-17 21:24:41 +03:00
|
|
|
}
|
|
|
|
|
2014-10-27 19:22:52 +11:00
|
|
|
p.expect(&token::LParen);
|
2013-07-31 17:59:59 -04:00
|
|
|
let input = p.parse_expr();
|
2014-10-27 19:22:52 +11:00
|
|
|
p.expect(&token::RParen);
|
2013-03-12 00:01:09 -07:00
|
|
|
|
2013-07-31 17:59:59 -04:00
|
|
|
inputs.push((constraint, input));
|
2013-03-12 00:01:09 -07:00
|
|
|
}
|
|
|
|
}
|
2014-05-28 09:24:28 -07:00
|
|
|
Clobbers => {
|
|
|
|
let mut clobs = Vec::new();
|
2014-10-27 19:22:52 +11:00
|
|
|
while p.token != token::Eof &&
|
|
|
|
p.token != token::Colon &&
|
|
|
|
p.token != token::ModSep {
|
2014-05-28 09:24:28 -07:00
|
|
|
|
|
|
|
if clobs.len() != 0 {
|
2014-10-27 19:22:52 +11:00
|
|
|
p.eat(&token::Comma);
|
2014-05-28 09:24:28 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
let (s, _str_style) = p.parse_str();
|
|
|
|
let clob = format!("~{{{}}}", s);
|
|
|
|
clobs.push(clob);
|
|
|
|
|
|
|
|
if OPTIONS.iter().any(|opt| s.equiv(opt)) {
|
2014-08-23 12:41:32 +02:00
|
|
|
cx.span_warn(p.last_span, "expected a clobber, found an option");
|
2014-05-28 09:24:28 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
cons = clobs.connect(",");
|
|
|
|
}
|
2013-03-12 00:01:09 -07:00
|
|
|
Options => {
|
2013-10-08 02:49:10 +02:00
|
|
|
let (option, _str_style) = p.parse_str();
|
2013-03-12 00:09:53 -07:00
|
|
|
|
2014-01-15 18:30:40 -08:00
|
|
|
if option.equiv(&("volatile")) {
|
2014-03-09 23:41:18 +01:00
|
|
|
// Indicates that the inline assembly has side effects
|
|
|
|
// and must not be optimized out along with its outputs.
|
2013-03-12 00:01:09 -07:00
|
|
|
volatile = true;
|
2014-01-15 18:30:40 -08:00
|
|
|
} else if option.equiv(&("alignstack")) {
|
2013-03-12 01:02:58 -07:00
|
|
|
alignstack = true;
|
2014-01-15 18:30:40 -08:00
|
|
|
} else if option.equiv(&("intel")) {
|
2014-01-09 15:05:33 +02:00
|
|
|
dialect = ast::AsmIntel;
|
2014-03-09 23:41:18 +01:00
|
|
|
} else {
|
|
|
|
cx.span_warn(p.last_span, "unrecognized option");
|
2013-03-12 00:01:09 -07:00
|
|
|
}
|
|
|
|
|
2014-10-27 19:22:52 +11:00
|
|
|
if p.token == token::Comma {
|
|
|
|
p.eat(&token::Comma);
|
2013-03-12 00:01:09 -07:00
|
|
|
}
|
|
|
|
}
|
2014-03-09 23:41:18 +01:00
|
|
|
StateNone => ()
|
2013-03-12 00:01:09 -07:00
|
|
|
}
|
|
|
|
|
2014-03-09 23:41:18 +01:00
|
|
|
loop {
|
|
|
|
// MOD_SEP is a double colon '::' without space in between.
|
|
|
|
// When encountered, the state must be advanced twice.
|
|
|
|
match (&p.token, state.next(), state.next().next()) {
|
2014-10-27 19:22:52 +11:00
|
|
|
(&token::Colon, StateNone, _) |
|
|
|
|
(&token::ModSep, _, StateNone) => {
|
2014-03-09 23:41:18 +01:00
|
|
|
p.bump();
|
|
|
|
break 'statement;
|
2013-03-12 00:01:09 -07:00
|
|
|
}
|
2014-10-27 19:22:52 +11:00
|
|
|
(&token::Colon, st, _) |
|
|
|
|
(&token::ModSep, _, st) => {
|
2014-03-09 23:41:18 +01:00
|
|
|
p.bump();
|
|
|
|
state = st;
|
2013-03-12 00:01:09 -07:00
|
|
|
}
|
2014-10-27 19:22:52 +11:00
|
|
|
(&token::Eof, _, _) => break 'statement,
|
2014-03-09 23:41:18 +01:00
|
|
|
_ => break
|
|
|
|
}
|
2013-03-12 00:01:09 -07:00
|
|
|
}
|
2013-03-10 22:08:38 -07:00
|
|
|
}
|
|
|
|
|
2014-09-28 09:25:48 -07:00
|
|
|
let expn_id = cx.codemap().record_expansion(codemap::ExpnInfo {
|
2014-09-27 01:33:36 -07:00
|
|
|
call_site: sp,
|
|
|
|
callee: codemap::NameAndSpan {
|
|
|
|
name: "asm".to_string(),
|
|
|
|
format: codemap::MacroBang,
|
|
|
|
span: None,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
2014-09-13 19:06:01 +03:00
|
|
|
MacExpr::new(P(ast::Expr {
|
2013-09-06 22:11:55 -04:00
|
|
|
id: ast::DUMMY_NODE_ID,
|
2014-01-09 15:05:33 +02:00
|
|
|
node: ast::ExprInlineAsm(ast::InlineAsm {
|
2014-01-15 18:30:40 -08:00
|
|
|
asm: token::intern_and_get_ident(asm.get()),
|
2013-10-08 02:49:10 +02:00
|
|
|
asm_str_style: asm_str_style.unwrap(),
|
2013-03-27 13:42:21 -07:00
|
|
|
outputs: outputs,
|
2014-08-19 20:39:26 +01:00
|
|
|
inputs: inputs,
|
|
|
|
clobbers: token::intern_and_get_ident(cons.as_slice()),
|
2013-03-27 13:42:21 -07:00
|
|
|
volatile: volatile,
|
2013-03-27 14:42:19 -07:00
|
|
|
alignstack: alignstack,
|
2014-09-27 01:33:36 -07:00
|
|
|
dialect: dialect,
|
|
|
|
expn_id: expn_id,
|
2013-03-27 13:42:21 -07:00
|
|
|
}),
|
2013-03-10 22:08:38 -07:00
|
|
|
span: sp
|
2014-09-13 19:06:01 +03:00
|
|
|
}))
|
2013-03-10 22:08:38 -07:00
|
|
|
}
|