Home | History | Annotate | Download | only in X86
      1 ; RUN: llc < %s -mtriple=i686-pc-linux | FileCheck %s -check-prefix=LINUX -check-prefix=CHECK
      2 ; RUN: llc < %s -mtriple=i686-apple-darwin | FileCheck %s -check-prefix=DARWIN -check-prefix=CHECK
      3 
      4 declare i32 @__gxx_personality_v0(...)
      5 declare void @good(i32 %a, i32 %b, i32 %c, i32 %d)
      6 declare void @large(i32 %a, i32 %b, i32 %c, i32 %d, i32 %e, i32 %f)
      7 declare void @empty()
      8 
      9 ; When we use an invoke, we expect a .cfi_escape GNU_ARGS_SIZE
     10 ; with size 16 before the invocation. Without FP, we also expect
     11 ; .cfi_adjust_cfa_offset after each push.
     12 ; Darwin should not generate pushes in either circumstance.
     13 ; CHECK-LABEL: test1_nofp:
     14 ; LINUX: .cfi_escape 0x2e, 0x10
     15 ; LINUX-NEXT: pushl   $4
     16 ; LINUX-NEXT: .cfi_adjust_cfa_offset 4
     17 ; LINUX-NEXT: pushl   $3
     18 ; LINUX-NEXT: .cfi_adjust_cfa_offset 4
     19 ; LINUX-NEXT: pushl   $2
     20 ; LINUX-NEXT: .cfi_adjust_cfa_offset 4
     21 ; LINUX-NEXT: pushl   $1
     22 ; LINUX-NEXT: .cfi_adjust_cfa_offset 4
     23 ; LINUX-NEXT: call
     24 ; LINUX-NEXT: addl $16, %esp
     25 ; LINUX: .cfi_adjust_cfa_offset -16
     26 ; DARWIN-NOT: .cfi_escape
     27 ; DARWIN-NOT: pushl
     28 define void @test1_nofp() #0 personality i8* bitcast (i32 (...)* @__gxx_personality_v0 to i8*) {
     29 entry:
     30   invoke void @good(i32 1, i32 2, i32 3, i32 4)
     31           to label %continue unwind label %cleanup
     32 continue:
     33   ret void
     34 cleanup:  
     35   landingpad { i8*, i32 }
     36      cleanup
     37   ret void
     38 }
     39 
     40 ; CHECK-LABEL: test1_fp:
     41 ; LINUX: .cfi_escape 0x2e, 0x10
     42 ; LINUX-NEXT: pushl   $4
     43 ; LINUX-NEXT: pushl   $3
     44 ; LINUX-NEXT: pushl   $2
     45 ; LINUX-NEXT: pushl   $1
     46 ; LINUX-NEXT: call
     47 ; LINUX-NEXT: addl $16, %esp
     48 ; DARWIN: pushl %ebp
     49 ; DARWIN-NOT: .cfi_escape
     50 ; DARWIN-NOT: pushl
     51 define void @test1_fp() #1 personality i8* bitcast (i32 (...)* @__gxx_personality_v0 to i8*) {
     52 entry:
     53   invoke void @good(i32 1, i32 2, i32 3, i32 4)
     54           to label %continue unwind label %cleanup
     55 continue:
     56   ret void
     57 cleanup:  
     58   landingpad { i8*, i32 }
     59      cleanup
     60   ret void
     61 }
     62 
     63 ; If the function has no handlers, we don't need to generate GNU_ARGS_SIZE,
     64 ; even if it has an unwind table. Without FP, we still need cfi_adjust_cfa_offset,
     65 ; so darwin should not generate pushes.
     66 ; CHECK-LABEL: test2_nofp:
     67 ; LINUX-NOT: .cfi_escape
     68 ; LINUX: pushl   $4
     69 ; LINUX-NEXT: .cfi_adjust_cfa_offset 4
     70 ; LINUX-NEXT: pushl   $3
     71 ; LINUX-NEXT: .cfi_adjust_cfa_offset 4
     72 ; LINUX-NEXT: pushl   $2
     73 ; LINUX-NEXT: .cfi_adjust_cfa_offset 4
     74 ; LINUX-NEXT: pushl   $1
     75 ; LINUX-NEXT: .cfi_adjust_cfa_offset 4
     76 ; LINUX-NEXT: call
     77 ; LINUX-NEXT: addl $28, %esp
     78 ; LINUX: .cfi_adjust_cfa_offset -28
     79 ; DARWIN-NOT: .cfi_escape
     80 ; DARWIN-NOT: pushl
     81 define void @test2_nofp() #0 personality i8* bitcast (i32 (...)* @__gxx_personality_v0 to i8*) {
     82 entry:
     83   call void @good(i32 1, i32 2, i32 3, i32 4)
     84   ret void
     85 }
     86 
     87 ; CHECK-LABEL: test2_fp:
     88 ; CHECK-NOT: .cfi_escape
     89 ; CHECK-NOT: .cfi_adjust_cfa_offset
     90 ; CHECK: pushl   $4
     91 ; CHECK-NEXT: pushl   $3
     92 ; CHECK-NEXT: pushl   $2
     93 ; CHECK-NEXT: pushl   $1
     94 ; CHECK-NEXT: call
     95 ; CHECK-NEXT: addl $24, %esp
     96 define void @test2_fp() #1 personality i8* bitcast (i32 (...)* @__gxx_personality_v0 to i8*) {
     97 entry:
     98   call void @good(i32 1, i32 2, i32 3, i32 4)
     99   ret void
    100 }
    101 
    102 ; If we did not end up using any pushes, no need for GNU_ARGS_SIZE or
    103 ; cfi_adjust_cfa_offset.
    104 ; CHECK-LABEL: test3_nofp:
    105 ; LINUX-NOT: .cfi_escape
    106 ; LINUX-NOT: .cfi_adjust_cfa_offset
    107 ; LINUX-NOT: pushl
    108 ; LINUX: retl
    109 define void @test3_nofp() #0 personality i8* bitcast (i32 (...)* @__gxx_personality_v0 to i8*) {
    110 entry:
    111   invoke void @empty()
    112           to label %continue unwind label %cleanup
    113 continue:
    114   ret void
    115 cleanup:  
    116   landingpad { i8*, i32 }
    117      cleanup
    118   ret void
    119 }
    120 
    121 ; If we did not end up using any pushes, no need for GNU_ARGS_SIZE or
    122 ; cfi_adjust_cfa_offset.
    123 ; CHECK-LABEL: test3_fp:
    124 ; LINUX: pushl %ebp
    125 ; LINUX-NOT: .cfi_escape
    126 ; LINUX-NOT: .cfi_adjust_cfa_offset
    127 ; LINUX-NOT: pushl
    128 ; LINUX: retl
    129 define void @test3_fp() #1 personality i8* bitcast (i32 (...)* @__gxx_personality_v0 to i8*) {
    130 entry:
    131   invoke void @empty()
    132           to label %continue unwind label %cleanup
    133 continue:
    134   ret void
    135 cleanup:  
    136   landingpad { i8*, i32 }
    137      cleanup
    138   ret void
    139 }
    140 
    141 ; Different sized stacks need different GNU_ARGS_SIZEs
    142 ; CHECK-LABEL: test4:
    143 ; LINUX: .cfi_escape 0x2e, 0x10
    144 ; LINUX-NEXT: pushl   $4
    145 ; LINUX-NEXT: pushl   $3
    146 ; LINUX-NEXT: pushl   $2
    147 ; LINUX-NEXT: pushl   $1
    148 ; LINUX-NEXT: call
    149 ; LINUX-NEXT: addl $16, %esp
    150 ; LINUX: .cfi_escape 0x2e, 0x20
    151 ; LINUX: subl    $8, %esp
    152 ; LINUX-NEXT: pushl   $11
    153 ; LINUX-NEXT: pushl   $10
    154 ; LINUX-NEXT: pushl   $9
    155 ; LINUX-NEXT: pushl   $8
    156 ; LINUX-NEXT: pushl   $7
    157 ; LINUX-NEXT: pushl   $6
    158 ; LINUX-NEXT: calll   large
    159 ; LINUX-NEXT: addl $32, %esp
    160 define void @test4() #1 personality i8* bitcast (i32 (...)* @__gxx_personality_v0 to i8*) {
    161 entry:
    162   invoke void @good(i32 1, i32 2, i32 3, i32 4)
    163           to label %continue1 unwind label %cleanup
    164 continue1:
    165   invoke void @large(i32 6, i32 7, i32 8, i32 9, i32 10, i32 11)
    166           to label %continue2 unwind label %cleanup
    167 continue2:
    168   ret void          
    169 cleanup:  
    170   landingpad { i8*, i32 }
    171      cleanup
    172   ret void
    173 }
    174 
    175 ; If we did use pushes, we need to reset GNU_ARGS_SIZE before a call
    176 ; without parameters, but don't need to adjust the cfa offset
    177 ; CHECK-LABEL: test5_nofp:
    178 ; LINUX: .cfi_escape 0x2e, 0x10
    179 ; LINUX-NEXT: pushl   $4
    180 ; LINUX-NEXT: .cfi_adjust_cfa_offset 4
    181 ; LINUX-NEXT: pushl   $3
    182 ; LINUX-NEXT: .cfi_adjust_cfa_offset 4
    183 ; LINUX-NEXT: pushl   $2
    184 ; LINUX-NEXT: .cfi_adjust_cfa_offset 4
    185 ; LINUX-NEXT: pushl   $1
    186 ; LINUX-NEXT: .cfi_adjust_cfa_offset 4
    187 ; LINUX-NEXT: call
    188 ; LINUX-NEXT: addl $16, %esp
    189 ; LINUX: .cfi_adjust_cfa_offset -16
    190 ; LINUX-NOT: .cfi_adjust_cfa_offset
    191 ; LINUX: .cfi_escape 0x2e, 0x00
    192 ; LINUX-NOT: .cfi_adjust_cfa_offset
    193 ; LINUX: call
    194 define void @test5_nofp() #0 personality i8* bitcast (i32 (...)* @__gxx_personality_v0 to i8*) {
    195 entry:
    196   invoke void @good(i32 1, i32 2, i32 3, i32 4)
    197           to label %continue1 unwind label %cleanup
    198 continue1:
    199   invoke void @empty()
    200           to label %continue2 unwind label %cleanup
    201 continue2:
    202   ret void          
    203 cleanup:  
    204   landingpad { i8*, i32 }
    205      cleanup
    206   ret void
    207 }
    208 
    209 ; CHECK-LABEL: test5_fp:
    210 ; LINUX: .cfi_escape 0x2e, 0x10
    211 ; LINUX-NEXT: pushl   $4
    212 ; LINUX-NEXT: pushl   $3
    213 ; LINUX-NEXT: pushl   $2
    214 ; LINUX-NEXT: pushl   $1
    215 ; LINUX-NEXT: call
    216 ; LINUX-NEXT: addl $16, %esp
    217 ; LINUX: .cfi_escape 0x2e, 0x00
    218 ; LINUX-NOT: .cfi_adjust_cfa_offset
    219 ; LINUX: call
    220 define void @test5_fp() #1 personality i8* bitcast (i32 (...)* @__gxx_personality_v0 to i8*) {
    221 entry:
    222   invoke void @good(i32 1, i32 2, i32 3, i32 4)
    223           to label %continue1 unwind label %cleanup
    224 continue1:
    225   invoke void @empty()
    226           to label %continue2 unwind label %cleanup
    227 continue2:
    228   ret void          
    229 cleanup:  
    230   landingpad { i8*, i32 }
    231      cleanup
    232   ret void
    233 }
    234 
    235 ; FIXME: This is actually inefficient - we don't need to repeat the .cfi_escape twice.
    236 ; CHECK-LABEL: test6:
    237 ; LINUX: .cfi_escape 0x2e, 0x10
    238 ; LINUX: call
    239 ; LINUX: .cfi_escape 0x2e, 0x10
    240 ; LINUX: call
    241 define void @test6() #1 personality i8* bitcast (i32 (...)* @__gxx_personality_v0 to i8*) {
    242 entry:
    243   invoke void @good(i32 1, i32 2, i32 3, i32 4)
    244           to label %continue1 unwind label %cleanup
    245 continue1:
    246   invoke void @good(i32 5, i32 6, i32 7, i32 8)
    247           to label %continue2 unwind label %cleanup
    248 continue2:
    249   ret void          
    250 cleanup:  
    251   landingpad { i8*, i32 }
    252      cleanup
    253   ret void
    254 }
    255 
    256 ; Darwin should generate pushes in the presense of FP and an unwind table,
    257 ; but not FP and invoke.
    258 ; CHECK-LABEL: test7:
    259 ; DARWIN: pushl %ebp
    260 ; DARWIN: movl %esp, %ebp
    261 ; DARWIN: .cfi_def_cfa_register %ebp
    262 ; DARWIN-NOT: .cfi_adjust_cfa_offset
    263 ; DARWIN: pushl   $4
    264 ; DARWIN-NEXT: pushl   $3
    265 ; DARWIN-NEXT: pushl   $2
    266 ; DARWIN-NEXT: pushl   $1
    267 ; DARWIN-NEXT: call
    268 define void @test7() #1 personality i8* bitcast (i32 (...)* @__gxx_personality_v0 to i8*) {
    269 entry:
    270   call void @good(i32 1, i32 2, i32 3, i32 4)
    271   ret void
    272 }
    273 
    274 ; CHECK-LABEL: test8:
    275 ; DARWIN: pushl %ebp
    276 ; DARWIN: movl %esp, %ebp
    277 ; DARWIN-NOT: .cfi_adjust_cfa_offset
    278 ; DARWIN-NOT: pushl
    279 define void @test8() #1 personality i8* bitcast (i32 (...)* @__gxx_personality_v0 to i8*) {
    280 entry:
    281   invoke void @good(i32 1, i32 2, i32 3, i32 4)
    282           to label %continue unwind label %cleanup
    283 continue:
    284   ret void
    285 cleanup:  
    286   landingpad { i8*, i32 }
    287      cleanup
    288   ret void
    289 }
    290 
    291 attributes #0 = { optsize }
    292 attributes #1 = { optsize "no-frame-pointer-elim"="true" }
    293