@@ -107,7 +107,7 @@ enum ConstraintOrRegister {
107
107
108
108
109
109
impl < ' a , ' gcc , ' tcx > AsmBuilderMethods < ' tcx > for Builder < ' a , ' gcc , ' tcx > {
110
- fn codegen_inline_asm ( & mut self , template : & [ InlineAsmTemplatePiece ] , rust_operands : & [ InlineAsmOperandRef < ' tcx , Self > ] , options : InlineAsmOptions , span : & [ Span ] , instance : Instance < ' _ > , _dest_catch_funclet : Option < ( Self :: BasicBlock , Self :: BasicBlock , Option < & Self :: Funclet > ) > ) {
110
+ fn codegen_inline_asm ( & mut self , template : & [ InlineAsmTemplatePiece ] , rust_operands : & [ InlineAsmOperandRef < ' tcx , Self > ] , options : InlineAsmOptions , span : & [ Span ] , instance : Instance < ' _ > , dest : Option < Self :: BasicBlock > , _catch_funclet : Option < ( Self :: BasicBlock , Option < & Self :: Funclet > ) > ) {
111
111
if options. contains ( InlineAsmOptions :: MAY_UNWIND ) {
112
112
self . sess ( ) . dcx ( )
113
113
. create_err ( UnwindingInlineAsm { span : span[ 0 ] } )
@@ -126,6 +126,10 @@ impl<'a, 'gcc, 'tcx> AsmBuilderMethods<'tcx> for Builder<'a, 'gcc, 'tcx> {
126
126
// added to `outputs.len()`
127
127
let mut inputs = vec ! [ ] ;
128
128
129
+ // GCC index of a label equals its position in the array added to
130
+ // `outputs.len() + inputs.len()`.
131
+ let mut labels = vec ! [ ] ;
132
+
129
133
// Clobbers collected from `out("explicit register") _` and `inout("expl_reg") var => _`
130
134
let mut clobbers = vec ! [ ] ;
131
135
@@ -269,6 +273,10 @@ impl<'a, 'gcc, 'tcx> AsmBuilderMethods<'tcx> for Builder<'a, 'gcc, 'tcx> {
269
273
// some targets to add a leading underscore (Mach-O).
270
274
constants_len += self . tcx . symbol_name ( Instance :: mono ( self . tcx , def_id) ) . name . len ( ) ;
271
275
}
276
+
277
+ InlineAsmOperandRef :: Label { label } => {
278
+ labels. push ( label) ;
279
+ }
272
280
}
273
281
}
274
282
@@ -368,6 +376,10 @@ impl<'a, 'gcc, 'tcx> AsmBuilderMethods<'tcx> for Builder<'a, 'gcc, 'tcx> {
368
376
InlineAsmOperandRef :: Const { .. } => {
369
377
// processed in the previous pass
370
378
}
379
+
380
+ InlineAsmOperandRef :: Label { .. } => {
381
+ // processed in the previous pass
382
+ }
371
383
}
372
384
}
373
385
@@ -454,6 +466,14 @@ impl<'a, 'gcc, 'tcx> AsmBuilderMethods<'tcx> for Builder<'a, 'gcc, 'tcx> {
454
466
InlineAsmOperandRef :: Const { ref string } => {
455
467
template_str. push_str ( string) ;
456
468
}
469
+
470
+ InlineAsmOperandRef :: Label { label } => {
471
+ let label_gcc_index = labels. iter ( )
472
+ . position ( |& l| l == label)
473
+ . expect ( "wrong rust index" ) ;
474
+ let gcc_index = label_gcc_index + outputs. len ( ) + inputs. len ( ) ;
475
+ push_to_template ( Some ( 'l' ) , gcc_index) ;
476
+ }
457
477
}
458
478
}
459
479
}
@@ -466,7 +486,12 @@ impl<'a, 'gcc, 'tcx> AsmBuilderMethods<'tcx> for Builder<'a, 'gcc, 'tcx> {
466
486
// 4. Generate Extended Asm block
467
487
468
488
let block = self . llbb ( ) ;
469
- let extended_asm = block. add_extended_asm ( None , & template_str) ;
489
+ let extended_asm = if let Some ( dest) = dest {
490
+ assert ! ( !labels. is_empty( ) ) ;
491
+ block. end_with_extended_asm_goto ( None , & template_str, & labels, Some ( dest) )
492
+ } else {
493
+ block. add_extended_asm ( None , & template_str)
494
+ } ;
470
495
471
496
for op in & outputs {
472
497
extended_asm. add_output_operand ( None , & op. to_constraint ( ) , op. tmp_var ) ;
@@ -494,7 +519,7 @@ impl<'a, 'gcc, 'tcx> AsmBuilderMethods<'tcx> for Builder<'a, 'gcc, 'tcx> {
494
519
if !options. contains ( InlineAsmOptions :: NOSTACK ) {
495
520
// TODO(@Commeownist): figure out how to align stack
496
521
}
497
- if options. contains ( InlineAsmOptions :: NORETURN ) {
522
+ if dest . is_none ( ) && options. contains ( InlineAsmOptions :: NORETURN ) {
498
523
let builtin_unreachable = self . context . get_builtin_function ( "__builtin_unreachable" ) ;
499
524
let builtin_unreachable: RValue < ' gcc > = unsafe { std:: mem:: transmute ( builtin_unreachable) } ;
500
525
self . call ( self . type_void ( ) , None , None , builtin_unreachable, & [ ] , None ) ;
0 commit comments