2018-02-25 09:58:54 -06:00
|
|
|
// Copyright 2014 The Rust Project Developers. See the COPYRIGHT
|
|
|
|
// file at the top-level directory of this distribution and at
|
|
|
|
// http://rust-lang.org/COPYRIGHT.
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
|
|
|
|
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
|
|
|
|
// option. This file may not be copied, modified, or distributed
|
|
|
|
// except according to those terms.
|
|
|
|
|
|
|
|
use rustc::infer::canonical::{Canonical, QueryResult};
|
2018-05-16 16:58:08 -05:00
|
|
|
use rustc::traits::{self, FulfillmentContext, ObligationCause, SelectionContext};
|
2018-02-25 09:58:54 -06:00
|
|
|
use rustc::traits::query::{CanonicalProjectionGoal, NoSolution, normalize::NormalizationResult};
|
|
|
|
use rustc::ty::{ParamEnvAnd, TyCtxt};
|
2018-03-14 14:11:23 -05:00
|
|
|
use rustc_data_structures::sync::Lrc;
|
2018-02-25 09:58:54 -06:00
|
|
|
use syntax::ast::DUMMY_NODE_ID;
|
|
|
|
use syntax_pos::DUMMY_SP;
|
|
|
|
use util;
|
2018-04-01 01:17:25 -05:00
|
|
|
use std::sync::atomic::Ordering;
|
2018-02-25 09:58:54 -06:00
|
|
|
|
|
|
|
crate fn normalize_projection_ty<'tcx>(
|
|
|
|
tcx: TyCtxt<'_, 'tcx, 'tcx>,
|
|
|
|
goal: CanonicalProjectionGoal<'tcx>,
|
2018-03-14 14:11:23 -05:00
|
|
|
) -> Result<Lrc<Canonical<'tcx, QueryResult<'tcx, NormalizationResult<'tcx>>>>, NoSolution> {
|
2018-02-25 09:58:54 -06:00
|
|
|
debug!("normalize_provider(goal={:#?})", goal);
|
|
|
|
|
2018-04-01 01:17:25 -05:00
|
|
|
tcx.sess.perf_stats.normalize_projection_ty.fetch_add(1, Ordering::Relaxed);
|
2018-02-25 09:58:54 -06:00
|
|
|
tcx.infer_ctxt().enter(|ref infcx| {
|
|
|
|
let (
|
|
|
|
ParamEnvAnd {
|
|
|
|
param_env,
|
|
|
|
value: goal,
|
|
|
|
},
|
|
|
|
canonical_inference_vars,
|
|
|
|
) = infcx.instantiate_canonical_with_fresh_inference_vars(DUMMY_SP, &goal);
|
|
|
|
let fulfill_cx = &mut FulfillmentContext::new();
|
|
|
|
let selcx = &mut SelectionContext::new(infcx);
|
|
|
|
let cause = ObligationCause::misc(DUMMY_SP, DUMMY_NODE_ID);
|
2018-05-16 16:58:08 -05:00
|
|
|
let mut obligations = vec![];
|
|
|
|
let answer =
|
|
|
|
traits::normalize_projection_type(selcx, param_env, goal, cause, 0, &mut obligations);
|
2018-02-25 09:58:54 -06:00
|
|
|
fulfill_cx.register_predicate_obligations(infcx, obligations);
|
|
|
|
|
|
|
|
// Now that we have fulfilled as much as we can, create a solution
|
|
|
|
// from what we've learned.
|
|
|
|
util::make_query_response(
|
|
|
|
infcx,
|
|
|
|
canonical_inference_vars,
|
|
|
|
NormalizationResult { normalized_ty: answer },
|
|
|
|
fulfill_cx,
|
|
|
|
)
|
|
|
|
})
|
|
|
|
}
|