llvm/llvm/test/Transforms/SafeStack/X86/coloring2.ll

; RUN: opt -safe-stack -safe-stack-coloring=1 -S -mtriple=i386-pc-linux-gnu < %s -o - | FileCheck %s
; RUN: opt -safe-stack -safe-stack-coloring=1 -S -mtriple=x86_64-pc-linux-gnu < %s -o - | FileCheck %s
; RUN: opt -passes=safe-stack -safe-stack-coloring=1 -S -mtriple=i386-pc-linux-gnu < %s -o - | FileCheck %s
; RUN: opt -passes=safe-stack -safe-stack-coloring=1 -S -mtriple=x86_64-pc-linux-gnu < %s -o - | FileCheck %s

; x and y share the stack slot.
define void @f() safestack {
; CHECK-LABEL: define void @f
entry:
; CHECK:  %[[USP:.*]] = load ptr, ptr @__safestack_unsafe_stack_ptr
; CHECK:   getelementptr i8, ptr %[[USP]], i32 -16

  %x = alloca i32, align 4
  %y = alloca i32, align 4
  %z = alloca i32, align 4

  call void @llvm.lifetime.start.p0(i64 -1, ptr %z)
  call void @llvm.lifetime.start.p0(i64 -1, ptr %x)

; CHECK:   getelementptr i8, ptr %[[USP]], i32 -4
  call void @capture32(ptr %x)
  call void @llvm.lifetime.end.p0(i64 -1, ptr %x)
  call void @llvm.lifetime.start.p0(i64 -1, ptr %y)

; CHECK:   getelementptr i8, ptr %[[USP]], i32 -4
  call void @capture32(ptr %y)
  call void @llvm.lifetime.end.p0(i64 -1, ptr %y)

; CHECK:   getelementptr i8, ptr %[[USP]], i32 -8
  call void @capture32(ptr %z)
  call void @llvm.lifetime.end.p0(i64 -1, ptr %z)

  ret void
}

define void @no_markers() safestack {
; CHECK-LABEL: define void @no_markers(
entry:
; CHECK:  %[[USP:.*]] = load ptr, ptr @__safestack_unsafe_stack_ptr
; CHECK:   getelementptr i8, ptr %[[USP]], i32 -16

  %x = alloca i32, align 4
  %y = alloca i32, align 4

  call void @llvm.lifetime.start.p0(i64 -1, ptr %x)

; CHECK:   getelementptr i8, ptr %[[USP]], i32 -4
  call void @capture32(ptr %x)
  call void @llvm.lifetime.end.p0(i64 -1, ptr %x)

; CHECK:   getelementptr i8, ptr %[[USP]], i32 -8
  call void @capture32(ptr %y)

  ret void
}

; x and y can't share memory, but they can split z's storage.
define void @g() safestack {
; CHECK-LABEL: define void @g
entry:
; CHECK:  %[[USP:.*]] = load ptr, ptr @__safestack_unsafe_stack_ptr
; CHECK:   getelementptr i8, ptr %[[USP]], i32 -16

  %x = alloca i32, align 4
  %y = alloca i32, align 4
  %z = alloca i64, align 4

  call void @llvm.lifetime.start.p0(i64 -1, ptr %x)
  call void @llvm.lifetime.start.p0(i64 -1, ptr %y)

; CHECK:   getelementptr i8, ptr %[[USP]], i32 -4
  call void @capture32(ptr %x)
  call void @llvm.lifetime.end.p0(i64 -1, ptr %x)

; CHECK:   getelementptr i8, ptr %[[USP]], i32 -8
  call void @capture32(ptr %y)
  call void @llvm.lifetime.end.p0(i64 -1, ptr %y)
  call void @llvm.lifetime.start.p0(i64 -1, ptr %z)

; CHECK:   getelementptr i8, ptr %[[USP]], i32 -8
  call void @capture64(ptr %z)
  call void @llvm.lifetime.end.p0(i64 -1, ptr %z)

  ret void
}

; Both y and z fit in x's alignment gap.
define void @h() safestack {
; CHECK-LABEL: define void @h
entry:
; CHECK:  %[[USP:.*]] = load ptr, ptr @__safestack_unsafe_stack_ptr
; CHECK:   getelementptr i8, ptr %[[USP]], i32 -16

  %x = alloca i32, align 16
  %z = alloca i64, align 4
  %y = alloca i32, align 4

  call void @llvm.lifetime.start.p0(i64 -1, ptr %x)
  call void @llvm.lifetime.start.p0(i64 -1, ptr %y)
  call void @llvm.lifetime.start.p0(i64 -1, ptr %z)

; CHECK:   getelementptr i8, ptr %[[USP]], i32 -16
  call void @capture32(ptr %x)

; CHECK:   getelementptr i8, ptr %[[USP]], i32 -12
  call void @capture32(ptr %y)

; CHECK:   getelementptr i8, ptr %[[USP]], i32 -8
  call void @capture64(ptr %z)

  call void @llvm.lifetime.end.p0(i64 -1, ptr %x)
  call void @llvm.lifetime.end.p0(i64 -1, ptr %y)
  call void @llvm.lifetime.end.p0(i64 -1, ptr %z)

  ret void
}

; void f(bool a, bool b) {
;   long x1, x2; capture64(&x1); capture64(&x2);
;   if (a) {
;     long y; capture64(&y);
;     if (b) {
;       long y1; capture64(&y1);
;     } else {
;       long y2; capture64(&y2);
;     }
;   } else {
;     long z; capture64(&z);
;     if (b) {
;       long z1; capture64(&z1);
;     } else {
;       long z2; capture64(&z2);
;     }
;   }
; }
; Everything fits in 4 x 64-bit slots.
define void @i(i1 zeroext %a, i1 zeroext %b) safestack {
; CHECK-LABEL: define void @i
entry:
; CHECK:        %[[USP:.*]] = load ptr, ptr @__safestack_unsafe_stack_ptr
; CHECK-NEXT:   getelementptr i8, ptr %[[USP]], i32 -32
  %x1 = alloca i64, align 8
  %x2 = alloca i64, align 8
  %y = alloca i64, align 8
  %y1 = alloca i64, align 8
  %y2 = alloca i64, align 8
  %z = alloca i64, align 8
  %z1 = alloca i64, align 8
  %z2 = alloca i64, align 8
  call void @llvm.lifetime.start.p0(i64 -1, ptr %x1)
  call void @llvm.lifetime.start.p0(i64 -1, ptr %x2)
; CHECK:   getelementptr i8, ptr %[[USP]], i32 -8
; CHECK:   call void @capture64(
  call void @capture64(ptr nonnull %x1)
; CHECK:   getelementptr i8, ptr %[[USP]], i32 -16
; CHECK:   call void @capture64(
  call void @capture64(ptr nonnull %x2)
  br i1 %a, label %if.then, label %if.else4

if.then:                                          ; preds = %entry
  call void @llvm.lifetime.start.p0(i64 -1, ptr %y)
; CHECK:   getelementptr i8, ptr %[[USP]], i32 -24
; CHECK:   call void @capture64(
  call void @capture64(ptr nonnull %y)
  br i1 %b, label %if.then3, label %if.else

if.then3:                                         ; preds = %if.then
  call void @llvm.lifetime.start.p0(i64 -1, ptr %y1)
; CHECK:   getelementptr i8, ptr %[[USP]], i32 -32
; CHECK:   call void @capture64(
  call void @capture64(ptr nonnull %y1)
  call void @llvm.lifetime.end.p0(i64 -1, ptr %y1)
  br label %if.end

if.else:                                          ; preds = %if.then
  call void @llvm.lifetime.start.p0(i64 -1, ptr %y2)
; CHECK:   getelementptr i8, ptr %[[USP]], i32 -32
; CHECK:   call void @capture64(
  call void @capture64(ptr nonnull %y2)
  call void @llvm.lifetime.end.p0(i64 -1, ptr %y2)
  br label %if.end

if.end:                                           ; preds = %if.else, %if.then3
  call void @llvm.lifetime.end.p0(i64 -1, ptr %y)
  br label %if.end9

if.else4:                                         ; preds = %entry
  call void @llvm.lifetime.start.p0(i64 -1, ptr %z)
; CHECK:   getelementptr i8, ptr %[[USP]], i32 -24
; CHECK:   call void @capture64(
  call void @capture64(ptr nonnull %z)
  br i1 %b, label %if.then6, label %if.else7

if.then6:                                         ; preds = %if.else4
  call void @llvm.lifetime.start.p0(i64 -1, ptr %z1)
; CHECK:   getelementptr i8, ptr %[[USP]], i32 -32
; CHECK:   call void @capture64(
  call void @capture64(ptr nonnull %z1)
  call void @llvm.lifetime.end.p0(i64 -1, ptr %z1)
  br label %if.end8

if.else7:                                         ; preds = %if.else4
  call void @llvm.lifetime.start.p0(i64 -1, ptr %z2)
; CHECK:   getelementptr i8, ptr %[[USP]], i32 -32
; CHECK:   call void @capture64(
  call void @capture64(ptr nonnull %z2)
  call void @llvm.lifetime.end.p0(i64 -1, ptr %z2)
  br label %if.end8

if.end8:                                          ; preds = %if.else7, %if.then6
  call void @llvm.lifetime.end.p0(i64 -1, ptr %z)
  br label %if.end9

if.end9:                                          ; preds = %if.end8, %if.end
  call void @llvm.lifetime.end.p0(i64 -1, ptr %x2)
  call void @llvm.lifetime.end.p0(i64 -1, ptr %x1)
  ret void
}

; lifetime for x ends in 2 different BBs
define void @no_merge1(i1 %d) safestack {
; CHECK-LABEL: define void @no_merge1(
entry:
; CHECK:        %[[USP:.*]] = load ptr, ptr @__safestack_unsafe_stack_ptr
; CHECK-NEXT:   getelementptr i8, ptr %[[USP]], i32 -16
  %x = alloca i32, align 4
  %y = alloca i32, align 4
  call void @llvm.lifetime.start.p0(i64 -1, ptr %x)
; CHECK:   getelementptr i8, ptr %[[USP]], i32 -4
; CHECK:   call void @capture32(
  call void @capture32(ptr %x)
  br i1 %d, label %bb2, label %bb3
bb2:
  call void @llvm.lifetime.start.p0(i64 -1, ptr %y)
; CHECK:   getelementptr i8, ptr %[[USP]], i32 -8
; CHECK:   call void @capture32(
  call void @capture32(ptr %y)
  call void @llvm.lifetime.end.p0(i64 -1, ptr %y)
  call void @llvm.lifetime.end.p0(i64 -1, ptr %x)
  ret void
bb3:
  call void @llvm.lifetime.end.p0(i64 -1, ptr %x)
  ret void
}

define void @merge1(i1 %d) safestack {
; CHECK-LABEL: define void @merge1(
entry:
; CHECK:        %[[USP:.*]] = load ptr, ptr @__safestack_unsafe_stack_ptr
; CHECK-NEXT:   getelementptr i8, ptr %[[USP]], i32 -16
  %x = alloca i32, align 4
  %y = alloca i32, align 4
  call void @llvm.lifetime.start.p0(i64 -1, ptr %x)
; CHECK:   getelementptr i8, ptr %[[USP]], i32 -4
; CHECK:   call void @capture32(
  call void @capture32(ptr %x)
  call void @llvm.lifetime.end.p0(i64 -1, ptr %x)
  br i1 %d, label %bb2, label %bb3
bb2:
  call void @llvm.lifetime.start.p0(i64 -1, ptr %y)
; CHECK:   getelementptr i8, ptr %[[USP]], i32 -4
; CHECK:   call void @capture32(
  call void @capture32(ptr %y)
  call void @llvm.lifetime.end.p0(i64 -1, ptr %y)
  ret void
bb3:
  ret void
}

; Missing lifetime.end
define void @merge2_noend(i1 %d) safestack {
; CHECK-LABEL: define void @merge2_noend(
entry:
; CHECK:        %[[USP:.*]] = load ptr, ptr @__safestack_unsafe_stack_ptr
; CHECK-NEXT:   getelementptr i8, ptr %[[USP]], i32 -16
  %x = alloca i32, align 4
  %y = alloca i32, align 4
  call void @llvm.lifetime.start.p0(i64 -1, ptr %x)
; CHECK:   getelementptr i8, ptr %[[USP]], i32 -4
; CHECK:   call void @capture32(
  call void @capture32(ptr %x)
  call void @llvm.lifetime.end.p0(i64 -1, ptr %x)
  br i1 %d, label %bb2, label %bb3
bb2:
  call void @llvm.lifetime.start.p0(i64 -1, ptr %y)
; CHECK:   getelementptr i8, ptr %[[USP]], i32 -4
; CHECK:   call void @capture32(
  call void @capture32(ptr %y)
  ret void
bb3:
  ret void
}

; Missing lifetime.end
define void @merge3_noend(i1 %d) safestack {
; CHECK-LABEL: define void @merge3_noend(
entry:
; CHECK:        %[[USP:.*]] = load ptr, ptr @__safestack_unsafe_stack_ptr
; CHECK-NEXT:   getelementptr i8, ptr %[[USP]], i32 -16
  %x = alloca i32, align 4
  %y = alloca i32, align 4
  call void @llvm.lifetime.start.p0(i64 -1, ptr %x)
; CHECK:   getelementptr i8, ptr %[[USP]], i32 -4
; CHECK:   call void @capture32(
  call void @capture32(ptr %x)
  br i1 %d, label %bb2, label %bb3
bb2:
  call void @llvm.lifetime.end.p0(i64 -1, ptr %x)
  call void @llvm.lifetime.start.p0(i64 -1, ptr %y)
; CHECK:   getelementptr i8, ptr %[[USP]], i32 -4
; CHECK:   call void @capture32(
  call void @capture32(ptr %y)
  ret void
bb3:
  ret void
}

; Missing lifetime.start
define void @nomerge4_nostart(i1 %d) safestack {
; CHECK-LABEL: define void @nomerge4_nostart(
entry:
; CHECK:        %[[USP:.*]] = load ptr, ptr @__safestack_unsafe_stack_ptr
; CHECK-NEXT:   getelementptr i8, ptr %[[USP]], i32 -16
  %x = alloca i32, align 4
  %y = alloca i32, align 4
; CHECK:   getelementptr i8, ptr %[[USP]], i32 -4
; CHECK:   call void @capture32(
  call void @capture32(ptr %x)
  call void @llvm.lifetime.end.p0(i64 -1, ptr %x)
  br i1 %d, label %bb2, label %bb3
bb2:
  call void @llvm.lifetime.start.p0(i64 -1, ptr %y)
; CHECK:   getelementptr i8, ptr %[[USP]], i32 -8
; CHECK:   call void @capture32(
  call void @capture32(ptr %y)
  ret void
bb3:
  ret void
}

define void @array_merge() safestack {
; CHECK-LABEL: define void @array_merge(
entry:
; CHECK:        %[[USP:.*]] = load ptr, ptr @__safestack_unsafe_stack_ptr
; CHECK-NEXT:   getelementptr i8, ptr %[[USP]], i32 -800
  %A.i1 = alloca [100 x i32], align 4
  %B.i2 = alloca [100 x i32], align 4
  %A.i = alloca [100 x i32], align 4
  %B.i = alloca [100 x i32], align 4
  call void @llvm.lifetime.start.p0(i64 -1, ptr %A.i)
  call void @llvm.lifetime.start.p0(i64 -1, ptr %B.i)
; CHECK:   getelementptr i8, ptr %[[USP]], i32 -400
; CHECK:   call void @capture100x32(
  call void @capture100x32(ptr %A.i)
; CHECK:   getelementptr i8, ptr %[[USP]], i32 -800
; CHECK:   call void @capture100x32(
  call void @capture100x32(ptr %B.i)
  call void @llvm.lifetime.end.p0(i64 -1, ptr %A.i)
  call void @llvm.lifetime.end.p0(i64 -1, ptr %B.i)
  call void @llvm.lifetime.start.p0(i64 -1, ptr %A.i1)
  call void @llvm.lifetime.start.p0(i64 -1, ptr %B.i2)
; CHECK:   getelementptr i8, ptr %[[USP]], i32 -400
; CHECK:   call void @capture100x32(
  call void @capture100x32(ptr %A.i1)
; CHECK:   getelementptr i8, ptr %[[USP]], i32 -800
; CHECK:   call void @capture100x32(
  call void @capture100x32(ptr %B.i2)
  call void @llvm.lifetime.end.p0(i64 -1, ptr %A.i1)
  call void @llvm.lifetime.end.p0(i64 -1, ptr %B.i2)
  ret void
}

define void @myCall_pr15707() safestack {
; CHECK-LABEL: define void @myCall_pr15707(
entry:
; CHECK:        %[[USP:.*]] = load ptr, ptr @__safestack_unsafe_stack_ptr
; CHECK-NEXT:   getelementptr i8, ptr %[[USP]], i32 -200000
  %buf1 = alloca i8, i32 100000, align 16
  %buf2 = alloca i8, i32 100000, align 16

  call void @llvm.lifetime.start.p0(i64 -1, ptr %buf1)
  call void @llvm.lifetime.end.p0(i64 -1, ptr %buf1)

  call void @llvm.lifetime.start.p0(i64 -1, ptr %buf1)
  call void @llvm.lifetime.start.p0(i64 -1, ptr %buf2)
  call void @capture8(ptr %buf1)
  call void @capture8(ptr %buf2)
  ret void
}

; Check that we don't assert and crash even when there are allocas
; outside the declared lifetime regions.
define void @bad_range() safestack {
; CHECK-LABEL: define void @bad_range(
entry:
; CHECK:        %[[USP:.*]] = load ptr, ptr @__safestack_unsafe_stack_ptr
; A.i and B.i unsafe, not merged
; CHECK-NEXT:   getelementptr i8, ptr %[[USP]], i32 -800
; A.i1 and B.i2 safe
; CHECK: = alloca [100 x i32], align 4
; CHECK: = alloca [100 x i32], align 4

  %A.i1 = alloca [100 x i32], align 4
  %B.i2 = alloca [100 x i32], align 4
  %A.i = alloca [100 x i32], align 4
  %B.i = alloca [100 x i32], align 4
  call void @llvm.lifetime.start.p0(i64 -1, ptr %A.i) nounwind
  call void @llvm.lifetime.start.p0(i64 -1, ptr %B.i) nounwind
  call void @capture100x32(ptr %A.i)
  call void @capture100x32(ptr %B.i)
  call void @llvm.lifetime.end.p0(i64 -1, ptr %A.i) nounwind
  call void @llvm.lifetime.end.p0(i64 -1, ptr %B.i) nounwind
  br label %block2

block2:
  ; I am used outside the marked lifetime.
  call void @capture100x32(ptr %A.i)
  call void @capture100x32(ptr %B.i)
  ret void
}

%struct.Klass = type { i32, i32 }

define i32 @shady_range(i32 %argc, ptr nocapture %argv) safestack {
; CHECK-LABEL: define i32 @shady_range(
entry:
; CHECK:        %[[USP:.*]] = load ptr, ptr @__safestack_unsafe_stack_ptr
; CHECK-NEXT:   getelementptr i8, ptr %[[USP]], i32 -64
  %a.i = alloca [4 x %struct.Klass], align 16
  %b.i = alloca [4 x %struct.Klass], align 16
  ; I am used outside the lifetime zone below:
  call void @llvm.lifetime.start.p0(i64 -1, ptr %a.i)
  call void @llvm.lifetime.start.p0(i64 -1, ptr %b.i)
  call void @capture8(ptr %a.i)
  call void @capture8(ptr %b.i)
  %z3 = load i32, ptr %a.i, align 16
  call void @llvm.lifetime.end.p0(i64 -1, ptr %a.i)
  call void @llvm.lifetime.end.p0(i64 -1, ptr %b.i)
  ret i32 %z3
}

define void @end_loop() safestack {
; CHECK-LABEL: define void @end_loop()
entry:
; CHECK:        %[[USP:.*]] = load ptr, ptr @__safestack_unsafe_stack_ptr
; CHECK-NEXT:   getelementptr i8, ptr %[[USP]], i32 -16
  %x = alloca i8, align 4
  call void @llvm.lifetime.start.p0(i64 4, ptr %x) nounwind
  br label %l2

l2:
  call void @capture8(ptr %x)
  call void @llvm.lifetime.end.p0(i64 4, ptr %x) nounwind
  br label %l2
}

; Check that @x and @y get distinct stack slots => @x lifetime does not break
; when control re-enters l2.
define void @start_loop() safestack {
; CHECK-LABEL: define void @start_loop()
entry:
; CHECK:        %[[USP:.*]] = load ptr, ptr @__safestack_unsafe_stack_ptr
; CHECK-NEXT:   getelementptr i8, ptr %[[USP]], i32 -16
  %x = alloca i8, align 4
  %y = alloca i8, align 4
  call void @llvm.lifetime.start.p0(i64 4, ptr %x) nounwind
  br label %l2

l2:
; CHECK:   getelementptr i8, ptr %[[USP]], i32 -8
  call void @llvm.lifetime.start.p0(i64 4, ptr %y) nounwind
  call void @capture8(ptr %y)
  call void @llvm.lifetime.end.p0(i64 4, ptr %y) nounwind

; CHECK:   getelementptr i8, ptr %[[USP]], i32 -4
  call void @llvm.lifetime.start.p0(i64 4, ptr %x) nounwind
  call void @capture8(ptr %x)
  br label %l2
}

; This test checks for a bug where the stack coloring algorithm was not tracking
; the live range of allocas through phi instructions, so it did not consider
; alloca and alloca2 to be live at the same time.  As a result it was using
; the same stack slot for both allocas.  To ensure this bug isn't present, we
; check that there are 64 bytes allocated for the unsafe stack which is enough
; space for both allocas.
; CHECK-LABEL: @stack_coloring_liveness_bug
define void @stack_coloring_liveness_bug(i32 %arg0) #0 {
entry:
; CHECK:        %[[USP:.*]] = load ptr, ptr @__safestack_unsafe_stack_ptr
; CHECK-NEXT:   getelementptr i8, ptr %[[USP]], i32 -64
  %alloca = alloca [32 x i8], align 16
  %alloca2 = alloca [32 x i8], align 16
  %cond = icmp eq i32 %arg0, 0
  br i1 %cond, label %if, label %else

if:
  br label %end

else:
; CHECK:   getelementptr i8, ptr %[[USP]], i32 -32
  call void @llvm.lifetime.start.p0(i64 32, ptr nonnull %alloca)
  call void @capture8(ptr %alloca)
  call void @llvm.lifetime.end.p0(i64 32, ptr nonnull %alloca)
  br label %end

end:
; CHECK:   getelementptr i8, ptr %[[USP]], i32 -64
  %alloca.end = phi ptr [ %alloca, %if], [%alloca, %else]
  call void @llvm.lifetime.start.p0(i64 32, ptr nonnull %alloca2)
  call void @llvm.lifetime.start.p0(i64 32, ptr nonnull %alloca.end)
  call void @capture2_8(ptr %alloca2, ptr %alloca.end)
  call void @llvm.lifetime.end.p0(i64 32, ptr nonnull %alloca2)
  call void @llvm.lifetime.end.p0(i64 32, ptr nonnull %alloca.end)
  ret void
}

attributes #0 = { safestack }

declare void @llvm.lifetime.start.p0(i64, ptr nocapture)
declare void @llvm.lifetime.end.p0(i64, ptr nocapture)
declare void @capture8(ptr)
declare void @capture32(ptr)
declare void @capture64(ptr)
declare void @capture100x32(ptr)
declare void @capture2_8(ptr, ptr)