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