Skip to main content

rustc_mir_build/builder/matches/
test.rs

1// Testing candidates
2//
3// After candidates have been simplified, the only match pairs that
4// remain are those that require some sort of test. The functions here
5// identify what tests are needed, perform the tests, and then filter
6// the candidates based on the result.
7
8use std::sync::Arc;
9
10use rustc_data_structures::fx::FxIndexMap;
11use rustc_hir::{LangItem, RangeEnd};
12use rustc_middle::bug;
13use rustc_middle::mir::*;
14use rustc_middle::ty::util::IntTypeExt;
15use rustc_middle::ty::{self, GenericArg, Ty, TyCtxt};
16use rustc_span::def_id::DefId;
17use rustc_span::{DUMMY_SP, Span, Spanned, Symbol, sym};
18use tracing::{debug, instrument};
19
20use crate::builder::Builder;
21use crate::builder::matches::{
22    MatchPairTree, PatConstKind, SliceLenOp, Test, TestBranch, TestKind, TestableCase,
23};
24
25impl<'a, 'tcx> Builder<'a, 'tcx> {
26    /// Identifies what test is needed to decide if `match_pair` is applicable.
27    ///
28    /// It is a bug to call this with a not-fully-simplified pattern.
29    pub(super) fn pick_test_for_match_pair(
30        &mut self,
31        match_pair: &MatchPairTree<'tcx>,
32    ) -> Test<'tcx> {
33        let kind = match match_pair.testable_case {
34            TestableCase::Variant { adt_def, variant_index: _ } => TestKind::Switch { adt_def },
35
36            TestableCase::Constant { value: _, kind: PatConstKind::Bool } => TestKind::If,
37            TestableCase::Constant { value: _, kind: PatConstKind::IntOrChar } => {
38                TestKind::SwitchInt
39            }
40            TestableCase::Constant { value, kind: PatConstKind::String } => {
41                TestKind::StringEq { value }
42            }
43            TestableCase::Constant { value, kind: PatConstKind::Float | PatConstKind::Other } => {
44                TestKind::ScalarEq { value }
45            }
46
47            TestableCase::Range(ref range) => {
48                match (&range.ty, &match_pair.pattern_ty) {
    (left_val, right_val) => {
        if !(*left_val == *right_val) {
            let kind = ::core::panicking::AssertKind::Eq;
            ::core::panicking::assert_failed(kind, &*left_val, &*right_val,
                ::core::option::Option::None);
        }
    }
};assert_eq!(range.ty, match_pair.pattern_ty);
49                TestKind::Range(Arc::clone(range))
50            }
51
52            TestableCase::Slice { len, op } => TestKind::SliceLen { len, op },
53
54            TestableCase::Deref { temp, mutability } => TestKind::Deref { temp, mutability },
55
56            TestableCase::Never => TestKind::Never,
57
58            // Or-patterns are not tested directly; instead they are expanded into subcandidates,
59            // which are then distinguished by testing whatever non-or patterns they contain.
60            TestableCase::Or { .. } => ::rustc_middle::util::bug::bug_fmt(format_args!("or-patterns should have already been handled"))bug!("or-patterns should have already been handled"),
61        };
62
63        Test { span: match_pair.pattern_span, kind }
64    }
65
66    #[allow(clippy :: suspicious_else_formatting)]
{
    let __tracing_attr_span;
    let __tracing_attr_guard;
    if ::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
                &&
                ::tracing::Level::DEBUG <=
                    ::tracing::level_filters::LevelFilter::current() ||
            { false } {
        __tracing_attr_span =
            {
                use ::tracing::__macro_support::Callsite as _;
                static __CALLSITE: ::tracing::callsite::DefaultCallsite =
                    {
                        static META: ::tracing::Metadata<'static> =
                            {
                                ::tracing_core::metadata::Metadata::new("perform_test",
                                    "rustc_mir_build::builder::matches::test",
                                    ::tracing::Level::DEBUG,
                                    ::tracing_core::__macro_support::Option::Some("compiler/rustc_mir_build/src/builder/matches/test.rs"),
                                    ::tracing_core::__macro_support::Option::Some(66u32),
                                    ::tracing_core::__macro_support::Option::Some("rustc_mir_build::builder::matches::test"),
                                    ::tracing_core::field::FieldSet::new(&["match_start_span",
                                                    "scrutinee_span", "block", "otherwise_block", "test"],
                                        ::tracing_core::callsite::Identifier(&__CALLSITE)),
                                    ::tracing::metadata::Kind::SPAN)
                            };
                        ::tracing::callsite::DefaultCallsite::new(&META)
                    };
                let mut interest = ::tracing::subscriber::Interest::never();
                if ::tracing::Level::DEBUG <=
                                    ::tracing::level_filters::STATIC_MAX_LEVEL &&
                                ::tracing::Level::DEBUG <=
                                    ::tracing::level_filters::LevelFilter::current() &&
                            { interest = __CALLSITE.interest(); !interest.is_never() }
                        &&
                        ::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
                            interest) {
                    let meta = __CALLSITE.metadata();
                    ::tracing::Span::new(meta,
                        &{
                                #[allow(unused_imports)]
                                use ::tracing::field::{debug, display, Value};
                                let mut iter = meta.fields().iter();
                                meta.fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                                    ::tracing::__macro_support::Option::Some(&::tracing::field::debug(&match_start_span)
                                                            as &dyn Value)),
                                                (&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                                    ::tracing::__macro_support::Option::Some(&::tracing::field::debug(&scrutinee_span)
                                                            as &dyn Value)),
                                                (&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                                    ::tracing::__macro_support::Option::Some(&::tracing::field::debug(&block)
                                                            as &dyn Value)),
                                                (&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                                    ::tracing::__macro_support::Option::Some(&::tracing::field::debug(&otherwise_block)
                                                            as &dyn Value)),
                                                (&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                                    ::tracing::__macro_support::Option::Some(&::tracing::field::debug(&test)
                                                            as &dyn Value))])
                            })
                } else {
                    let span =
                        ::tracing::__macro_support::__disabled_span(__CALLSITE.metadata());
                    {};
                    span
                }
            };
        __tracing_attr_guard = __tracing_attr_span.enter();
    }

    #[warn(clippy :: suspicious_else_formatting)]
    {

        #[allow(unknown_lints, unreachable_code, clippy ::
        diverging_sub_expression, clippy :: empty_loop, clippy ::
        let_unit_value, clippy :: let_with_type_underscore, clippy ::
        needless_return, clippy :: unreachable)]
        if false {
            let __tracing_attr_fake_return: () = loop {};
            return __tracing_attr_fake_return;
        }
        {
            let place_ty = place.ty(&self.local_decls, self.tcx);
            {
                use ::tracing::__macro_support::Callsite as _;
                static __CALLSITE: ::tracing::callsite::DefaultCallsite =
                    {
                        static META: ::tracing::Metadata<'static> =
                            {
                                ::tracing_core::metadata::Metadata::new("event compiler/rustc_mir_build/src/builder/matches/test.rs:78",
                                    "rustc_mir_build::builder::matches::test",
                                    ::tracing::Level::DEBUG,
                                    ::tracing_core::__macro_support::Option::Some("compiler/rustc_mir_build/src/builder/matches/test.rs"),
                                    ::tracing_core::__macro_support::Option::Some(78u32),
                                    ::tracing_core::__macro_support::Option::Some("rustc_mir_build::builder::matches::test"),
                                    ::tracing_core::field::FieldSet::new(&["place", "place_ty"],
                                        ::tracing_core::callsite::Identifier(&__CALLSITE)),
                                    ::tracing::metadata::Kind::EVENT)
                            };
                        ::tracing::callsite::DefaultCallsite::new(&META)
                    };
                let enabled =
                    ::tracing::Level::DEBUG <=
                                ::tracing::level_filters::STATIC_MAX_LEVEL &&
                            ::tracing::Level::DEBUG <=
                                ::tracing::level_filters::LevelFilter::current() &&
                        {
                            let interest = __CALLSITE.interest();
                            !interest.is_never() &&
                                ::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
                                    interest)
                        };
                if enabled {
                    (|value_set: ::tracing::field::ValueSet|
                                {
                                    let meta = __CALLSITE.metadata();
                                    ::tracing::Event::dispatch(meta, &value_set);
                                    ;
                                })({
                            #[allow(unused_imports)]
                            use ::tracing::field::{debug, display, Value};
                            let mut iter = __CALLSITE.metadata().fields().iter();
                            __CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                                ::tracing::__macro_support::Option::Some(&debug(&place) as
                                                        &dyn Value)),
                                            (&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                                ::tracing::__macro_support::Option::Some(&debug(&place_ty)
                                                        as &dyn Value))])
                        });
                } else { ; }
            };
            let target_block =
                |branch|
                    target_blocks.get(&branch).copied().unwrap_or(otherwise_block);
            let source_info = self.source_info(test.span);
            match test.kind {
                TestKind::Switch { adt_def } => {
                    let otherwise_block = target_block(TestBranch::Failure);
                    let switch_targets =
                        SwitchTargets::new(adt_def.discriminants(self.tcx).filter_map(|(idx,
                                        discr)|
                                    {
                                        if let Some(&block) =
                                                target_blocks.get(&TestBranch::Variant(idx)) {
                                            Some((discr.val, block))
                                        } else { None }
                                    }), otherwise_block);
                    {
                        use ::tracing::__macro_support::Callsite as _;
                        static __CALLSITE: ::tracing::callsite::DefaultCallsite =
                            {
                                static META: ::tracing::Metadata<'static> =
                                    {
                                        ::tracing_core::metadata::Metadata::new("event compiler/rustc_mir_build/src/builder/matches/test.rs:95",
                                            "rustc_mir_build::builder::matches::test",
                                            ::tracing::Level::DEBUG,
                                            ::tracing_core::__macro_support::Option::Some("compiler/rustc_mir_build/src/builder/matches/test.rs"),
                                            ::tracing_core::__macro_support::Option::Some(95u32),
                                            ::tracing_core::__macro_support::Option::Some("rustc_mir_build::builder::matches::test"),
                                            ::tracing_core::field::FieldSet::new(&["message"],
                                                ::tracing_core::callsite::Identifier(&__CALLSITE)),
                                            ::tracing::metadata::Kind::EVENT)
                                    };
                                ::tracing::callsite::DefaultCallsite::new(&META)
                            };
                        let enabled =
                            ::tracing::Level::DEBUG <=
                                        ::tracing::level_filters::STATIC_MAX_LEVEL &&
                                    ::tracing::Level::DEBUG <=
                                        ::tracing::level_filters::LevelFilter::current() &&
                                {
                                    let interest = __CALLSITE.interest();
                                    !interest.is_never() &&
                                        ::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
                                            interest)
                                };
                        if enabled {
                            (|value_set: ::tracing::field::ValueSet|
                                        {
                                            let meta = __CALLSITE.metadata();
                                            ::tracing::Event::dispatch(meta, &value_set);
                                            ;
                                        })({
                                    #[allow(unused_imports)]
                                    use ::tracing::field::{debug, display, Value};
                                    let mut iter = __CALLSITE.metadata().fields().iter();
                                    __CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
                                                        ::tracing::__macro_support::Option::Some(&format_args!("num_enum_variants: {0}",
                                                                        adt_def.variants().len()) as &dyn Value))])
                                });
                        } else { ; }
                    };
                    let discr_ty = adt_def.repr().discr_type().to_ty(self.tcx);
                    let discr = self.temp(discr_ty, test.span);
                    self.cfg.push_assign(block,
                        self.source_info(scrutinee_span), discr,
                        Rvalue::Discriminant(place));
                    self.cfg.terminate(block,
                        self.source_info(match_start_span),
                        TerminatorKind::SwitchInt {
                            discr: Operand::Move(discr),
                            targets: switch_targets,
                        });
                }
                TestKind::SwitchInt => {
                    let otherwise_block = target_block(TestBranch::Failure);
                    let switch_targets =
                        SwitchTargets::new(target_blocks.iter().filter_map(|(&branch,
                                        &block)|
                                    {
                                        if let TestBranch::Constant(value) = branch {
                                            let bits = value.to_leaf().to_bits_unchecked();
                                            Some((bits, block))
                                        } else { None }
                                    }), otherwise_block);
                    let terminator =
                        TerminatorKind::SwitchInt {
                            discr: Operand::Copy(place),
                            targets: switch_targets,
                        };
                    self.cfg.terminate(block,
                        self.source_info(match_start_span), terminator);
                }
                TestKind::If => {
                    let success_block = target_block(TestBranch::Success);
                    let fail_block = target_block(TestBranch::Failure);
                    let terminator =
                        TerminatorKind::if_(Operand::Copy(place), success_block,
                            fail_block);
                    self.cfg.terminate(block,
                        self.source_info(match_start_span), terminator);
                }
                TestKind::StringEq { value } => {
                    let tcx = self.tcx;
                    let success_block = target_block(TestBranch::Success);
                    let fail_block = target_block(TestBranch::Failure);
                    let ref_str_ty =
                        Ty::new_imm_ref(tcx, tcx.lifetimes.re_erased,
                            tcx.types.str_);
                    if !ref_str_ty.is_imm_ref_str() {
                        {
                            ::core::panicking::panic_fmt(format_args!("{0:?}",
                                    ref_str_ty));
                        }
                    };
                    if !value.ty.is_str() {
                        {
                            ::core::panicking::panic_fmt(format_args!("unexpected value type for StringEq test: {0:?}",
                                    value));
                        }
                    };
                    let expected_value =
                        ty::Value { ty: ref_str_ty, valtree: value.valtree };
                    let expected_value_operand =
                        self.literal_operand(test.span,
                            Const::from_ty_value(tcx, expected_value));
                    let actual_value_ref_place =
                        self.temp(ref_str_ty, test.span);
                    self.cfg.push_assign(block, self.source_info(test.span),
                        actual_value_ref_place,
                        Rvalue::Ref(tcx.lifetimes.re_erased, BorrowKind::Shared,
                            place));
                    self.string_compare(block, success_block, fail_block,
                        source_info, expected_value_operand,
                        Operand::Copy(actual_value_ref_place));
                }
                TestKind::ScalarEq { value } => {
                    let tcx = self.tcx;
                    let success_block = target_block(TestBranch::Success);
                    let fail_block = target_block(TestBranch::Failure);
                    let mut expected_value_ty = value.ty;
                    let mut expected_value_operand =
                        self.literal_operand(test.span,
                            Const::from_ty_value(tcx, value));
                    let mut actual_value_place = place;
                    match value.ty.kind() {
                        &ty::Pat(base, _) => {
                            if !base.is_trivially_pure_clone_copy() {
                                ::core::panicking::panic("assertion failed: base.is_trivially_pure_clone_copy()")
                            };
                            let transmuted_place = self.temp(base, test.span);
                            self.cfg.push_assign(block,
                                self.source_info(scrutinee_span), transmuted_place,
                                Rvalue::Cast(CastKind::Transmute,
                                    Operand::Copy(actual_value_place), base));
                            let transmuted_expect = self.temp(base, test.span);
                            self.cfg.push_assign(block, self.source_info(test.span),
                                transmuted_expect,
                                Rvalue::Cast(CastKind::Transmute, expected_value_operand,
                                    base));
                            actual_value_place = transmuted_place;
                            expected_value_operand = Operand::Copy(transmuted_expect);
                            expected_value_ty = base;
                        }
                        _ => {}
                    }
                    if !expected_value_ty.is_scalar() {
                        ::core::panicking::panic("assertion failed: expected_value_ty.is_scalar()")
                    };
                    self.compare(block, success_block, fail_block, source_info,
                        BinOp::Eq, expected_value_operand,
                        Operand::Copy(actual_value_place));
                }
                TestKind::Range(ref range) => {
                    let success = target_block(TestBranch::Success);
                    let fail = target_block(TestBranch::Failure);
                    let val = Operand::Copy(place);
                    let intermediate_block =
                        if !range.lo.is_finite() {
                            block
                        } else if !range.hi.is_finite() {
                            success
                        } else { self.cfg.start_new_block() };
                    if let Some(lo) = range.lo.as_finite() {
                        let lo = ty::Value { ty: range.ty, valtree: lo };
                        let lo =
                            self.literal_operand(test.span,
                                Const::from_ty_value(self.tcx, lo));
                        self.compare(block, intermediate_block, fail, source_info,
                            BinOp::Le, lo, val.clone());
                    };
                    if let Some(hi) = range.hi.as_finite() {
                        let hi = ty::Value { ty: range.ty, valtree: hi };
                        let hi =
                            self.literal_operand(test.span,
                                Const::from_ty_value(self.tcx, hi));
                        let op =
                            match range.end {
                                RangeEnd::Included => BinOp::Le,
                                RangeEnd::Excluded => BinOp::Lt,
                            };
                        self.compare(intermediate_block, success, fail, source_info,
                            op, val, hi);
                    }
                }
                TestKind::SliceLen { len, op } => {
                    let usize_ty = self.tcx.types.usize;
                    let actual = self.temp(usize_ty, test.span);
                    let length_op =
                        self.len_of_slice_or_array(block, place, test.span,
                            source_info);
                    self.cfg.push_assign(block, source_info, actual,
                        Rvalue::Use(length_op));
                    let expected = self.push_usize(block, source_info, len);
                    let success_block = target_block(TestBranch::Success);
                    let fail_block = target_block(TestBranch::Failure);
                    self.compare(block, success_block, fail_block, source_info,
                        match op {
                            SliceLenOp::Equal => BinOp::Eq,
                            SliceLenOp::GreaterOrEqual => BinOp::Ge,
                        }, Operand::Move(actual), Operand::Move(expected));
                }
                TestKind::Deref { temp, mutability } => {
                    let ty = place_ty.ty;
                    let target = target_block(TestBranch::Success);
                    self.call_deref(block, target, place, mutability, ty, temp,
                        test.span);
                }
                TestKind::Never => {
                    self.cfg.push_fake_read(block, source_info,
                        FakeReadCause::ForMatchedPlace(None), place);
                    self.cfg.terminate(block, source_info,
                        TerminatorKind::Unreachable);
                }
            }
        }
    }
}#[instrument(skip(self, target_blocks, place), level = "debug")]
67    pub(super) fn perform_test(
68        &mut self,
69        match_start_span: Span,
70        scrutinee_span: Span,
71        block: BasicBlock,
72        otherwise_block: BasicBlock,
73        place: Place<'tcx>,
74        test: &Test<'tcx>,
75        target_blocks: FxIndexMap<TestBranch<'tcx>, BasicBlock>,
76    ) {
77        let place_ty = place.ty(&self.local_decls, self.tcx);
78        debug!(?place, ?place_ty);
79        let target_block = |branch| target_blocks.get(&branch).copied().unwrap_or(otherwise_block);
80
81        let source_info = self.source_info(test.span);
82        match test.kind {
83            TestKind::Switch { adt_def } => {
84                let otherwise_block = target_block(TestBranch::Failure);
85                let switch_targets = SwitchTargets::new(
86                    adt_def.discriminants(self.tcx).filter_map(|(idx, discr)| {
87                        if let Some(&block) = target_blocks.get(&TestBranch::Variant(idx)) {
88                            Some((discr.val, block))
89                        } else {
90                            None
91                        }
92                    }),
93                    otherwise_block,
94                );
95                debug!("num_enum_variants: {}", adt_def.variants().len());
96                let discr_ty = adt_def.repr().discr_type().to_ty(self.tcx);
97                let discr = self.temp(discr_ty, test.span);
98                self.cfg.push_assign(
99                    block,
100                    self.source_info(scrutinee_span),
101                    discr,
102                    Rvalue::Discriminant(place),
103                );
104                self.cfg.terminate(
105                    block,
106                    self.source_info(match_start_span),
107                    TerminatorKind::SwitchInt {
108                        discr: Operand::Move(discr),
109                        targets: switch_targets,
110                    },
111                );
112            }
113
114            TestKind::SwitchInt => {
115                // The switch may be inexhaustive so we have a catch-all block
116                let otherwise_block = target_block(TestBranch::Failure);
117                let switch_targets = SwitchTargets::new(
118                    target_blocks.iter().filter_map(|(&branch, &block)| {
119                        if let TestBranch::Constant(value) = branch {
120                            let bits = value.to_leaf().to_bits_unchecked();
121                            Some((bits, block))
122                        } else {
123                            None
124                        }
125                    }),
126                    otherwise_block,
127                );
128                let terminator = TerminatorKind::SwitchInt {
129                    discr: Operand::Copy(place),
130                    targets: switch_targets,
131                };
132                self.cfg.terminate(block, self.source_info(match_start_span), terminator);
133            }
134
135            TestKind::If => {
136                let success_block = target_block(TestBranch::Success);
137                let fail_block = target_block(TestBranch::Failure);
138                let terminator =
139                    TerminatorKind::if_(Operand::Copy(place), success_block, fail_block);
140                self.cfg.terminate(block, self.source_info(match_start_span), terminator);
141            }
142
143            TestKind::StringEq { value } => {
144                let tcx = self.tcx;
145                let success_block = target_block(TestBranch::Success);
146                let fail_block = target_block(TestBranch::Failure);
147
148                let ref_str_ty = Ty::new_imm_ref(tcx, tcx.lifetimes.re_erased, tcx.types.str_);
149                assert!(ref_str_ty.is_imm_ref_str(), "{ref_str_ty:?}");
150
151                // The string constant we're testing against has type `str`, but
152                // calling `<str as PartialEq>::eq` requires `&str` operands.
153                //
154                // Because `str` and `&str` have the same valtree representation,
155                // we can "cast" to the desired type by just replacing the type.
156                assert!(value.ty.is_str(), "unexpected value type for StringEq test: {value:?}");
157                let expected_value = ty::Value { ty: ref_str_ty, valtree: value.valtree };
158                let expected_value_operand =
159                    self.literal_operand(test.span, Const::from_ty_value(tcx, expected_value));
160
161                // Similarly, the scrutinized place has type `str`, but we need `&str`.
162                // Get a reference by doing `let actual_value_ref_place: &str = &place`.
163                let actual_value_ref_place = self.temp(ref_str_ty, test.span);
164                self.cfg.push_assign(
165                    block,
166                    self.source_info(test.span),
167                    actual_value_ref_place,
168                    Rvalue::Ref(tcx.lifetimes.re_erased, BorrowKind::Shared, place),
169                );
170
171                // Compare two strings using `<str as std::cmp::PartialEq>::eq`.
172                // (Interestingly this means that exhaustiveness analysis relies, for soundness,
173                // on the `PartialEq` impl for `str` to be correct!)
174                self.string_compare(
175                    block,
176                    success_block,
177                    fail_block,
178                    source_info,
179                    expected_value_operand,
180                    Operand::Copy(actual_value_ref_place),
181                );
182            }
183
184            TestKind::ScalarEq { value } => {
185                let tcx = self.tcx;
186                let success_block = target_block(TestBranch::Success);
187                let fail_block = target_block(TestBranch::Failure);
188
189                let mut expected_value_ty = value.ty;
190                let mut expected_value_operand =
191                    self.literal_operand(test.span, Const::from_ty_value(tcx, value));
192
193                let mut actual_value_place = place;
194
195                match value.ty.kind() {
196                    &ty::Pat(base, _) => {
197                        assert!(base.is_trivially_pure_clone_copy());
198
199                        let transmuted_place = self.temp(base, test.span);
200                        self.cfg.push_assign(
201                            block,
202                            self.source_info(scrutinee_span),
203                            transmuted_place,
204                            Rvalue::Cast(
205                                CastKind::Transmute,
206                                Operand::Copy(actual_value_place),
207                                base,
208                            ),
209                        );
210
211                        let transmuted_expect = self.temp(base, test.span);
212                        self.cfg.push_assign(
213                            block,
214                            self.source_info(test.span),
215                            transmuted_expect,
216                            Rvalue::Cast(CastKind::Transmute, expected_value_operand, base),
217                        );
218
219                        actual_value_place = transmuted_place;
220                        expected_value_operand = Operand::Copy(transmuted_expect);
221                        expected_value_ty = base;
222                    }
223                    _ => {}
224                }
225
226                assert!(expected_value_ty.is_scalar());
227
228                self.compare(
229                    block,
230                    success_block,
231                    fail_block,
232                    source_info,
233                    BinOp::Eq,
234                    expected_value_operand,
235                    Operand::Copy(actual_value_place),
236                );
237            }
238
239            TestKind::Range(ref range) => {
240                let success = target_block(TestBranch::Success);
241                let fail = target_block(TestBranch::Failure);
242                // Test `val` by computing `lo <= val && val <= hi`, using primitive comparisons.
243                let val = Operand::Copy(place);
244
245                let intermediate_block = if !range.lo.is_finite() {
246                    block
247                } else if !range.hi.is_finite() {
248                    success
249                } else {
250                    self.cfg.start_new_block()
251                };
252
253                if let Some(lo) = range.lo.as_finite() {
254                    let lo = ty::Value { ty: range.ty, valtree: lo };
255                    let lo = self.literal_operand(test.span, Const::from_ty_value(self.tcx, lo));
256                    self.compare(
257                        block,
258                        intermediate_block,
259                        fail,
260                        source_info,
261                        BinOp::Le,
262                        lo,
263                        val.clone(),
264                    );
265                };
266
267                if let Some(hi) = range.hi.as_finite() {
268                    let hi = ty::Value { ty: range.ty, valtree: hi };
269                    let hi = self.literal_operand(test.span, Const::from_ty_value(self.tcx, hi));
270                    let op = match range.end {
271                        RangeEnd::Included => BinOp::Le,
272                        RangeEnd::Excluded => BinOp::Lt,
273                    };
274                    self.compare(intermediate_block, success, fail, source_info, op, val, hi);
275                }
276            }
277
278            TestKind::SliceLen { len, op } => {
279                let usize_ty = self.tcx.types.usize;
280                let actual = self.temp(usize_ty, test.span);
281
282                // actual = len(place)
283                let length_op = self.len_of_slice_or_array(block, place, test.span, source_info);
284                self.cfg.push_assign(block, source_info, actual, Rvalue::Use(length_op));
285
286                // expected = <N>
287                let expected = self.push_usize(block, source_info, len);
288
289                let success_block = target_block(TestBranch::Success);
290                let fail_block = target_block(TestBranch::Failure);
291                // result = actual == expected OR result = actual < expected
292                // branch based on result
293                self.compare(
294                    block,
295                    success_block,
296                    fail_block,
297                    source_info,
298                    match op {
299                        SliceLenOp::Equal => BinOp::Eq,
300                        SliceLenOp::GreaterOrEqual => BinOp::Ge,
301                    },
302                    Operand::Move(actual),
303                    Operand::Move(expected),
304                );
305            }
306
307            TestKind::Deref { temp, mutability } => {
308                let ty = place_ty.ty;
309                let target = target_block(TestBranch::Success);
310                self.call_deref(block, target, place, mutability, ty, temp, test.span);
311            }
312
313            TestKind::Never => {
314                // Check that the place is initialized.
315                // FIXME(never_patterns): Also assert validity of the data at `place`.
316                self.cfg.push_fake_read(
317                    block,
318                    source_info,
319                    FakeReadCause::ForMatchedPlace(None),
320                    place,
321                );
322                // A never pattern is only allowed on an uninhabited type, so validity of the data
323                // implies unreachability.
324                self.cfg.terminate(block, source_info, TerminatorKind::Unreachable);
325            }
326        }
327    }
328
329    /// Perform `let temp = <ty as Deref>::deref(&place)`.
330    /// or `let temp = <ty as DerefMut>::deref_mut(&mut place)`.
331    pub(super) fn call_deref(
332        &mut self,
333        block: BasicBlock,
334        target_block: BasicBlock,
335        place: Place<'tcx>,
336        mutability: Mutability,
337        ty: Ty<'tcx>,
338        temp: Place<'tcx>,
339        span: Span,
340    ) {
341        let (trait_item, method) = match mutability {
342            Mutability::Not => (LangItem::Deref, sym::deref),
343            Mutability::Mut => (LangItem::DerefMut, sym::deref_mut),
344        };
345        let borrow_kind = super::util::ref_pat_borrow_kind(mutability);
346        let source_info = self.source_info(span);
347        let re_erased = self.tcx.lifetimes.re_erased;
348        let trait_item = self.tcx.require_lang_item(trait_item, span);
349        let method = trait_method(self.tcx, trait_item, method, [ty]);
350        let ref_src = self.temp(Ty::new_ref(self.tcx, re_erased, ty, mutability), span);
351        // `let ref_src = &src_place;`
352        // or `let ref_src = &mut src_place;`
353        self.cfg.push_assign(
354            block,
355            source_info,
356            ref_src,
357            Rvalue::Ref(re_erased, borrow_kind, place),
358        );
359        // `let temp = <Ty as Deref>::deref(ref_src);`
360        // or `let temp = <Ty as DerefMut>::deref_mut(ref_src);`
361        self.cfg.terminate(
362            block,
363            source_info,
364            TerminatorKind::Call {
365                func: Operand::Constant(Box::new(ConstOperand {
366                    span,
367                    user_ty: None,
368                    const_: method,
369                })),
370                args: [Spanned { node: Operand::Move(ref_src), span }].into(),
371                destination: temp,
372                target: Some(target_block),
373                unwind: UnwindAction::Continue,
374                call_source: CallSource::Misc,
375                fn_span: source_info.span,
376            },
377        );
378    }
379
380    /// Compare using the provided built-in comparison operator
381    fn compare(
382        &mut self,
383        block: BasicBlock,
384        success_block: BasicBlock,
385        fail_block: BasicBlock,
386        source_info: SourceInfo,
387        op: BinOp,
388        left: Operand<'tcx>,
389        right: Operand<'tcx>,
390    ) {
391        let bool_ty = self.tcx.types.bool;
392        let result = self.temp(bool_ty, source_info.span);
393
394        // result = op(left, right)
395        self.cfg.push_assign(
396            block,
397            source_info,
398            result,
399            Rvalue::BinaryOp(op, Box::new((left, right))),
400        );
401
402        // branch based on result
403        self.cfg.terminate(
404            block,
405            source_info,
406            TerminatorKind::if_(Operand::Move(result), success_block, fail_block),
407        );
408    }
409
410    /// Compare two values of type `&str` using `<str as std::cmp::PartialEq>::eq`.
411    fn string_compare(
412        &mut self,
413        block: BasicBlock,
414        success_block: BasicBlock,
415        fail_block: BasicBlock,
416        source_info: SourceInfo,
417        expect: Operand<'tcx>,
418        val: Operand<'tcx>,
419    ) {
420        let str_ty = self.tcx.types.str_;
421        let eq_def_id = self.tcx.require_lang_item(LangItem::PartialEq, source_info.span);
422        let method = trait_method(self.tcx, eq_def_id, sym::eq, [str_ty, str_ty]);
423
424        let bool_ty = self.tcx.types.bool;
425        let eq_result = self.temp(bool_ty, source_info.span);
426        let eq_block = self.cfg.start_new_block();
427        self.cfg.terminate(
428            block,
429            source_info,
430            TerminatorKind::Call {
431                func: Operand::Constant(Box::new(ConstOperand {
432                    span: source_info.span,
433
434                    // FIXME(#54571): This constant comes from user input (a
435                    // constant in a pattern). Are there forms where users can add
436                    // type annotations here?  For example, an associated constant?
437                    // Need to experiment.
438                    user_ty: None,
439
440                    const_: method,
441                })),
442                args: [
443                    Spanned { node: val, span: DUMMY_SP },
444                    Spanned { node: expect, span: DUMMY_SP },
445                ]
446                .into(),
447                destination: eq_result,
448                target: Some(eq_block),
449                unwind: UnwindAction::Continue,
450                call_source: CallSource::MatchCmp,
451                fn_span: source_info.span,
452            },
453        );
454        self.diverge_from(block);
455
456        // check the result
457        self.cfg.terminate(
458            eq_block,
459            source_info,
460            TerminatorKind::if_(Operand::Move(eq_result), success_block, fail_block),
461        );
462    }
463}
464
465fn trait_method<'tcx>(
466    tcx: TyCtxt<'tcx>,
467    trait_def_id: DefId,
468    method_name: Symbol,
469    args: impl IntoIterator<Item: Into<GenericArg<'tcx>>>,
470) -> Const<'tcx> {
471    // The unhygienic comparison here is acceptable because this is only
472    // used on known traits.
473    let item = tcx
474        .associated_items(trait_def_id)
475        .filter_by_name_unhygienic(method_name)
476        .find(|item| item.is_fn())
477        .expect("trait method not found");
478
479    let method_ty = Ty::new_fn_def(tcx, item.def_id, args);
480
481    Const::zero_sized(method_ty)
482}