Add range attribute to scalar function results and arguments
This commit is contained in:
parent
9bad7ba324
commit
cfadfabfcd
@ -415,9 +415,32 @@ fn apply_attrs_llfn(&self, cx: &CodegenCx<'ll, 'tcx>, llfn: &'ll Value) {
|
|||||||
i += 1;
|
i += 1;
|
||||||
i - 1
|
i - 1
|
||||||
};
|
};
|
||||||
|
|
||||||
|
let apply_range_attr = |idx: AttributePlace, scalar: rustc_target::abi::Scalar| {
|
||||||
|
if cx.sess().opts.optimize != config::OptLevel::No
|
||||||
|
&& llvm_util::get_version() >= (19, 0, 0)
|
||||||
|
&& matches!(scalar.primitive(), Int(..))
|
||||||
|
// If the value is a boolean, the range is 0..2 and that ultimately
|
||||||
|
// become 0..0 when the type becomes i1, which would be rejected
|
||||||
|
// by the LLVM verifier.
|
||||||
|
&& !scalar.is_bool()
|
||||||
|
// LLVM also rejects full range.
|
||||||
|
&& !scalar.is_always_valid(cx)
|
||||||
|
{
|
||||||
|
attributes::apply_to_llfn(
|
||||||
|
llfn,
|
||||||
|
idx,
|
||||||
|
&[llvm::CreateRangeAttr(cx.llcx, scalar.size(cx), scalar.valid_range(cx))],
|
||||||
|
);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
match &self.ret.mode {
|
match &self.ret.mode {
|
||||||
PassMode::Direct(attrs) => {
|
PassMode::Direct(attrs) => {
|
||||||
attrs.apply_attrs_to_llfn(llvm::AttributePlace::ReturnValue, cx, llfn);
|
attrs.apply_attrs_to_llfn(llvm::AttributePlace::ReturnValue, cx, llfn);
|
||||||
|
if let abi::Abi::Scalar(scalar) = self.ret.layout.abi {
|
||||||
|
apply_range_attr(llvm::AttributePlace::ReturnValue, scalar);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
PassMode::Indirect { attrs, meta_attrs: _, on_stack } => {
|
PassMode::Indirect { attrs, meta_attrs: _, on_stack } => {
|
||||||
assert!(!on_stack);
|
assert!(!on_stack);
|
||||||
@ -456,8 +479,13 @@ fn apply_attrs_llfn(&self, cx: &CodegenCx<'ll, 'tcx>, llfn: &'ll Value) {
|
|||||||
);
|
);
|
||||||
attributes::apply_to_llfn(llfn, llvm::AttributePlace::Argument(i), &[byval]);
|
attributes::apply_to_llfn(llfn, llvm::AttributePlace::Argument(i), &[byval]);
|
||||||
}
|
}
|
||||||
PassMode::Direct(attrs)
|
PassMode::Direct(attrs) => {
|
||||||
| PassMode::Indirect { attrs, meta_attrs: None, on_stack: false } => {
|
let i = apply(attrs);
|
||||||
|
if let abi::Abi::Scalar(scalar) = arg.layout.abi {
|
||||||
|
apply_range_attr(llvm::AttributePlace::Argument(i), scalar);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
PassMode::Indirect { attrs, meta_attrs: None, on_stack: false } => {
|
||||||
apply(attrs);
|
apply(attrs);
|
||||||
}
|
}
|
||||||
PassMode::Indirect { attrs, meta_attrs: Some(meta_attrs), on_stack } => {
|
PassMode::Indirect { attrs, meta_attrs: Some(meta_attrs), on_stack } => {
|
||||||
@ -466,8 +494,12 @@ fn apply_attrs_llfn(&self, cx: &CodegenCx<'ll, 'tcx>, llfn: &'ll Value) {
|
|||||||
apply(meta_attrs);
|
apply(meta_attrs);
|
||||||
}
|
}
|
||||||
PassMode::Pair(a, b) => {
|
PassMode::Pair(a, b) => {
|
||||||
apply(a);
|
let i = apply(a);
|
||||||
apply(b);
|
let ii = apply(b);
|
||||||
|
if let abi::Abi::ScalarPair(scalar_a, scalar_b) = arg.layout.abi {
|
||||||
|
apply_range_attr(llvm::AttributePlace::Argument(i), scalar_a);
|
||||||
|
apply_range_attr(llvm::AttributePlace::Argument(ii), scalar_b);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
PassMode::Cast { cast, pad_i32 } => {
|
PassMode::Cast { cast, pad_i32 } => {
|
||||||
if *pad_i32 {
|
if *pad_i32 {
|
||||||
@ -517,15 +549,18 @@ fn apply_attrs_callsite(&self, bx: &mut Builder<'_, 'll, 'tcx>, callsite: &'ll V
|
|||||||
}
|
}
|
||||||
_ => {}
|
_ => {}
|
||||||
}
|
}
|
||||||
if let abi::Abi::Scalar(scalar) = self.ret.layout.abi {
|
if bx.cx.sess().opts.optimize != config::OptLevel::No
|
||||||
// If the value is a boolean, the range is 0..2 and that ultimately
|
&& llvm_util::get_version() < (19, 0, 0)
|
||||||
// become 0..0 when the type becomes i1, which would be rejected
|
&& let abi::Abi::Scalar(scalar) = self.ret.layout.abi
|
||||||
// by the LLVM verifier.
|
&& matches!(scalar.primitive(), Int(..))
|
||||||
if let Int(..) = scalar.primitive() {
|
// If the value is a boolean, the range is 0..2 and that ultimately
|
||||||
if !scalar.is_bool() && !scalar.is_always_valid(bx) {
|
// become 0..0 when the type becomes i1, which would be rejected
|
||||||
bx.range_metadata(callsite, scalar.valid_range(bx));
|
// by the LLVM verifier.
|
||||||
}
|
&& !scalar.is_bool()
|
||||||
}
|
// LLVM also rejects full range.
|
||||||
|
&& !scalar.is_always_valid(bx)
|
||||||
|
{
|
||||||
|
bx.range_metadata(callsite, scalar.valid_range(bx));
|
||||||
}
|
}
|
||||||
for arg in self.args.iter() {
|
for arg in self.args.iter() {
|
||||||
match &arg.mode {
|
match &arg.mode {
|
||||||
|
@ -1575,6 +1575,12 @@ pub fn LLVMRustGetNamedValue(
|
|||||||
pub fn LLVMRustCreateAllocSizeAttr(C: &Context, size_arg: u32) -> &Attribute;
|
pub fn LLVMRustCreateAllocSizeAttr(C: &Context, size_arg: u32) -> &Attribute;
|
||||||
pub fn LLVMRustCreateAllocKindAttr(C: &Context, size_arg: u64) -> &Attribute;
|
pub fn LLVMRustCreateAllocKindAttr(C: &Context, size_arg: u64) -> &Attribute;
|
||||||
pub fn LLVMRustCreateMemoryEffectsAttr(C: &Context, effects: MemoryEffects) -> &Attribute;
|
pub fn LLVMRustCreateMemoryEffectsAttr(C: &Context, effects: MemoryEffects) -> &Attribute;
|
||||||
|
pub fn LLVMRustCreateRangeAttribute(
|
||||||
|
C: &Context,
|
||||||
|
num_bits: c_uint,
|
||||||
|
lower_words: *const u64,
|
||||||
|
upper_words: *const u64,
|
||||||
|
) -> &Attribute;
|
||||||
|
|
||||||
// Operations on functions
|
// Operations on functions
|
||||||
pub fn LLVMRustGetOrInsertFunction<'a>(
|
pub fn LLVMRustGetOrInsertFunction<'a>(
|
||||||
|
@ -8,7 +8,7 @@
|
|||||||
use libc::c_uint;
|
use libc::c_uint;
|
||||||
use rustc_data_structures::small_c_str::SmallCStr;
|
use rustc_data_structures::small_c_str::SmallCStr;
|
||||||
use rustc_llvm::RustString;
|
use rustc_llvm::RustString;
|
||||||
use rustc_target::abi::Align;
|
use rustc_target::abi::{Align, Size, WrappingRange};
|
||||||
|
|
||||||
pub use self::AtomicRmwBinOp::*;
|
pub use self::AtomicRmwBinOp::*;
|
||||||
pub use self::CallConv::*;
|
pub use self::CallConv::*;
|
||||||
@ -105,6 +105,21 @@ pub fn CreateAllocKindAttr(llcx: &Context, kind_arg: AllocKindFlags) -> &Attribu
|
|||||||
unsafe { LLVMRustCreateAllocKindAttr(llcx, kind_arg.bits()) }
|
unsafe { LLVMRustCreateAllocKindAttr(llcx, kind_arg.bits()) }
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn CreateRangeAttr(llcx: &Context, size: Size, range: WrappingRange) -> &Attribute {
|
||||||
|
let lower = range.start;
|
||||||
|
let upper = range.end.wrapping_add(1);
|
||||||
|
let lower_words = [lower as u64, (lower >> 64) as u64];
|
||||||
|
let upper_words = [upper as u64, (upper >> 64) as u64];
|
||||||
|
unsafe {
|
||||||
|
LLVMRustCreateRangeAttribute(
|
||||||
|
llcx,
|
||||||
|
size.bits().try_into().unwrap(),
|
||||||
|
lower_words.as_ptr(),
|
||||||
|
upper_words.as_ptr(),
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#[derive(Copy, Clone)]
|
#[derive(Copy, Clone)]
|
||||||
pub enum AttributePlace {
|
pub enum AttributePlace {
|
||||||
ReturnValue,
|
ReturnValue,
|
||||||
|
@ -397,6 +397,18 @@ LLVMRustCreateAllocSizeAttr(LLVMContextRef C, uint32_t ElementSizeArg) {
|
|||||||
std::nullopt));
|
std::nullopt));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
extern "C" LLVMAttributeRef
|
||||||
|
LLVMRustCreateRangeAttribute(LLVMContextRef C, unsigned NumBits,
|
||||||
|
const uint64_t LowerWords[],
|
||||||
|
const uint64_t UpperWords[]) {
|
||||||
|
#if LLVM_VERSION_GE(19, 0)
|
||||||
|
return LLVMCreateConstantRangeAttribute(C, Attribute::Range, NumBits,
|
||||||
|
LowerWords, UpperWords);
|
||||||
|
#else
|
||||||
|
report_fatal_error("LLVM 19.0 is required for Range Attribute");
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
// These values **must** match ffi::AllocKindFlags.
|
// These values **must** match ffi::AllocKindFlags.
|
||||||
// It _happens_ to match the LLVM values of llvm::AllocFnKind,
|
// It _happens_ to match the LLVM values of llvm::AllocFnKind,
|
||||||
// but that's happenstance and we do explicit conversions before
|
// but that's happenstance and we do explicit conversions before
|
||||||
|
@ -2,6 +2,7 @@
|
|||||||
// scalar value.
|
// scalar value.
|
||||||
|
|
||||||
//@ compile-flags: -O -C no-prepopulate-passes
|
//@ compile-flags: -O -C no-prepopulate-passes
|
||||||
|
//@ ignore-llvm-version: 19 - 99
|
||||||
|
|
||||||
#![crate_type = "lib"]
|
#![crate_type = "lib"]
|
||||||
|
|
||||||
|
@ -20,8 +20,6 @@
|
|||||||
// CHECK-LABEL: @char_as_u32_index
|
// CHECK-LABEL: @char_as_u32_index
|
||||||
#[no_mangle]
|
#[no_mangle]
|
||||||
pub fn char_as_u32_index(c: char) -> [bool; 22] {
|
pub fn char_as_u32_index(c: char) -> [bool; 22] {
|
||||||
// CHECK: %[[B:.+]] = icmp ult i32 %c, 1114112
|
|
||||||
// CHECK: call void @llvm.assume(i1 %[[B]])
|
|
||||||
let c = c as u32;
|
let c = c as u32;
|
||||||
|
|
||||||
let mut array = [false; 22];
|
let mut array = [false; 22];
|
||||||
|
@ -28,7 +28,7 @@ pub fn insert_box(x: Box<()>) -> Result<usize, Box<()>> {
|
|||||||
|
|
||||||
// CHECK-LABEL: @extract_int
|
// CHECK-LABEL: @extract_int
|
||||||
// CHECK-NOT: nonnull
|
// CHECK-NOT: nonnull
|
||||||
// CHECK-SAME: (i{{[0-9]+}} {{[^,]+}} [[DISCRIMINANT:%[0-9]+]], ptr {{[^,]+}} [[PAYLOAD:%[0-9]+]])
|
// CHECK-SAME: (i{{[0-9]+}} {{[^%]+}} [[DISCRIMINANT:%[0-9]+]], ptr {{[^,]+}} [[PAYLOAD:%[0-9]+]])
|
||||||
#[no_mangle]
|
#[no_mangle]
|
||||||
pub unsafe fn extract_int(x: Result<usize, Box<()>>) -> usize {
|
pub unsafe fn extract_int(x: Result<usize, Box<()>>) -> usize {
|
||||||
// CHECK: [[TEMP:%.+]] = ptrtoint ptr [[PAYLOAD]] to [[USIZE:i[0-9]+]]
|
// CHECK: [[TEMP:%.+]] = ptrtoint ptr [[PAYLOAD]] to [[USIZE:i[0-9]+]]
|
||||||
@ -40,7 +40,7 @@ pub unsafe fn extract_int(x: Result<usize, Box<()>>) -> usize {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// CHECK-LABEL: @extract_box
|
// CHECK-LABEL: @extract_box
|
||||||
// CHECK-SAME: (i{{[0-9]+}} {{[^,]+}} [[DISCRIMINANT:%[0-9]+]], ptr {{[^,]+}} [[PAYLOAD:%[0-9]+]])
|
// CHECK-SAME: (i{{[0-9]+}} {{[^%]+}} [[DISCRIMINANT:%[0-9]+]], ptr {{[^,]+}} [[PAYLOAD:%[0-9]+]])
|
||||||
#[no_mangle]
|
#[no_mangle]
|
||||||
pub unsafe fn extract_box(x: Result<usize, Box<i32>>) -> Box<i32> {
|
pub unsafe fn extract_box(x: Result<usize, Box<i32>>) -> Box<i32> {
|
||||||
// CHECK: ret ptr [[PAYLOAD]]
|
// CHECK: ret ptr [[PAYLOAD]]
|
||||||
|
@ -34,7 +34,7 @@ pub enum Enum1 {
|
|||||||
|
|
||||||
// CHECK: define noundef{{( range\(i8 [0-9]+, [0-9]+\))?}} i8 @match1{{.*}}
|
// CHECK: define noundef{{( range\(i8 [0-9]+, [0-9]+\))?}} i8 @match1{{.*}}
|
||||||
// CHECK-NEXT: start:
|
// CHECK-NEXT: start:
|
||||||
// CHECK-NEXT: %1 = add i8 %0, -2
|
// CHECK-NEXT: %1 = add{{( nsw)?}} i8 %0, -2
|
||||||
// CHECK-NEXT: %2 = zext i8 %1 to i64
|
// CHECK-NEXT: %2 = zext i8 %1 to i64
|
||||||
// CHECK-NEXT: %3 = icmp ult i8 %1, 2
|
// CHECK-NEXT: %3 = icmp ult i8 %1, 2
|
||||||
// CHECK-NEXT: %4 = add nuw nsw i64 %2, 1
|
// CHECK-NEXT: %4 = add nuw nsw i64 %2, 1
|
||||||
|
@ -50,7 +50,7 @@ pub fn maybeuninit_enum_bool(x: MaybeUninit<MyBool>) -> MaybeUninit<MyBool> {
|
|||||||
x
|
x
|
||||||
}
|
}
|
||||||
|
|
||||||
// CHECK: noundef i32 @char(i32 noundef %x)
|
// CHECK: noundef{{( range\(i32 0, 1114112\))?}} i32 @char(i32 noundef{{( range\(i32 0, 1114112\))?}} %x)
|
||||||
#[no_mangle]
|
#[no_mangle]
|
||||||
pub fn char(x: char) -> char {
|
pub fn char(x: char) -> char {
|
||||||
x
|
x
|
||||||
@ -68,7 +68,7 @@ pub fn int(x: u64) -> u64 {
|
|||||||
x
|
x
|
||||||
}
|
}
|
||||||
|
|
||||||
// CHECK: noundef i64 @nonzero_int(i64 noundef %x)
|
// CHECK: noundef{{( range\(i64 1, 0\))?}} i64 @nonzero_int(i64 noundef{{( range\(i64 1, 0\))?}} %x)
|
||||||
#[no_mangle]
|
#[no_mangle]
|
||||||
pub fn nonzero_int(x: NonZero<u64>) -> NonZero<u64> {
|
pub fn nonzero_int(x: NonZero<u64>) -> NonZero<u64> {
|
||||||
x
|
x
|
||||||
@ -250,7 +250,7 @@ pub fn return_slice(x: &[u16]) -> &[u16] {
|
|||||||
x
|
x
|
||||||
}
|
}
|
||||||
|
|
||||||
// CHECK: { i16, i16 } @enum_id_1(i16 noundef %x.0, i16 %x.1)
|
// CHECK: { i16, i16 } @enum_id_1(i16 noundef{{( range\(i16 0, 3\))?}} %x.0, i16 %x.1)
|
||||||
#[no_mangle]
|
#[no_mangle]
|
||||||
pub fn enum_id_1(x: Option<Result<u16, u16>>) -> Option<Result<u16, u16>> {
|
pub fn enum_id_1(x: Option<Result<u16, u16>>) -> Option<Result<u16, u16>> {
|
||||||
x
|
x
|
||||||
|
@ -5,7 +5,7 @@
|
|||||||
// MIR inlining now optimizes this code.
|
// MIR inlining now optimizes this code.
|
||||||
|
|
||||||
// CHECK-LABEL: @unwrap_combinators
|
// CHECK-LABEL: @unwrap_combinators
|
||||||
// CHECK: icmp
|
// CHECK: {{icmp|trunc}}
|
||||||
// CHECK-NEXT: icmp
|
// CHECK-NEXT: icmp
|
||||||
// CHECK-NEXT: select i1
|
// CHECK-NEXT: select i1
|
||||||
// CHECK-NEXT: ret i1
|
// CHECK-NEXT: ret i1
|
||||||
|
68
tests/codegen/range-attribute.rs
Normal file
68
tests/codegen/range-attribute.rs
Normal file
@ -0,0 +1,68 @@
|
|||||||
|
// Checks that range metadata gets emitted on functions result and arguments
|
||||||
|
// with scalar value.
|
||||||
|
|
||||||
|
//@ compile-flags: -O -C no-prepopulate-passes
|
||||||
|
//@ min-llvm-version: 19
|
||||||
|
|
||||||
|
#![crate_type = "lib"]
|
||||||
|
|
||||||
|
use std::num::NonZero;
|
||||||
|
|
||||||
|
// Hack to get the correct size for usize
|
||||||
|
// CHECK: @helper([[USIZE:i[0-9]+]] noundef %_1)
|
||||||
|
#[no_mangle]
|
||||||
|
pub fn helper(_: usize) {}
|
||||||
|
|
||||||
|
// CHECK: noundef range(i128 1, 0) i128 @nonzero_int(i128 noundef range(i128 1, 0) %x)
|
||||||
|
#[no_mangle]
|
||||||
|
pub fn nonzero_int(x: NonZero<u128>) -> NonZero<u128> {
|
||||||
|
x
|
||||||
|
}
|
||||||
|
|
||||||
|
// CHECK: noundef range(i8 0, 3) i8 @optional_bool(i8 noundef range(i8 0, 3) %x)
|
||||||
|
#[no_mangle]
|
||||||
|
pub fn optional_bool(x: Option<bool>) -> Option<bool> {
|
||||||
|
x
|
||||||
|
}
|
||||||
|
|
||||||
|
pub enum Enum0 {
|
||||||
|
A(bool),
|
||||||
|
B,
|
||||||
|
C,
|
||||||
|
}
|
||||||
|
|
||||||
|
// CHECK: noundef range(i8 0, 4) i8 @enum0_value(i8 noundef range(i8 0, 4) %x)
|
||||||
|
#[no_mangle]
|
||||||
|
pub fn enum0_value(x: Enum0) -> Enum0 {
|
||||||
|
x
|
||||||
|
}
|
||||||
|
|
||||||
|
pub enum Enum1 {
|
||||||
|
A(u64),
|
||||||
|
B(u64),
|
||||||
|
C(u64),
|
||||||
|
}
|
||||||
|
|
||||||
|
// CHECK: { [[ENUM1_TYP:i[0-9]+]], i64 } @enum1_value([[ENUM1_TYP]] noundef range([[ENUM1_TYP]] 0, 3) %x.0, i64 noundef %x.1)
|
||||||
|
#[no_mangle]
|
||||||
|
pub fn enum1_value(x: Enum1) -> Enum1 {
|
||||||
|
x
|
||||||
|
}
|
||||||
|
|
||||||
|
pub enum Enum2 {
|
||||||
|
A(Enum0),
|
||||||
|
B(Enum0),
|
||||||
|
C(Enum0),
|
||||||
|
}
|
||||||
|
|
||||||
|
// CHECK: { i8, i8 } @enum2_value(i8 noundef range(i8 0, 3) %x.0, i8 noundef %x.1)
|
||||||
|
#[no_mangle]
|
||||||
|
pub fn enum2_value(x: Enum2) -> Enum2 {
|
||||||
|
x
|
||||||
|
}
|
||||||
|
|
||||||
|
// CHECK: noundef [[USIZE]] @takes_slice(ptr noalias noundef nonnull readonly align 4 %x.0, [[USIZE]] noundef %x.1)
|
||||||
|
#[no_mangle]
|
||||||
|
pub fn takes_slice(x: &[i32]) -> usize {
|
||||||
|
x.len()
|
||||||
|
}
|
@ -74,7 +74,7 @@ pub enum Bool {
|
|||||||
FileNotFound,
|
FileNotFound,
|
||||||
}
|
}
|
||||||
|
|
||||||
// CHECK: define{{( dso_local)?}} noundef{{( zeroext)?}} i8 @test_Gpz(i8 noundef{{( zeroext)?}} %_1)
|
// CHECK: define{{( dso_local)?}} noundef{{( zeroext)?( range\(i8 0, 3\))?}} i8 @test_Gpz(i8 noundef{{( zeroext)?( range\(i8 0, 3\))?}} %_1)
|
||||||
#[no_mangle]
|
#[no_mangle]
|
||||||
pub extern "C" fn test_Gpz(_: GenericPlusZst<Bool>) -> GenericPlusZst<Bool> {
|
pub extern "C" fn test_Gpz(_: GenericPlusZst<Bool>) -> GenericPlusZst<Bool> {
|
||||||
loop {}
|
loop {}
|
||||||
|
Loading…
Reference in New Issue
Block a user