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::{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 pub(super) fn pick_test_for_match_pair(
30 &mut self,
31 match_pair: &MatchPairTree<'tcx>,
32 ) -> Test<'tcx> {
33 let kind = match match_pair.testable_case {
34 TestableCase::Variant { adt_def, variant_index: _ } => TestKind::Switch { adt_def },
35
36 TestableCase::Constant { value: _, kind: PatConstKind::Bool } => TestKind::If,
37 TestableCase::Constant { value: _, kind: PatConstKind::IntOrChar } => {
38 TestKind::SwitchInt
39 }
40 TestableCase::Constant { value, kind: PatConstKind::String } => {
41 TestKind::StringEq { value }
42 }
43 TestableCase::Constant { value, kind: PatConstKind::Float | PatConstKind::Other } => {
44 TestKind::ScalarEq { value }
45 }
46
47 TestableCase::Range(ref range) => {
48 match (&range.ty, &match_pair.pattern_ty) {
(left_val, right_val) => {
if !(*left_val == *right_val) {
let kind = ::core::panicking::AssertKind::Eq;
::core::panicking::assert_failed(kind, &*left_val, &*right_val,
::core::option::Option::None);
}
}
};assert_eq!(range.ty, match_pair.pattern_ty);
49 TestKind::Range(Arc::clone(range))
50 }
51
52 TestableCase::Slice { len, op } => TestKind::SliceLen { len, op },
53
54 TestableCase::Deref { temp, mutability } => TestKind::Deref { temp, mutability },
55
56 TestableCase::Never => TestKind::Never,
57
58 TestableCase::Or { .. } => ::rustc_middle::util::bug::bug_fmt(format_args!("or-patterns should have already been handled"))bug!("or-patterns should have already been handled"),
61 };
62
63 Test { span: match_pair.pattern_span, kind }
64 }
65
66 #[allow(clippy :: suspicious_else_formatting)]
{
let __tracing_attr_span;
let __tracing_attr_guard;
if ::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() ||
{ false } {
__tracing_attr_span =
{
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("perform_test",
"rustc_mir_build::builder::matches::test",
::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_mir_build/src/builder/matches/test.rs"),
::tracing_core::__macro_support::Option::Some(66u32),
::tracing_core::__macro_support::Option::Some("rustc_mir_build::builder::matches::test"),
::tracing_core::field::FieldSet::new(&["match_start_span",
"scrutinee_span", "block", "otherwise_block", "test"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::SPAN)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let mut interest = ::tracing::subscriber::Interest::never();
if ::tracing::Level::DEBUG <=
::tracing::level_filters::STATIC_MAX_LEVEL &&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() &&
{ interest = __CALLSITE.interest(); !interest.is_never() }
&&
::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
interest) {
let meta = __CALLSITE.metadata();
::tracing::Span::new(meta,
&{
#[allow(unused_imports)]
use ::tracing::field::{debug, display, Value};
let mut iter = meta.fields().iter();
meta.fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&::tracing::field::debug(&match_start_span)
as &dyn Value)),
(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&::tracing::field::debug(&scrutinee_span)
as &dyn Value)),
(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&::tracing::field::debug(&block)
as &dyn Value)),
(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&::tracing::field::debug(&otherwise_block)
as &dyn Value)),
(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&::tracing::field::debug(&test)
as &dyn Value))])
})
} else {
let span =
::tracing::__macro_support::__disabled_span(__CALLSITE.metadata());
{};
span
}
};
__tracing_attr_guard = __tracing_attr_span.enter();
}
#[warn(clippy :: suspicious_else_formatting)]
{
#[allow(unknown_lints, unreachable_code, clippy ::
diverging_sub_expression, clippy :: empty_loop, clippy ::
let_unit_value, clippy :: let_with_type_underscore, clippy ::
needless_return, clippy :: unreachable)]
if false {
let __tracing_attr_fake_return: () = loop {};
return __tracing_attr_fake_return;
}
{
let place_ty = place.ty(&self.local_decls, self.tcx);
{
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_mir_build/src/builder/matches/test.rs:78",
"rustc_mir_build::builder::matches::test",
::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_mir_build/src/builder/matches/test.rs"),
::tracing_core::__macro_support::Option::Some(78u32),
::tracing_core::__macro_support::Option::Some("rustc_mir_build::builder::matches::test"),
::tracing_core::field::FieldSet::new(&["place", "place_ty"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::DEBUG <=
::tracing::level_filters::STATIC_MAX_LEVEL &&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() &&
{
let interest = __CALLSITE.interest();
!interest.is_never() &&
::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
interest)
};
if enabled {
(|value_set: ::tracing::field::ValueSet|
{
let meta = __CALLSITE.metadata();
::tracing::Event::dispatch(meta, &value_set);
;
})({
#[allow(unused_imports)]
use ::tracing::field::{debug, display, Value};
let mut iter = __CALLSITE.metadata().fields().iter();
__CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&debug(&place) as
&dyn Value)),
(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&debug(&place_ty)
as &dyn Value))])
});
} else { ; }
};
let target_block =
|branch|
target_blocks.get(&branch).copied().unwrap_or(otherwise_block);
let source_info = self.source_info(test.span);
match test.kind {
TestKind::Switch { adt_def } => {
let otherwise_block = target_block(TestBranch::Failure);
let switch_targets =
SwitchTargets::new(adt_def.discriminants(self.tcx).filter_map(|(idx,
discr)|
{
if let Some(&block) =
target_blocks.get(&TestBranch::Variant(idx)) {
Some((discr.val, block))
} else { None }
}), otherwise_block);
{
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_mir_build/src/builder/matches/test.rs:95",
"rustc_mir_build::builder::matches::test",
::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_mir_build/src/builder/matches/test.rs"),
::tracing_core::__macro_support::Option::Some(95u32),
::tracing_core::__macro_support::Option::Some("rustc_mir_build::builder::matches::test"),
::tracing_core::field::FieldSet::new(&["message"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::DEBUG <=
::tracing::level_filters::STATIC_MAX_LEVEL &&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() &&
{
let interest = __CALLSITE.interest();
!interest.is_never() &&
::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
interest)
};
if enabled {
(|value_set: ::tracing::field::ValueSet|
{
let meta = __CALLSITE.metadata();
::tracing::Event::dispatch(meta, &value_set);
;
})({
#[allow(unused_imports)]
use ::tracing::field::{debug, display, Value};
let mut iter = __CALLSITE.metadata().fields().iter();
__CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&format_args!("num_enum_variants: {0}",
adt_def.variants().len()) as &dyn Value))])
});
} else { ; }
};
let discr_ty = adt_def.repr().discr_type().to_ty(self.tcx);
let discr = self.temp(discr_ty, test.span);
self.cfg.push_assign(block,
self.source_info(scrutinee_span), discr,
Rvalue::Discriminant(place));
self.cfg.terminate(block,
self.source_info(match_start_span),
TerminatorKind::SwitchInt {
discr: Operand::Move(discr),
targets: switch_targets,
});
}
TestKind::SwitchInt => {
let otherwise_block = target_block(TestBranch::Failure);
let switch_targets =
SwitchTargets::new(target_blocks.iter().filter_map(|(&branch,
&block)|
{
if let TestBranch::Constant(value) = branch {
let bits = value.to_leaf().to_bits_unchecked();
Some((bits, block))
} else { None }
}), otherwise_block);
let terminator =
TerminatorKind::SwitchInt {
discr: Operand::Copy(place),
targets: switch_targets,
};
self.cfg.terminate(block,
self.source_info(match_start_span), terminator);
}
TestKind::If => {
let success_block = target_block(TestBranch::Success);
let fail_block = target_block(TestBranch::Failure);
let terminator =
TerminatorKind::if_(Operand::Copy(place), success_block,
fail_block);
self.cfg.terminate(block,
self.source_info(match_start_span), terminator);
}
TestKind::StringEq { value } => {
let tcx = self.tcx;
let success_block = target_block(TestBranch::Success);
let fail_block = target_block(TestBranch::Failure);
let ref_str_ty =
Ty::new_imm_ref(tcx, tcx.lifetimes.re_erased,
tcx.types.str_);
if !ref_str_ty.is_imm_ref_str() {
{
::core::panicking::panic_fmt(format_args!("{0:?}",
ref_str_ty));
}
};
if !value.ty.is_str() {
{
::core::panicking::panic_fmt(format_args!("unexpected value type for StringEq test: {0:?}",
value));
}
};
let expected_value =
ty::Value { ty: ref_str_ty, valtree: value.valtree };
let expected_value_operand =
self.literal_operand(test.span,
Const::from_ty_value(tcx, expected_value));
let actual_value_ref_place =
self.temp(ref_str_ty, test.span);
self.cfg.push_assign(block, self.source_info(test.span),
actual_value_ref_place,
Rvalue::Ref(tcx.lifetimes.re_erased, BorrowKind::Shared,
place));
self.string_compare(block, success_block, fail_block,
source_info, expected_value_operand,
Operand::Copy(actual_value_ref_place));
}
TestKind::ScalarEq { value } => {
let tcx = self.tcx;
let success_block = target_block(TestBranch::Success);
let fail_block = target_block(TestBranch::Failure);
let mut expected_value_ty = value.ty;
let mut expected_value_operand =
self.literal_operand(test.span,
Const::from_ty_value(tcx, value));
let mut actual_value_place = place;
match value.ty.kind() {
&ty::Pat(base, _) => {
if !base.is_trivially_pure_clone_copy() {
::core::panicking::panic("assertion failed: base.is_trivially_pure_clone_copy()")
};
let transmuted_place = self.temp(base, test.span);
self.cfg.push_assign(block,
self.source_info(scrutinee_span), transmuted_place,
Rvalue::Cast(CastKind::Transmute,
Operand::Copy(actual_value_place), base));
let transmuted_expect = self.temp(base, test.span);
self.cfg.push_assign(block, self.source_info(test.span),
transmuted_expect,
Rvalue::Cast(CastKind::Transmute, expected_value_operand,
base));
actual_value_place = transmuted_place;
expected_value_operand = Operand::Copy(transmuted_expect);
expected_value_ty = base;
}
_ => {}
}
if !expected_value_ty.is_scalar() {
::core::panicking::panic("assertion failed: expected_value_ty.is_scalar()")
};
self.compare(block, success_block, fail_block, source_info,
BinOp::Eq, expected_value_operand,
Operand::Copy(actual_value_place));
}
TestKind::Range(ref range) => {
let success = target_block(TestBranch::Success);
let fail = target_block(TestBranch::Failure);
let val = Operand::Copy(place);
let intermediate_block =
if !range.lo.is_finite() {
block
} else if !range.hi.is_finite() {
success
} else { self.cfg.start_new_block() };
if let Some(lo) = range.lo.as_finite() {
let lo = ty::Value { ty: range.ty, valtree: lo };
let lo =
self.literal_operand(test.span,
Const::from_ty_value(self.tcx, lo));
self.compare(block, intermediate_block, fail, source_info,
BinOp::Le, lo, val.clone());
};
if let Some(hi) = range.hi.as_finite() {
let hi = ty::Value { ty: range.ty, valtree: hi };
let hi =
self.literal_operand(test.span,
Const::from_ty_value(self.tcx, hi));
let op =
match range.end {
RangeEnd::Included => BinOp::Le,
RangeEnd::Excluded => BinOp::Lt,
};
self.compare(intermediate_block, success, fail, source_info,
op, val, hi);
}
}
TestKind::SliceLen { len, op } => {
let usize_ty = self.tcx.types.usize;
let actual = self.temp(usize_ty, test.span);
let length_op =
self.len_of_slice_or_array(block, place, test.span,
source_info);
self.cfg.push_assign(block, source_info, actual,
Rvalue::Use(length_op));
let expected = self.push_usize(block, source_info, len);
let success_block = target_block(TestBranch::Success);
let fail_block = target_block(TestBranch::Failure);
self.compare(block, success_block, fail_block, source_info,
match op {
SliceLenOp::Equal => BinOp::Eq,
SliceLenOp::GreaterOrEqual => BinOp::Ge,
}, Operand::Move(actual), Operand::Move(expected));
}
TestKind::Deref { temp, mutability } => {
let ty = place_ty.ty;
let target = target_block(TestBranch::Success);
self.call_deref(block, target, place, mutability, ty, temp,
test.span);
}
TestKind::Never => {
self.cfg.push_fake_read(block, source_info,
FakeReadCause::ForMatchedPlace(None), place);
self.cfg.terminate(block, source_info,
TerminatorKind::Unreachable);
}
}
}
}
}#[instrument(skip(self, target_blocks, place), level = "debug")]
67 pub(super) fn perform_test(
68 &mut self,
69 match_start_span: Span,
70 scrutinee_span: Span,
71 block: BasicBlock,
72 otherwise_block: BasicBlock,
73 place: Place<'tcx>,
74 test: &Test<'tcx>,
75 target_blocks: FxIndexMap<TestBranch<'tcx>, BasicBlock>,
76 ) {
77 let place_ty = place.ty(&self.local_decls, self.tcx);
78 debug!(?place, ?place_ty);
79 let target_block = |branch| target_blocks.get(&branch).copied().unwrap_or(otherwise_block);
80
81 let source_info = self.source_info(test.span);
82 match test.kind {
83 TestKind::Switch { adt_def } => {
84 let otherwise_block = target_block(TestBranch::Failure);
85 let switch_targets = SwitchTargets::new(
86 adt_def.discriminants(self.tcx).filter_map(|(idx, discr)| {
87 if let Some(&block) = target_blocks.get(&TestBranch::Variant(idx)) {
88 Some((discr.val, block))
89 } else {
90 None
91 }
92 }),
93 otherwise_block,
94 );
95 debug!("num_enum_variants: {}", adt_def.variants().len());
96 let discr_ty = adt_def.repr().discr_type().to_ty(self.tcx);
97 let discr = self.temp(discr_ty, test.span);
98 self.cfg.push_assign(
99 block,
100 self.source_info(scrutinee_span),
101 discr,
102 Rvalue::Discriminant(place),
103 );
104 self.cfg.terminate(
105 block,
106 self.source_info(match_start_span),
107 TerminatorKind::SwitchInt {
108 discr: Operand::Move(discr),
109 targets: switch_targets,
110 },
111 );
112 }
113
114 TestKind::SwitchInt => {
115 let otherwise_block = target_block(TestBranch::Failure);
117 let switch_targets = SwitchTargets::new(
118 target_blocks.iter().filter_map(|(&branch, &block)| {
119 if let TestBranch::Constant(value) = branch {
120 let bits = value.to_leaf().to_bits_unchecked();
121 Some((bits, block))
122 } else {
123 None
124 }
125 }),
126 otherwise_block,
127 );
128 let terminator = TerminatorKind::SwitchInt {
129 discr: Operand::Copy(place),
130 targets: switch_targets,
131 };
132 self.cfg.terminate(block, self.source_info(match_start_span), terminator);
133 }
134
135 TestKind::If => {
136 let success_block = target_block(TestBranch::Success);
137 let fail_block = target_block(TestBranch::Failure);
138 let terminator =
139 TerminatorKind::if_(Operand::Copy(place), success_block, fail_block);
140 self.cfg.terminate(block, self.source_info(match_start_span), terminator);
141 }
142
143 TestKind::StringEq { value } => {
144 let tcx = self.tcx;
145 let success_block = target_block(TestBranch::Success);
146 let fail_block = target_block(TestBranch::Failure);
147
148 let ref_str_ty = Ty::new_imm_ref(tcx, tcx.lifetimes.re_erased, tcx.types.str_);
149 assert!(ref_str_ty.is_imm_ref_str(), "{ref_str_ty:?}");
150
151 assert!(value.ty.is_str(), "unexpected value type for StringEq test: {value:?}");
157 let expected_value = ty::Value { ty: ref_str_ty, valtree: value.valtree };
158 let expected_value_operand =
159 self.literal_operand(test.span, Const::from_ty_value(tcx, expected_value));
160
161 let actual_value_ref_place = self.temp(ref_str_ty, test.span);
164 self.cfg.push_assign(
165 block,
166 self.source_info(test.span),
167 actual_value_ref_place,
168 Rvalue::Ref(tcx.lifetimes.re_erased, BorrowKind::Shared, place),
169 );
170
171 self.string_compare(
175 block,
176 success_block,
177 fail_block,
178 source_info,
179 expected_value_operand,
180 Operand::Copy(actual_value_ref_place),
181 );
182 }
183
184 TestKind::ScalarEq { value } => {
185 let tcx = self.tcx;
186 let success_block = target_block(TestBranch::Success);
187 let fail_block = target_block(TestBranch::Failure);
188
189 let mut expected_value_ty = value.ty;
190 let mut expected_value_operand =
191 self.literal_operand(test.span, Const::from_ty_value(tcx, value));
192
193 let mut actual_value_place = place;
194
195 match value.ty.kind() {
196 &ty::Pat(base, _) => {
197 assert!(base.is_trivially_pure_clone_copy());
198
199 let transmuted_place = self.temp(base, test.span);
200 self.cfg.push_assign(
201 block,
202 self.source_info(scrutinee_span),
203 transmuted_place,
204 Rvalue::Cast(
205 CastKind::Transmute,
206 Operand::Copy(actual_value_place),
207 base,
208 ),
209 );
210
211 let transmuted_expect = self.temp(base, test.span);
212 self.cfg.push_assign(
213 block,
214 self.source_info(test.span),
215 transmuted_expect,
216 Rvalue::Cast(CastKind::Transmute, expected_value_operand, base),
217 );
218
219 actual_value_place = transmuted_place;
220 expected_value_operand = Operand::Copy(transmuted_expect);
221 expected_value_ty = base;
222 }
223 _ => {}
224 }
225
226 assert!(expected_value_ty.is_scalar());
227
228 self.compare(
229 block,
230 success_block,
231 fail_block,
232 source_info,
233 BinOp::Eq,
234 expected_value_operand,
235 Operand::Copy(actual_value_place),
236 );
237 }
238
239 TestKind::Range(ref range) => {
240 let success = target_block(TestBranch::Success);
241 let fail = target_block(TestBranch::Failure);
242 let val = Operand::Copy(place);
244
245 let intermediate_block = if !range.lo.is_finite() {
246 block
247 } else if !range.hi.is_finite() {
248 success
249 } else {
250 self.cfg.start_new_block()
251 };
252
253 if let Some(lo) = range.lo.as_finite() {
254 let lo = ty::Value { ty: range.ty, valtree: lo };
255 let lo = self.literal_operand(test.span, Const::from_ty_value(self.tcx, lo));
256 self.compare(
257 block,
258 intermediate_block,
259 fail,
260 source_info,
261 BinOp::Le,
262 lo,
263 val.clone(),
264 );
265 };
266
267 if let Some(hi) = range.hi.as_finite() {
268 let hi = ty::Value { ty: range.ty, valtree: hi };
269 let hi = self.literal_operand(test.span, Const::from_ty_value(self.tcx, hi));
270 let op = match range.end {
271 RangeEnd::Included => BinOp::Le,
272 RangeEnd::Excluded => BinOp::Lt,
273 };
274 self.compare(intermediate_block, success, fail, source_info, op, val, hi);
275 }
276 }
277
278 TestKind::SliceLen { len, op } => {
279 let usize_ty = self.tcx.types.usize;
280 let actual = self.temp(usize_ty, test.span);
281
282 let length_op = self.len_of_slice_or_array(block, place, test.span, source_info);
284 self.cfg.push_assign(block, source_info, actual, Rvalue::Use(length_op));
285
286 let expected = self.push_usize(block, source_info, len);
288
289 let success_block = target_block(TestBranch::Success);
290 let fail_block = target_block(TestBranch::Failure);
291 self.compare(
294 block,
295 success_block,
296 fail_block,
297 source_info,
298 match op {
299 SliceLenOp::Equal => BinOp::Eq,
300 SliceLenOp::GreaterOrEqual => BinOp::Ge,
301 },
302 Operand::Move(actual),
303 Operand::Move(expected),
304 );
305 }
306
307 TestKind::Deref { temp, mutability } => {
308 let ty = place_ty.ty;
309 let target = target_block(TestBranch::Success);
310 self.call_deref(block, target, place, mutability, ty, temp, test.span);
311 }
312
313 TestKind::Never => {
314 self.cfg.push_fake_read(
317 block,
318 source_info,
319 FakeReadCause::ForMatchedPlace(None),
320 place,
321 );
322 self.cfg.terminate(block, source_info, TerminatorKind::Unreachable);
325 }
326 }
327 }
328
329 pub(super) fn call_deref(
332 &mut self,
333 block: BasicBlock,
334 target_block: BasicBlock,
335 place: Place<'tcx>,
336 mutability: Mutability,
337 ty: Ty<'tcx>,
338 temp: Place<'tcx>,
339 span: Span,
340 ) {
341 let (trait_item, method) = match mutability {
342 Mutability::Not => (LangItem::Deref, sym::deref),
343 Mutability::Mut => (LangItem::DerefMut, sym::deref_mut),
344 };
345 let borrow_kind = super::util::ref_pat_borrow_kind(mutability);
346 let source_info = self.source_info(span);
347 let re_erased = self.tcx.lifetimes.re_erased;
348 let trait_item = self.tcx.require_lang_item(trait_item, span);
349 let method = trait_method(self.tcx, trait_item, method, [ty]);
350 let ref_src = self.temp(Ty::new_ref(self.tcx, re_erased, ty, mutability), span);
351 self.cfg.push_assign(
354 block,
355 source_info,
356 ref_src,
357 Rvalue::Ref(re_erased, borrow_kind, place),
358 );
359 self.cfg.terminate(
362 block,
363 source_info,
364 TerminatorKind::Call {
365 func: Operand::Constant(Box::new(ConstOperand {
366 span,
367 user_ty: None,
368 const_: method,
369 })),
370 args: [Spanned { node: Operand::Move(ref_src), span }].into(),
371 destination: temp,
372 target: Some(target_block),
373 unwind: UnwindAction::Continue,
374 call_source: CallSource::Misc,
375 fn_span: source_info.span,
376 },
377 );
378 }
379
380 fn compare(
382 &mut self,
383 block: BasicBlock,
384 success_block: BasicBlock,
385 fail_block: BasicBlock,
386 source_info: SourceInfo,
387 op: BinOp,
388 left: Operand<'tcx>,
389 right: Operand<'tcx>,
390 ) {
391 let bool_ty = self.tcx.types.bool;
392 let result = self.temp(bool_ty, source_info.span);
393
394 self.cfg.push_assign(
396 block,
397 source_info,
398 result,
399 Rvalue::BinaryOp(op, Box::new((left, right))),
400 );
401
402 self.cfg.terminate(
404 block,
405 source_info,
406 TerminatorKind::if_(Operand::Move(result), success_block, fail_block),
407 );
408 }
409
410 fn string_compare(
412 &mut self,
413 block: BasicBlock,
414 success_block: BasicBlock,
415 fail_block: BasicBlock,
416 source_info: SourceInfo,
417 expect: Operand<'tcx>,
418 val: Operand<'tcx>,
419 ) {
420 let str_ty = self.tcx.types.str_;
421 let eq_def_id = self.tcx.require_lang_item(LangItem::PartialEq, source_info.span);
422 let method = trait_method(self.tcx, eq_def_id, sym::eq, [str_ty, str_ty]);
423
424 let bool_ty = self.tcx.types.bool;
425 let eq_result = self.temp(bool_ty, source_info.span);
426 let eq_block = self.cfg.start_new_block();
427 self.cfg.terminate(
428 block,
429 source_info,
430 TerminatorKind::Call {
431 func: Operand::Constant(Box::new(ConstOperand {
432 span: source_info.span,
433
434 user_ty: None,
439
440 const_: method,
441 })),
442 args: [
443 Spanned { node: val, span: DUMMY_SP },
444 Spanned { node: expect, span: DUMMY_SP },
445 ]
446 .into(),
447 destination: eq_result,
448 target: Some(eq_block),
449 unwind: UnwindAction::Continue,
450 call_source: CallSource::MatchCmp,
451 fn_span: source_info.span,
452 },
453 );
454 self.diverge_from(block);
455
456 self.cfg.terminate(
458 eq_block,
459 source_info,
460 TerminatorKind::if_(Operand::Move(eq_result), success_block, fail_block),
461 );
462 }
463}
464
465fn trait_method<'tcx>(
466 tcx: TyCtxt<'tcx>,
467 trait_def_id: DefId,
468 method_name: Symbol,
469 args: impl IntoIterator<Item: Into<GenericArg<'tcx>>>,
470) -> Const<'tcx> {
471 let item = tcx
474 .associated_items(trait_def_id)
475 .filter_by_name_unhygienic(method_name)
476 .find(|item| item.is_fn())
477 .expect("trait method not found");
478
479 let method_ty = Ty::new_fn_def(tcx, item.def_id, args);
480
481 Const::zero_sized(method_ty)
482}