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