blob: 5083c790011c6d0403c7661db46b71a1d94087df [file] [log] [blame] [edit]
; RUN: opt < %s -passes=tsan -S | FileCheck %s
target datalayout = "e-p:64:64:64-i1:8:8-i8:8:8-i16:16:16-i32:32:32-i64:64:64-f32:32:32-f64:64:64-v64:64:64-v128:128:128-a0:0:64-s0:64:64-f80:128:128-n8:16:32:64-S128"
declare void @escape(ptr)
@sink = global ptr null, align 4
define void @captured0() nounwind uwtable sanitize_thread {
entry:
%ptr = alloca i32, align 4
; escapes due to call
call void @escape(ptr %ptr)
store i32 42, ptr %ptr, align 4
ret void
}
; CHECK-LABEL: define void @captured0
; CHECK: __tsan_write
; CHECK: ret void
define void @captured1() nounwind uwtable sanitize_thread {
entry:
%ptr = alloca i32, align 4
; escapes due to store into global
store ptr %ptr, ptr @sink, align 8
store i32 42, ptr %ptr, align 4
ret void
}
; CHECK-LABEL: define void @captured1
; CHECK: __tsan_write
; CHECK: __tsan_write
; CHECK: ret void
define void @captured2() nounwind uwtable sanitize_thread {
entry:
%ptr = alloca i32, align 4
%tmp = alloca ptr, align 8
; transitive escape
store ptr %ptr, ptr %tmp, align 8
%0 = load ptr, ptr %tmp, align 8
store ptr %0, ptr @sink, align 8
store i32 42, ptr %ptr, align 4
ret void
}
; CHECK-LABEL: define void @captured2
; CHECK: __tsan_write
; CHECK: __tsan_write
; CHECK-NOT: __tsan_write
; CHECK: ret void
define void @captured3() nounwind uwtable sanitize_thread {
entry:
%stkobj = alloca [2 x i32], align 8
; escapes due to store into global
store ptr %stkobj, ptr @sink, align 8
; derived is captured as its base object is captured
%derived = getelementptr inbounds i32, ptr %stkobj, i64 1
store i32 42, ptr %derived, align 4
ret void
}
; CHECK-LABEL: define void @captured3
; CHECK: __tsan_write
; CHECK: __tsan_write
; CHECK: ret void
define void @notcaptured0() nounwind uwtable sanitize_thread {
entry:
%ptr = alloca i32, align 4
store i32 42, ptr %ptr, align 4
; escapes due to call
call void @escape(ptr %ptr)
ret void
}
; CHECK-LABEL: define void @notcaptured0
; CHECK: __tsan_write
; CHECK: ret void
define void @notcaptured1() nounwind uwtable sanitize_thread {
entry:
%ptr = alloca i32, align 4
store i32 42, ptr %ptr, align 4
; escapes due to store into global
store ptr %ptr, ptr @sink, align 8
ret void
}
; CHECK-LABEL: define void @notcaptured1
; CHECK: __tsan_write
; CHECK: __tsan_write
; CHECK: ret void
define void @notcaptured2() nounwind uwtable sanitize_thread {
entry:
%ptr = alloca i32, align 4
%tmp = alloca ptr, align 8
store i32 42, ptr %ptr, align 4
; transitive escape
store ptr %ptr, ptr %tmp, align 8
%0 = load ptr, ptr %tmp, align 8
store ptr %0, ptr @sink, align 8
ret void
}
; CHECK-LABEL: define void @notcaptured2
; CHECK: __tsan_write
; CHECK: __tsan_write
; CHECK-NOT: __tsan_write
; CHECK: ret void
define void @notcaptured3(i1 %cond) nounwind uwtable sanitize_thread {
entry:
%stkobj = alloca [2 x i32], align 8
%derived = getelementptr inbounds i32, ptr %stkobj, i64 1
%ptr = select i1 %cond, ptr %derived, ptr %stkobj
store i32 42, ptr %ptr, align 4
ret void
}
; CHECK-LABEL: define void @notcaptured3
; CHECK-NOT: call void @__tsan_write4(ptr %ptr)
; CHECK: ret void
define void @notcaptured4() nounwind uwtable sanitize_thread {
entry:
%stkobj = alloca [10 x i8], align 1
br label %loop
exit:
ret void
loop:
%count = phi i32 [ 0, %entry ], [ %addone, %loop ]
%derived = phi ptr [ %stkobj, %entry ], [ %ptraddone, %loop ]
store i32 %count, ptr %derived, align 4
%ptraddone = getelementptr inbounds i32, ptr %derived, i64 1
%addone = add nuw nsw i32 %count, 1
%eq10 = icmp eq i32 %addone, 10
br i1 %eq10, label %exit, label %loop
}
; CHECK-LABEL: define void @notcaptured4
; CHECK: ret void
; CHECK-NOT: call void @__tsan_write4(ptr %derived)