1 // Copyright 2014 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 super::check_argument_types;
13 use super::check_expr;
14 use super::check_method_argument_types;
16 use super::DeferredCallResolution;
18 use super::Expectation;
19 use super::expected_types_for_fn_args;
21 use super::LvaluePreference;
23 use super::structurally_resolved_type;
24 use super::TupleArgumentsFlag;
25 use super::UnresolvedTypeAction;
26 use super::write_call;
30 use middle::ty::{self, Ty};
32 use syntax::codemap::Span;
33 use syntax::parse::token;
36 /// Check that it is legal to call methods of the trait corresponding
37 /// to `trait_id` (this only cares about the trait, not the specific
38 /// method that is called)
39 pub fn check_legal_trait_for_method_call(ccx: &CrateCtxt, span: Span, trait_id: ast::DefId) {
41 let did = Some(trait_id);
42 let li = &tcx.lang_items;
44 if did == li.drop_trait() {
45 span_err!(tcx.sess, span, E0040, "explicit use of destructor method");
46 } else if !tcx.sess.features.borrow().unboxed_closures {
47 // the #[feature(unboxed_closures)] feature isn't
48 // activated so we need to enforce the closure
51 let method = if did == li.fn_trait() {
53 } else if did == li.fn_mut_trait() {
55 } else if did == li.fn_once_trait() {
58 return // not a closure method, everything is OK.
61 span_err!(tcx.sess, span, E0174,
62 "explicit use of unboxed closure method `{}` is experimental",
64 fileline_help!(tcx.sess, span,
65 "add `#![feature(unboxed_closures)]` to the crate attributes to enable");
69 pub fn check_call<'a, 'tcx>(fcx: &FnCtxt<'a, 'tcx>,
70 call_expr: &'tcx ast::Expr,
71 callee_expr: &'tcx ast::Expr,
72 arg_exprs: &'tcx [P<ast::Expr>],
73 expected: Expectation<'tcx>)
75 check_expr(fcx, callee_expr);
76 let original_callee_ty = fcx.expr_ty(callee_expr);
77 let (callee_ty, _, result) =
82 UnresolvedTypeAction::Error,
83 LvaluePreference::NoPreference,
85 try_overloaded_call_step(fcx, call_expr, callee_expr, adj_ty, idx)
90 // this will report an error since original_callee_ty is not a fn
91 confirm_builtin_call(fcx, call_expr, original_callee_ty, arg_exprs, expected);
94 Some(CallStep::Builtin) => {
95 confirm_builtin_call(fcx, call_expr, callee_ty, arg_exprs, expected);
98 Some(CallStep::DeferredClosure(fn_sig)) => {
99 confirm_deferred_closure_call(fcx, call_expr, arg_exprs, expected, fn_sig);
102 Some(CallStep::Overloaded(method_callee)) => {
103 confirm_overloaded_call(fcx, call_expr, callee_expr,
104 arg_exprs, expected, method_callee);
109 enum CallStep<'tcx> {
111 DeferredClosure(ty::FnSig<'tcx>),
112 Overloaded(ty::MethodCallee<'tcx>)
115 fn try_overloaded_call_step<'a, 'tcx>(fcx: &FnCtxt<'a, 'tcx>,
116 call_expr: &'tcx ast::Expr,
117 callee_expr: &'tcx ast::Expr,
118 adjusted_ty: Ty<'tcx>,
120 -> Option<CallStep<'tcx>>
122 debug!("try_overloaded_call_step(call_expr={:?}, adjusted_ty={:?}, autoderefs={})",
127 // If the callee is a bare function or a closure, then we're all set.
128 match structurally_resolved_type(fcx, callee_expr.span, adjusted_ty).sty {
129 ty::TyBareFn(..) => {
130 fcx.write_autoderef_adjustment(callee_expr.id, autoderefs);
131 return Some(CallStep::Builtin);
134 ty::TyClosure(def_id, ref substs) => {
135 assert_eq!(def_id.krate, ast::LOCAL_CRATE);
137 // Check whether this is a call to a closure where we
138 // haven't yet decided on whether the closure is fn vs
139 // fnmut vs fnonce. If so, we have to defer further processing.
140 if fcx.infcx().closure_kind(def_id).is_none() {
142 fcx.infcx().closure_type(def_id, substs);
144 fcx.infcx().replace_late_bound_regions_with_fresh_var(call_expr.span,
147 fcx.record_deferred_call_resolution(def_id, Box::new(CallResolution {
148 call_expr: call_expr,
149 callee_expr: callee_expr,
150 adjusted_ty: adjusted_ty,
151 autoderefs: autoderefs,
152 fn_sig: fn_sig.clone(),
153 closure_def_id: def_id
155 return Some(CallStep::DeferredClosure(fn_sig));
159 // Hack: we know that there are traits implementing Fn for &F
160 // where F:Fn and so forth. In the particular case of types
161 // like `x: &mut FnMut()`, if there is a call `x()`, we would
162 // normally translate to `FnMut::call_mut(&mut x, ())`, but
163 // that winds up requiring `mut x: &mut FnMut()`. A little
164 // over the top. The simplest fix by far is to just ignore
165 // this case and deref again, so we wind up with
166 // `FnMut::call_mut(&mut *x, ())`.
167 ty::TyRef(..) if autoderefs == 0 => {
174 try_overloaded_call_traits(fcx, call_expr, callee_expr, adjusted_ty, autoderefs)
175 .map(|method_callee| CallStep::Overloaded(method_callee))
178 fn try_overloaded_call_traits<'a,'tcx>(fcx: &FnCtxt<'a, 'tcx>,
179 call_expr: &ast::Expr,
180 callee_expr: &ast::Expr,
181 adjusted_ty: Ty<'tcx>,
183 -> Option<ty::MethodCallee<'tcx>>
185 // Try the options that are least restrictive on the caller first.
186 for &(opt_trait_def_id, method_name) in &[
187 (fcx.tcx().lang_items.fn_trait(), token::intern("call")),
188 (fcx.tcx().lang_items.fn_mut_trait(), token::intern("call_mut")),
189 (fcx.tcx().lang_items.fn_once_trait(), token::intern("call_once")),
191 let trait_def_id = match opt_trait_def_id {
192 Some(def_id) => def_id,
196 match method::lookup_in_trait_adjusted(fcx,
206 Some(method_callee) => {
207 return Some(method_callee);
215 fn confirm_builtin_call<'a,'tcx>(fcx: &FnCtxt<'a,'tcx>,
216 call_expr: &ast::Expr,
218 arg_exprs: &'tcx [P<ast::Expr>],
219 expected: Expectation<'tcx>)
223 let fn_sig = match callee_ty.sty {
224 ty::TyBareFn(_, &ty::BareFnTy {ref sig, ..}) => {
228 fcx.type_error_message(call_expr.span, |actual| {
229 format!("expected function, found `{}`", actual)
232 // This is the "default" function signature, used in case of error.
233 // In that case, we check each argument against "error" in order to
234 // set up all the node type bindings.
235 error_fn_sig = ty::Binder(ty::FnSig {
236 inputs: err_args(fcx.tcx(), arg_exprs.len()),
237 output: ty::FnConverging(fcx.tcx().types.err),
245 // Replace any late-bound regions that appear in the function
246 // signature with region variables. We also have to
247 // renormalize the associated types at this point, since they
248 // previously appeared within a `Binder<>` and hence would not
249 // have been normalized before.
251 fcx.infcx().replace_late_bound_regions_with_fresh_var(call_expr.span,
255 fcx.normalize_associated_types_in(call_expr.span, &fn_sig);
257 // Call the generic checker.
258 let expected_arg_tys = expected_types_for_fn_args(fcx,
263 check_argument_types(fcx,
266 &expected_arg_tys[..],
269 TupleArgumentsFlag::DontTupleArguments);
271 write_call(fcx, call_expr, fn_sig.output);
274 fn confirm_deferred_closure_call<'a,'tcx>(fcx: &FnCtxt<'a,'tcx>,
275 call_expr: &ast::Expr,
276 arg_exprs: &'tcx [P<ast::Expr>],
277 expected: Expectation<'tcx>,
278 fn_sig: ty::FnSig<'tcx>)
280 // `fn_sig` is the *signature* of the cosure being called. We
281 // don't know the full details yet (`Fn` vs `FnMut` etc), but we
282 // do know the types expected for each argument and the return
285 let expected_arg_tys =
286 expected_types_for_fn_args(fcx,
289 fn_sig.output.clone(),
292 check_argument_types(fcx,
298 TupleArgumentsFlag::TupleArguments);
300 write_call(fcx, call_expr, fn_sig.output);
303 fn confirm_overloaded_call<'a,'tcx>(fcx: &FnCtxt<'a, 'tcx>,
304 call_expr: &ast::Expr,
305 callee_expr: &'tcx ast::Expr,
306 arg_exprs: &'tcx [P<ast::Expr>],
307 expected: Expectation<'tcx>,
308 method_callee: ty::MethodCallee<'tcx>)
311 check_method_argument_types(fcx,
316 TupleArgumentsFlag::TupleArguments,
318 write_call(fcx, call_expr, output_type);
320 write_overloaded_call_method_map(fcx, call_expr, method_callee);
323 fn write_overloaded_call_method_map<'a,'tcx>(fcx: &FnCtxt<'a, 'tcx>,
324 call_expr: &ast::Expr,
325 method_callee: ty::MethodCallee<'tcx>) {
326 let method_call = ty::MethodCall::expr(call_expr.id);
327 fcx.inh.tables.borrow_mut().method_map.insert(method_call, method_callee);
331 struct CallResolution<'tcx> {
332 call_expr: &'tcx ast::Expr,
333 callee_expr: &'tcx ast::Expr,
334 adjusted_ty: Ty<'tcx>,
336 fn_sig: ty::FnSig<'tcx>,
337 closure_def_id: ast::DefId,
340 impl<'tcx> DeferredCallResolution<'tcx> for CallResolution<'tcx> {
341 fn resolve<'a>(&mut self, fcx: &FnCtxt<'a,'tcx>) {
342 debug!("DeferredCallResolution::resolve() {:?}",
345 // we should not be invoked until the closure kind has been
346 // determined by upvar inference
347 assert!(fcx.infcx().closure_kind(self.closure_def_id).is_some());
349 // We may now know enough to figure out fn vs fnmut etc.
350 match try_overloaded_call_traits(fcx, self.call_expr, self.callee_expr,
351 self.adjusted_ty, self.autoderefs) {
352 Some(method_callee) => {
353 // One problem is that when we get here, we are going
354 // to have a newly instantiated function signature
355 // from the call trait. This has to be reconciled with
356 // the older function signature we had before. In
357 // principle we *should* be able to fn_sigs(), but we
358 // can't because of the annoying need for a TypeTrace.
359 // (This always bites me, should find a way to
361 let method_sig = fcx.tcx().no_late_bound_regions(method_callee.ty.fn_sig())
364 debug!("attempt_resolution: method_callee={:?}",
367 for (&method_arg_ty, &self_arg_ty) in
368 method_sig.inputs[1..].iter().zip(&self.fn_sig.inputs)
370 demand::eqtype(fcx, self.call_expr.span, self_arg_ty, method_arg_ty);
373 let nilty = fcx.tcx().mk_nil();
376 method_sig.output.unwrap_or(nilty),
377 self.fn_sig.output.unwrap_or(nilty));
379 write_overloaded_call_method_map(fcx, self.call_expr, method_callee);
382 fcx.tcx().sess.span_bug(
384 "failed to find an overloaded call trait for closure call");