Skip to content

Commit 9545976

Browse files
committed
Revert "[Clang] Propagate guaranteed alignment for malloc and others"
The above change assumed that malloc (and friends) would always allocate memory to getNewAlign(), even for allocations which have a smaller size. This is not actually required by spec (a 1-byte allocation may validly have 1-byte alignment). Some real-world malloc implementations do not provide this guarantee, and thus this optimization is breaking programs. Fixes #53540 This reverts commit c229754. Differential Revision: https://reviews.llvm.org/D118804
1 parent 34d557f commit 9545976

File tree

3 files changed

+25
-65
lines changed

3 files changed

+25
-65
lines changed

clang/include/clang/Basic/TargetInfo.h

Lines changed: 2 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -646,8 +646,8 @@ class TargetInfo : public virtual TransferrableTargetInfo,
646646
}
647647

648648
/// Return the largest alignment for which a suitably-sized allocation with
649-
/// '::operator new(size_t)' or 'malloc' is guaranteed to produce a
650-
/// correctly-aligned pointer.
649+
/// '::operator new(size_t)' is guaranteed to produce a correctly-aligned
650+
/// pointer.
651651
unsigned getNewAlign() const {
652652
return NewAlign ? NewAlign : std::max(LongDoubleAlign, LongLongAlign);
653653
}

clang/lib/Sema/SemaDecl.cpp

Lines changed: 0 additions & 16 deletions
Original file line numberDiff line numberDiff line change
@@ -15320,23 +15320,7 @@ void Sema::AddKnownFunctionAttributes(FunctionDecl *FD) {
1532015320
if (!FD->hasAttr<AllocAlignAttr>())
1532115321
FD->addAttr(AllocAlignAttr::CreateImplicit(Context, ParamIdx(1, FD),
1532215322
FD->getLocation()));
15323-
LLVM_FALLTHROUGH;
15324-
case Builtin::BIcalloc:
15325-
case Builtin::BImalloc:
15326-
case Builtin::BIrealloc:
15327-
case Builtin::BIstrdup:
15328-
case Builtin::BIstrndup: {
15329-
if (!FD->hasAttr<AssumeAlignedAttr>()) {
15330-
unsigned NewAlign = Context.getTargetInfo().getNewAlign() /
15331-
Context.getTargetInfo().getCharWidth();
15332-
IntegerLiteral *Alignment = IntegerLiteral::Create(
15333-
Context, Context.MakeIntValue(NewAlign, Context.UnsignedIntTy),
15334-
Context.UnsignedIntTy, FD->getLocation());
15335-
FD->addAttr(AssumeAlignedAttr::CreateImplicit(
15336-
Context, Alignment, /*Offset=*/nullptr, FD->getLocation()));
15337-
}
1533815323
break;
15339-
}
1534015324
default:
1534115325
break;
1534215326
}
Lines changed: 23 additions & 47 deletions
Original file line numberDiff line numberDiff line change
@@ -1,12 +1,9 @@
1-
// RUN: %clang_cc1 -triple x86_64-unknown-linux-gnu -emit-llvm < %s | FileCheck %s --check-prefix=ALIGN16
2-
// RUN: %clang_cc1 -triple x86_64-windows-msvc -emit-llvm < %s | FileCheck %s --check-prefix=ALIGN16
3-
// RUN: %clang_cc1 -triple i386-apple-darwin -emit-llvm < %s | FileCheck %s --check-prefix=ALIGN16
4-
// RUN: %clang_cc1 -triple i386-unknown-linux-gnu -emit-llvm < %s | FileCheck %s --check-prefix=ALIGN8
5-
// RUN: %clang_cc1 -triple x86_64-unknown-linux-gnu -fno-builtin-malloc -emit-llvm < %s | FileCheck %s --check-prefix=NOBUILTIN-MALLOC
6-
// RUN: %clang_cc1 -triple x86_64-unknown-linux-gnu -fno-builtin-calloc -emit-llvm < %s | FileCheck %s --check-prefix=NOBUILTIN-CALLOC
7-
// RUN: %clang_cc1 -triple x86_64-unknown-linux-gnu -fno-builtin-realloc -emit-llvm < %s | FileCheck %s --check-prefix=NOBUILTIN-REALLOC
8-
// RUN: %clang_cc1 -triple x86_64-unknown-linux-gnu -fno-builtin-aligned_alloc -emit-llvm < %s | FileCheck %s --check-prefix=NOBUILTIN-ALIGNED_ALLOC
9-
// RUN: %clang_cc1 -triple x86_64-unknown-linux-gnu -fno-builtin-memalign -emit-llvm < %s | FileCheck %s --check-prefix=NOBUILTIN-MEMALIGN
1+
// RUN: %clang_cc1 -triple x86_64-unknown-linux-gnu -emit-llvm < %s | FileCheck %s
2+
3+
// Note: this test originally asserted that malloc/calloc/realloc got alignment
4+
// attributes on their return pointer. However, that was reverted in
5+
// https://reviews.llvm.org/D118804 and it now asserts that they do _NOT_ get
6+
// align attributes.
107

118
typedef __SIZE_TYPE__ size_t;
129

@@ -49,57 +46,36 @@ void *memalign_large_constant_test(size_t n) {
4946
}
5047

5148
// CHECK-LABEL: @malloc_test
52-
// ALIGN16: align 16 i8* @malloc
53-
54-
// CHECK-LABEL: @calloc_test
55-
// ALIGN16: align 16 i8* @calloc
56-
57-
// CHECK-LABEL: @realloc_test
58-
// ALIGN16: align 16 i8* @realloc
49+
// CHECK: call i8* @malloc
5950

60-
// CHECK-LABEL: @aligned_alloc_variable_test
61-
// ALIGN16: %[[ALLOCATED:.*]] = call align 16 i8* @aligned_alloc({{i32|i64}} noundef %[[ALIGN:.*]], {{i32|i64}} noundef %[[NBYTES:.*]])
62-
// ALIGN16-NEXT: call void @llvm.assume(i1 true) [ "align"(i8* %[[ALLOCATED]], {{i32|i64}} %[[ALIGN]]) ]
63-
64-
// CHECK-LABEL: @memalign_variable_test
65-
// ALIGN16: %[[ALLOCATED:.*]] = call align 16 i8* @memalign({{i32|i64}} noundef %[[ALIGN:.*]], {{i32|i64}} noundef %[[NBYTES:.*]])
66-
// ALIGN16-NEXT: call void @llvm.assume(i1 true) [ "align"(i8* %[[ALLOCATED]], {{i32|i64}} %[[ALIGN]]) ]
67-
68-
// CHECK-LABEL: @aligned_alloc_constant_test
69-
// ALIGN16: align 16 i8* @aligned_alloc
70-
71-
// CHECK-LABEL: @aligned_alloc_large_constant_test
72-
// ALIGN16: align 4096 i8* @aligned_alloc
73-
74-
// CHECK-LABEL: @memalign_large_constant_test
75-
// ALIGN16: align 4096 i8* @memalign
76-
77-
// CHECK-LABEL: @malloc_test
78-
// ALIGN8: align 8 i8* @malloc
51+
// CHECK: declare i8* @malloc
7952

8053
// CHECK-LABEL: @calloc_test
81-
// ALIGN8: align 8 i8* @calloc
54+
// CHECK: call i8* @calloc
55+
56+
// CHECK: declare i8* @calloc
8257

8358
// CHECK-LABEL: @realloc_test
84-
// ALIGN8: align 8 i8* @realloc
59+
// CHECK: call i8* @realloc
60+
61+
// CHECK: declare i8* @realloc
8562

8663
// CHECK-LABEL: @aligned_alloc_variable_test
87-
// ALIGN8: align 8 i8* @aligned_alloc
64+
// CHECK: %[[ALLOCATED:.*]] = call i8* @aligned_alloc({{i32|i64}} noundef %[[ALIGN:.*]], {{i32|i64}} noundef %[[NBYTES:.*]])
65+
// CHECK-NEXT: call void @llvm.assume(i1 true) [ "align"(i8* %[[ALLOCATED]], {{i32|i64}} %[[ALIGN]]) ]
66+
67+
// CHECK: declare i8* @aligned_alloc
8868

8969
// CHECK-LABEL: @memalign_variable_test
90-
// ALIGN8: align 8 i8* @memalign
70+
// CHECK: %[[ALLOCATED:.*]] = call i8* @memalign({{i32|i64}} noundef %[[ALIGN:.*]], {{i32|i64}} noundef %[[NBYTES:.*]])
71+
// CHECK-NEXT: call void @llvm.assume(i1 true) [ "align"(i8* %[[ALLOCATED]], {{i32|i64}} %[[ALIGN]]) ]
9172

9273
// CHECK-LABEL: @aligned_alloc_constant_test
93-
// ALIGN8: align 8 i8* @aligned_alloc
74+
// CHECK: call align 8 i8* @aligned_alloc
9475

9576
// CHECK-LABEL: @aligned_alloc_large_constant_test
96-
// ALIGN8: align 4096 i8* @aligned_alloc
77+
// CHECK: call align 4096 i8* @aligned_alloc
9778

9879
// CHECK-LABEL: @memalign_large_constant_test
99-
// ALIGN8: align 4096 i8* @memalign
80+
// CHECK: align 4096 i8* @memalign
10081

101-
// NOBUILTIN-MALLOC: declare i8* @malloc
102-
// NOBUILTIN-CALLOC: declare i8* @calloc
103-
// NOBUILTIN-REALLOC: declare i8* @realloc
104-
// NOBUILTIN-ALIGNED_ALLOC: declare i8* @aligned_alloc
105-
// NOBUILTIN-MEMALIGN: declare i8* @memalign

0 commit comments

Comments
 (0)