b5d3580843
module and into `hair/cx`, now that we don't have a trait defining the interface
150 lines
4.7 KiB
Rust
150 lines
4.7 KiB
Rust
// Copyright 2015 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.
|
|
|
|
/*!
|
|
* This module contains the code to convert from the wacky tcx data
|
|
* structures into the hair. The `builder` is generally ignorant of
|
|
* the tcx etc, and instead goes through the `Cx` for most of its
|
|
* work.
|
|
*/
|
|
|
|
use hair::*;
|
|
use repr::*;
|
|
|
|
use rustc::middle::const_eval::ConstVal;
|
|
use rustc::middle::def_id::DefId;
|
|
use rustc::middle::infer::InferCtxt;
|
|
use rustc::middle::subst::{Subst, Substs};
|
|
use rustc::middle::ty::{self, Ty};
|
|
use syntax::codemap::Span;
|
|
use syntax::parse::token;
|
|
|
|
#[derive(Copy, Clone)]
|
|
pub struct Cx<'a, 'tcx: 'a> {
|
|
tcx: &'a ty::ctxt<'tcx>,
|
|
infcx: &'a InferCtxt<'a, 'tcx>,
|
|
}
|
|
|
|
impl<'a,'tcx> Cx<'a,'tcx> {
|
|
pub fn new(infcx: &'a InferCtxt<'a, 'tcx>) -> Cx<'a, 'tcx> {
|
|
Cx {
|
|
tcx: infcx.tcx,
|
|
infcx: infcx,
|
|
}
|
|
}
|
|
}
|
|
|
|
pub use self::pattern::PatNode;
|
|
|
|
impl<'a,'tcx:'a> Cx<'a, 'tcx> {
|
|
/// Normalizes `ast` into the appropriate `mirror` type.
|
|
pub fn mirror<M: Mirror<'tcx>>(&mut self, ast: M) -> M::Output {
|
|
ast.make_mirror(self)
|
|
}
|
|
|
|
pub fn unit_ty(&mut self) -> Ty<'tcx> {
|
|
self.tcx.mk_nil()
|
|
}
|
|
|
|
pub fn usize_ty(&mut self) -> Ty<'tcx> {
|
|
self.tcx.types.usize
|
|
}
|
|
|
|
pub fn usize_literal(&mut self, value: usize) -> Literal<'tcx> {
|
|
Literal::Value { value: ConstVal::Uint(value as u64) }
|
|
}
|
|
|
|
pub fn bool_ty(&mut self) -> Ty<'tcx> {
|
|
self.tcx.types.bool
|
|
}
|
|
|
|
pub fn true_literal(&mut self) -> Literal<'tcx> {
|
|
Literal::Value { value: ConstVal::Bool(true) }
|
|
}
|
|
|
|
pub fn false_literal(&mut self) -> Literal<'tcx> {
|
|
Literal::Value { value: ConstVal::Bool(false) }
|
|
}
|
|
|
|
pub fn partial_eq(&mut self, ty: Ty<'tcx>) -> ItemRef<'tcx> {
|
|
let eq_def_id = self.tcx.lang_items.eq_trait().unwrap();
|
|
self.cmp_method_ref(eq_def_id, "eq", ty)
|
|
}
|
|
|
|
pub fn partial_le(&mut self, ty: Ty<'tcx>) -> ItemRef<'tcx> {
|
|
let ord_def_id = self.tcx.lang_items.ord_trait().unwrap();
|
|
self.cmp_method_ref(ord_def_id, "le", ty)
|
|
}
|
|
|
|
pub fn num_variants(&mut self, adt_def: ty::AdtDef<'tcx>) -> usize {
|
|
adt_def.variants.len()
|
|
}
|
|
|
|
pub fn all_fields(&mut self, adt_def: ty::AdtDef<'tcx>, variant_index: usize) -> Vec<Field> {
|
|
(0..adt_def.variants[variant_index].fields.len())
|
|
.map(Field::new)
|
|
.collect()
|
|
}
|
|
|
|
pub fn needs_drop(&mut self, ty: Ty<'tcx>, span: Span) -> bool {
|
|
if self.infcx.type_moves_by_default(ty, span) {
|
|
// FIXME(#21859) we should do an add'l check here to determine if
|
|
// any dtor will execute, but the relevant fn
|
|
// (`type_needs_drop`) is currently factored into
|
|
// `librustc_trans`, so we can't easily do so.
|
|
true
|
|
} else {
|
|
// if type implements Copy, cannot require drop
|
|
false
|
|
}
|
|
}
|
|
|
|
pub fn span_bug(&mut self, span: Span, message: &str) -> ! {
|
|
self.tcx.sess.span_bug(span, message)
|
|
}
|
|
|
|
pub fn tcx(&self) -> &'a ty::ctxt<'tcx> {
|
|
self.tcx
|
|
}
|
|
|
|
fn cmp_method_ref(&mut self,
|
|
trait_def_id: DefId,
|
|
method_name: &str,
|
|
arg_ty: Ty<'tcx>)
|
|
-> ItemRef<'tcx> {
|
|
let method_name = token::intern(method_name);
|
|
let substs = Substs::new_trait(vec![arg_ty], vec![], arg_ty);
|
|
for trait_item in self.tcx.trait_items(trait_def_id).iter() {
|
|
match *trait_item {
|
|
ty::ImplOrTraitItem::MethodTraitItem(ref method) => {
|
|
if method.name == method_name {
|
|
let method_ty = self.tcx.lookup_item_type(method.def_id);
|
|
let method_ty = method_ty.ty.subst(self.tcx, &substs);
|
|
return ItemRef {
|
|
ty: method_ty,
|
|
def_id: method.def_id,
|
|
substs: self.tcx.mk_substs(substs),
|
|
};
|
|
}
|
|
}
|
|
ty::ImplOrTraitItem::ConstTraitItem(..) |
|
|
ty::ImplOrTraitItem::TypeTraitItem(..) => {}
|
|
}
|
|
}
|
|
|
|
self.tcx.sess.bug(&format!("found no method `{}` in `{:?}`", method_name, trait_def_id));
|
|
}
|
|
}
|
|
|
|
mod block;
|
|
mod expr;
|
|
mod pattern;
|
|
mod to_ref;
|