OLD | NEW |
(Empty) | |
| 1 ; RUN: opt %s -expand-getelementptr -replace-ptrs-with-ints \ |
| 2 ; RUN: -minsfi-sandbox-memory-accesses -S \ |
| 3 ; RUN: | FileCheck %s -check-prefix=CHECK-GEP |
| 4 ; RUN: opt %s -expand-getelementptr -replace-ptrs-with-ints \ |
| 5 ; RUN: -minsfi-sandbox-memory-accesses -minsfi-ptrsize=20 -S \ |
| 6 ; RUN: | FileCheck %s -check-prefix=CHECK-GEP-MASK |
| 7 ; RUN: opt %s -expand-getelementptr -minsfi-sandbox-memory-accesses \ |
| 8 ; RUN: -minsfi-ptrsize=20 -S \ |
| 9 ; RUN: | FileCheck %s |
| 10 |
| 11 target datalayout = "p:32:32:32" |
| 12 target triple = "le32-unknown-nacl" |
| 13 |
| 14 declare void @llvm.memcpy.p0i8.p0i8.i32(i8* nocapture, i8* nocapture readonly, i
32, i32, i1) |
| 15 declare void @llvm.memmove.p0i8.p0i8.i32(i8* nocapture, i8* nocapture readonly,
i32, i32, i1) |
| 16 declare void @llvm.memset.p0i8.i32(i8* nocapture, i8, i32, i32, i1) |
| 17 |
| 18 ; This test verifies that the pass recognizes the pointer arithmetic pattern |
| 19 ; produced by the ExpandGetElementPtr pass and that it emits a more efficient |
| 20 ; address sandboxing than in the general case. |
| 21 |
| 22 define i32 @test_load_elementptr([100 x i32]* %foo) { |
| 23 %elem = getelementptr inbounds [100 x i32]* %foo, i32 0, i32 97 |
| 24 %val = load i32* %elem |
| 25 ret i32 %val |
| 26 } |
| 27 |
| 28 ; CHECK-GEP-LABEL: define i32 @test_load_elementptr(i32 %foo) { |
| 29 ; CHECK-GEP-NEXT: %mem_base = load i64* @__sfi_memory_base |
| 30 ; CHECK-GEP-NEXT: %1 = zext i32 %foo to i64 |
| 31 ; CHECK-GEP-NEXT: %2 = add i64 %mem_base, %1 |
| 32 ; CHECK-GEP-NEXT: %3 = add i64 %2, 388 |
| 33 ; CHECK-GEP-NEXT: %4 = inttoptr i64 %3 to i32* |
| 34 ; CHECK-GEP-NEXT: %val = load i32* %4 |
| 35 ; CHECK-GEP-NEXT: ret i32 %val |
| 36 ; CHECK-GEP-NEXT: } |
| 37 |
| 38 ; CHECK-GEP-MASK-LABEL: define i32 @test_load_elementptr(i32 %foo) { |
| 39 ; CHECK-GEP-MASK-NEXT: %mem_base = load i64* @__sfi_memory_base |
| 40 ; CHECK-GEP-MASK-NEXT: %1 = and i32 %foo, 1048575 |
| 41 ; CHECK-GEP-MASK-NEXT: %2 = zext i32 %1 to i64 |
| 42 ; CHECK-GEP-MASK-NEXT: %3 = add i64 %mem_base, %2 |
| 43 ; CHECK-GEP-MASK-NEXT: %4 = add i64 %3, 388 |
| 44 ; CHECK-GEP-MASK-NEXT: %5 = inttoptr i64 %4 to i32* |
| 45 ; CHECK-GEP-MASK-NEXT: %val = load i32* %5 |
| 46 ; CHECK-GEP-MASK-NEXT: ret i32 %val |
| 47 ; CHECK-GEP-MASK-NEXT: } |
| 48 |
| 49 define <4 x float> @test_max_offset(i32 %x) { |
| 50 %1 = add i32 %x, 1048560 ; 1MB - 16B |
| 51 %ptr = inttoptr i32 %1 to <4 x float>* |
| 52 %val = load <4 x float>* %ptr |
| 53 ret <4 x float> %val |
| 54 } |
| 55 |
| 56 ; CHECK-LABEL: define <4 x float> @test_max_offset(i32 %x) { |
| 57 ; CHECK-NEXT: %mem_base = load i64* @__sfi_memory_base |
| 58 ; CHECK-NEXT: %1 = and i32 %x, 1048575 |
| 59 ; CHECK-NEXT: %2 = zext i32 %1 to i64 |
| 60 ; CHECK-NEXT: %3 = add i64 %mem_base, %2 |
| 61 ; CHECK-NEXT: %4 = add i64 %3, 1048560 |
| 62 ; CHECK-NEXT: %5 = inttoptr i64 %4 to <4 x float>* |
| 63 ; CHECK-NEXT: %val = load <4 x float>* %5 |
| 64 ; CHECK-NEXT: ret <4 x float> %val |
| 65 ; CHECK-NEXT: } |
| 66 |
| 67 ; This will not get optimized as it could access memory past the guard region. |
| 68 define <4 x float> @test_offset_overflow(i32 %x) { |
| 69 %1 = add i32 %x, 1048561 |
| 70 %ptr = inttoptr i32 %1 to <4 x float>* |
| 71 %val = load <4 x float>* %ptr |
| 72 ret <4 x float> %val |
| 73 } |
| 74 |
| 75 ; CHECK-LABEL: define <4 x float> @test_offset_overflow(i32 %x) { |
| 76 ; CHECK-NEXT: %mem_base = load i64* @__sfi_memory_base |
| 77 ; CHECK-NEXT: %1 = add i32 %x, 1048561 |
| 78 ; CHECK-NEXT: %ptr = inttoptr i32 %1 to <4 x float>* |
| 79 ; CHECK-NEXT: %2 = ptrtoint <4 x float>* %ptr to i32 |
| 80 ; CHECK-NEXT: %3 = and i32 %2, 1048575 |
| 81 ; CHECK-NEXT: %4 = zext i32 %3 to i64 |
| 82 ; CHECK-NEXT: %5 = add i64 %mem_base, %4 |
| 83 ; CHECK-NEXT: %6 = inttoptr i64 %5 to <4 x float>* |
| 84 ; CHECK-NEXT: %val = load <4 x float>* %6 |
| 85 ; CHECK-NEXT: ret <4 x float> %val |
| 86 ; CHECK-NEXT: } |
| 87 |
| 88 define void @test_not_applied_on_memcpy(i32 %x) { |
| 89 %1 = add i32 %x, 1024 |
| 90 %ptr = inttoptr i32 %1 to i8* |
| 91 call void @llvm.memcpy.p0i8.p0i8.i32(i8* %ptr, i8* %ptr, i32 2048, i32 4, i1 f
alse); |
| 92 ret void |
| 93 } |
| 94 |
| 95 ; CHECK-LABEL: define void @test_not_applied_on_memcpy(i32 %x) { |
| 96 ; CHECK: [[IPTR1:%[0-9]+]] = ptrtoint i8* %ptr to i32 |
| 97 ; CHECK-NEXT: [[AND1:%[0-9]+]] = and i32 [[IPTR1]], 1048575 |
| 98 ; CHECK-NEXT: [[ZEXT1:%[0-9]+]] = zext i32 [[AND1]] to i64 |
| 99 ; CHECK-NEXT: [[BASE1:%[0-9]+]] = add i64 %mem_base, [[ZEXT1]] |
| 100 ; CHECK-NEXT: inttoptr i64 [[BASE1]] to i8* |
| 101 ; CHECK: [[IPTR2:%[0-9]+]] = ptrtoint i8* %ptr to i32 |
| 102 ; CHECK-NEXT: [[AND2:%[0-9]+]] = and i32 [[IPTR2]], 1048575 |
| 103 ; CHECK-NEXT: [[ZEXT2:%[0-9]+]] = zext i32 [[AND2]] to i64 |
| 104 ; CHECK-NEXT: [[BASE2:%[0-9]+]] = add i64 %mem_base, [[ZEXT2]] |
| 105 ; CHECK-NEXT: inttoptr i64 [[BASE2]] to i8* |
| 106 ; CHECK: call void @llvm.memcpy.p0i8.p0i8.i32 |
| 107 |
| 108 define void @test_not_applied_on_memmove(i32 %x) { |
| 109 %1 = add i32 %x, 1024 |
| 110 %ptr = inttoptr i32 %1 to i8* |
| 111 call void @llvm.memmove.p0i8.p0i8.i32(i8* %ptr, i8* %ptr, i32 2048, i32 4, i1
false); |
| 112 ret void |
| 113 } |
| 114 |
| 115 ; CHECK-LABEL: define void @test_not_applied_on_memmove(i32 %x) { |
| 116 ; CHECK: [[IPTR1:%[0-9]+]] = ptrtoint i8* %ptr to i32 |
| 117 ; CHECK-NEXT: [[AND1:%[0-9]+]] = and i32 [[IPTR1]], 1048575 |
| 118 ; CHECK-NEXT: [[ZEXT1:%[0-9]+]] = zext i32 [[AND1]] to i64 |
| 119 ; CHECK-NEXT: [[BASE1:%[0-9]+]] = add i64 %mem_base, [[ZEXT1]] |
| 120 ; CHECK-NEXT: inttoptr i64 [[BASE1]] to i8* |
| 121 ; CHECK: [[IPTR2:%[0-9]+]] = ptrtoint i8* %ptr to i32 |
| 122 ; CHECK-NEXT: [[AND2:%[0-9]+]] = and i32 [[IPTR2]], 1048575 |
| 123 ; CHECK-NEXT: [[ZEXT2:%[0-9]+]] = zext i32 [[AND2]] to i64 |
| 124 ; CHECK-NEXT: [[BASE2:%[0-9]+]] = add i64 %mem_base, [[ZEXT2]] |
| 125 ; CHECK-NEXT: inttoptr i64 [[BASE2]] to i8* |
| 126 ; CHECK: call void @llvm.memmove.p0i8.p0i8.i32 |
| 127 |
| 128 define void @test_not_applied_on_memset(i32 %x) { |
| 129 %1 = add i32 %x, 1024 |
| 130 %ptr = inttoptr i32 %1 to i8* |
| 131 call void @llvm.memset.p0i8.i32(i8* %ptr, i8 3, i32 2048, i32 4, i1 false); |
| 132 ret void |
| 133 } |
| 134 |
| 135 ; CHECK-LABEL: define void @test_not_applied_on_memset(i32 %x) { |
| 136 ; CHECK: [[IPTR:%[0-9]+]] = ptrtoint i8* %ptr to i32 |
| 137 ; CHECK-NEXT: [[AND:%[0-9]+]] = and i32 [[IPTR]], 1048575 |
| 138 ; CHECK-NEXT: [[ZEXT:%[0-9]+]] = zext i32 [[AND]] to i64 |
| 139 ; CHECK-NEXT: [[BASE:%[0-9]+]] = add i64 %mem_base, [[ZEXT]] |
| 140 ; CHECK-NEXT: inttoptr i64 [[BASE]] to i8* |
| 141 ; CHECK: call void @llvm.memset.p0i8.i32 |
OLD | NEW |