Skip to content

Commit d0443bb

Browse files
committed
Add a "raw" option for asm! which ignores format string specifiers
1 parent 1e13a9b commit d0443bb

File tree

8 files changed

+48
-23
lines changed

8 files changed

+48
-23
lines changed

compiler/rustc_ast/src/ast.rs

+1
Original file line numberDiff line numberDiff line change
@@ -1933,6 +1933,7 @@ bitflags::bitflags! {
19331933
const NORETURN = 1 << 4;
19341934
const NOSTACK = 1 << 5;
19351935
const ATT_SYNTAX = 1 << 6;
1936+
const RAW = 1 << 7;
19361937
}
19371938
}
19381939

compiler/rustc_ast_pretty/src/pprust/state.rs

+3
Original file line numberDiff line numberDiff line change
@@ -2274,6 +2274,9 @@ impl<'a> State<'a> {
22742274
if opts.contains(InlineAsmOptions::ATT_SYNTAX) {
22752275
options.push("att_syntax");
22762276
}
2277+
if opts.contains(InlineAsmOptions::RAW) {
2278+
options.push("raw");
2279+
}
22772280
s.commasep(Inconsistent, &options, |s, &opt| {
22782281
s.word(opt);
22792282
});

compiler/rustc_builtin_macros/src/asm.rs

+10
Original file line numberDiff line numberDiff line change
@@ -356,6 +356,8 @@ fn parse_options<'a>(
356356
try_set_option(p, args, sym::nostack, ast::InlineAsmOptions::NOSTACK);
357357
} else if p.eat_keyword(sym::att_syntax) {
358358
try_set_option(p, args, sym::att_syntax, ast::InlineAsmOptions::ATT_SYNTAX);
359+
} else if p.eat_keyword(kw::Raw) {
360+
try_set_option(p, args, kw::Raw, ast::InlineAsmOptions::RAW);
359361
} else {
360362
return p.unexpected();
361363
}
@@ -467,6 +469,14 @@ fn expand_preparsed_asm(ecx: &mut ExtCtxt<'_>, args: AsmArgs) -> Option<ast::Inl
467469
}
468470
}
469471

472+
// Don't treat raw asm as a format string.
473+
if args.options.contains(ast::InlineAsmOptions::RAW) {
474+
template.push(ast::InlineAsmTemplatePiece::String(template_str.to_string()));
475+
let template_num_lines = 1 + template_str.matches('\n').count();
476+
line_spans.extend(std::iter::repeat(template_sp).take(template_num_lines));
477+
continue;
478+
}
479+
470480
let mut parser = parse::Parser::new(
471481
template_str,
472482
str_style,

compiler/rustc_hir_pretty/src/lib.rs

+3
Original file line numberDiff line numberDiff line change
@@ -1446,6 +1446,9 @@ impl<'a> State<'a> {
14461446
if opts.contains(ast::InlineAsmOptions::ATT_SYNTAX) {
14471447
options.push("att_syntax");
14481448
}
1449+
if opts.contains(ast::InlineAsmOptions::RAW) {
1450+
options.push("raw");
1451+
}
14491452
s.commasep(Inconsistent, &options, |s, &opt| {
14501453
s.word(opt);
14511454
});

src/doc/unstable-book/src/library-features/asm.md

+2-1
Original file line numberDiff line numberDiff line change
@@ -455,7 +455,7 @@ reg_spec := <register class> / "<explicit register>"
455455
operand_expr := expr / "_" / expr "=>" expr / expr "=>" "_"
456456
reg_operand := dir_spec "(" reg_spec ")" operand_expr
457457
operand := reg_operand / "const" const_expr / "sym" path
458-
option := "pure" / "nomem" / "readonly" / "preserves_flags" / "noreturn" / "nostack" / "att_syntax"
458+
option := "pure" / "nomem" / "readonly" / "preserves_flags" / "noreturn" / "nostack" / "att_syntax" / "raw"
459459
options := "options(" option *["," option] [","] ")"
460460
asm := "asm!(" format_string *("," format_string) *("," [ident "="] operand) ["," options] [","] ")"
461461
```
@@ -775,6 +775,7 @@ Currently the following options are defined:
775775
- `noreturn`: The `asm` block never returns, and its return type is defined as `!` (never). Behavior is undefined if execution falls through past the end of the asm code. A `noreturn` asm block behaves just like a function which doesn't return; notably, local variables in scope are not dropped before it is invoked.
776776
- `nostack`: The `asm` block does not push data to the stack, or write to the stack red-zone (if supported by the target). If this option is *not* used then the stack pointer is guaranteed to be suitably aligned (according to the target ABI) for a function call.
777777
- `att_syntax`: This option is only valid on x86, and causes the assembler to use the `.att_syntax prefix` mode of the GNU assembler. Register operands are substituted in with a leading `%`.
778+
- `raw`: This causes the template string to be parsed as a raw assembly string, with no special handling for `{` and `}`. This is primarily useful when including raw assembly code from an external file using `include_str!`.
778779

779780
The compiler performs some additional checks on options:
780781
- The `nomem` and `readonly` options are mutually exclusive: it is a compile-time error to specify both.

src/test/codegen/asm-options.rs

+7
Original file line numberDiff line numberDiff line change
@@ -94,3 +94,10 @@ pub unsafe fn dont_remove_nonpure() {
9494
asm!("", options(nomem));
9595
asm!("", options(readonly));
9696
}
97+
98+
// CHECK-LABEL: @raw
99+
// CHECK: call void asm sideeffect inteldialect "{} {}", ""()
100+
#[no_mangle]
101+
pub unsafe fn raw() {
102+
asm!("{} {}", options(nostack, nomem, preserves_flags, raw));
103+
}

src/test/ui/asm/bad-options.stderr

+12-12
Original file line numberDiff line numberDiff line change
@@ -28,41 +28,41 @@ error: asm outputs are not allowed with the `noreturn` option
2828
LL | asm!("{}", out(reg) foo, options(noreturn));
2929
| ^^^^^^^^^^^^
3030

31-
error: expected one of `)` or `att_syntax`, found `nomem`
31+
error: expected one of `)`, `att_syntax`, or `raw`, found `nomem`
3232
--> $DIR/bad-options.rs:20:25
3333
|
3434
LL | global_asm!("", options(nomem));
35-
| ^^^^^ expected one of `)` or `att_syntax`
35+
| ^^^^^ expected one of `)`, `att_syntax`, or `raw`
3636

37-
error: expected one of `)` or `att_syntax`, found `readonly`
37+
error: expected one of `)`, `att_syntax`, or `raw`, found `readonly`
3838
--> $DIR/bad-options.rs:22:25
3939
|
4040
LL | global_asm!("", options(readonly));
41-
| ^^^^^^^^ expected one of `)` or `att_syntax`
41+
| ^^^^^^^^ expected one of `)`, `att_syntax`, or `raw`
4242

43-
error: expected one of `)` or `att_syntax`, found `noreturn`
43+
error: expected one of `)`, `att_syntax`, or `raw`, found `noreturn`
4444
--> $DIR/bad-options.rs:24:25
4545
|
4646
LL | global_asm!("", options(noreturn));
47-
| ^^^^^^^^ expected one of `)` or `att_syntax`
47+
| ^^^^^^^^ expected one of `)`, `att_syntax`, or `raw`
4848

49-
error: expected one of `)` or `att_syntax`, found `pure`
49+
error: expected one of `)`, `att_syntax`, or `raw`, found `pure`
5050
--> $DIR/bad-options.rs:26:25
5151
|
5252
LL | global_asm!("", options(pure));
53-
| ^^^^ expected one of `)` or `att_syntax`
53+
| ^^^^ expected one of `)`, `att_syntax`, or `raw`
5454

55-
error: expected one of `)` or `att_syntax`, found `nostack`
55+
error: expected one of `)`, `att_syntax`, or `raw`, found `nostack`
5656
--> $DIR/bad-options.rs:28:25
5757
|
5858
LL | global_asm!("", options(nostack));
59-
| ^^^^^^^ expected one of `)` or `att_syntax`
59+
| ^^^^^^^ expected one of `)`, `att_syntax`, or `raw`
6060

61-
error: expected one of `)` or `att_syntax`, found `preserves_flags`
61+
error: expected one of `)`, `att_syntax`, or `raw`, found `preserves_flags`
6262
--> $DIR/bad-options.rs:30:25
6363
|
6464
LL | global_asm!("", options(preserves_flags));
65-
| ^^^^^^^^^^^^^^^ expected one of `)` or `att_syntax`
65+
| ^^^^^^^^^^^^^^^ expected one of `)`, `att_syntax`, or `raw`
6666

6767
error: aborting due to 11 previous errors
6868

src/test/ui/asm/parse-error.stderr

+10-10
Original file line numberDiff line numberDiff line change
@@ -64,23 +64,23 @@ error: argument to `sym` must be a path expression
6464
LL | asm!("{}", sym foo + bar);
6565
| ^^^^^^^^^
6666

67-
error: expected one of `)`, `att_syntax`, `nomem`, `noreturn`, `nostack`, `preserves_flags`, `pure`, or `readonly`, found `foo`
67+
error: expected one of `)`, `att_syntax`, `nomem`, `noreturn`, `nostack`, `preserves_flags`, `pure`, `raw`, or `readonly`, found `foo`
6868
--> $DIR/parse-error.rs:31:26
6969
|
7070
LL | asm!("", options(foo));
71-
| ^^^ expected one of 8 possible tokens
71+
| ^^^ expected one of 9 possible tokens
7272

7373
error: expected one of `)` or `,`, found `foo`
7474
--> $DIR/parse-error.rs:33:32
7575
|
7676
LL | asm!("", options(nomem foo));
7777
| ^^^ expected one of `)` or `,`
7878

79-
error: expected one of `)`, `att_syntax`, `nomem`, `noreturn`, `nostack`, `preserves_flags`, `pure`, or `readonly`, found `foo`
79+
error: expected one of `)`, `att_syntax`, `nomem`, `noreturn`, `nostack`, `preserves_flags`, `pure`, `raw`, or `readonly`, found `foo`
8080
--> $DIR/parse-error.rs:35:33
8181
|
8282
LL | asm!("", options(nomem, foo));
83-
| ^^^ expected one of 8 possible tokens
83+
| ^^^ expected one of 9 possible tokens
8484

8585
error: arguments are not allowed after options
8686
--> $DIR/parse-error.rs:37:31
@@ -200,23 +200,23 @@ error: expected one of `,`, `.`, `?`, or an operator, found `FOO`
200200
LL | global_asm!("{}", const(reg) FOO);
201201
| ^^^ expected one of `,`, `.`, `?`, or an operator
202202

203-
error: expected one of `)` or `att_syntax`, found `FOO`
203+
error: expected one of `)`, `att_syntax`, or `raw`, found `FOO`
204204
--> $DIR/parse-error.rs:81:25
205205
|
206206
LL | global_asm!("", options(FOO));
207-
| ^^^ expected one of `)` or `att_syntax`
207+
| ^^^ expected one of `)`, `att_syntax`, or `raw`
208208

209-
error: expected one of `)` or `att_syntax`, found `nomem`
209+
error: expected one of `)`, `att_syntax`, or `raw`, found `nomem`
210210
--> $DIR/parse-error.rs:83:25
211211
|
212212
LL | global_asm!("", options(nomem FOO));
213-
| ^^^^^ expected one of `)` or `att_syntax`
213+
| ^^^^^ expected one of `)`, `att_syntax`, or `raw`
214214

215-
error: expected one of `)` or `att_syntax`, found `nomem`
215+
error: expected one of `)`, `att_syntax`, or `raw`, found `nomem`
216216
--> $DIR/parse-error.rs:85:25
217217
|
218218
LL | global_asm!("", options(nomem, FOO));
219-
| ^^^^^ expected one of `)` or `att_syntax`
219+
| ^^^^^ expected one of `)`, `att_syntax`, or `raw`
220220

221221
error: arguments are not allowed after options
222222
--> $DIR/parse-error.rs:87:30

0 commit comments

Comments
 (0)