Home | History | Annotate | Download | only in InstCombine
      1 ; RUN: opt < %s -instcombine -S | FileCheck %s
      2 ;
      3 ; Test that instcombine folds allocsize function calls properly.
      4 ; Dummy arguments are inserted to verify that allocsize is picking the right
      5 ; args, and to prove that arbitrary unfoldable values don't interfere with
      6 ; allocsize if they're not used by allocsize.
      7 
      8 declare i8* @my_malloc(i8*, i32) allocsize(1)
      9 declare i8* @my_calloc(i8*, i8*, i32, i32) allocsize(2, 3)
     10 
     11 ; CHECK-LABEL: define void @test_malloc
     12 define void @test_malloc(i8** %p, i64* %r) {
     13   %1 = call i8* @my_malloc(i8* null, i32 100)
     14   store i8* %1, i8** %p, align 8 ; To ensure objectsize isn't killed
     15 
     16   %2 = call i64 @llvm.objectsize.i64.p0i8(i8* %1, i1 false)
     17   ; CHECK: store i64 100
     18   store i64 %2, i64* %r, align 8
     19   ret void
     20 }
     21 
     22 ; CHECK-LABEL: define void @test_calloc
     23 define void @test_calloc(i8** %p, i64* %r) {
     24   %1 = call i8* @my_calloc(i8* null, i8* null, i32 100, i32 5)
     25   store i8* %1, i8** %p, align 8 ; To ensure objectsize isn't killed
     26 
     27   %2 = call i64 @llvm.objectsize.i64.p0i8(i8* %1, i1 false)
     28   ; CHECK: store i64 500
     29   store i64 %2, i64* %r, align 8
     30   ret void
     31 }
     32 
     33 ; Failure cases with non-constant values...
     34 ; CHECK-LABEL: define void @test_malloc_fails
     35 define void @test_malloc_fails(i8** %p, i64* %r, i32 %n) {
     36   %1 = call i8* @my_malloc(i8* null, i32 %n)
     37   store i8* %1, i8** %p, align 8 ; To ensure objectsize isn't killed
     38 
     39   ; CHECK: @llvm.objectsize.i64.p0i8
     40   %2 = call i64 @llvm.objectsize.i64.p0i8(i8* %1, i1 false)
     41   store i64 %2, i64* %r, align 8
     42   ret void
     43 }
     44 
     45 ; CHECK-LABEL: define void @test_calloc_fails
     46 define void @test_calloc_fails(i8** %p, i64* %r, i32 %n) {
     47   %1 = call i8* @my_calloc(i8* null, i8* null, i32 %n, i32 5)
     48   store i8* %1, i8** %p, align 8 ; To ensure objectsize isn't killed
     49 
     50   ; CHECK: @llvm.objectsize.i64.p0i8
     51   %2 = call i64 @llvm.objectsize.i64.p0i8(i8* %1, i1 false)
     52   store i64 %2, i64* %r, align 8
     53 
     54 
     55   %3 = call i8* @my_calloc(i8* null, i8* null, i32 100, i32 %n)
     56   store i8* %3, i8** %p, align 8 ; To ensure objectsize isn't killed
     57 
     58   ; CHECK: @llvm.objectsize.i64.p0i8
     59   %4 = call i64 @llvm.objectsize.i64.p0i8(i8* %3, i1 false)
     60   store i64 %4, i64* %r, align 8
     61   ret void
     62 }
     63 
     64 declare i8* @my_malloc_outofline(i8*, i32) #0
     65 declare i8* @my_calloc_outofline(i8*, i8*, i32, i32) #1
     66 
     67 ; Verifying that out of line allocsize is parsed correctly
     68 ; CHECK-LABEL: define void @test_outofline
     69 define void @test_outofline(i8** %p, i64* %r) {
     70   %1 = call i8* @my_malloc_outofline(i8* null, i32 100)
     71   store i8* %1, i8** %p, align 8 ; To ensure objectsize isn't killed
     72 
     73   %2 = call i64 @llvm.objectsize.i64.p0i8(i8* %1, i1 false)
     74   ; CHECK: store i64 100
     75   store i64 %2, i64* %r, align 8
     76 
     77 
     78   %3 = call i8* @my_calloc_outofline(i8* null, i8* null, i32 100, i32 5)
     79   store i8* %3, i8** %p, align 8 ; To ensure objectsize isn't killed
     80 
     81   %4 = call i64 @llvm.objectsize.i64.p0i8(i8* %3, i1 false)
     82   ; CHECK: store i64 500
     83   store i64 %4, i64* %r, align 8
     84   ret void
     85 }
     86 
     87 declare i8* @my_malloc_i64(i8*, i64) #0
     88 declare i8* @my_tiny_calloc(i8*, i8*, i8, i8) #1
     89 declare i8* @my_varied_calloc(i8*, i8*, i32, i8) #1
     90 
     91 ; CHECK-LABEL: define void @test_overflow
     92 define void @test_overflow(i8** %p, i32* %r) {
     93   %r64 = bitcast i32* %r to i64*
     94 
     95   ; (2**31 + 1) * 2 > 2**31. So overflow. Yay.
     96   %big_malloc = call i8* @my_calloc(i8* null, i8* null, i32 2147483649, i32 2)
     97   store i8* %big_malloc, i8** %p, align 8
     98 
     99   ; CHECK: @llvm.objectsize
    100   %1 = call i32 @llvm.objectsize.i32.p0i8(i8* %big_malloc, i1 false)
    101   store i32 %1, i32* %r, align 4
    102 
    103 
    104   %big_little_malloc = call i8* @my_tiny_calloc(i8* null, i8* null, i8 127, i8 4)
    105   store i8* %big_little_malloc, i8** %p, align 8
    106 
    107   ; CHECK: store i32 508
    108   %2 = call i32 @llvm.objectsize.i32.p0i8(i8* %big_little_malloc, i1 false)
    109   store i32 %2, i32* %r, align 4
    110 
    111 
    112   ; malloc(2**33)
    113   %big_malloc_i64 = call i8* @my_malloc_i64(i8* null, i64 8589934592)
    114   store i8* %big_malloc_i64, i8** %p, align 8
    115 
    116   ; CHECK: @llvm.objectsize
    117   %3 = call i32 @llvm.objectsize.i32.p0i8(i8* %big_malloc_i64, i1 false)
    118   store i32 %3, i32* %r, align 4
    119 
    120 
    121   %4 = call i64 @llvm.objectsize.i64.p0i8(i8* %big_malloc_i64, i1 false)
    122   ; CHECK: store i64 8589934592
    123   store i64 %4, i64* %r64, align 8
    124 
    125 
    126   ; Just intended to ensure that we properly handle args of different types...
    127   %varied_calloc = call i8* @my_varied_calloc(i8* null, i8* null, i32 1000, i8 5)
    128   store i8* %varied_calloc, i8** %p, align 8
    129 
    130   ; CHECK: store i32 5000
    131   %5 = call i32 @llvm.objectsize.i32.p0i8(i8* %varied_calloc, i1 false)
    132   store i32 %5, i32* %r, align 4
    133 
    134   ret void
    135 }
    136 
    137 attributes #0 = { allocsize(1) }
    138 attributes #1 = { allocsize(2, 3) }
    139 
    140 declare i32 @llvm.objectsize.i32.p0i8(i8*, i1)
    141 declare i64 @llvm.objectsize.i64.p0i8(i8*, i1)
    142