1 // Copyright 2015 The Rust Project Developers. See the COPYRIGHT
2 // file at the top-level directory of this distribution and at
3 // http://rust-lang.org/COPYRIGHT.
5 // Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
6 // http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
7 // <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
8 // option. This file may not be copied, modified, or distributed
9 // except according to those terms.
12 use rustc_data_structures::indexed_vec::Idx;
13 use rustc_const_math::ConstInt;
16 use hair::cx::to_ref::ToRef;
18 use rustc::hir::def::Def;
19 use rustc::middle::const_val::ConstVal;
20 use rustc_const_eval as const_eval;
21 use rustc::middle::region::CodeExtent;
22 use rustc::ty::{self, VariantDef, Ty};
23 use rustc::ty::cast::CastKind as TyCastKind;
24 use rustc::mir::repr::*;
28 impl<'tcx> Mirror<'tcx> for &'tcx hir::Expr {
29 type Output = Expr<'tcx>;
31 fn make_mirror<'a, 'gcx>(self, cx: &mut Cx<'a, 'gcx, 'tcx>) -> Expr<'tcx> {
32 let temp_lifetime = cx.tcx.region_maps.temporary_scope(self.id);
33 let expr_extent = cx.tcx.region_maps.node_extent(self.id);
35 debug!("Expr::make_mirror(): id={}, span={:?}", self.id, self.span);
37 let mut expr = make_mirror_unadjusted(cx, self);
39 debug!("make_mirror: unadjusted-expr={:?} applying adjustments={:?}",
40 expr, cx.tcx.tables.borrow().adjustments.get(&self.id));
42 // Now apply adjustments, if any.
43 match cx.tcx.tables.borrow().adjustments.get(&self.id) {
45 Some(&ty::adjustment::AdjustReifyFnPointer) => {
46 let adjusted_ty = cx.tcx.expr_ty_adjusted(self);
48 temp_lifetime: temp_lifetime,
51 kind: ExprKind::ReifyFnPointer { source: expr.to_ref() },
54 Some(&ty::adjustment::AdjustUnsafeFnPointer) => {
55 let adjusted_ty = cx.tcx.expr_ty_adjusted(self);
57 temp_lifetime: temp_lifetime,
60 kind: ExprKind::UnsafeFnPointer { source: expr.to_ref() },
63 Some(&ty::adjustment::AdjustMutToConstPointer) => {
64 let adjusted_ty = cx.tcx.expr_ty_adjusted(self);
66 temp_lifetime: temp_lifetime,
69 kind: ExprKind::Cast { source: expr.to_ref() },
72 Some(&ty::adjustment::AdjustDerefRef(ref adj)) => {
73 for i in 0..adj.autoderefs {
76 expr.ty.adjust_for_autoderef(
81 |mc| cx.tcx.tables.borrow().method_map.get(&mc).map(|m| m.ty));
82 debug!("make_mirror: autoderef #{}, adjusted_ty={:?}", i, adjusted_ty);
83 let method_key = ty::MethodCall::autoderef(self.id, i);
85 cx.tcx.tables.borrow().method_map.get(&method_key).map(|m| m.ty);
86 let kind = if let Some(meth_ty) = meth_ty {
87 debug!("make_mirror: overloaded autoderef (meth_ty={:?})", meth_ty);
89 let ref_ty = cx.tcx.no_late_bound_regions(&meth_ty.fn_ret());
90 let (region, mutbl) = match ref_ty {
91 Some(ty::FnConverging(&ty::TyS {
92 sty: ty::TyRef(region, mt), ..
93 })) => (region, mt.mutbl),
94 _ => span_bug!(expr.span, "autoderef returned bad type")
98 temp_lifetime: temp_lifetime,
100 region, ty::TypeAndMut { ty: expr.ty, mutbl: mutbl }),
102 kind: ExprKind::Borrow {
104 borrow_kind: to_borrow_kind(mutbl),
109 overloaded_lvalue(cx, self, method_key,
110 PassArgs::ByRef, expr.to_ref(), vec![])
112 debug!("make_mirror: built-in autoderef");
113 ExprKind::Deref { arg: expr.to_ref() }
116 temp_lifetime: temp_lifetime,
123 if let Some(autoref) = adj.autoref {
124 let adjusted_ty = expr.ty.adjust_for_autoref(cx.tcx, Some(autoref));
126 ty::adjustment::AutoPtr(r, m) => {
128 temp_lifetime: temp_lifetime,
131 kind: ExprKind::Borrow {
133 borrow_kind: to_borrow_kind(m),
138 ty::adjustment::AutoUnsafe(m) => {
139 // Convert this to a suitable `&foo` and
140 // then an unsafe coercion. Limit the region to be just this
142 let region = ty::ReScope(expr_extent);
143 let region = cx.tcx.mk_region(region);
145 temp_lifetime: temp_lifetime,
146 ty: cx.tcx.mk_ref(region, ty::TypeAndMut { ty: expr.ty, mutbl: m }),
148 kind: ExprKind::Borrow {
150 borrow_kind: to_borrow_kind(m),
155 temp_lifetime: temp_lifetime,
158 kind: ExprKind::Cast { source: expr.to_ref() },
164 if let Some(target) = adj.unsize {
166 temp_lifetime: temp_lifetime,
169 kind: ExprKind::Unsize { source: expr.to_ref() },
175 // Next, wrap this up in the expr's scope.
177 temp_lifetime: temp_lifetime,
180 kind: ExprKind::Scope {
182 value: expr.to_ref(),
186 // Finally, create a destruction scope, if any.
187 if let Some(extent) = cx.tcx.region_maps.opt_destruction_extent(self.id) {
189 temp_lifetime: temp_lifetime,
192 kind: ExprKind::Scope {
194 value: expr.to_ref(),
204 fn make_mirror_unadjusted<'a, 'gcx, 'tcx>(cx: &mut Cx<'a, 'gcx, 'tcx>,
205 expr: &'tcx hir::Expr)
207 let expr_ty = cx.tcx.expr_ty(expr);
208 let temp_lifetime = cx.tcx.region_maps.temporary_scope(expr.id);
210 let kind = match expr.node {
211 // Here comes the interesting stuff:
212 hir::ExprMethodCall(_, _, ref args) => {
213 // Rewrite a.b(c) into UFCS form like Trait::b(a, c)
214 let expr = method_callee(cx, expr, ty::MethodCall::expr(expr.id));
215 let args = args.iter()
225 hir::ExprCall(ref fun, ref args) => {
226 if cx.tcx.is_method_call(expr.id) {
227 // The callee is something implementing Fn, FnMut, or FnOnce.
228 // Find the actual method implementation being called and
229 // build the appropriate UFCS call expression with the
230 // callee-object as expr parameter.
232 // rewrite f(u, v) into FnOnce::call_once(f, (u, v))
234 let method = method_callee(cx, expr, ty::MethodCall::expr(expr.id));
236 let sig = match method.ty.sty {
237 ty::TyFnDef(_, _, fn_ty) => &fn_ty.sig,
238 _ => span_bug!(expr.span, "type of method is not an fn")
241 let sig = cx.tcx.no_late_bound_regions(sig).unwrap_or_else(|| {
242 span_bug!(expr.span, "method call has late-bound regions")
245 assert_eq!(sig.inputs.len(), 2);
247 let tupled_args = Expr {
249 temp_lifetime: temp_lifetime,
251 kind: ExprKind::Tuple {
252 fields: args.iter().map(ToRef::to_ref).collect()
258 fun: method.to_ref(),
259 args: vec![fun.to_ref(), tupled_args.to_ref()]
262 let adt_data = if let hir::ExprPath(..) = fun.node {
263 // Tuple-like ADTs are represented as ExprCall. We convert them here.
264 expr_ty.ty_adt_def().and_then(|adt_def|{
265 match cx.tcx.expect_def(fun.id) {
266 Def::Variant(_, variant_id) => {
267 Some((adt_def, adt_def.variant_index_with_id(variant_id)))
276 if let Some((adt_def, index)) = adt_data {
277 let substs = cx.tcx.node_id_item_substs(fun.id).substs;
278 let field_refs = args.iter().enumerate().map(|(idx, e)| FieldExprRef {
279 name: Field::new(idx),
285 variant_index: index,
291 ty: cx.tcx.node_id_to_type(fun.id),
299 hir::ExprAddrOf(mutbl, ref expr) => {
300 let region = match expr_ty.sty {
301 ty::TyRef(r, _) => r,
302 _ => span_bug!(expr.span, "type of & not region"),
306 borrow_kind: to_borrow_kind(mutbl),
311 hir::ExprBlock(ref blk) => {
312 ExprKind::Block { body: &blk }
315 hir::ExprAssign(ref lhs, ref rhs) => {
322 hir::ExprAssignOp(op, ref lhs, ref rhs) => {
323 if cx.tcx.is_method_call(expr.id) {
324 let pass_args = if op.node.is_by_value() {
329 overloaded_operator(cx, expr, ty::MethodCall::expr(expr.id),
330 pass_args, lhs.to_ref(), vec![rhs])
340 hir::ExprLit(..) => ExprKind::Literal {
341 literal: cx.const_eval_literal(expr)
344 hir::ExprBinary(op, ref lhs, ref rhs) => {
345 if cx.tcx.is_method_call(expr.id) {
346 let pass_args = if op.node.is_by_value() {
351 overloaded_operator(cx, expr, ty::MethodCall::expr(expr.id),
352 pass_args, lhs.to_ref(), vec![rhs])
355 match (op.node, cx.constness) {
356 // FIXME(eddyb) use logical ops in constants when
357 // they can handle that kind of control-flow.
358 (hir::BinOp_::BiAnd, hir::Constness::Const) => {
365 (hir::BinOp_::BiOr, hir::Constness::Const) => {
373 (hir::BinOp_::BiAnd, hir::Constness::NotConst) => {
374 ExprKind::LogicalOp {
380 (hir::BinOp_::BiOr, hir::Constness::NotConst) => {
381 ExprKind::LogicalOp {
389 let op = bin_op(op.node);
400 hir::ExprIndex(ref lhs, ref index) => {
401 if cx.tcx.is_method_call(expr.id) {
402 overloaded_lvalue(cx, expr, ty::MethodCall::expr(expr.id),
403 PassArgs::ByValue, lhs.to_ref(), vec![index])
407 index: index.to_ref(),
412 hir::ExprUnary(hir::UnOp::UnDeref, ref arg) => {
413 if cx.tcx.is_method_call(expr.id) {
414 overloaded_lvalue(cx, expr, ty::MethodCall::expr(expr.id),
415 PassArgs::ByValue, arg.to_ref(), vec![])
417 ExprKind::Deref { arg: arg.to_ref() }
421 hir::ExprUnary(hir::UnOp::UnNot, ref arg) => {
422 if cx.tcx.is_method_call(expr.id) {
423 overloaded_operator(cx, expr, ty::MethodCall::expr(expr.id),
424 PassArgs::ByValue, arg.to_ref(), vec![])
433 hir::ExprUnary(hir::UnOp::UnNeg, ref arg) => {
434 if cx.tcx.is_method_call(expr.id) {
435 overloaded_operator(cx, expr, ty::MethodCall::expr(expr.id),
436 PassArgs::ByValue, arg.to_ref(), vec![])
438 // FIXME runtime-overflow
439 if let hir::ExprLit(_) = arg.node {
441 literal: cx.const_eval_literal(expr),
452 hir::ExprStruct(_, ref fields, ref base) => {
454 ty::TyStruct(adt, substs) => {
455 let field_refs = field_refs(&adt.variants[0], fields);
461 base: base.as_ref().map(|base| {
464 field_types: cx.tcx.tables
466 .fru_field_types[&expr.id]
472 ty::TyEnum(adt, substs) => {
473 match cx.tcx.expect_def(expr.id) {
474 Def::Variant(enum_id, variant_id) => {
475 debug_assert!(adt.did == enum_id);
476 assert!(base.is_none());
478 let index = adt.variant_index_with_id(variant_id);
479 let field_refs = field_refs(&adt.variants[index], fields);
482 variant_index: index,
491 "unexpected def: {:?}",
499 "unexpected type for struct literal: {:?}",
505 hir::ExprClosure(..) => {
506 let closure_ty = cx.tcx.expr_ty(expr);
507 let (def_id, substs) = match closure_ty.sty {
508 ty::TyClosure(def_id, substs) => (def_id, substs),
511 "closure expr w/o closure type: {:?}",
515 let upvars = cx.tcx.with_freevars(expr.id, |freevars| {
518 .map(|(i, fv)| capture_freevar(cx, expr, fv, substs.upvar_tys[i]))
528 hir::ExprPath(..) => {
529 convert_path_expr(cx, expr)
532 hir::ExprInlineAsm(ref asm, ref outputs, ref inputs) => {
533 ExprKind::InlineAsm {
535 outputs: outputs.to_ref(),
536 inputs: inputs.to_ref()
540 // Now comes the rote stuff:
542 hir::ExprRepeat(ref v, ref c) => ExprKind::Repeat {
544 count: TypedConstVal {
545 ty: cx.tcx.expr_ty(c),
547 value: match const_eval::eval_const_expr(cx.tcx.global_tcx(), c) {
548 ConstVal::Integral(ConstInt::Usize(u)) => u,
549 other => bug!("constant evaluation of repeat count yielded {:?}", other),
553 hir::ExprRet(ref v) =>
554 ExprKind::Return { value: v.to_ref() },
555 hir::ExprBreak(label) =>
556 ExprKind::Break { label: label.map(|_| loop_label(cx, expr)) },
557 hir::ExprAgain(label) =>
558 ExprKind::Continue { label: label.map(|_| loop_label(cx, expr)) },
559 hir::ExprMatch(ref discr, ref arms, _) =>
560 ExprKind::Match { discriminant: discr.to_ref(),
561 arms: arms.iter().map(|a| convert_arm(cx, a)).collect() },
562 hir::ExprIf(ref cond, ref then, ref otherwise) =>
563 ExprKind::If { condition: cond.to_ref(),
564 then: block::to_expr_ref(cx, then),
565 otherwise: otherwise.to_ref() },
566 hir::ExprWhile(ref cond, ref body, _) =>
567 ExprKind::Loop { condition: Some(cond.to_ref()),
568 body: block::to_expr_ref(cx, body) },
569 hir::ExprLoop(ref body, _) =>
570 ExprKind::Loop { condition: None,
571 body: block::to_expr_ref(cx, body) },
572 hir::ExprField(ref source, name) => {
573 let index = match cx.tcx.expr_ty_adjusted(source).sty {
574 ty::TyStruct(adt_def, _) =>
575 adt_def.variants[0].index_of_field_named(name.node),
579 "field of non-struct: {:?}",
582 let index = index.unwrap_or_else(|| {
585 "no index found for field `{}`",
588 ExprKind::Field { lhs: source.to_ref(), name: Field::new(index) }
590 hir::ExprTupField(ref source, index) =>
591 ExprKind::Field { lhs: source.to_ref(),
592 name: Field::new(index.node as usize) },
593 hir::ExprCast(ref source, _) => {
594 // Check to see if this cast is a "coercion cast", where the cast is actually done
595 // using a coercion (or is a no-op).
596 if let Some(&TyCastKind::CoercionCast) = cx.tcx.cast_kinds.borrow().get(&source.id) {
597 // Skip the actual cast itexpr, as it's now a no-op.
598 return source.make_mirror(cx);
600 ExprKind::Cast { source: source.to_ref() }
603 hir::ExprType(ref source, _) =>
604 return source.make_mirror(cx),
605 hir::ExprBox(ref value) =>
607 value: value.to_ref(),
608 value_extents: cx.tcx.region_maps.node_extent(value.id)
610 hir::ExprVec(ref fields) =>
611 ExprKind::Vec { fields: fields.to_ref() },
612 hir::ExprTup(ref fields) =>
613 ExprKind::Tuple { fields: fields.to_ref() },
617 temp_lifetime: temp_lifetime,
624 fn method_callee<'a, 'gcx, 'tcx>(cx: &mut Cx<'a, 'gcx, 'tcx>,
626 method_call: ty::MethodCall)
628 let tables = cx.tcx.tables.borrow();
629 let callee = &tables.method_map[&method_call];
630 let temp_lifetime = cx.tcx.region_maps.temporary_scope(expr.id);
632 temp_lifetime: temp_lifetime,
635 kind: ExprKind::Literal {
636 literal: Literal::Item {
637 def_id: callee.def_id,
638 substs: callee.substs,
644 fn to_borrow_kind(m: hir::Mutability) -> BorrowKind {
646 hir::MutMutable => BorrowKind::Mut,
647 hir::MutImmutable => BorrowKind::Shared,
651 fn convert_arm<'a, 'gcx, 'tcx>(cx: &mut Cx<'a, 'gcx, 'tcx>,
652 arm: &'tcx hir::Arm) -> Arm<'tcx> {
654 patterns: arm.pats.iter().map(|p| cx.refutable_pat(p)).collect(),
655 guard: arm.guard.to_ref(),
656 body: arm.body.to_ref(),
660 fn convert_path_expr<'a, 'gcx, 'tcx>(cx: &mut Cx<'a, 'gcx, 'tcx>,
661 expr: &'tcx hir::Expr)
663 let substs = cx.tcx.node_id_item_substs(expr.id).substs;
664 // Otherwise there may be def_map borrow conflicts
665 let def = cx.tcx.expect_def(expr.id);
666 let def_id = match def {
667 // A regular function.
668 Def::Fn(def_id) | Def::Method(def_id) => def_id,
669 Def::Struct(def_id) => match cx.tcx.node_id_to_type(expr.id).sty {
670 // A tuple-struct constructor. Should only be reached if not called in the same
672 ty::TyFnDef(..) => def_id,
673 // A unit struct which is used as a value. We return a completely different ExprKind
674 // here to account for this special case.
675 ty::TyStruct(adt_def, substs) => return ExprKind::Adt {
682 ref sty => bug!("unexpected sty: {:?}", sty)
684 Def::Variant(enum_id, variant_id) => match cx.tcx.node_id_to_type(expr.id).sty {
685 // A variant constructor. Should only be reached if not called in the same
687 ty::TyFnDef(..) => variant_id,
688 // A unit variant, similar special case to the struct case above.
689 ty::TyEnum(adt_def, substs) => {
690 debug_assert!(adt_def.did == enum_id);
691 let index = adt_def.variant_index_with_id(variant_id);
692 return ExprKind::Adt {
695 variant_index: index,
700 ref sty => bug!("unexpected sty: {:?}", sty)
703 Def::AssociatedConst(def_id) => def_id,
705 Def::Static(node_id, _) => return ExprKind::StaticRef {
709 Def::Local(..) | Def::Upvar(..) => return convert_var(cx, expr, def),
711 _ => span_bug!(expr.span, "def `{:?}` not yet implemented", def),
714 literal: Literal::Item { def_id: def_id, substs: substs }
718 fn convert_var<'a, 'gcx, 'tcx>(cx: &mut Cx<'a, 'gcx, 'tcx>,
719 expr: &'tcx hir::Expr,
722 let temp_lifetime = cx.tcx.region_maps.temporary_scope(expr.id);
725 Def::Local(_, node_id) => {
731 Def::Upvar(_, id_var, index, closure_expr_id) => {
732 debug!("convert_var(upvar({:?}, {:?}, {:?}))", id_var, index, closure_expr_id);
733 let var_ty = cx.tcx.node_id_to_type(id_var);
735 let body_id = match cx.tcx.map.find(closure_expr_id) {
736 Some(map::NodeExpr(expr)) => {
738 hir::ExprClosure(_, _, ref body, _) => body.id,
740 span_bug!(expr.span, "closure expr is not a closure expr");
745 span_bug!(expr.span, "ast-map has garbage for closure expr");
749 // FIXME free regions in closures are not right
750 let closure_ty = cx.tcx.node_id_to_type(closure_expr_id);
752 // FIXME we're just hard-coding the idea that the
753 // signature will be &self or &mut self and hence will
754 // have a bound region with number 0
755 let region = ty::Region::ReFree(ty::FreeRegion {
756 scope: cx.tcx.region_maps.node_extent(body_id),
757 bound_region: ty::BoundRegion::BrAnon(0),
759 let region = cx.tcx.mk_region(region);
761 let self_expr = match cx.tcx.closure_kind(cx.tcx.map.local_def_id(closure_expr_id)) {
762 ty::ClosureKind::Fn => {
764 cx.tcx.mk_ref(region,
765 ty::TypeAndMut { ty: closure_ty,
766 mutbl: hir::MutImmutable });
769 temp_lifetime: temp_lifetime,
771 kind: ExprKind::Deref {
774 temp_lifetime: temp_lifetime,
776 kind: ExprKind::SelfRef
781 ty::ClosureKind::FnMut => {
783 cx.tcx.mk_ref(region,
784 ty::TypeAndMut { ty: closure_ty,
785 mutbl: hir::MutMutable });
788 temp_lifetime: temp_lifetime,
790 kind: ExprKind::Deref {
793 temp_lifetime: temp_lifetime,
795 kind: ExprKind::SelfRef
800 ty::ClosureKind::FnOnce => {
803 temp_lifetime: temp_lifetime,
805 kind: ExprKind::SelfRef,
810 // at this point we have `self.n`, which loads up the upvar
811 let field_kind = ExprKind::Field {
812 lhs: self_expr.to_ref(),
813 name: Field::new(index),
816 // ...but the upvar might be an `&T` or `&mut T` capture, at which
817 // point we need an implicit deref
818 let upvar_id = ty::UpvarId {
820 closure_expr_id: closure_expr_id,
822 let upvar_capture = match cx.tcx.upvar_capture(upvar_id) {
827 "no upvar_capture for {:?}",
831 match upvar_capture {
832 ty::UpvarCapture::ByValue => field_kind,
833 ty::UpvarCapture::ByRef(borrow) => {
836 temp_lifetime: temp_lifetime,
838 cx.tcx.mk_region(borrow.region),
841 mutbl: borrow.kind.to_mutbl_lossy()
851 _ => span_bug!(expr.span, "type of & not region"),
856 fn bin_op(op: hir::BinOp_) -> BinOp {
858 hir::BinOp_::BiAdd => BinOp::Add,
859 hir::BinOp_::BiSub => BinOp::Sub,
860 hir::BinOp_::BiMul => BinOp::Mul,
861 hir::BinOp_::BiDiv => BinOp::Div,
862 hir::BinOp_::BiRem => BinOp::Rem,
863 hir::BinOp_::BiBitXor => BinOp::BitXor,
864 hir::BinOp_::BiBitAnd => BinOp::BitAnd,
865 hir::BinOp_::BiBitOr => BinOp::BitOr,
866 hir::BinOp_::BiShl => BinOp::Shl,
867 hir::BinOp_::BiShr => BinOp::Shr,
868 hir::BinOp_::BiEq => BinOp::Eq,
869 hir::BinOp_::BiLt => BinOp::Lt,
870 hir::BinOp_::BiLe => BinOp::Le,
871 hir::BinOp_::BiNe => BinOp::Ne,
872 hir::BinOp_::BiGe => BinOp::Ge,
873 hir::BinOp_::BiGt => BinOp::Gt,
874 _ => bug!("no equivalent for ast binop {:?}", op),
883 fn overloaded_operator<'a, 'gcx, 'tcx>(cx: &mut Cx<'a, 'gcx, 'tcx>,
884 expr: &'tcx hir::Expr,
885 method_call: ty::MethodCall,
887 receiver: ExprRef<'tcx>,
888 args: Vec<&'tcx P<hir::Expr>>)
890 // the receiver has all the adjustments that are needed, so we can
891 // just push a reference to it
892 let mut argrefs = vec![receiver];
894 // the arguments, unfortunately, do not, so if this is a ByRef
895 // operator, we have to gin up the autorefs (but by value is easy)
897 PassArgs::ByValue => {
898 argrefs.extend(args.iter().map(|arg| arg.to_ref()))
902 let scope = cx.tcx.region_maps.node_extent(expr.id);
903 let region = cx.tcx.mk_region(ty::ReScope(scope));
904 let temp_lifetime = cx.tcx.region_maps.temporary_scope(expr.id);
908 let arg_ty = cx.tcx.expr_ty_adjusted(arg);
910 cx.tcx.mk_ref(region,
911 ty::TypeAndMut { ty: arg_ty,
912 mutbl: hir::MutImmutable });
914 temp_lifetime: temp_lifetime,
917 kind: ExprKind::Borrow { region: *region,
918 borrow_kind: BorrowKind::Shared,
925 // now create the call itself
926 let fun = method_callee(cx, expr, method_call);
934 fn overloaded_lvalue<'a, 'gcx, 'tcx>(cx: &mut Cx<'a, 'gcx, 'tcx>,
935 expr: &'tcx hir::Expr,
936 method_call: ty::MethodCall,
938 receiver: ExprRef<'tcx>,
939 args: Vec<&'tcx P<hir::Expr>>)
941 // For an overloaded *x or x[y] expression of type T, the method
942 // call returns an &T and we must add the deref so that the types
943 // line up (this is because `*x` and `x[y]` represent lvalues):
945 // to find the type &T of the content returned by the method;
946 let tables = cx.tcx.tables.borrow();
947 let callee = &tables.method_map[&method_call];
948 let ref_ty = callee.ty.fn_ret();
949 let ref_ty = cx.tcx.no_late_bound_regions(&ref_ty).unwrap().unwrap();
951 // (1) callees always have all late-bound regions fully instantiated,
952 // (2) overloaded methods don't return `!`
954 // construct the complete expression `foo()` for the overloaded call,
955 // which will yield the &T type
956 let temp_lifetime = cx.tcx.region_maps.temporary_scope(expr.id);
957 let ref_kind = overloaded_operator(cx, expr, method_call, pass_args, receiver, args);
958 let ref_expr = Expr {
959 temp_lifetime: temp_lifetime,
965 // construct and return a deref wrapper `*foo()`
966 ExprKind::Deref { arg: ref_expr.to_ref() }
969 fn capture_freevar<'a, 'gcx, 'tcx>(cx: &mut Cx<'a, 'gcx, 'tcx>,
970 closure_expr: &'tcx hir::Expr,
971 freevar: &hir::Freevar,
972 freevar_ty: Ty<'tcx>)
974 let id_var = freevar.def.var_id();
975 let upvar_id = ty::UpvarId {
977 closure_expr_id: closure_expr.id,
979 let upvar_capture = cx.tcx.upvar_capture(upvar_id).unwrap();
980 let temp_lifetime = cx.tcx.region_maps.temporary_scope(closure_expr.id);
981 let var_ty = cx.tcx.node_id_to_type(id_var);
982 let captured_var = Expr {
983 temp_lifetime: temp_lifetime,
985 span: closure_expr.span,
986 kind: convert_var(cx, closure_expr, freevar.def),
988 match upvar_capture {
989 ty::UpvarCapture::ByValue => {
990 captured_var.to_ref()
992 ty::UpvarCapture::ByRef(upvar_borrow) => {
993 let borrow_kind = match upvar_borrow.kind {
994 ty::BorrowKind::ImmBorrow => BorrowKind::Shared,
995 ty::BorrowKind::UniqueImmBorrow => BorrowKind::Unique,
996 ty::BorrowKind::MutBorrow => BorrowKind::Mut,
999 temp_lifetime: temp_lifetime,
1001 span: closure_expr.span,
1002 kind: ExprKind::Borrow { region: upvar_borrow.region,
1003 borrow_kind: borrow_kind,
1004 arg: captured_var.to_ref() }
1010 fn loop_label<'a, 'gcx, 'tcx>(cx: &mut Cx<'a, 'gcx, 'tcx>,
1011 expr: &'tcx hir::Expr) -> CodeExtent {
1012 match cx.tcx.expect_def(expr.id) {
1013 Def::Label(loop_id) => cx.tcx.region_maps.node_extent(loop_id),
1014 d => span_bug!(expr.span, "loop scope resolved to {:?}", d),
1018 /// Converts a list of named fields (i.e. for struct-like struct/enum ADTs) into FieldExprRef.
1019 fn field_refs<'tcx>(variant: VariantDef<'tcx>,
1020 fields: &'tcx [hir::Field])
1021 -> Vec<FieldExprRef<'tcx>>
1024 .map(|field| FieldExprRef {
1025 name: Field::new(variant.index_of_field_named(field.name.node).unwrap()),
1026 expr: field.expr.to_ref(),