1; NOTE: Assertions have been autogenerated by utils/update_llc_test_checks.py 2; RUN: llc < %s -mcpu=generic -mtriple=i386-apple-darwin -verify-machineinstrs -no-integrated-as | FileCheck %s 3 4; There should be no stack manipulations between the inline asm and ret. 5define x86_fp80 @test1() { 6; CHECK-LABEL: test1: 7; CHECK: ## BB#0: 8; CHECK-NEXT: ## InlineAsm Start 9; CHECK-NEXT: fld0 10; CHECK-NEXT: ## InlineAsm End 11; CHECK-NEXT: retl 12 %tmp85 = call x86_fp80 asm sideeffect "fld0", "={st(0)}"() 13 ret x86_fp80 %tmp85 14} 15 16define double @test2() { 17; CHECK-LABEL: test2: 18; CHECK: ## BB#0: 19; CHECK-NEXT: ## InlineAsm Start 20; CHECK-NEXT: fld0 21; CHECK-NEXT: ## InlineAsm End 22; CHECK-NEXT: retl 23 %tmp85 = call double asm sideeffect "fld0", "={st(0)}"() 24 ret double %tmp85 25} 26 27; Setting up argument in st(0) should be a single fld. 28; Asm consumes stack, nothing should be popped. 29define void @test3(x86_fp80 %X) { 30; CHECK-LABEL: test3: 31; CHECK: ## BB#0: 32; CHECK-NEXT: fldt {{[0-9]+}}(%esp) 33; CHECK-NEXT: ## InlineAsm Start 34; CHECK-NEXT: frob 35; CHECK-NEXT: ## InlineAsm End 36; CHECK-NEXT: retl 37 call void asm sideeffect "frob ", "{st(0)},~{st},~{dirflag},~{fpsr},~{flags}"( x86_fp80 %X) 38 ret void 39} 40 41define void @test4(double %X) { 42; CHECK-LABEL: test4: 43; CHECK: ## BB#0: 44; CHECK-NEXT: fldl {{[0-9]+}}(%esp) 45; CHECK-NEXT: ## InlineAsm Start 46; CHECK-NEXT: frob 47; CHECK-NEXT: ## InlineAsm End 48; CHECK-NEXT: retl 49 call void asm sideeffect "frob ", "{st(0)},~{st},~{dirflag},~{fpsr},~{flags}"( double %X) 50 ret void 51} 52 53; Same as test3/4, but using value from fadd. 54; The fadd can be done in xmm or x87 regs - we don't test that. 55define void @test5(double %X) { 56; CHECK-LABEL: test5: 57; CHECK: ## BB#0: 58; CHECK-NEXT: fldl {{[0-9]+}}(%esp) 59; CHECK-NEXT: fadds LCPI4_0 60; CHECK-NEXT: ## InlineAsm Start 61; CHECK-NEXT: frob 62; CHECK-NEXT: ## InlineAsm End 63; CHECK-NEXT: retl 64 %Y = fadd double %X, 123.0 65 call void asm sideeffect "frob ", "{st(0)},~{st},~{dirflag},~{fpsr},~{flags}"( double %Y) 66 ret void 67} 68 69define void @test6(double %A, double %B, double %C, double %D, double %E) nounwind { 70; CHECK-LABEL: test6: 71; CHECK: ## BB#0: ## %entry 72; CHECK-NEXT: fldl {{[0-9]+}}(%esp) 73; CHECK-NEXT: fldl {{[0-9]+}}(%esp) 74; CHECK-NEXT: fldl {{[0-9]+}}(%esp) 75; CHECK-NEXT: fldl {{[0-9]+}}(%esp) 76; CHECK-NEXT: fldl {{[0-9]+}}(%esp) 77; CHECK-NEXT: ## InlineAsm Start 78; CHECK-NEXT: foo %st(0) %st(0) 79; CHECK-NEXT: ## InlineAsm End 80; CHECK-NEXT: fstp %st(0) 81; CHECK-NEXT: ## InlineAsm Start 82; CHECK-NEXT: bar %st(1) %st(0) 83; CHECK-NEXT: ## InlineAsm End 84; CHECK-NEXT: fstp %st(1) 85; CHECK-NEXT: fstp %st(0) 86; CHECK-NEXT: ## InlineAsm Start 87; CHECK-NEXT: baz %st(1) %st(0) 88; CHECK-NEXT: ## InlineAsm End 89; CHECK-NEXT: fstp %st(0) 90; CHECK-NEXT: ## InlineAsm Start 91; CHECK-NEXT: baz %st(0) 92; CHECK-NEXT: ## InlineAsm End 93; CHECK-NEXT: fstp %st(0) 94; CHECK-NEXT: retl 95; CHECK-NEXT: ## -- End function 96entry: 97; Uses the same value twice, should have one fstp after the asm. 98 tail call void asm sideeffect "foo $0 $1", "f,f,~{dirflag},~{fpsr},~{flags}"( double %A, double %A ) nounwind 99; Uses two different values, should be in st(0)/st(1) and both be popped. 100 tail call void asm sideeffect "bar $0 $1", "f,f,~{dirflag},~{fpsr},~{flags}"( double %B, double %C ) nounwind 101; Uses two different values, one of which isn't killed in this asm, it should not be popped after the asm. 102 tail call void asm sideeffect "baz $0 $1", "f,f,~{dirflag},~{fpsr},~{flags}"( double %D, double %E ) nounwind 103; This is the last use of %D, so it should be popped after. 104 tail call void asm sideeffect "baz $0", "f,~{dirflag},~{fpsr},~{flags}"( double %D ) nounwind 105 ret void 106} 107 108; PR4185 109; Passing a non-killed value to asm in {st}. 110; Make sure it is duped before. 111; asm kills st(0), so we shouldn't pop anything 112; A valid alternative would be to remat the constant pool load before each 113; inline asm. 114define void @testPR4185() { 115; CHECK-LABEL: testPR4185: 116; CHECK: ## BB#0: ## %return 117; CHECK-NEXT: flds LCPI6_0 118; CHECK-NEXT: fld %st(0) 119; CHECK-NEXT: ## InlineAsm Start 120; CHECK-NEXT: fistpl %st(0) 121; CHECK-NEXT: ## InlineAsm End 122; CHECK-NEXT: ## InlineAsm Start 123; CHECK-NEXT: fistpl %st(0) 124; CHECK-NEXT: ## InlineAsm End 125; CHECK-NEXT: retl 126return: 127 call void asm sideeffect "fistpl $0", "{st},~{st}"(double 1.000000e+06) 128 call void asm sideeffect "fistpl $0", "{st},~{st}"(double 1.000000e+06) 129 ret void 130} 131 132; Passing a non-killed value through asm in {st}. 133; Make sure it is not duped before. 134; Second asm kills st(0), so we shouldn't pop anything 135; A valid alternative would be to remat the constant pool load before each inline asm. 136define void @testPR4185b() { 137; CHECK-LABEL: testPR4185b: 138; CHECK: ## BB#0: ## %return 139; CHECK-NEXT: flds LCPI7_0 140; CHECK-NEXT: ## InlineAsm Start 141; CHECK-NEXT: fistl %st(0) 142; CHECK-NEXT: ## InlineAsm End 143; CHECK-NEXT: ## InlineAsm Start 144; CHECK-NEXT: fistpl %st(0) 145; CHECK-NEXT: ## InlineAsm End 146; CHECK-NEXT: retl 147return: 148 call void asm sideeffect "fistl $0", "{st}"(double 1.000000e+06) 149 call void asm sideeffect "fistpl $0", "{st},~{st}"(double 1.000000e+06) 150 ret void 151} 152 153; PR4459 154; The return value from ceil must be duped before being consumed by asm. 155define void @testPR4459(x86_fp80 %a) { 156; CHECK-LABEL: testPR4459: 157; CHECK: ## BB#0: ## %entry 158; CHECK-NEXT: subl $28, %esp 159; CHECK-NEXT: Lcfi0: 160; CHECK-NEXT: .cfi_def_cfa_offset 32 161; CHECK-NEXT: fldt {{[0-9]+}}(%esp) 162; CHECK-NEXT: fstpt (%esp) 163; CHECK-NEXT: calll _ceil 164; CHECK-NEXT: fld %st(0) 165; CHECK-NEXT: ## InlineAsm Start 166; CHECK-NEXT: fistpl %st(0) 167; CHECK-NEXT: ## InlineAsm End 168; CHECK-NEXT: fstpt (%esp) 169; CHECK-NEXT: calll _test3 170; CHECK-NEXT: addl $28, %esp 171; CHECK-NEXT: retl 172entry: 173 %0 = call x86_fp80 @ceil(x86_fp80 %a) 174 call void asm sideeffect "fistpl $0", "{st},~{st}"( x86_fp80 %0) 175 call void @test3(x86_fp80 %0 ) 176 ret void 177} 178declare x86_fp80 @ceil(x86_fp80) 179 180; PR4484 181; test1 leaves a value on the stack that is needed after the asm. 182; Load %a from stack after ceil 183; Set up call to test. 184define void @testPR4484(x86_fp80 %a) { 185; CHECK-LABEL: testPR4484: 186; CHECK: ## BB#0: ## %entry 187; CHECK-NEXT: subl $28, %esp 188; CHECK-NEXT: Lcfi1: 189; CHECK-NEXT: .cfi_def_cfa_offset 32 190; CHECK-NEXT: fldt {{[0-9]+}}(%esp) 191; CHECK-NEXT: fstpt {{[0-9]+}}(%esp) ## 10-byte Folded Spill 192; CHECK-NEXT: calll _test1 193; CHECK-NEXT: fldt {{[0-9]+}}(%esp) ## 10-byte Folded Reload 194; CHECK-NEXT: ## InlineAsm Start 195; CHECK-NEXT: fistpl %st(0) 196; CHECK-NEXT: ## InlineAsm End 197; CHECK-NEXT: fstpt (%esp) 198; CHECK-NEXT: calll _test3 199; CHECK-NEXT: addl $28, %esp 200; CHECK-NEXT: retl 201entry: 202 %0 = call x86_fp80 @test1() 203 call void asm sideeffect "fistpl $0", "{st},~{st}"(x86_fp80 %a) 204 call void @test3(x86_fp80 %0) 205 ret void 206} 207 208; PR4485 209define void @testPR4485(x86_fp80* %a) { 210; CHECK-LABEL: testPR4485: 211; CHECK: ## BB#0: ## %entry 212; CHECK-NEXT: movl {{[0-9]+}}(%esp), %eax 213; CHECK-NEXT: fldt (%eax) 214; CHECK-NEXT: flds LCPI10_0 215; CHECK-NEXT: fmul %st(0), %st(1) 216; CHECK-NEXT: flds LCPI10_1 217; CHECK-NEXT: fmul %st(0), %st(2) 218; CHECK-NEXT: fxch %st(2) 219; CHECK-NEXT: ## InlineAsm Start 220; CHECK-NEXT: fistpl %st(0) 221; CHECK-NEXT: ## InlineAsm End 222; CHECK-NEXT: fldt (%eax) 223; CHECK-NEXT: fmulp %st(1) 224; CHECK-NEXT: fmulp %st(1) 225; CHECK-NEXT: ## InlineAsm Start 226; CHECK-NEXT: fistpl %st(0) 227; CHECK-NEXT: ## InlineAsm End 228; CHECK-NEXT: retl 229entry: 230 %0 = load x86_fp80, x86_fp80* %a, align 16 231 %1 = fmul x86_fp80 %0, 0xK4006B400000000000000 232 %2 = fmul x86_fp80 %1, 0xK4012F424000000000000 233 tail call void asm sideeffect "fistpl $0", "{st},~{st}"(x86_fp80 %2) 234 %3 = load x86_fp80, x86_fp80* %a, align 16 235 %4 = fmul x86_fp80 %3, 0xK4006B400000000000000 236 %5 = fmul x86_fp80 %4, 0xK4012F424000000000000 237 tail call void asm sideeffect "fistpl $0", "{st},~{st}"(x86_fp80 %5) 238 ret void 239} 240 241; An input argument in a fixed position is implicitly popped by the asm only if 242; the input argument is tied to an output register, or it is in the clobber list. 243; The clobber list case is tested above. 244; 245; This doesn't implicitly pop the stack: 246; 247; void fist1(long double x, int *p) { 248; asm volatile ("fistl %1" : : "t"(x), "m"(*p)); 249; } 250define void @fist1(x86_fp80 %x, i32* %p) nounwind ssp { 251; CHECK-LABEL: fist1: 252; CHECK: ## BB#0: ## %entry 253; CHECK-NEXT: fldt {{[0-9]+}}(%esp) 254; CHECK-NEXT: movl {{[0-9]+}}(%esp), %eax 255; CHECK-NEXT: ## InlineAsm Start 256; CHECK-NEXT: fistl (%eax) 257; CHECK-NEXT: ## InlineAsm End 258; CHECK-NEXT: fstp %st(0) 259; CHECK-NEXT: retl 260; CHECK-NEXT: ## -- End function 261entry: 262 tail call void asm sideeffect "fistl $1", "{st},*m,~{memory},~{dirflag},~{fpsr},~{flags}"(x86_fp80 %x, i32* %p) nounwind 263 ret void 264} 265 266; Here, the input operand is tied to an output which means that is is 267; implicitly popped (and then the output is implicitly pushed). 268; 269; long double fist2(long double x, int *p) { 270; long double y; 271; asm ("fistl %1" : "=&t"(y) : "0"(x), "m"(*p) : "memory"); 272; return y; 273; } 274define x86_fp80 @fist2(x86_fp80 %x, i32* %p) nounwind ssp { 275; CHECK-LABEL: fist2: 276; CHECK: ## BB#0: ## %entry 277; CHECK-NEXT: fldt {{[0-9]+}}(%esp) 278; CHECK-NEXT: movl {{[0-9]+}}(%esp), %eax 279; CHECK-NEXT: ## InlineAsm Start 280; CHECK-NEXT: fistl (%eax) 281; CHECK-NEXT: ## InlineAsm End 282; CHECK-NEXT: retl 283; CHECK-NEXT: ## -- End function 284entry: 285 %0 = tail call x86_fp80 asm "fistl $2", "=&{st},0,*m,~{memory},~{dirflag},~{fpsr},~{flags}"(x86_fp80 %x, i32* %p) nounwind 286 ret x86_fp80 %0 287} 288 289; An 'f' constraint is never implicitly popped: 290; 291; void fucomp1(long double x, long double y) { 292; asm volatile ("fucomp %1" : : "t"(x), "f"(y) : "st"); 293; } 294define void @fucomp1(x86_fp80 %x, x86_fp80 %y) nounwind ssp { 295; CHECK-LABEL: fucomp1: 296; CHECK: ## BB#0: ## %entry 297; CHECK-NEXT: fldt {{[0-9]+}}(%esp) 298; CHECK-NEXT: fldt {{[0-9]+}}(%esp) 299; CHECK-NEXT: fxch %st(1) 300; CHECK-NEXT: ## InlineAsm Start 301; CHECK-NEXT: fucomp %st(1) 302; CHECK-NEXT: ## InlineAsm End 303; CHECK-NEXT: fstp %st(0) 304; CHECK-NEXT: retl 305; CHECK-NEXT: ## -- End function 306entry: 307 tail call void asm sideeffect "fucomp $1", "{st},f,~{st},~{dirflag},~{fpsr},~{flags}"(x86_fp80 %x, x86_fp80 %y) nounwind 308 ret void 309} 310 311; The 'u' constraint is only popped implicitly when clobbered: 312; 313; void fucomp2(long double x, long double y) { 314; asm volatile ("fucomp %1" : : "t"(x), "u"(y) : "st"); 315; } 316; 317; void fucomp3(long double x, long double y) { 318; asm volatile ("fucompp %1" : : "t"(x), "u"(y) : "st", "st(1)"); 319; } 320; 321define void @fucomp2(x86_fp80 %x, x86_fp80 %y) nounwind ssp { 322; CHECK-LABEL: fucomp2: 323; CHECK: ## BB#0: ## %entry 324; CHECK-NEXT: fldt {{[0-9]+}}(%esp) 325; CHECK-NEXT: fldt {{[0-9]+}}(%esp) 326; CHECK-NEXT: fxch %st(1) 327; CHECK-NEXT: ## InlineAsm Start 328; CHECK-NEXT: fucomp %st(1) 329; CHECK-NEXT: ## InlineAsm End 330; CHECK-NEXT: fstp %st(0) 331; CHECK-NEXT: retl 332; CHECK-NEXT: ## -- End function 333entry: 334 tail call void asm sideeffect "fucomp $1", "{st},{st(1)},~{st},~{dirflag},~{fpsr},~{flags}"(x86_fp80 %x, x86_fp80 %y) nounwind 335 ret void 336} 337 338define void @fucomp3(x86_fp80 %x, x86_fp80 %y) nounwind ssp { 339; CHECK-LABEL: fucomp3: 340; CHECK: ## BB#0: ## %entry 341; CHECK-NEXT: fldt {{[0-9]+}}(%esp) 342; CHECK-NEXT: fldt {{[0-9]+}}(%esp) 343; CHECK-NEXT: fxch %st(1) 344; CHECK-NEXT: ## InlineAsm Start 345; CHECK-NEXT: fucompp %st(1) 346; CHECK-NEXT: ## InlineAsm End 347; CHECK-NEXT: retl 348; CHECK-NEXT: ## -- End function 349entry: 350 tail call void asm sideeffect "fucompp $1", "{st},{st(1)},~{st},~{st(1)},~{dirflag},~{fpsr},~{flags}"(x86_fp80 %x, x86_fp80 %y) nounwind 351 ret void 352} 353 354; One input, two outputs, one dead output. 355%complex = type { float, float } 356define float @sincos1(float %x) nounwind ssp { 357; CHECK-LABEL: sincos1: 358; CHECK: ## BB#0: ## %entry 359; CHECK-NEXT: flds {{[0-9]+}}(%esp) 360; CHECK-NEXT: ## InlineAsm Start 361; CHECK-NEXT: sincos 362; CHECK-NEXT: ## InlineAsm End 363; CHECK-NEXT: fstp %st(1) 364; CHECK-NEXT: retl 365; CHECK-NEXT: ## -- End function 366entry: 367 %0 = tail call %complex asm "sincos", "={st},={st(1)},0,~{dirflag},~{fpsr},~{flags}"(float %x) nounwind 368 %asmresult = extractvalue %complex %0, 0 369 ret float %asmresult 370} 371 372; Same thing, swapped output operands. 373define float @sincos2(float %x) nounwind ssp { 374; CHECK-LABEL: sincos2: 375; CHECK: ## BB#0: ## %entry 376; CHECK-NEXT: flds {{[0-9]+}}(%esp) 377; CHECK-NEXT: ## InlineAsm Start 378; CHECK-NEXT: sincos 379; CHECK-NEXT: ## InlineAsm End 380; CHECK-NEXT: fstp %st(1) 381; CHECK-NEXT: retl 382; CHECK-NEXT: ## -- End function 383entry: 384 %0 = tail call %complex asm "sincos", "={st(1)},={st},1,~{dirflag},~{fpsr},~{flags}"(float %x) nounwind 385 %asmresult = extractvalue %complex %0, 1 386 ret float %asmresult 387} 388 389; Clobber st(0) after it was live-out/dead from the previous asm. 390; Load x, make a copy for the second asm. 391; Discard dead result in st(0), bring x to the top. 392; x is now in st(0) for the second asm 393; Discard both results. 394define float @sincos3(float %x) nounwind ssp { 395; CHECK-LABEL: sincos3: 396; CHECK: ## BB#0: ## %entry 397; CHECK-NEXT: flds {{[0-9]+}}(%esp) 398; CHECK-NEXT: fld %st(0) 399; CHECK-NEXT: ## InlineAsm Start 400; CHECK-NEXT: sincos 401; CHECK-NEXT: ## InlineAsm End 402; CHECK-NEXT: fstp %st(0) 403; CHECK-NEXT: fxch %st(1) 404; CHECK-NEXT: ## InlineAsm Start 405; CHECK-NEXT: sincos 406; CHECK-NEXT: ## InlineAsm End 407; CHECK-NEXT: fstp %st(1) 408; CHECK-NEXT: fstp %st(0) 409; CHECK-NEXT: retl 410; CHECK-NEXT: ## -- End function 411entry: 412 %0 = tail call %complex asm sideeffect "sincos", "={st(1)},={st},1,~{dirflag},~{fpsr},~{flags}"(float %x) nounwind 413 %1 = tail call %complex asm sideeffect "sincos", "={st(1)},={st},1,~{dirflag},~{fpsr},~{flags}"(float %x) nounwind 414 %asmresult = extractvalue %complex %0, 0 415 ret float %asmresult 416} 417 418; Pass the same value in two fixed stack slots. 419define i32 @PR10602() nounwind ssp { 420; CHECK-LABEL: PR10602: 421; CHECK: ## BB#0: ## %entry 422; CHECK-NEXT: flds LCPI19_0 423; CHECK-NEXT: fld %st(0) 424; CHECK-NEXT: fxch %st(1) 425; CHECK-NEXT: ## InlineAsm Start 426; CHECK-NEXT: fcomi %st(1), %st(0); pushf; pop %eax 427; CHECK-NEXT: ## InlineAsm End 428; CHECK-NEXT: fstp %st(0) 429; CHECK-NEXT: fstp %st(0) 430; CHECK-NEXT: retl 431; CHECK-NEXT: ## -- End function 432entry: 433 %0 = tail call i32 asm "fcomi $2, $1; pushf; pop $0", "=r,{st},{st(1)},~{dirflag},~{fpsr},~{flags}"(double 2.000000e+00, double 2.000000e+00) nounwind 434 ret i32 %0 435} 436 437; <rdar://problem/16952634> 438; X87 stackifier asserted when there was an ST register defined by an 439; inline-asm instruction and the ST register was live across another 440; inline-asm instruction. 441; 442; INLINEASM <es:frndint> [sideeffect] [attdialect], $0:[regdef], %ST0<imp-def,tied5>, $1:[reguse tiedto:$0], %ST0<tied3>, $2:[clobber], %EFLAGS<earlyclobber,imp-def,dead> 443; INLINEASM <es:fldcw $0> [sideeffect] [mayload] [attdialect], $0:[mem], %EAX<undef>, 1, %noreg, 0, %noreg, $1:[clobber], %EFLAGS<earlyclobber,imp-def,dead> 444; %FP0<def> = COPY %ST0 445 446%struct.fpu_t = type { [8 x x86_fp80], x86_fp80, %struct.anon1, %struct.anon2, i32, i8, [15 x i8] } 447%struct.anon1 = type { i32, i32, i32 } 448%struct.anon2 = type { i32, i32, i32, i32 } 449 450@fpu = external global %struct.fpu_t, align 16 451 452; Function Attrs: ssp 453define void @test_live_st(i32 %a1) { 454; CHECK-LABEL: test_live_st: 455; CHECK: ## BB#0: ## %entry 456; CHECK-NEXT: subl $12, %esp 457; CHECK-NEXT: Lcfi2: 458; CHECK-NEXT: .cfi_def_cfa_offset 16 459; CHECK-NEXT: fldt (%eax) 460; CHECK-NEXT: cmpl $1, {{[0-9]+}}(%esp) 461; CHECK-NEXT: jne LBB20_2 462; CHECK-NEXT: ## BB#1: ## %sw.bb4.i 463; CHECK-NEXT: ## InlineAsm Start 464; CHECK-NEXT: frndint 465; CHECK-NEXT: ## InlineAsm End 466; CHECK-NEXT: ## InlineAsm Start 467; CHECK-NEXT: fldcw (%eax) 468; CHECK-NEXT: ## InlineAsm End 469; CHECK-NEXT: LBB20_2: ## %_Z5tointRKe.exit 470; CHECK-NEXT: fnstcw {{[0-9]+}}(%esp) 471; CHECK-NEXT: movzwl {{[0-9]+}}(%esp), %eax 472; CHECK-NEXT: movw $3199, {{[0-9]+}}(%esp) ## imm = 0xC7F 473; CHECK-NEXT: fldcw {{[0-9]+}}(%esp) 474; CHECK-NEXT: movw %ax, {{[0-9]+}}(%esp) 475; CHECK-NEXT: fistpl {{[0-9]+}}(%esp) 476; CHECK-NEXT: fldcw {{[0-9]+}}(%esp) 477; CHECK-NEXT: movl {{[0-9]+}}(%esp), %eax 478; CHECK-NEXT: movl %eax, {{[0-9]+}}(%esp) 479; CHECK-NEXT: fildl {{[0-9]+}}(%esp) 480; CHECK-NEXT: movl L_fpu$non_lazy_ptr, %eax 481; CHECK-NEXT: fstpt 128(%eax) 482; CHECK-NEXT: addl $12, %esp 483; CHECK-NEXT: retl 484entry: 485 %0 = load x86_fp80, x86_fp80* undef, align 16 486 %cond = icmp eq i32 %a1, 1 487 br i1 %cond, label %sw.bb4.i, label %_Z5tointRKe.exit 488 489sw.bb4.i: 490 %1 = call x86_fp80 asm sideeffect "frndint", "={st},0,~{dirflag},~{fpsr},~{flags}"(x86_fp80 %0) 491 call void asm sideeffect "fldcw $0", "*m,~{dirflag},~{fpsr},~{flags}"(i32* undef) 492 br label %_Z5tointRKe.exit 493 494_Z5tointRKe.exit: 495 %result.0.i = phi x86_fp80 [ %1, %sw.bb4.i ], [ %0, %entry ] 496 %conv.i1814 = fptosi x86_fp80 %result.0.i to i32 497 %conv626 = sitofp i32 %conv.i1814 to x86_fp80 498 store x86_fp80 %conv626, x86_fp80* getelementptr inbounds (%struct.fpu_t, %struct.fpu_t* @fpu, i32 0, i32 1) 499 br label %return 500 501return: 502 ret void 503} 504 505; Check that x87 stackifier is correctly rewriting FP registers to ST registers. 506define double @test_operand_rewrite() { 507; CHECK-LABEL: test_operand_rewrite: 508; CHECK: ## BB#0: ## %entry 509; CHECK-NEXT: ## InlineAsm Start 510; CHECK-NEXT: foo %st(0), %st(1) 511; CHECK-NEXT: ## InlineAsm End 512; CHECK-NEXT: fsubp %st(1) 513; CHECK-NEXT: retl 514entry: 515 %0 = tail call { double, double } asm sideeffect "foo $0, $1", "={st},={st(1)},~{dirflag},~{fpsr},~{flags}"() 516 %asmresult = extractvalue { double, double } %0, 0 517 %asmresult1 = extractvalue { double, double } %0, 1 518 %sub = fsub double %asmresult, %asmresult1 519 ret double %sub 520} 521