1use 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 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 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 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 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 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 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 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 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 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 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 self.cfg.push_fake_read(
318 block,
319 source_info,
320 FakeReadCause::ForMatchedPlace(None),
321 place,
322 );
323 self.cfg.terminate(block, source_info, TerminatorKind::Unreachable);
326 }
327 }
328 }
329
330 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 self.cfg.push_assign(
355 block,
356 source_info,
357 ref_src,
358 Rvalue::Ref(re_erased, borrow_kind, place),
359 );
360 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 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 self.cfg.push_assign(
397 block,
398 source_info,
399 result,
400 Rvalue::BinaryOp(op, Box::new((left, right))),
401 );
402
403 self.cfg.terminate(
405 block,
406 source_info,
407 TerminatorKind::if_(Operand::Move(result), success_block, fail_block),
408 );
409 }
410
411 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 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 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 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}