Skip to content

Commit 32c8d13

Browse files
author
serge-sans-paille
committed
Improve support of GNU mempcpy
- Lower to the memcpy intrinsic - Raise warnings when size/bounds are known
1 parent 3995129 commit 32c8d13

File tree

6 files changed

+39
-4
lines changed

6 files changed

+39
-4
lines changed

clang/include/clang/Basic/Builtins.def

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -984,6 +984,7 @@ LIBBUILTIN(longjmp, "vJi", "fr", "setjmp.h", ALL_LANGUAGES)
984984
LIBBUILTIN(alloca, "v*z", "f", "stdlib.h", ALL_GNU_LANGUAGES)
985985
// POSIX string.h
986986
LIBBUILTIN(memccpy, "v*v*vC*iz", "f", "string.h", ALL_GNU_LANGUAGES)
987+
LIBBUILTIN(mempcpy, "v*v*vC*z", "f", "string.h", ALL_GNU_LANGUAGES)
987988
LIBBUILTIN(stpcpy, "c*c*cC*", "f", "string.h", ALL_GNU_LANGUAGES)
988989
LIBBUILTIN(stpncpy, "c*c*cC*z", "f", "string.h", ALL_GNU_LANGUAGES)
989990
LIBBUILTIN(strdup, "c*cC*", "f", "string.h", ALL_GNU_LANGUAGES)

clang/lib/AST/Decl.cpp

Lines changed: 7 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -3866,6 +3866,11 @@ unsigned FunctionDecl::getMemoryFunctionKind() const {
38663866
case Builtin::BImemcpy:
38673867
return Builtin::BImemcpy;
38683868

3869+
case Builtin::BI__builtin_mempcpy:
3870+
case Builtin::BI__builtin___mempcpy_chk:
3871+
case Builtin::BImempcpy:
3872+
return Builtin::BImempcpy;
3873+
38693874
case Builtin::BI__builtin_memmove:
38703875
case Builtin::BI__builtin___memmove_chk:
38713876
case Builtin::BImemmove:
@@ -3923,6 +3928,8 @@ unsigned FunctionDecl::getMemoryFunctionKind() const {
39233928
return Builtin::BImemset;
39243929
else if (FnInfo->isStr("memcpy"))
39253930
return Builtin::BImemcpy;
3931+
else if (FnInfo->isStr("mempcpy"))
3932+
return Builtin::BImempcpy;
39263933
else if (FnInfo->isStr("memmove"))
39273934
return Builtin::BImemmove;
39283935
else if (FnInfo->isStr("memcmp"))

clang/lib/CodeGen/CGBuiltin.cpp

Lines changed: 8 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -2506,7 +2506,9 @@ RValue CodeGenFunction::EmitBuiltinExpr(const GlobalDecl GD, unsigned BuiltinID,
25062506
return RValue::get(nullptr);
25072507
}
25082508
case Builtin::BImemcpy:
2509-
case Builtin::BI__builtin_memcpy: {
2509+
case Builtin::BI__builtin_memcpy:
2510+
case Builtin::BImempcpy:
2511+
case Builtin::BI__builtin_mempcpy: {
25102512
Address Dest = EmitPointerWithAlignment(E->getArg(0));
25112513
Address Src = EmitPointerWithAlignment(E->getArg(1));
25122514
Value *SizeVal = EmitScalarExpr(E->getArg(2));
@@ -2515,7 +2517,11 @@ RValue CodeGenFunction::EmitBuiltinExpr(const GlobalDecl GD, unsigned BuiltinID,
25152517
EmitNonNullArgCheck(RValue::get(Src.getPointer()), E->getArg(1)->getType(),
25162518
E->getArg(1)->getExprLoc(), FD, 1);
25172519
Builder.CreateMemCpy(Dest, Src, SizeVal, false);
2518-
return RValue::get(Dest.getPointer());
2520+
if (BuiltinID == Builtin::BImempcpy ||
2521+
BuiltinID == Builtin::BI__builtin_mempcpy)
2522+
return RValue::get(Builder.CreateInBoundsGEP(Dest.getPointer(), SizeVal));
2523+
else
2524+
return RValue::get(Dest.getPointer());
25192525
}
25202526

25212527
case Builtin::BI__builtin_char_memchr:

clang/lib/Sema/SemaChecking.cpp

Lines changed: 5 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -340,7 +340,8 @@ void Sema::checkFortifiedBuiltinMemoryFunction(FunctionDecl *FD,
340340
case Builtin::BI__builtin___strncat_chk:
341341
case Builtin::BI__builtin___strncpy_chk:
342342
case Builtin::BI__builtin___stpncpy_chk:
343-
case Builtin::BI__builtin___memccpy_chk: {
343+
case Builtin::BI__builtin___memccpy_chk:
344+
case Builtin::BI__builtin___mempcpy_chk: {
344345
DiagID = diag::warn_builtin_chk_overflow;
345346
IsChkVariant = true;
346347
SizeIndex = TheCall->getNumArgs() - 2;
@@ -379,7 +380,9 @@ void Sema::checkFortifiedBuiltinMemoryFunction(FunctionDecl *FD,
379380
case Builtin::BImemmove:
380381
case Builtin::BI__builtin_memmove:
381382
case Builtin::BImemset:
382-
case Builtin::BI__builtin_memset: {
383+
case Builtin::BI__builtin_memset:
384+
case Builtin::BImempcpy:
385+
case Builtin::BI__builtin_mempcpy: {
383386
DiagID = diag::warn_fortify_source_overflow;
384387
SizeIndex = TheCall->getNumArgs() - 1;
385388
ObjectIndex = 0;

clang/test/Analysis/bstring.c

Lines changed: 6 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -222,6 +222,9 @@ void mempcpy2 () {
222222
char dst[1];
223223

224224
mempcpy(dst, src, 4); // expected-warning{{Memory copy function overflows destination buffer}}
225+
#ifndef VARIANT
226+
// expected-warning@-2{{'mempcpy' will always overflow; destination buffer has size 1, but size argument is 4}}
227+
#endif
225228
}
226229

227230
void mempcpy3 () {
@@ -243,6 +246,9 @@ void mempcpy5() {
243246
char dst[3];
244247

245248
mempcpy(dst+2, src+2, 2); // expected-warning{{Memory copy function overflows destination buffer}}
249+
#ifndef VARIANT
250+
// expected-warning@-2{{'mempcpy' will always overflow; destination buffer has size 1, but size argument is 2}}
251+
#endif
246252
}
247253

248254
void mempcpy6() {
Lines changed: 12 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
// RUN: %clang_cc1 -emit-llvm < %s| FileCheck %s
2+
3+
typedef __SIZE_TYPE__ size_t;
4+
5+
void *mempcpy(void *, void const *, size_t);
6+
7+
char *test(char *d, char *s, size_t n) {
8+
// CHECK: call void @llvm.memcpy.p0i8.p0i8.i64(i8* {{.*}} %[[REG1:[^ ]+]], i8* {{.*}} %1, i64 %[[REG2:[^ ]+]], i1 false)
9+
// CHECK-NEXT: %[[REGr:[^ ]+]] = getelementptr inbounds i8, i8* %[[REG1]], i64 %[[REG2]]
10+
// CHECK-NEXT: ret i8* %[[REGr]]
11+
return mempcpy(d, s, n);
12+
}

0 commit comments

Comments
 (0)