diff --git a/clang/include/clang/Basic/Builtins.def b/clang/include/clang/Basic/Builtins.def --- a/clang/include/clang/Basic/Builtins.def +++ b/clang/include/clang/Basic/Builtins.def @@ -979,6 +979,7 @@ // when these functions are used in non-GNU mode. PR16138. LIBBUILTIN(alloca, "v*z", "f", "stdlib.h", ALL_GNU_LANGUAGES) // POSIX string.h +LIBBUILTIN(memccpy, "v*v*vC*iz", "f", "string.h", ALL_GNU_LANGUAGES) LIBBUILTIN(stpcpy, "c*c*cC*", "f", "string.h", ALL_GNU_LANGUAGES) LIBBUILTIN(stpncpy, "c*c*cC*z", "f", "string.h", ALL_GNU_LANGUAGES) LIBBUILTIN(strdup, "c*cC*", "f", "string.h", ALL_GNU_LANGUAGES) diff --git a/clang/test/CodeGen/builtin-memfns.c b/clang/test/CodeGen/builtin-memfns.c --- a/clang/test/CodeGen/builtin-memfns.c +++ b/clang/test/CodeGen/builtin-memfns.c @@ -4,6 +4,7 @@ typedef __SIZE_TYPE__ size_t; void *memcpy(void *, void const *, size_t); +void *memccpy(void *, void const *, int, size_t); // CHECK: @test1 // CHECK: call void @llvm.memset.p0i8.i32 @@ -118,3 +119,9 @@ // CHECK: call void @llvm.memcpy{{.*}}( memcpy(&dest_array, &dest_array, 2); } + +// CHECK-LABEL: @test13 +void test13(char *d, char *s, int c, size_t n) { + // CHECK: call i8* @memccpy + memccpy(d, s, c, n); +} diff --git a/clang/test/CodeGen/memccpy-libcall.c b/clang/test/CodeGen/memccpy-libcall.c new file mode 100644 --- /dev/null +++ b/clang/test/CodeGen/memccpy-libcall.c @@ -0,0 +1,12 @@ +// RUN: %clang_cc1 -fno-builtin-memccpy -emit-llvm < %s| FileCheck %s + +typedef __SIZE_TYPE__ size_t; + +void *memccpy(void *, void const *, int, size_t); + +void test13(char *d, char *s, int c, size_t n) { + // CHECK: call i8* @memccpy{{.*}} #2 + memccpy(d, s, c, n); +} + +// CHECK: attributes #2 = { nobuiltin } \ No newline at end of file