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