From 2cae7025c1d9e83d971f670c9a66497a8c1094ec Mon Sep 17 00:00:00 2001 From: =?utf8?q?D=C3=A1vid=20Bolvansk=C3=BD?= Date: Fri, 23 Apr 2021 14:05:57 +0200 Subject: [PATCH] Reland "[Clang] Propagate guaranteed alignment for malloc and others" This relands commit 6914a0ed2b30924b188968e59a83efa07ac5fe57. Crash in InstCombine was fixed. --- clang/include/clang/Basic/TargetInfo.h | 4 ++-- clang/lib/CodeGen/CGCall.cpp | 18 ++++++++++++++++ clang/test/CodeGen/alloc-fns-alignment.c | 35 ++++++++++++++++++++++++++++++++ 3 files changed, 55 insertions(+), 2 deletions(-) create mode 100644 clang/test/CodeGen/alloc-fns-alignment.c diff --git a/clang/include/clang/Basic/TargetInfo.h b/clang/include/clang/Basic/TargetInfo.h index 3bcaace..449c026 100644 --- a/clang/include/clang/Basic/TargetInfo.h +++ b/clang/include/clang/Basic/TargetInfo.h @@ -612,8 +612,8 @@ public: } /// Return the largest alignment for which a suitably-sized allocation with - /// '::operator new(size_t)' is guaranteed to produce a correctly-aligned - /// pointer. + /// '::operator new(size_t)' or 'malloc' is guaranteed to produce a + /// correctly-aligned pointer. unsigned getNewAlign() const { return NewAlign ? NewAlign : std::max(LongDoubleAlign, LongLongAlign); } diff --git a/clang/lib/CodeGen/CGCall.cpp b/clang/lib/CodeGen/CGCall.cpp index 366c83e..6c77f18 100644 --- a/clang/lib/CodeGen/CGCall.cpp +++ b/clang/lib/CodeGen/CGCall.cpp @@ -2048,6 +2048,24 @@ void CodeGenModule::ConstructAttributeList( // allows it to work on indirect virtual function calls. if (AttrOnCallSite && TargetDecl->hasAttr()) FuncAttrs.addAttribute(llvm::Attribute::NoMerge); + + // Add known guaranteed alignment for allocation functions. + if (unsigned BuiltinID = Fn->getBuiltinID()) { + switch (BuiltinID) { + case Builtin::BIaligned_alloc: + case Builtin::BIcalloc: + case Builtin::BImalloc: + case Builtin::BImemalign: + case Builtin::BIrealloc: + case Builtin::BIstrdup: + case Builtin::BIstrndup: + RetAttrs.addAlignmentAttr(Context.getTargetInfo().getNewAlign() / + Context.getTargetInfo().getCharWidth()); + break; + default: + break; + } + } } // 'const', 'pure' and 'noalias' attributed functions are also nounwind. diff --git a/clang/test/CodeGen/alloc-fns-alignment.c b/clang/test/CodeGen/alloc-fns-alignment.c new file mode 100644 index 0000000..d2f9b46 --- /dev/null +++ b/clang/test/CodeGen/alloc-fns-alignment.c @@ -0,0 +1,35 @@ +// RUN: %clang_cc1 -triple x86_64-unknown-linux-gnu -emit-llvm < %s | FileCheck %s --check-prefix=ALIGN16 +// RUN: %clang_cc1 -triple x86_64-windows-msvc -emit-llvm < %s | FileCheck %s --check-prefix=ALIGN16 +// RUN: %clang_cc1 -triple i386-apple-darwin -emit-llvm < %s | FileCheck %s --check-prefix=ALIGN16 +// RUN: %clang_cc1 -triple i386-unknown-linux-gnu -emit-llvm < %s | FileCheck %s --check-prefix=ALIGN8 +// RUN: %clang_cc1 -triple x86_64-unknown-linux-gnu -fno-builtin-malloc -emit-llvm < %s | FileCheck %s --check-prefix=NOBUILTIN-MALLOC +// RUN: %clang_cc1 -triple x86_64-unknown-linux-gnu -fno-builtin-calloc -emit-llvm < %s | FileCheck %s --check-prefix=NOBUILTIN-CALLOC +// RUN: %clang_cc1 -triple x86_64-unknown-linux-gnu -fno-builtin-realloc -emit-llvm < %s | FileCheck %s --check-prefix=NOBUILTIN-REALLOC + +typedef __SIZE_TYPE__ size_t; + +void *malloc(size_t); +void *calloc(size_t, size_t); +void *realloc(void *, size_t); + +void *malloc_test(size_t n) { + return malloc(n); +} + +void *calloc_test(size_t n) { + return calloc(1, n); +} + +void *raalloc_test(void *p, size_t n) { + return realloc(p, n); +} + +// ALIGN16: align 16 i8* @malloc +// ALIGN16: align 16 i8* @calloc +// ALIGN16: align 16 i8* @realloc +// ALIGN8: align 8 i8* @malloc +// ALIGN8: align 8 i8* @calloc +// ALIGN8: align 8 i8* @realloc +// NOBUILTIN-MALLOC: declare i8* @malloc +// NOBUILTIN-CALLOC: declare i8* @calloc +// NOBUILTIN-REALLOC: declare i8* @realloc -- 2.7.4