Skip to content

Commit 816104f

Browse files
committed
Auto merge of rust-lang#134626 - veera-sivarajan:five-codegen-tests, r=Mark-Simulacrum
Add Four Codegen Tests Closes rust-lang#74615 Closes rust-lang#123216 Closes rust-lang#49572 Closes rust-lang#93514 This PR adds four codegen tests. The FileCheck assertions were generated with the help of `update_test_checks.py` and `update_llc_test_checks.py` from the LLVM project.
2 parents 39dc268 + c00d7e2 commit 816104f

4 files changed

+145
-0
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,31 @@
1+
//@ assembly-output: emit-asm
2+
//@ only-x86_64
3+
//@ ignore-sgx Test incompatible with LVI mitigations
4+
//@ compile-flags: -O
5+
//! Ensure that indexing a slice with `bool` does not
6+
//! generate any redundant `jmp` and `and` instructions.
7+
//! Discovered in issue #123216.
8+
9+
#![crate_type = "lib"]
10+
11+
#[no_mangle]
12+
fn f(a: u32, b: bool, c: bool, d: &mut [u128; 2]) {
13+
// CHECK-LABEL: f:
14+
// CHECK: testl %esi, %esi
15+
// CHECK: je
16+
// CHECK: xorb %dl, %dil
17+
// CHECK: orb $1, (%rcx)
18+
// CHECK: movzbl %dil, %eax
19+
// CHECK: andl $1, %eax
20+
// CHECK: shll $4, %eax
21+
// CHECK: orb $1, (%rcx,%rax)
22+
// CHECK-NOT: jmp
23+
// CHECK-NOT: andl %dil, $1
24+
// CHECK: retq
25+
let mut a = a & 1 != 0;
26+
if b {
27+
a ^= c;
28+
d[0] |= 1;
29+
}
30+
d[a as usize] |= 1;
31+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,53 @@
1+
//@ compile-flags: -O
2+
//! Ensure that classifying `FpCategory` based on `usize` compiles to
3+
//! a handful of instructions without any `switch` or branches.
4+
//! This regressed between rustc 1.37 and 1.38 and was discovered
5+
//! in issue #74615.
6+
7+
#![crate_type = "lib"]
8+
9+
use std::num::FpCategory;
10+
11+
fn conv(input: u32) -> FpCategory {
12+
match input {
13+
0b10000000 | 0b00000001 => FpCategory::Infinite,
14+
0b01000000 | 0b00000010 => FpCategory::Normal,
15+
0b00100000 | 0b00000100 => FpCategory::Subnormal,
16+
0b00010000 | 0b00001000 => FpCategory::Zero,
17+
0b100000000 | 0b1000000000 => FpCategory::Nan,
18+
_ => unsafe { std::hint::unreachable_unchecked() },
19+
}
20+
}
21+
22+
#[no_mangle]
23+
pub fn complex_test(input: u32) -> bool {
24+
// CHECK-LABEL: @complex_test(
25+
// CHECK-SAME: i32 noundef [[INPUT:%.*]]) unnamed_addr #[[ATTR0:[0-9]+]] {
26+
// CHECK: [[TMP0:%.*]] = tail call
27+
// CHECK: [[DOTOFF:%.*]] = add nsw i32 [[TMP0]], -3
28+
// CHECK: [[SWITCH:%.*]] = icmp ult i32 [[DOTOFF]], 2
29+
// CHECK-NOT: switch i32
30+
// CHECK-NOT: br
31+
// CHECK-NOT: label
32+
// CHECK: ret i1 [[SWITCH]]
33+
//
34+
conv(input) == FpCategory::Zero
35+
}
36+
37+
#[no_mangle]
38+
pub fn simpler_test(input: u32) -> bool {
39+
// CHECK-LABEL: @simpler_test(
40+
// CHECK-SAME: i32 noundef [[INPUT:%.*]]) unnamed_addr #[[ATTR0]] {
41+
// CHECK: [[TMP0:%.*]] = add i32 [[INPUT]], -8
42+
// CHECK: [[SWITCH_AND:%.*]] = and i32 [[TMP0]], -9
43+
// CHECK: [[SWITCH_SELECTCMP:%.*]] = icmp eq i32 [[SWITCH_AND]], 0
44+
// CHECK-NOT: switch i32
45+
// CHECK-NOT: br
46+
// CHECK-NOT: label
47+
// CHECK: ret i1 [[SWITCH_SELECTCMP]]
48+
//
49+
match input {
50+
0b00010000 | 0b00001000 => true,
51+
_ => false,
52+
}
53+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,20 @@
1+
//@ compile-flags: -O
2+
//! Ensure that `.get()` on `std::num::NonZero*` types do not
3+
//! check for zero equivalency.
4+
//! Discovered in issue #49572.
5+
6+
#![crate_type = "lib"]
7+
8+
#[no_mangle]
9+
pub fn foo(x: std::num::NonZeroU32) -> bool {
10+
// CHECK-LABEL: @foo(
11+
// CHECK: ret i1 true
12+
x.get() != 0
13+
}
14+
15+
#[no_mangle]
16+
pub fn bar(x: std::num::NonZeroI64) -> bool {
17+
// CHECK-LABEL: @bar(
18+
// CHECK: ret i1 true
19+
x.get() != 0
20+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,41 @@
1+
//@ compile-flags: -O
2+
//! Ensure that matching on `x % 5` generates an unreachable
3+
//! branch for values greater than 4.
4+
//! Discovered in issue #93514.
5+
6+
#![crate_type = "lib"]
7+
8+
#[no_mangle]
9+
pub unsafe fn parse0(x: u32) -> u32 {
10+
// CHECK-LABEL: i32 @parse0(
11+
// CHECK-SAME: i32 noundef [[X:%.*]]) unnamed_addr #[[ATTR0:[0-9]+]] {
12+
// CHECK-NEXT: [[START:.*:]]
13+
// CHECK-NEXT: [[_2:%.*]] = urem i32 [[X]], 5
14+
// CHECK-NEXT: switch i32 [[_2]], label %[[DEFAULT_UNREACHABLE1:.*]] [
15+
// CHECK-NEXT: i32 0, label %[[BB6:.*]]
16+
// CHECK-NEXT: i32 1, label %[[BB5:.*]]
17+
// CHECK-NEXT: i32 2, label %[[BB4:.*]]
18+
// CHECK-NEXT: i32 3, label %[[BB3:.*]]
19+
// CHECK-NEXT: i32 4, label %[[BB2:.*]]
20+
// CHECK-NEXT: ]
21+
// CHECK: [[DEFAULT_UNREACHABLE1]]:
22+
// CHECK-NEXT: unreachable
23+
// CHECK: ret i32
24+
match x % 5 {
25+
0 => f1(x),
26+
1 => f2(x),
27+
2 => f3(x),
28+
3 => f4(x),
29+
4 => f5(x),
30+
_ => eliminate_me(),
31+
}
32+
}
33+
34+
extern "Rust" {
35+
fn eliminate_me() -> u32;
36+
fn f1(x: u32) -> u32;
37+
fn f2(x: u32) -> u32;
38+
fn f3(x: u32) -> u32;
39+
fn f4(x: u32) -> u32;
40+
fn f5(x: u32) -> u32;
41+
}

0 commit comments

Comments
 (0)