|  | // RUN: %clang_cc1 -triple x86_64-apple-darwin -emit-llvm %s -o - 2>&1 | FileCheck %s | 
|  |  | 
|  | #define strcpy(dest, src) \ | 
|  | ((__builtin_object_size(dest, 0) != -1ULL) \ | 
|  | ? __builtin___strcpy_chk (dest, src, __builtin_object_size(dest, 1)) \ | 
|  | : __inline_strcpy_chk(dest, src)) | 
|  |  | 
|  | static char *__inline_strcpy_chk (char *dest, const char *src) { | 
|  | return __builtin___strcpy_chk(dest, src, __builtin_object_size(dest, 1)); | 
|  | } | 
|  |  | 
|  | char gbuf[63]; | 
|  | char *gp; | 
|  | int gi, gj; | 
|  |  | 
|  | // CHECK: define void @test1 | 
|  | void test1() { | 
|  | // CHECK:     = call i8* @__strcpy_chk(i8* getelementptr inbounds ([63 x i8]* @gbuf, i32 0, i64 4), i8* getelementptr inbounds ([9 x i8]* @.str, i32 0, i32 0), i64 59) | 
|  | strcpy(&gbuf[4], "Hi there"); | 
|  | } | 
|  |  | 
|  | // CHECK: define void @test2 | 
|  | void test2() { | 
|  | // CHECK:     = call i8* @__strcpy_chk(i8* getelementptr inbounds ([63 x i8]* @gbuf, i32 0, i32 0), i8* getelementptr inbounds ([9 x i8]* @.str, i32 0, i32 0), i64 63) | 
|  | strcpy(gbuf, "Hi there"); | 
|  | } | 
|  |  | 
|  | // CHECK: define void @test3 | 
|  | void test3() { | 
|  | // CHECK:     = call i8* @__strcpy_chk(i8* getelementptr inbounds ([63 x i8]* @gbuf, i64 1, i64 37), i8* getelementptr inbounds ([9 x i8]* @.str, i32 0, i32 0), i64 0) | 
|  | strcpy(&gbuf[100], "Hi there"); | 
|  | } | 
|  |  | 
|  | // CHECK: define void @test4 | 
|  | void test4() { | 
|  | // CHECK:     = call i8* @__strcpy_chk(i8* getelementptr inbounds ([63 x i8]* @gbuf, i32 0, i64 -1), i8* getelementptr inbounds ([9 x i8]* @.str, i32 0, i32 0), i64 0) | 
|  | strcpy((char*)(void*)&gbuf[-1], "Hi there"); | 
|  | } | 
|  |  | 
|  | // CHECK: define void @test5 | 
|  | void test5() { | 
|  | // CHECK:     = load i8** @gp | 
|  | // CHECK-NEXT:= call i64 @llvm.objectsize.i64(i8* %{{.*}}, i1 false) | 
|  | strcpy(gp, "Hi there"); | 
|  | } | 
|  |  | 
|  | // CHECK: define void @test6 | 
|  | void test6() { | 
|  | char buf[57]; | 
|  |  | 
|  | // CHECK:       = call i8* @__strcpy_chk(i8* %{{.*}}, i8* getelementptr inbounds ([9 x i8]* @.str, i32 0, i32 0), i64 53) | 
|  | strcpy(&buf[4], "Hi there"); | 
|  | } | 
|  |  | 
|  | // CHECK: define void @test7 | 
|  | void test7() { | 
|  | int i; | 
|  | // Ensure we only evaluate the side-effect once. | 
|  | // CHECK:     = add | 
|  | // CHECK-NOT: = add | 
|  | // CHECK:     = call i8* @__strcpy_chk(i8* getelementptr inbounds ([63 x i8]* @gbuf, i32 0, i32 0), i8* getelementptr inbounds ([9 x i8]* @.str, i32 0, i32 0), i64 63) | 
|  | strcpy((++i, gbuf), "Hi there"); | 
|  | } | 
|  |  | 
|  | // CHECK: define void @test8 | 
|  | void test8() { | 
|  | char *buf[50]; | 
|  | // CHECK-NOT:   __strcpy_chk | 
|  | // CHECK:       = call i8* @__inline_strcpy_chk(i8* %{{.*}}, i8* getelementptr inbounds ([9 x i8]* @.str, i32 0, i32 0)) | 
|  | strcpy(buf[++gi], "Hi there"); | 
|  | } | 
|  |  | 
|  | // CHECK: define void @test9 | 
|  | void test9() { | 
|  | // CHECK-NOT:   __strcpy_chk | 
|  | // CHECK:       = call i8* @__inline_strcpy_chk(i8* %{{.*}}, i8* getelementptr inbounds ([9 x i8]* @.str, i32 0, i32 0)) | 
|  | strcpy((char *)((++gi) + gj), "Hi there"); | 
|  | } | 
|  |  | 
|  | // CHECK: define void @test10 | 
|  | char **p; | 
|  | void test10() { | 
|  | // CHECK-NOT:   __strcpy_chk | 
|  | // CHECK:       = call i8* @__inline_strcpy_chk(i8* %{{.*}}, i8* getelementptr inbounds ([9 x i8]* @.str, i32 0, i32 0)) | 
|  | strcpy(*(++p), "Hi there"); | 
|  | } | 
|  |  | 
|  | // CHECK: define void @test11 | 
|  | void test11() { | 
|  | // CHECK-NOT:   __strcpy_chk | 
|  | // CHECK:       = call i8* @__inline_strcpy_chk(i8* getelementptr inbounds ([63 x i8]* @gbuf, i32 0, i32 0), i8* getelementptr inbounds ([9 x i8]* @.str, i32 0, i32 0)) | 
|  | strcpy(gp = gbuf, "Hi there"); | 
|  | } | 
|  |  | 
|  | // CHECK: define void @test12 | 
|  | void test12() { | 
|  | // CHECK-NOT:   __strcpy_chk | 
|  | // CHECK:       = call i8* @__inline_strcpy_chk(i8* %{{.*}}, i8* getelementptr inbounds ([9 x i8]* @.str, i32 0, i32 0)) | 
|  | strcpy(++gp, "Hi there"); | 
|  | } | 
|  |  | 
|  | // CHECK: define void @test13 | 
|  | void test13() { | 
|  | // CHECK-NOT:   __strcpy_chk | 
|  | // CHECK:       = call i8* @__inline_strcpy_chk(i8* %{{.*}}, i8* getelementptr inbounds ([9 x i8]* @.str, i32 0, i32 0)) | 
|  | strcpy(gp++, "Hi there"); | 
|  | } | 
|  |  | 
|  | // CHECK: define void @test14 | 
|  | void test14() { | 
|  | // CHECK-NOT:   __strcpy_chk | 
|  | // CHECK:       = call i8* @__inline_strcpy_chk(i8* %{{.*}}, i8* getelementptr inbounds ([9 x i8]* @.str, i32 0, i32 0)) | 
|  | strcpy(--gp, "Hi there"); | 
|  | } | 
|  |  | 
|  | // CHECK: define void @test15 | 
|  | void test15() { | 
|  | // CHECK-NOT:   __strcpy_chk | 
|  | // CHECK:       = call i8* @__inline_strcpy_chk(i8* %{{..*}}, i8* getelementptr inbounds ([9 x i8]* @.str, i32 0, i32 0)) | 
|  | strcpy(gp--, "Hi there"); | 
|  | } | 
|  |  | 
|  | // CHECK: define void @test16 | 
|  | void test16() { | 
|  | // CHECK-NOT:   __strcpy_chk | 
|  | // CHECK:       = call i8* @__inline_strcpy_chk(i8* %{{.*}}, i8* getelementptr inbounds ([9 x i8]* @.str, i32 0, i32 0)) | 
|  | strcpy(gp += 1, "Hi there"); | 
|  | } | 
|  |  | 
|  | // CHECK: @test17 | 
|  | void test17() { | 
|  | // CHECK: store i32 -1 | 
|  | gi = __builtin_object_size(gp++, 0); | 
|  | // CHECK: store i32 -1 | 
|  | gi = __builtin_object_size(gp++, 1); | 
|  | // CHECK: store i32 0 | 
|  | gi = __builtin_object_size(gp++, 2); | 
|  | // CHECK: store i32 0 | 
|  | gi = __builtin_object_size(gp++, 3); | 
|  | } | 
|  |  | 
|  | // CHECK: @test18 | 
|  | unsigned test18(int cond) { | 
|  | int a[4], b[4]; | 
|  | // CHECK: phi i32* | 
|  | // CHECK: call i64 @llvm.objectsize.i64 | 
|  | return __builtin_object_size(cond ? a : b, 0); | 
|  | } |