1; NOTE: Assertions have been autogenerated by utils/update_llc_test_checks.py 2; RUN: llc < %s -disable-wasm-fallthrough-return-opt -wasm-keep-registers | FileCheck %s 3 4; Test a subset of compiler-rt/libm libcalls expected to be emitted by the wasm backend 5 6target triple = "wasm32-unknown-unknown" 7 8declare fp128 @llvm.sqrt.f128(fp128) 9declare fp128 @llvm.floor.f128(fp128) 10declare fp128 @llvm.trunc.f128(fp128) 11declare fp128 @llvm.nearbyint.f128(fp128) 12declare fp128 @llvm.pow.f128(fp128, fp128) 13declare fp128 @llvm.powi.f128.i32(fp128, i32) 14 15declare double @llvm.tan.f64(double) 16declare double @llvm.cos.f64(double) 17declare double @llvm.log10.f64(double) 18declare double @llvm.pow.f64(double, double) 19declare double @llvm.powi.f64.i32(double, i32) 20declare double @llvm.log.f64(double) 21declare double @llvm.exp.f64(double) 22declare double @llvm.exp10.f64(double) 23declare double @llvm.ldexp.f64.i32(double, i32) 24declare {double, i32} @llvm.frexp.f64.i32(double) 25declare i32 @llvm.lround(double) 26 27declare void @escape_value(i32) 28 29define fp128 @fp128libcalls(fp128 %x, fp128 %y, i32 %z) { 30 ; compiler-rt call 31; CHECK-LABEL: fp128libcalls: 32; CHECK: .functype fp128libcalls (i32, i64, i64, i64, i64, i32) -> () 33; CHECK-NEXT: .local i32 34; CHECK-NEXT: # %bb.0: 35; CHECK-NEXT: global.get $push18=, __stack_pointer 36; CHECK-NEXT: i32.const $push19=, 144 37; CHECK-NEXT: i32.sub $push39=, $pop18, $pop19 38; CHECK-NEXT: local.tee $push38=, 6, $pop39 39; CHECK-NEXT: global.set __stack_pointer, $pop38 40; CHECK-NEXT: local.get $push40=, 6 41; CHECK-NEXT: i32.const $push36=, 128 42; CHECK-NEXT: i32.add $push37=, $pop40, $pop36 43; CHECK-NEXT: local.get $push44=, 1 44; CHECK-NEXT: local.get $push43=, 2 45; CHECK-NEXT: local.get $push42=, 3 46; CHECK-NEXT: local.get $push41=, 4 47; CHECK-NEXT: call __addtf3, $pop37, $pop44, $pop43, $pop42, $pop41 48; CHECK-NEXT: local.get $push45=, 6 49; CHECK-NEXT: i32.const $push34=, 112 50; CHECK-NEXT: i32.add $push35=, $pop45, $pop34 51; CHECK-NEXT: local.get $push46=, 6 52; CHECK-NEXT: i64.load $push1=, 128($pop46) 53; CHECK-NEXT: local.get $push47=, 6 54; CHECK-NEXT: i64.load $push0=, 136($pop47) 55; CHECK-NEXT: local.get $push49=, 3 56; CHECK-NEXT: local.get $push48=, 4 57; CHECK-NEXT: call __multf3, $pop35, $pop1, $pop0, $pop49, $pop48 58; CHECK-NEXT: local.get $push50=, 6 59; CHECK-NEXT: i32.const $push32=, 96 60; CHECK-NEXT: i32.add $push33=, $pop50, $pop32 61; CHECK-NEXT: local.get $push51=, 6 62; CHECK-NEXT: i64.load $push3=, 112($pop51) 63; CHECK-NEXT: local.get $push52=, 6 64; CHECK-NEXT: i64.load $push2=, 120($pop52) 65; CHECK-NEXT: local.get $push54=, 3 66; CHECK-NEXT: local.get $push53=, 4 67; CHECK-NEXT: call __divtf3, $pop33, $pop3, $pop2, $pop54, $pop53 68; CHECK-NEXT: local.get $push55=, 6 69; CHECK-NEXT: i32.const $push30=, 80 70; CHECK-NEXT: i32.add $push31=, $pop55, $pop30 71; CHECK-NEXT: local.get $push56=, 6 72; CHECK-NEXT: i64.load $push5=, 96($pop56) 73; CHECK-NEXT: local.get $push57=, 6 74; CHECK-NEXT: i64.load $push4=, 104($pop57) 75; CHECK-NEXT: call sqrtl, $pop31, $pop5, $pop4 76; CHECK-NEXT: local.get $push58=, 6 77; CHECK-NEXT: i32.const $push28=, 64 78; CHECK-NEXT: i32.add $push29=, $pop58, $pop28 79; CHECK-NEXT: local.get $push59=, 6 80; CHECK-NEXT: i64.load $push7=, 80($pop59) 81; CHECK-NEXT: local.get $push60=, 6 82; CHECK-NEXT: i64.load $push6=, 88($pop60) 83; CHECK-NEXT: call floorl, $pop29, $pop7, $pop6 84; CHECK-NEXT: local.get $push61=, 6 85; CHECK-NEXT: i32.const $push26=, 48 86; CHECK-NEXT: i32.add $push27=, $pop61, $pop26 87; CHECK-NEXT: local.get $push62=, 6 88; CHECK-NEXT: i64.load $push9=, 64($pop62) 89; CHECK-NEXT: local.get $push63=, 6 90; CHECK-NEXT: i64.load $push8=, 72($pop63) 91; CHECK-NEXT: local.get $push65=, 3 92; CHECK-NEXT: local.get $push64=, 4 93; CHECK-NEXT: call powl, $pop27, $pop9, $pop8, $pop65, $pop64 94; CHECK-NEXT: local.get $push66=, 6 95; CHECK-NEXT: i32.const $push24=, 32 96; CHECK-NEXT: i32.add $push25=, $pop66, $pop24 97; CHECK-NEXT: local.get $push67=, 6 98; CHECK-NEXT: i64.load $push11=, 48($pop67) 99; CHECK-NEXT: local.get $push68=, 6 100; CHECK-NEXT: i64.load $push10=, 56($pop68) 101; CHECK-NEXT: local.get $push69=, 5 102; CHECK-NEXT: call __powitf2, $pop25, $pop11, $pop10, $pop69 103; CHECK-NEXT: local.get $push70=, 6 104; CHECK-NEXT: i32.const $push22=, 16 105; CHECK-NEXT: i32.add $push23=, $pop70, $pop22 106; CHECK-NEXT: local.get $push71=, 6 107; CHECK-NEXT: i64.load $push13=, 32($pop71) 108; CHECK-NEXT: local.get $push72=, 6 109; CHECK-NEXT: i64.load $push12=, 40($pop72) 110; CHECK-NEXT: call truncl, $pop23, $pop13, $pop12 111; CHECK-NEXT: local.get $push75=, 6 112; CHECK-NEXT: local.get $push73=, 6 113; CHECK-NEXT: i64.load $push15=, 16($pop73) 114; CHECK-NEXT: local.get $push74=, 6 115; CHECK-NEXT: i64.load $push14=, 24($pop74) 116; CHECK-NEXT: call nearbyintl, $pop75, $pop15, $pop14 117; CHECK-NEXT: local.get $push77=, 0 118; CHECK-NEXT: local.get $push76=, 6 119; CHECK-NEXT: i64.load $push16=, 8($pop76) 120; CHECK-NEXT: i64.store 8($pop77), $pop16 121; CHECK-NEXT: local.get $push79=, 0 122; CHECK-NEXT: local.get $push78=, 6 123; CHECK-NEXT: i64.load $push17=, 0($pop78) 124; CHECK-NEXT: i64.store 0($pop79), $pop17 125; CHECK-NEXT: local.get $push80=, 6 126; CHECK-NEXT: i32.const $push20=, 144 127; CHECK-NEXT: i32.add $push21=, $pop80, $pop20 128; CHECK-NEXT: global.set __stack_pointer, $pop21 129; CHECK-NEXT: return 130 %a = fadd fp128 %x, %y 131 %b = fmul fp128 %a, %y 132 %c = fdiv fp128 %b, %y 133 ; libm calls 134 %d = call fp128 @llvm.sqrt.f128(fp128 %c) 135 %e = call fp128 @llvm.floor.f128(fp128 %d) 136 %f = call fp128 @llvm.pow.f128(fp128 %e, fp128 %y) 137 %g = call fp128 @llvm.powi.f128.i32(fp128 %f, i32 %z) 138 %h = call fp128 @llvm.trunc.f128(fp128 %g) 139 %i = call fp128 @llvm.nearbyint.f128(fp128 %h) 140 ret fp128 %i 141} 142 143define i128 @i128libcalls(i128 %x, i128 %y) { 144 ; Basic ops should be expanded 145; CHECK-LABEL: i128libcalls: 146; CHECK: .functype i128libcalls (i32, i64, i64, i64, i64) -> () 147; CHECK-NEXT: .local i32, i64 148; CHECK-NEXT: # %bb.0: 149; CHECK-NEXT: global.get $push8=, __stack_pointer 150; CHECK-NEXT: i32.const $push9=, 32 151; CHECK-NEXT: i32.sub $push17=, $pop8, $pop9 152; CHECK-NEXT: local.tee $push16=, 5, $pop17 153; CHECK-NEXT: global.set __stack_pointer, $pop16 154; CHECK-NEXT: local.get $push18=, 5 155; CHECK-NEXT: i32.const $push12=, 16 156; CHECK-NEXT: i32.add $push13=, $pop18, $pop12 157; CHECK-NEXT: local.get $push20=, 1 158; CHECK-NEXT: local.get $push19=, 3 159; CHECK-NEXT: i64.add $push15=, $pop20, $pop19 160; CHECK-NEXT: local.tee $push14=, 6, $pop15 161; CHECK-NEXT: local.get $push22=, 2 162; CHECK-NEXT: local.get $push21=, 4 163; CHECK-NEXT: i64.add $push0=, $pop22, $pop21 164; CHECK-NEXT: local.get $push24=, 6 165; CHECK-NEXT: local.get $push23=, 1 166; CHECK-NEXT: i64.lt_u $push1=, $pop24, $pop23 167; CHECK-NEXT: i64.extend_i32_u $push2=, $pop1 168; CHECK-NEXT: i64.add $push3=, $pop0, $pop2 169; CHECK-NEXT: local.get $push26=, 3 170; CHECK-NEXT: local.get $push25=, 4 171; CHECK-NEXT: call __multi3, $pop13, $pop14, $pop3, $pop26, $pop25 172; CHECK-NEXT: local.get $push31=, 5 173; CHECK-NEXT: local.get $push27=, 5 174; CHECK-NEXT: i64.load $push5=, 16($pop27) 175; CHECK-NEXT: local.get $push28=, 5 176; CHECK-NEXT: i64.load $push4=, 24($pop28) 177; CHECK-NEXT: local.get $push30=, 3 178; CHECK-NEXT: local.get $push29=, 4 179; CHECK-NEXT: call __umodti3, $pop31, $pop5, $pop4, $pop30, $pop29 180; CHECK-NEXT: local.get $push33=, 0 181; CHECK-NEXT: local.get $push32=, 5 182; CHECK-NEXT: i64.load $push6=, 8($pop32) 183; CHECK-NEXT: i64.store 8($pop33), $pop6 184; CHECK-NEXT: local.get $push35=, 0 185; CHECK-NEXT: local.get $push34=, 5 186; CHECK-NEXT: i64.load $push7=, 0($pop34) 187; CHECK-NEXT: i64.store 0($pop35), $pop7 188; CHECK-NEXT: local.get $push36=, 5 189; CHECK-NEXT: i32.const $push10=, 32 190; CHECK-NEXT: i32.add $push11=, $pop36, $pop10 191; CHECK-NEXT: global.set __stack_pointer, $pop11 192; CHECK-NEXT: return 193 %a = add i128 %x, %y 194 %b = mul i128 %a, %y 195 %c = urem i128 %b, %y 196 ret i128 %c 197} 198 199define double @f64libcalls(double %x, double %y, i32 %z) { 200; CHECK-LABEL: f64libcalls: 201; CHECK: .functype f64libcalls (f64, f64, i32) -> (f64) 202; CHECK-NEXT: .local i32 203; CHECK-NEXT: # %bb.0: 204; CHECK-NEXT: global.get $push12=, __stack_pointer 205; CHECK-NEXT: i32.const $push13=, 16 206; CHECK-NEXT: i32.sub $push19=, $pop12, $pop13 207; CHECK-NEXT: local.tee $push18=, 3, $pop19 208; CHECK-NEXT: global.set __stack_pointer, $pop18 209; CHECK-NEXT: local.get $push23=, 0 210; CHECK-NEXT: local.get $push20=, 0 211; CHECK-NEXT: call $push0=, tan, $pop20 212; CHECK-NEXT: call $push1=, cos, $pop0 213; CHECK-NEXT: call $push2=, log10, $pop1 214; CHECK-NEXT: local.get $push21=, 1 215; CHECK-NEXT: call $push3=, pow, $pop2, $pop21 216; CHECK-NEXT: local.get $push22=, 2 217; CHECK-NEXT: call $push4=, __powidf2, $pop3, $pop22 218; CHECK-NEXT: call $push5=, log, $pop4 219; CHECK-NEXT: call $push6=, exp, $pop5 220; CHECK-NEXT: call $push7=, exp10, $pop6 221; CHECK-NEXT: call $push8=, cbrt, $pop7 222; CHECK-NEXT: call $push9=, lround, $pop8 223; CHECK-NEXT: call $push10=, ldexp, $pop23, $pop9 224; CHECK-NEXT: local.get $push24=, 3 225; CHECK-NEXT: i32.const $push16=, 12 226; CHECK-NEXT: i32.add $push17=, $pop24, $pop16 227; CHECK-NEXT: call $push25=, frexp, $pop10, $pop17 228; CHECK-NEXT: local.set 0, $pop25 229; CHECK-NEXT: local.get $push26=, 3 230; CHECK-NEXT: i32.load $push11=, 12($pop26) 231; CHECK-NEXT: call escape_value, $pop11 232; CHECK-NEXT: local.get $push27=, 3 233; CHECK-NEXT: i32.const $push14=, 16 234; CHECK-NEXT: i32.add $push15=, $pop27, $pop14 235; CHECK-NEXT: global.set __stack_pointer, $pop15 236; CHECK-NEXT: local.get $push28=, 0 237; CHECK-NEXT: return $pop28 238 239 240 %k = call double @llvm.tan.f64(double %x) 241 %a = call double @llvm.cos.f64(double %k) 242 %b = call double @llvm.log10.f64(double %a) 243 %c = call double @llvm.pow.f64(double %b, double %y) 244 %d = call double @llvm.powi.f64.i32(double %c, i32 %z) 245 %e = call double @llvm.log.f64(double %d) 246 %f = call double @llvm.exp.f64(double %e) 247 %g = call double @llvm.exp10.f64(double %f) 248 %h = call fast double @llvm.pow.f64(double %g, double 0x3FD5555555555555) 249 %i = call i32 @llvm.lround(double %h) 250 %j = call double @llvm.ldexp.f64.i32(double %x, i32 %i); 251 %result = call {double, i32} @llvm.frexp.f64.i32(double %j) 252 %result.0 = extractvalue { double, i32 } %result, 0 253 %result.1 = extractvalue { double, i32 } %result, 1 254 call void @escape_value(i32 %result.1) 255 ret double %result.0 256} 257 258; fcmp ord and unord (RTLIB::O_F32 / RTLIB::UO_F32 etc) are a special case (see 259; comment in WebAssemblyRunimeLibcallSignatures.cpp) so check them separately. 260; no libcalls are needed for f32 and f64 261 262define i1 @unordd(double %x, double %y) { 263; CHECK-LABEL: unordd: 264; CHECK: .functype unordd (f64, f64) -> (i32) 265; CHECK-NEXT: # %bb.0: 266; CHECK-NEXT: local.get $push8=, 0 267; CHECK-NEXT: local.get $push7=, 0 268; CHECK-NEXT: f64.ne $push4=, $pop8, $pop7 269; CHECK-NEXT: local.get $push10=, 1 270; CHECK-NEXT: local.get $push9=, 1 271; CHECK-NEXT: f64.ne $push3=, $pop10, $pop9 272; CHECK-NEXT: i32.or $push5=, $pop4, $pop3 273; CHECK-NEXT: local.get $push12=, 0 274; CHECK-NEXT: local.get $push11=, 0 275; CHECK-NEXT: f64.eq $push1=, $pop12, $pop11 276; CHECK-NEXT: local.get $push14=, 1 277; CHECK-NEXT: local.get $push13=, 1 278; CHECK-NEXT: f64.eq $push0=, $pop14, $pop13 279; CHECK-NEXT: i32.and $push2=, $pop1, $pop0 280; CHECK-NEXT: i32.xor $push6=, $pop5, $pop2 281; CHECK-NEXT: return $pop6 282 %a = fcmp uno double %x, %y 283 %b = fcmp ord double %x, %y 284 %c = xor i1 %a, %b 285 ret i1 %c 286} 287 288define i1 @unordf(float %x, float %y) { 289; CHECK-LABEL: unordf: 290; CHECK: .functype unordf (f32, f32) -> (i32) 291; CHECK-NEXT: # %bb.0: 292; CHECK-NEXT: local.get $push8=, 0 293; CHECK-NEXT: local.get $push7=, 0 294; CHECK-NEXT: f32.ne $push4=, $pop8, $pop7 295; CHECK-NEXT: local.get $push10=, 1 296; CHECK-NEXT: local.get $push9=, 1 297; CHECK-NEXT: f32.ne $push3=, $pop10, $pop9 298; CHECK-NEXT: i32.or $push5=, $pop4, $pop3 299; CHECK-NEXT: local.get $push12=, 0 300; CHECK-NEXT: local.get $push11=, 0 301; CHECK-NEXT: f32.eq $push1=, $pop12, $pop11 302; CHECK-NEXT: local.get $push14=, 1 303; CHECK-NEXT: local.get $push13=, 1 304; CHECK-NEXT: f32.eq $push0=, $pop14, $pop13 305; CHECK-NEXT: i32.and $push2=, $pop1, $pop0 306; CHECK-NEXT: i32.xor $push6=, $pop5, $pop2 307; CHECK-NEXT: return $pop6 308 %a = fcmp uno float %x, %y 309 %b = fcmp ord float %x, %y 310 %c = xor i1 %a, %b 311 ret i1 %c 312} 313 314define i1 @unordt(fp128 %x, fp128 %y) { 315; CHECK-LABEL: unordt: 316; CHECK: .functype unordt (i64, i64, i64, i64) -> (i32) 317; CHECK-NEXT: # %bb.0: 318; CHECK-NEXT: local.get $push6=, 0 319; CHECK-NEXT: local.get $push5=, 1 320; CHECK-NEXT: local.get $push4=, 2 321; CHECK-NEXT: local.get $push3=, 3 322; CHECK-NEXT: call $push1=, __unordtf2, $pop6, $pop5, $pop4, $pop3 323; CHECK-NEXT: i32.const $push0=, 0 324; CHECK-NEXT: i32.ne $push2=, $pop1, $pop0 325; CHECK-NEXT: return $pop2 326 %a = fcmp uno fp128 %x, %y 327 ret i1 %a 328} 329 330define i1 @ordt(fp128 %x, fp128 %y) { 331; CHECK-LABEL: ordt: 332; CHECK: .functype ordt (i64, i64, i64, i64) -> (i32) 333; CHECK-NEXT: # %bb.0: 334; CHECK-NEXT: local.get $push5=, 0 335; CHECK-NEXT: local.get $push4=, 1 336; CHECK-NEXT: local.get $push3=, 2 337; CHECK-NEXT: local.get $push2=, 3 338; CHECK-NEXT: call $push0=, __unordtf2, $pop5, $pop4, $pop3, $pop2 339; CHECK-NEXT: i32.eqz $push1=, $pop0 340; CHECK-NEXT: return $pop1 341 %a = fcmp ord fp128 %x, %y 342 ret i1 %a 343} 344