| //@ compile-flags: -Copt-level=3 -C no-prepopulate-passes |
| //@ only-64bit (because these discriminants are isize) |
| |
| #![crate_type = "lib"] |
| |
| // This directly tests what we emit for these matches, rather than what happens |
| // after optimization, so it doesn't need to worry about extra flags on the |
| // instructions and is less susceptible to being broken on LLVM updates. |
| |
| // CHECK-LABEL: @option_match |
| #[no_mangle] |
| pub fn option_match(x: Option<i32>) -> u16 { |
| // CHECK-NOT: %x = alloca |
| // CHECK: %[[OUT:.+]] = alloca [2 x i8] |
| // CHECK-NOT: %x = alloca |
| |
| // CHECK: %[[DISCR:.+]] = zext i32 %x.0 to i64 |
| // CHECK: %[[COND:.+]] = trunc nuw i64 %[[DISCR]] to i1 |
| // CHECK: br i1 %[[COND]], label %[[TRUE:[a-z0-9]+]], label %[[FALSE:[a-z0-9]+]] |
| |
| // CHECK: [[TRUE]]: |
| // CHECK: store i16 13, ptr %[[OUT]] |
| |
| // CHECK: [[FALSE]]: |
| // CHECK: store i16 42, ptr %[[OUT]] |
| |
| // CHECK: %[[RET:.+]] = load i16, ptr %[[OUT]] |
| // CHECK: ret i16 %[[RET]] |
| match x { |
| Some(_) => 13, |
| None => 42, |
| } |
| } |
| |
| // CHECK-LABEL: @result_match |
| #[no_mangle] |
| pub fn result_match(x: Result<u64, i64>) -> u16 { |
| // CHECK-NOT: %x = alloca |
| // CHECK: %[[OUT:.+]] = alloca [2 x i8] |
| // CHECK-NOT: %x = alloca |
| |
| // CHECK: %[[COND:.+]] = trunc nuw i64 %x.0 to i1 |
| // CHECK: br i1 %[[COND]], label %[[TRUE:[a-z0-9]+]], label %[[FALSE:[a-z0-9]+]] |
| |
| // CHECK: [[TRUE]]: |
| // CHECK: store i16 13, ptr %[[OUT]] |
| |
| // CHECK: [[FALSE]]: |
| // CHECK: store i16 42, ptr %[[OUT]] |
| |
| // CHECK: %[[RET:.+]] = load i16, ptr %[[OUT]] |
| // CHECK: ret i16 %[[RET]] |
| match x { |
| Err(_) => 13, |
| Ok(_) => 42, |
| } |
| } |
| |
| // CHECK-LABEL: @option_bool_match( |
| #[no_mangle] |
| pub fn option_bool_match(x: Option<bool>) -> char { |
| // CHECK: %[[RAW:.+]] = load i8, ptr %x |
| // CHECK: %[[IS_NONE:.+]] = icmp eq i8 %[[RAW]], 2 |
| // CHECK: %[[OPT_DISCR:.+]] = select i1 %[[IS_NONE]], i64 0, i64 1 |
| // CHECK: %[[OPT_DISCR_T:.+]] = trunc nuw i64 %[[OPT_DISCR]] to i1 |
| // CHECK: br i1 %[[OPT_DISCR_T]], label %[[BB_SOME:.+]], label %[[BB_NONE:.+]] |
| |
| // CHECK: [[BB_SOME]]: |
| // CHECK: %[[FIELD:.+]] = load i8, ptr %x |
| // CHECK: %[[FIELD_T:.+]] = trunc nuw i8 %[[FIELD]] to i1 |
| // CHECK: br i1 %[[FIELD_T]] |
| match x { |
| None => 'n', |
| Some(false) => 'f', |
| Some(true) => 't', |
| } |
| } |
| |
| use std::cmp::Ordering::{self, *}; |
| // CHECK-LABEL: @option_ordering_match( |
| #[no_mangle] |
| pub fn option_ordering_match(x: Option<Ordering>) -> char { |
| // CHECK: %[[RAW:.+]] = load i8, ptr %x |
| // CHECK: %[[IS_NONE:.+]] = icmp eq i8 %[[RAW]], 2 |
| // CHECK: %[[OPT_DISCR:.+]] = select i1 %[[IS_NONE]], i64 0, i64 1 |
| // CHECK: %[[OPT_DISCR_T:.+]] = trunc nuw i64 %[[OPT_DISCR]] to i1 |
| // CHECK: br i1 %[[OPT_DISCR_T]], label %[[BB_SOME:.+]], label %[[BB_NONE:.+]] |
| |
| // CHECK: [[BB_SOME]]: |
| // CHECK: %[[FIELD:.+]] = load i8, ptr %x |
| // CHECK: switch i8 %[[FIELD]], label %[[UNREACHABLE:.+]] [ |
| // CHECK-NEXT: i8 -1, label |
| // CHECK-NEXT: i8 0, label |
| // CHECK-NEXT: i8 1, label |
| // CHECK-NEXT: ] |
| |
| // CHECK: [[UNREACHABLE]]: |
| // CHECK-NEXT: unreachable |
| match x { |
| None => '?', |
| Some(Less) => '<', |
| Some(Equal) => '=', |
| Some(Greater) => '>', |
| } |
| } |
| |
| // CHECK-LABEL: @option_nonzero_match( |
| #[no_mangle] |
| pub fn option_nonzero_match(x: Option<std::num::NonZero<u16>>) -> u16 { |
| // CHECK: %[[OUT:.+]] = alloca [2 x i8] |
| |
| // CHECK: %[[IS_NONE:.+]] = icmp eq i16 %x, 0 |
| // CHECK: %[[OPT_DISCR:.+]] = select i1 %[[IS_NONE]], i64 0, i64 1 |
| // CHECK: %[[OPT_DISCR_T:.+]] = trunc nuw i64 %[[OPT_DISCR]] to i1 |
| // CHECK: br i1 %[[OPT_DISCR_T]], label %[[BB_SOME:.+]], label %[[BB_NONE:.+]] |
| |
| // CHECK: [[BB_SOME]]: |
| // CHECK: store i16 987, ptr %[[OUT]] |
| |
| // CHECK: [[BB_NONE]]: |
| // CHECK: store i16 123, ptr %[[OUT]] |
| |
| // CHECK: %[[RET:.+]] = load i16, ptr %[[OUT]] |
| // CHECK: ret i16 %[[RET]] |
| |
| match x { |
| None => 123, |
| Some(_) => 987, |
| } |
| } |