-
Notifications
You must be signed in to change notification settings - Fork 12.5k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
[llvm] Improve llvm.objectsize computation by computing GEP, alloca a… (
#117849) …nd malloc parameters bound Using a naive expression walker, it is possible to compute valuable information for allocation functions, GEP and alloca, even in the presence of some dynamic information. We don't rely on computeConstantRange to avoid taking advantage of undefined behavior, which would be counter-productive wrt. usual llvm.objectsize usage. llvm.objectsize plays an important role in _FORTIFY_SOURCE definitions, so improving its diagnostic in turns improves the security of compiled application. As a side note, as a result of recent optimization improvements, clang no longer passes https://github.com/serge-sans-paille/builtin_object_size-test-suite This commit restores the situation and greatly improves the scope of code handled by the static version of __builtin_object_size. This is a recommit of #115522 with fix applied.
- Loading branch information
1 parent
ef2e590
commit f8c1a22
Showing
3 changed files
with
217 additions
and
8 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
109 changes: 109 additions & 0 deletions
109
llvm/test/Transforms/LowerConstantIntrinsics/builtin-object-size-range.ll
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,109 @@ | ||
; NOTE: Assertions have been autogenerated by utils/update_test_checks.py | ||
; RUN: opt -passes=lower-constant-intrinsics -S < %s | FileCheck %s | ||
|
||
target datalayout = "e-m:e-p270:32:32-p271:32:32-p272:64:64-i64:64-f80:128-n8:16:32:64-S128" | ||
target triple = "x86_64-unknown-linux-gnu" | ||
|
||
declare i64 @llvm.objectsize.i64.p0(ptr, i1 immarg, i1 immarg, i1 immarg) | ||
declare noalias ptr @malloc(i64 noundef) #0 | ||
|
||
define i64 @select_alloc_size(i1 %cond) { | ||
; CHECK-LABEL: @select_alloc_size( | ||
; CHECK-NEXT: [[SIZE:%.*]] = select i1 [[COND:%.*]], i64 3, i64 4 | ||
; CHECK-NEXT: [[PTR:%.*]] = alloca i8, i64 [[SIZE]], align 1 | ||
; CHECK-NEXT: [[RES:%.*]] = select i1 [[COND]], i64 4, i64 3 | ||
; CHECK-NEXT: ret i64 [[RES]] | ||
; | ||
%size = select i1 %cond, i64 3, i64 4 | ||
%ptr = alloca i8, i64 %size | ||
%objsize_max = call i64 @llvm.objectsize.i64.p0(ptr %ptr, i1 false, i1 true, i1 false) | ||
%objsize_min = call i64 @llvm.objectsize.i64.p0(ptr %ptr, i1 true, i1 true, i1 false) | ||
%res = select i1 %cond, i64 %objsize_max, i64 %objsize_min | ||
ret i64 %res | ||
} | ||
|
||
define i64 @select_malloc_size(i1 %cond) { | ||
; CHECK-LABEL: @select_malloc_size( | ||
; CHECK-NEXT: [[SIZE:%.*]] = select i1 [[COND:%.*]], i64 3, i64 4 | ||
; CHECK-NEXT: [[PTR:%.*]] = call noalias ptr @malloc(i64 noundef [[SIZE]]) | ||
; CHECK-NEXT: [[RES:%.*]] = select i1 [[COND]], i64 4, i64 3 | ||
; CHECK-NEXT: ret i64 [[RES]] | ||
; | ||
%size = select i1 %cond, i64 3, i64 4 | ||
%ptr = call noalias ptr @malloc(i64 noundef %size) | ||
%objsize_max = call i64 @llvm.objectsize.i64.p0(ptr %ptr, i1 false, i1 true, i1 false) | ||
%objsize_min = call i64 @llvm.objectsize.i64.p0(ptr %ptr, i1 true, i1 true, i1 false) | ||
%res = select i1 %cond, i64 %objsize_max, i64 %objsize_min | ||
ret i64 %res | ||
} | ||
|
||
define i64 @select_gep_offset(i1 %cond) { | ||
; CHECK-LABEL: @select_gep_offset( | ||
; CHECK-NEXT: [[PTR:%.*]] = alloca i8, i64 10, align 1 | ||
; CHECK-NEXT: [[OFFSET:%.*]] = select i1 [[COND:%.*]], i64 3, i64 4 | ||
; CHECK-NEXT: [[PTR_SLIDE:%.*]] = getelementptr inbounds i8, ptr [[PTR]], i64 [[OFFSET]] | ||
; CHECK-NEXT: [[RES:%.*]] = select i1 [[COND]], i64 7, i64 6 | ||
; CHECK-NEXT: ret i64 [[RES]] | ||
; | ||
%ptr = alloca i8, i64 10 | ||
%offset = select i1 %cond, i64 3, i64 4 | ||
%ptr.slide = getelementptr inbounds i8, ptr %ptr, i64 %offset | ||
%objsize_max = call i64 @llvm.objectsize.i64.p0(ptr %ptr.slide, i1 false, i1 true, i1 false) | ||
%objsize_min = call i64 @llvm.objectsize.i64.p0(ptr %ptr.slide, i1 true, i1 true, i1 false) | ||
%res = select i1 %cond, i64 %objsize_max, i64 %objsize_min | ||
ret i64 %res | ||
} | ||
|
||
define i64 @select_gep_neg_offset(i1 %c0, i1 %c1) { | ||
; CHECK-LABEL: @select_gep_neg_offset( | ||
; CHECK-NEXT: [[PTR:%.*]] = alloca i8, i64 10, align 1 | ||
; CHECK-NEXT: [[PTR_SLIDE_1:%.*]] = getelementptr inbounds i8, ptr [[PTR]], i64 5 | ||
; CHECK-NEXT: [[OFFSET:%.*]] = select i1 [[COND:%.*]], i64 -3, i64 -4 | ||
; CHECK-NEXT: [[PTR_SLIDE_2:%.*]] = getelementptr inbounds i8, ptr [[PTR_SLIDE_1]], i64 [[OFFSET]] | ||
; CHECK-NEXT: [[RES:%.*]] = select i1 [[C1:%.*]], i64 9, i64 8 | ||
; CHECK-NEXT: ret i64 [[RES]] | ||
; | ||
%ptr = alloca i8, i64 10 | ||
%ptr.slide.1 = getelementptr inbounds i8, ptr %ptr, i64 5 | ||
%offset = select i1 %c0, i64 -3, i64 -4 | ||
%ptr.slide.2 = getelementptr inbounds i8, ptr %ptr.slide.1, i64 %offset | ||
%objsize_max = call i64 @llvm.objectsize.i64.p0(ptr %ptr.slide.2, i1 false, i1 true, i1 false) | ||
%objsize_min = call i64 @llvm.objectsize.i64.p0(ptr %ptr.slide.2, i1 true, i1 true, i1 false) | ||
%res = select i1 %c1, i64 %objsize_max, i64 %objsize_min | ||
ret i64 %res | ||
} | ||
|
||
define i64 @select_neg_oob_offset(i1 %c0, i1 %c1) { | ||
; CHECK-LABEL: @select_neg_oob_offset( | ||
; CHECK-NEXT: [[PTR:%.*]] = alloca i8, i64 10, align 1 | ||
; CHECK-NEXT: [[OFFSET:%.*]] = select i1 [[C0:%.*]], i64 -3, i64 -4 | ||
; CHECK-NEXT: [[PTR_SLIDE:%.*]] = getelementptr inbounds i8, ptr [[PTR]], i64 [[OFFSET]] | ||
; CHECK-NEXT: ret i64 0 | ||
; | ||
%ptr = alloca i8, i64 10 | ||
%offset = select i1 %c0, i64 -3, i64 -4 | ||
%ptr.slide = getelementptr inbounds i8, ptr %ptr, i64 %offset | ||
%objsize_max = call i64 @llvm.objectsize.i64.p0(ptr %ptr.slide, i1 false, i1 true, i1 false) | ||
%objsize_min = call i64 @llvm.objectsize.i64.p0(ptr %ptr.slide, i1 true, i1 true, i1 false) | ||
%res = select i1 %c1, i64 %objsize_max, i64 %objsize_min | ||
ret i64 %res | ||
} | ||
|
||
define i64 @select_gep_offsets(i1 %cond) { | ||
; CHECK-LABEL: @select_gep_offsets( | ||
; CHECK-NEXT: [[PTR:%.*]] = alloca [10 x i8], i64 2, align 1 | ||
; CHECK-NEXT: [[OFFSET:%.*]] = select i1 [[COND:%.*]], i32 0, i32 1 | ||
; CHECK-NEXT: [[PTR_SLIDE:%.*]] = getelementptr inbounds [10 x i8], ptr [[PTR]], i32 [[OFFSET]], i32 5 | ||
; CHECK-NEXT: [[RES:%.*]] = select i1 [[COND]], i64 15, i64 5 | ||
; CHECK-NEXT: ret i64 [[RES]] | ||
; | ||
%ptr = alloca [10 x i8], i64 2 | ||
%offset = select i1 %cond, i32 0, i32 1 | ||
%ptr.slide = getelementptr inbounds [10 x i8], ptr %ptr, i32 %offset, i32 5 | ||
%objsize_max = call i64 @llvm.objectsize.i64.p0(ptr %ptr.slide, i1 false, i1 true, i1 false) | ||
%objsize_min = call i64 @llvm.objectsize.i64.p0(ptr %ptr.slide, i1 true, i1 true, i1 false) | ||
%res = select i1 %cond, i64 %objsize_max, i64 %objsize_min | ||
ret i64 %res | ||
} | ||
|
||
attributes #0 = { nounwind allocsize(0) } |