rust/src/librustc_metadata/index_builder.rs

96 lines
3.1 KiB
Rust
Raw Normal View History

2016-08-10 13:27:06 -05:00
// Copyright 2012-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.
2016-08-10 20:37:53 -05:00
use common::tag_items_data_item;
2016-08-10 13:27:06 -05:00
use encoder::EncodeContext;
use index::IndexData;
use rbml::writer::Encoder;
2016-08-10 20:37:53 -05:00
use rustc::dep_graph::DepNode;
2016-08-10 13:27:06 -05:00
use rustc::hir::def_id::DefId;
use rustc::ty;
use rustc_data_structures::fnv::FnvHashMap;
use std::ops::{Deref, DerefMut};
2016-08-10 13:27:06 -05:00
/// Builder that can encode new items, adding them into the index.
/// Item encoding cannot be nested.
pub struct IndexBuilder<'a, 'tcx: 'a> {
2016-08-10 13:27:06 -05:00
items: IndexData,
builder: ItemContentBuilder<'a, 'tcx>,
}
/// Builder that can encode the content of items, but can't start a
/// new item itself. Most code is attached to here.
pub struct ItemContentBuilder<'a, 'tcx: 'a> {
2016-08-10 13:27:06 -05:00
xrefs: FnvHashMap<XRef<'tcx>, u32>, // sequentially-assigned
ecx: &'a EncodeContext<'a, 'tcx>,
2016-08-10 13:27:06 -05:00
}
/// "interned" entries referenced by id
#[derive(PartialEq, Eq, Hash)]
pub enum XRef<'tcx> { Predicate(ty::Predicate<'tcx>) }
2016-08-10 13:27:36 -05:00
impl<'a, 'tcx> IndexBuilder<'a, 'tcx> {
pub fn new(ecx: &'a EncodeContext<'a, 'tcx>) -> Self {
2016-08-10 13:27:36 -05:00
IndexBuilder {
2016-08-10 13:27:06 -05:00
items: IndexData::new(ecx.tcx.map.num_local_def_ids()),
builder: ItemContentBuilder {
ecx: ecx,
xrefs: FnvHashMap(),
},
2016-08-10 13:27:06 -05:00
}
}
/// Records that `id` is being emitted at the current offset.
/// This data is later used to construct the item index in the
/// metadata so we can quickly find the data for a given item.
///
/// Returns a dep-graph task that you should keep live as long as
/// the data for this item is being emitted.
2016-08-10 20:37:53 -05:00
pub fn record<OP>(&mut self, id: DefId, rbml_w: &mut Encoder, op: OP)
where OP: FnOnce(&mut ItemContentBuilder<'a, 'tcx>, &mut Encoder)
{
2016-08-10 13:27:06 -05:00
let position = rbml_w.mark_stable_position();
self.items.record(id, position);
2016-08-10 20:37:53 -05:00
let _task = self.ecx.tcx.dep_graph.in_task(DepNode::MetaData(id));
rbml_w.start_tag(tag_items_data_item).unwrap();
op(self, rbml_w);
rbml_w.end_tag().unwrap();
2016-08-10 13:27:06 -05:00
}
pub fn into_fields(self) -> (IndexData, FnvHashMap<XRef<'tcx>, u32>) {
(self.items, self.builder.xrefs)
2016-08-10 13:27:06 -05:00
}
}
2016-08-10 13:27:06 -05:00
impl<'a, 'tcx> Deref for IndexBuilder<'a, 'tcx> {
type Target = ItemContentBuilder<'a, 'tcx>;
fn deref(&self) -> &Self::Target {
&self.builder
}
}
impl<'a, 'tcx> DerefMut for IndexBuilder<'a, 'tcx> {
fn deref_mut(&mut self) -> &mut Self::Target {
&mut self.builder
2016-08-10 13:27:06 -05:00
}
}
impl<'a, 'tcx> ItemContentBuilder<'a, 'tcx> {
pub fn ecx(&self) -> &'a EncodeContext<'a, 'tcx> {
self.ecx
}
pub fn add_xref(&mut self, xref: XRef<'tcx>) -> u32 {
let old_len = self.xrefs.len() as u32;
*self.xrefs.entry(xref).or_insert(old_len)
}
}