2012-12-03 16:48:01 -08:00
|
|
|
// Copyright 2012 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.
|
|
|
|
|
2012-07-25 18:36:18 -07:00
|
|
|
// Detecting language items.
|
|
|
|
//
|
|
|
|
// Language items are items that represent concepts intrinsic to the language
|
|
|
|
// itself. Examples are:
|
|
|
|
//
|
2013-07-10 17:31:53 -07:00
|
|
|
// * Traits that specify "kinds"; e.g. "Freeze", "Send".
|
2012-07-25 18:36:18 -07:00
|
|
|
//
|
2013-06-05 14:52:27 -07:00
|
|
|
// * Traits that represent operators; e.g. "Add", "Sub", "Index".
|
2012-07-25 18:36:18 -07:00
|
|
|
//
|
|
|
|
// * Functions called by the compiler itself.
|
|
|
|
|
2013-05-17 15:28:44 -07:00
|
|
|
|
2012-10-15 14:56:42 -07:00
|
|
|
use driver::session::Session;
|
2013-03-26 16:38:07 -04:00
|
|
|
use metadata::csearch::each_lang_item;
|
|
|
|
use metadata::cstore::iter_crate_data;
|
2013-08-16 16:57:42 -04:00
|
|
|
use middle::ty::{BuiltinBound, BoundFreeze, BoundSend, BoundSized};
|
2013-10-22 15:13:18 -07:00
|
|
|
use syntax::ast;
|
2013-03-26 16:38:07 -04:00
|
|
|
use syntax::ast_util::local_def;
|
2013-07-19 21:51:37 +10:00
|
|
|
use syntax::attr::AttrMetaMethods;
|
2013-08-13 15:30:08 +02:00
|
|
|
use syntax::visit;
|
|
|
|
use syntax::visit::Visitor;
|
2012-07-25 18:36:18 -07:00
|
|
|
|
2013-06-28 18:32:26 -04:00
|
|
|
use std::hashmap::HashMap;
|
2013-09-30 02:34:36 +02:00
|
|
|
use std::iter::Enumerate;
|
|
|
|
use std::vec;
|
2012-07-25 18:36:18 -07:00
|
|
|
|
2013-01-06 12:05:34 -08:00
|
|
|
pub enum LangItem {
|
2013-06-30 03:22:18 -04:00
|
|
|
FreezeTraitLangItem, // 0
|
2013-07-10 17:31:53 -07:00
|
|
|
SendTraitLangItem, // 1
|
|
|
|
SizedTraitLangItem, // 2
|
|
|
|
|
|
|
|
DropTraitLangItem, // 3
|
|
|
|
|
|
|
|
AddTraitLangItem, // 4
|
|
|
|
SubTraitLangItem, // 5
|
|
|
|
MulTraitLangItem, // 6
|
|
|
|
DivTraitLangItem, // 7
|
|
|
|
RemTraitLangItem, // 8
|
|
|
|
NegTraitLangItem, // 9
|
|
|
|
NotTraitLangItem, // 10
|
2013-06-30 03:22:18 -04:00
|
|
|
BitXorTraitLangItem, // 11
|
2013-07-10 17:31:53 -07:00
|
|
|
BitAndTraitLangItem, // 12
|
|
|
|
BitOrTraitLangItem, // 13
|
|
|
|
ShlTraitLangItem, // 14
|
|
|
|
ShrTraitLangItem, // 15
|
|
|
|
IndexTraitLangItem, // 16
|
|
|
|
|
|
|
|
EqTraitLangItem, // 17
|
|
|
|
OrdTraitLangItem, // 18
|
|
|
|
|
|
|
|
StrEqFnLangItem, // 19
|
|
|
|
UniqStrEqFnLangItem, // 20
|
2013-09-18 01:37:34 -07:00
|
|
|
FailFnLangItem, // 21
|
|
|
|
FailBoundsCheckFnLangItem, // 22
|
|
|
|
ExchangeMallocFnLangItem, // 23
|
|
|
|
ClosureExchangeMallocFnLangItem, // 24
|
|
|
|
ExchangeFreeFnLangItem, // 25
|
|
|
|
MallocFnLangItem, // 26
|
|
|
|
FreeFnLangItem, // 27
|
|
|
|
BorrowAsImmFnLangItem, // 28
|
|
|
|
BorrowAsMutFnLangItem, // 29
|
|
|
|
ReturnToMutFnLangItem, // 30
|
|
|
|
CheckNotBorrowedFnLangItem, // 31
|
|
|
|
StrDupUniqFnLangItem, // 32
|
|
|
|
RecordBorrowFnLangItem, // 33
|
|
|
|
UnrecordBorrowFnLangItem, // 34
|
|
|
|
|
|
|
|
StartFnLangItem, // 35
|
|
|
|
|
|
|
|
TyDescStructLangItem, // 36
|
|
|
|
TyVisitorTraitLangItem, // 37
|
|
|
|
OpaqueStructLangItem, // 38
|
2013-10-22 15:13:18 -07:00
|
|
|
|
|
|
|
EventLoopFactoryLangItem, // 39
|
2013-01-06 12:05:34 -08:00
|
|
|
}
|
|
|
|
|
2013-01-30 13:44:24 -08:00
|
|
|
pub struct LanguageItems {
|
2013-10-22 15:13:18 -07:00
|
|
|
items: [Option<ast::DefId>, ..40]
|
2012-08-27 14:08:37 -07:00
|
|
|
}
|
|
|
|
|
2013-05-31 15:17:22 -07:00
|
|
|
impl LanguageItems {
|
2013-03-21 19:07:54 -07:00
|
|
|
pub fn new() -> LanguageItems {
|
2012-08-27 14:08:37 -07:00
|
|
|
LanguageItems {
|
2013-10-22 15:13:18 -07:00
|
|
|
items: [ None, ..40 ]
|
2012-08-27 14:08:37 -07:00
|
|
|
}
|
2012-07-25 18:36:18 -07:00
|
|
|
}
|
|
|
|
|
2013-10-22 15:13:18 -07:00
|
|
|
pub fn items<'a>(&'a self) -> Enumerate<vec::VecIterator<'a, Option<ast::DefId>>> {
|
2013-09-30 02:34:36 +02:00
|
|
|
self.items.iter().enumerate()
|
2013-05-03 13:08:08 -04:00
|
|
|
}
|
2013-01-07 10:51:53 -08:00
|
|
|
|
2013-03-21 19:07:54 -07:00
|
|
|
pub fn item_name(index: uint) -> &'static str {
|
2013-01-07 10:51:53 -08:00
|
|
|
match index {
|
2013-06-05 14:52:27 -07:00
|
|
|
0 => "freeze",
|
2013-07-10 17:31:53 -07:00
|
|
|
1 => "send",
|
|
|
|
2 => "sized",
|
|
|
|
|
|
|
|
3 => "drop",
|
|
|
|
|
|
|
|
4 => "add",
|
|
|
|
5 => "sub",
|
|
|
|
6 => "mul",
|
|
|
|
7 => "div",
|
|
|
|
8 => "rem",
|
|
|
|
9 => "neg",
|
|
|
|
10 => "not",
|
|
|
|
11 => "bitxor",
|
|
|
|
12 => "bitand",
|
|
|
|
13 => "bitor",
|
|
|
|
14 => "shl",
|
|
|
|
15 => "shr",
|
|
|
|
16 => "index",
|
|
|
|
17 => "eq",
|
|
|
|
18 => "ord",
|
|
|
|
|
|
|
|
19 => "str_eq",
|
|
|
|
20 => "uniq_str_eq",
|
2013-09-18 01:37:34 -07:00
|
|
|
21 => "fail_",
|
|
|
|
22 => "fail_bounds_check",
|
|
|
|
23 => "exchange_malloc",
|
|
|
|
24 => "closure_exchange_malloc",
|
|
|
|
25 => "exchange_free",
|
|
|
|
26 => "malloc",
|
|
|
|
27 => "free",
|
|
|
|
28 => "borrow_as_imm",
|
|
|
|
29 => "borrow_as_mut",
|
|
|
|
30 => "return_to_mut",
|
|
|
|
31 => "check_not_borrowed",
|
|
|
|
32 => "strdup_uniq",
|
|
|
|
33 => "record_borrow",
|
|
|
|
34 => "unrecord_borrow",
|
|
|
|
|
|
|
|
35 => "start",
|
|
|
|
|
|
|
|
36 => "ty_desc",
|
|
|
|
37 => "ty_visitor",
|
|
|
|
38 => "opaque",
|
2013-06-20 11:39:49 +02:00
|
|
|
|
2013-10-22 15:13:18 -07:00
|
|
|
39 => "event_loop_factory",
|
|
|
|
|
2013-01-07 10:51:53 -08:00
|
|
|
_ => "???"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-01-24 18:49:51 -08:00
|
|
|
// FIXME #4621: Method macros sure would be nice here.
|
2013-01-06 12:05:34 -08:00
|
|
|
|
2013-10-22 15:13:18 -07:00
|
|
|
pub fn require(&self, it: LangItem) -> Result<ast::DefId, ~str> {
|
2013-07-15 20:42:13 -07:00
|
|
|
match self.items[it as uint] {
|
|
|
|
Some(id) => Ok(id),
|
2013-09-27 22:38:08 -07:00
|
|
|
None => Err(format!("requires `{}` lang_item",
|
2013-07-15 20:42:13 -07:00
|
|
|
LanguageItems::item_name(it as uint)))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-10-22 15:13:18 -07:00
|
|
|
pub fn to_builtin_kind(&self, id: ast::DefId) -> Option<BuiltinBound> {
|
2013-08-16 16:57:42 -04:00
|
|
|
if Some(id) == self.freeze_trait() {
|
|
|
|
Some(BoundFreeze)
|
|
|
|
} else if Some(id) == self.send_trait() {
|
|
|
|
Some(BoundSend)
|
|
|
|
} else if Some(id) == self.sized_trait() {
|
|
|
|
Some(BoundSized)
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
2013-08-14 18:13:16 -04:00
|
|
|
}
|
|
|
|
|
2013-10-22 15:13:18 -07:00
|
|
|
pub fn freeze_trait(&self) -> Option<ast::DefId> {
|
2013-07-15 20:42:13 -07:00
|
|
|
self.items[FreezeTraitLangItem as uint]
|
2013-01-06 12:05:34 -08:00
|
|
|
}
|
2013-10-22 15:13:18 -07:00
|
|
|
pub fn send_trait(&self) -> Option<ast::DefId> {
|
2013-07-15 20:42:13 -07:00
|
|
|
self.items[SendTraitLangItem as uint]
|
2013-01-06 12:05:34 -08:00
|
|
|
}
|
2013-10-22 15:13:18 -07:00
|
|
|
pub fn sized_trait(&self) -> Option<ast::DefId> {
|
2013-07-15 20:42:13 -07:00
|
|
|
self.items[SizedTraitLangItem as uint]
|
2013-05-30 20:03:01 -04:00
|
|
|
}
|
2013-01-06 12:05:34 -08:00
|
|
|
|
2013-10-22 15:13:18 -07:00
|
|
|
pub fn drop_trait(&self) -> Option<ast::DefId> {
|
2013-07-15 20:42:13 -07:00
|
|
|
self.items[DropTraitLangItem as uint]
|
2013-01-06 12:05:34 -08:00
|
|
|
}
|
|
|
|
|
2013-10-22 15:13:18 -07:00
|
|
|
pub fn add_trait(&self) -> Option<ast::DefId> {
|
2013-07-15 20:42:13 -07:00
|
|
|
self.items[AddTraitLangItem as uint]
|
2013-01-06 12:05:34 -08:00
|
|
|
}
|
2013-10-22 15:13:18 -07:00
|
|
|
pub fn sub_trait(&self) -> Option<ast::DefId> {
|
2013-07-15 20:42:13 -07:00
|
|
|
self.items[SubTraitLangItem as uint]
|
2013-01-06 12:05:34 -08:00
|
|
|
}
|
2013-10-22 15:13:18 -07:00
|
|
|
pub fn mul_trait(&self) -> Option<ast::DefId> {
|
2013-07-15 20:42:13 -07:00
|
|
|
self.items[MulTraitLangItem as uint]
|
2013-01-06 12:05:34 -08:00
|
|
|
}
|
2013-10-22 15:13:18 -07:00
|
|
|
pub fn div_trait(&self) -> Option<ast::DefId> {
|
2013-07-15 20:42:13 -07:00
|
|
|
self.items[DivTraitLangItem as uint]
|
2013-01-06 12:05:34 -08:00
|
|
|
}
|
2013-10-22 15:13:18 -07:00
|
|
|
pub fn rem_trait(&self) -> Option<ast::DefId> {
|
2013-07-15 20:42:13 -07:00
|
|
|
self.items[RemTraitLangItem as uint]
|
2013-01-06 12:05:34 -08:00
|
|
|
}
|
2013-10-22 15:13:18 -07:00
|
|
|
pub fn neg_trait(&self) -> Option<ast::DefId> {
|
2013-07-15 20:42:13 -07:00
|
|
|
self.items[NegTraitLangItem as uint]
|
2013-01-06 12:05:34 -08:00
|
|
|
}
|
2013-10-22 15:13:18 -07:00
|
|
|
pub fn not_trait(&self) -> Option<ast::DefId> {
|
2013-07-15 20:42:13 -07:00
|
|
|
self.items[NotTraitLangItem as uint]
|
2013-01-11 12:16:58 -05:00
|
|
|
}
|
2013-10-22 15:13:18 -07:00
|
|
|
pub fn bitxor_trait(&self) -> Option<ast::DefId> {
|
2013-07-15 20:42:13 -07:00
|
|
|
self.items[BitXorTraitLangItem as uint]
|
2013-01-06 12:05:34 -08:00
|
|
|
}
|
2013-10-22 15:13:18 -07:00
|
|
|
pub fn bitand_trait(&self) -> Option<ast::DefId> {
|
2013-07-15 20:42:13 -07:00
|
|
|
self.items[BitAndTraitLangItem as uint]
|
2013-01-06 12:05:34 -08:00
|
|
|
}
|
2013-10-22 15:13:18 -07:00
|
|
|
pub fn bitor_trait(&self) -> Option<ast::DefId> {
|
2013-07-15 20:42:13 -07:00
|
|
|
self.items[BitOrTraitLangItem as uint]
|
2013-01-06 12:05:34 -08:00
|
|
|
}
|
2013-10-22 15:13:18 -07:00
|
|
|
pub fn shl_trait(&self) -> Option<ast::DefId> {
|
2013-07-15 20:42:13 -07:00
|
|
|
self.items[ShlTraitLangItem as uint]
|
2013-01-06 12:05:34 -08:00
|
|
|
}
|
2013-10-22 15:13:18 -07:00
|
|
|
pub fn shr_trait(&self) -> Option<ast::DefId> {
|
2013-07-15 20:42:13 -07:00
|
|
|
self.items[ShrTraitLangItem as uint]
|
2013-01-06 12:05:34 -08:00
|
|
|
}
|
2013-10-22 15:13:18 -07:00
|
|
|
pub fn index_trait(&self) -> Option<ast::DefId> {
|
2013-07-15 20:42:13 -07:00
|
|
|
self.items[IndexTraitLangItem as uint]
|
2013-01-06 12:05:34 -08:00
|
|
|
}
|
|
|
|
|
2013-10-22 15:13:18 -07:00
|
|
|
pub fn eq_trait(&self) -> Option<ast::DefId> {
|
2013-07-15 20:42:13 -07:00
|
|
|
self.items[EqTraitLangItem as uint]
|
2013-01-06 12:05:34 -08:00
|
|
|
}
|
2013-10-22 15:13:18 -07:00
|
|
|
pub fn ord_trait(&self) -> Option<ast::DefId> {
|
2013-07-15 20:42:13 -07:00
|
|
|
self.items[OrdTraitLangItem as uint]
|
2013-01-06 12:05:34 -08:00
|
|
|
}
|
2012-09-05 15:58:43 -07:00
|
|
|
|
2013-10-22 15:13:18 -07:00
|
|
|
pub fn str_eq_fn(&self) -> Option<ast::DefId> {
|
2013-07-15 20:42:13 -07:00
|
|
|
self.items[StrEqFnLangItem as uint]
|
2013-01-06 12:05:34 -08:00
|
|
|
}
|
2013-10-22 15:13:18 -07:00
|
|
|
pub fn uniq_str_eq_fn(&self) -> Option<ast::DefId> {
|
2013-07-15 20:42:13 -07:00
|
|
|
self.items[UniqStrEqFnLangItem as uint]
|
2013-01-06 12:05:34 -08:00
|
|
|
}
|
2013-10-22 15:13:18 -07:00
|
|
|
pub fn fail_fn(&self) -> Option<ast::DefId> {
|
2013-07-15 20:42:13 -07:00
|
|
|
self.items[FailFnLangItem as uint]
|
2013-01-07 12:21:34 -08:00
|
|
|
}
|
2013-10-22 15:13:18 -07:00
|
|
|
pub fn fail_bounds_check_fn(&self) -> Option<ast::DefId> {
|
2013-07-15 20:42:13 -07:00
|
|
|
self.items[FailBoundsCheckFnLangItem as uint]
|
2013-01-07 12:21:34 -08:00
|
|
|
}
|
2013-10-22 15:13:18 -07:00
|
|
|
pub fn exchange_malloc_fn(&self) -> Option<ast::DefId> {
|
2013-07-15 20:42:13 -07:00
|
|
|
self.items[ExchangeMallocFnLangItem as uint]
|
2013-01-07 12:21:34 -08:00
|
|
|
}
|
2013-10-22 15:13:18 -07:00
|
|
|
pub fn closure_exchange_malloc_fn(&self) -> Option<ast::DefId> {
|
2013-07-15 20:42:13 -07:00
|
|
|
self.items[ClosureExchangeMallocFnLangItem as uint]
|
2013-06-30 03:22:18 -04:00
|
|
|
}
|
2013-10-22 15:13:18 -07:00
|
|
|
pub fn exchange_free_fn(&self) -> Option<ast::DefId> {
|
2013-07-15 20:42:13 -07:00
|
|
|
self.items[ExchangeFreeFnLangItem as uint]
|
2013-01-07 12:21:34 -08:00
|
|
|
}
|
2013-10-22 15:13:18 -07:00
|
|
|
pub fn malloc_fn(&self) -> Option<ast::DefId> {
|
2013-07-15 20:42:13 -07:00
|
|
|
self.items[MallocFnLangItem as uint]
|
2013-01-07 12:21:34 -08:00
|
|
|
}
|
2013-10-22 15:13:18 -07:00
|
|
|
pub fn free_fn(&self) -> Option<ast::DefId> {
|
2013-07-15 20:42:13 -07:00
|
|
|
self.items[FreeFnLangItem as uint]
|
2013-01-07 12:21:34 -08:00
|
|
|
}
|
2013-10-22 15:13:18 -07:00
|
|
|
pub fn borrow_as_imm_fn(&self) -> Option<ast::DefId> {
|
2013-07-15 20:42:13 -07:00
|
|
|
self.items[BorrowAsImmFnLangItem as uint]
|
2013-01-11 21:01:42 -08:00
|
|
|
}
|
2013-10-22 15:13:18 -07:00
|
|
|
pub fn borrow_as_mut_fn(&self) -> Option<ast::DefId> {
|
2013-07-15 20:42:13 -07:00
|
|
|
self.items[BorrowAsMutFnLangItem as uint]
|
2013-05-01 13:48:00 -04:00
|
|
|
}
|
2013-10-22 15:13:18 -07:00
|
|
|
pub fn return_to_mut_fn(&self) -> Option<ast::DefId> {
|
2013-07-15 20:42:13 -07:00
|
|
|
self.items[ReturnToMutFnLangItem as uint]
|
2013-01-11 21:01:42 -08:00
|
|
|
}
|
2013-10-22 15:13:18 -07:00
|
|
|
pub fn check_not_borrowed_fn(&self) -> Option<ast::DefId> {
|
2013-07-15 20:42:13 -07:00
|
|
|
self.items[CheckNotBorrowedFnLangItem as uint]
|
2013-01-11 21:01:42 -08:00
|
|
|
}
|
2013-10-22 15:13:18 -07:00
|
|
|
pub fn strdup_uniq_fn(&self) -> Option<ast::DefId> {
|
2013-07-15 20:42:13 -07:00
|
|
|
self.items[StrDupUniqFnLangItem as uint]
|
2013-01-29 19:59:52 -08:00
|
|
|
}
|
2013-10-22 15:13:18 -07:00
|
|
|
pub fn record_borrow_fn(&self) -> Option<ast::DefId> {
|
2013-07-15 20:42:13 -07:00
|
|
|
self.items[RecordBorrowFnLangItem as uint]
|
2013-05-03 05:42:00 -04:00
|
|
|
}
|
2013-10-22 15:13:18 -07:00
|
|
|
pub fn unrecord_borrow_fn(&self) -> Option<ast::DefId> {
|
2013-07-15 20:42:13 -07:00
|
|
|
self.items[UnrecordBorrowFnLangItem as uint]
|
2013-05-03 05:42:00 -04:00
|
|
|
}
|
2013-10-22 15:13:18 -07:00
|
|
|
pub fn start_fn(&self) -> Option<ast::DefId> {
|
2013-07-15 20:42:13 -07:00
|
|
|
self.items[StartFnLangItem as uint]
|
2013-02-26 19:40:39 -08:00
|
|
|
}
|
2013-10-22 15:13:18 -07:00
|
|
|
pub fn ty_desc(&self) -> Option<ast::DefId> {
|
2013-07-15 20:42:13 -07:00
|
|
|
self.items[TyDescStructLangItem as uint]
|
2013-06-20 11:39:49 +02:00
|
|
|
}
|
2013-10-22 15:13:18 -07:00
|
|
|
pub fn ty_visitor(&self) -> Option<ast::DefId> {
|
2013-07-15 20:42:13 -07:00
|
|
|
self.items[TyVisitorTraitLangItem as uint]
|
2013-06-20 11:39:49 +02:00
|
|
|
}
|
2013-10-22 15:13:18 -07:00
|
|
|
pub fn opaque(&self) -> Option<ast::DefId> {
|
2013-07-15 20:42:13 -07:00
|
|
|
self.items[OpaqueStructLangItem as uint]
|
2013-06-20 11:39:49 +02:00
|
|
|
}
|
2013-10-22 15:13:18 -07:00
|
|
|
pub fn event_loop_factory(&self) -> Option<ast::DefId> {
|
|
|
|
self.items[EventLoopFactoryLangItem as uint]
|
|
|
|
}
|
2013-01-06 12:05:34 -08:00
|
|
|
}
|
|
|
|
|
2013-10-22 15:13:18 -07:00
|
|
|
struct LanguageItemCollector {
|
2013-03-15 15:24:24 -04:00
|
|
|
items: LanguageItems,
|
2012-07-25 18:36:18 -07:00
|
|
|
|
2012-10-15 14:56:42 -07:00
|
|
|
session: Session,
|
2012-07-25 18:36:18 -07:00
|
|
|
|
2013-09-30 02:34:23 +02:00
|
|
|
item_refs: HashMap<&'static str, uint>,
|
2012-09-07 19:04:40 -07:00
|
|
|
}
|
|
|
|
|
2013-08-13 15:30:08 +02:00
|
|
|
struct LanguageItemVisitor<'self> {
|
2013-10-22 15:13:18 -07:00
|
|
|
this: &'self mut LanguageItemCollector,
|
2013-08-13 15:30:08 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
impl<'self> Visitor<()> for LanguageItemVisitor<'self> {
|
2013-10-22 15:13:18 -07:00
|
|
|
fn visit_item(&mut self, item: @ast::item, _: ()) {
|
|
|
|
match extract(item.attrs) {
|
|
|
|
Some(value) => {
|
|
|
|
let item_index = self.this.item_refs.find_equiv(&value).map(|x| *x);
|
|
|
|
|
|
|
|
match item_index {
|
|
|
|
Some(item_index) => {
|
|
|
|
self.this.collect_item(item_index, local_def(item.id))
|
2013-08-13 15:30:08 +02:00
|
|
|
}
|
2013-10-22 15:13:18 -07:00
|
|
|
None => {}
|
2013-08-13 15:30:08 +02:00
|
|
|
}
|
2013-10-22 15:13:18 -07:00
|
|
|
}
|
|
|
|
None => {}
|
|
|
|
}
|
2013-08-13 15:30:08 +02:00
|
|
|
|
|
|
|
visit::walk_item(self, item, ());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-10-22 15:13:18 -07:00
|
|
|
impl LanguageItemCollector {
|
|
|
|
pub fn new(session: Session) -> LanguageItemCollector {
|
2013-06-27 15:04:22 +02:00
|
|
|
let mut item_refs = HashMap::new();
|
|
|
|
|
2013-09-30 02:34:23 +02:00
|
|
|
item_refs.insert("freeze", FreezeTraitLangItem as uint);
|
|
|
|
item_refs.insert("send", SendTraitLangItem as uint);
|
|
|
|
item_refs.insert("sized", SizedTraitLangItem as uint);
|
|
|
|
|
|
|
|
item_refs.insert("drop", DropTraitLangItem as uint);
|
|
|
|
|
|
|
|
item_refs.insert("add", AddTraitLangItem as uint);
|
|
|
|
item_refs.insert("sub", SubTraitLangItem as uint);
|
|
|
|
item_refs.insert("mul", MulTraitLangItem as uint);
|
|
|
|
item_refs.insert("div", DivTraitLangItem as uint);
|
|
|
|
item_refs.insert("rem", RemTraitLangItem as uint);
|
|
|
|
item_refs.insert("neg", NegTraitLangItem as uint);
|
|
|
|
item_refs.insert("not", NotTraitLangItem as uint);
|
|
|
|
item_refs.insert("bitxor", BitXorTraitLangItem as uint);
|
|
|
|
item_refs.insert("bitand", BitAndTraitLangItem as uint);
|
|
|
|
item_refs.insert("bitor", BitOrTraitLangItem as uint);
|
|
|
|
item_refs.insert("shl", ShlTraitLangItem as uint);
|
|
|
|
item_refs.insert("shr", ShrTraitLangItem as uint);
|
|
|
|
item_refs.insert("index", IndexTraitLangItem as uint);
|
|
|
|
|
|
|
|
item_refs.insert("eq", EqTraitLangItem as uint);
|
|
|
|
item_refs.insert("ord", OrdTraitLangItem as uint);
|
|
|
|
|
|
|
|
item_refs.insert("str_eq", StrEqFnLangItem as uint);
|
|
|
|
item_refs.insert("uniq_str_eq", UniqStrEqFnLangItem as uint);
|
|
|
|
item_refs.insert("fail_", FailFnLangItem as uint);
|
|
|
|
item_refs.insert("fail_bounds_check",
|
2013-06-27 15:04:22 +02:00
|
|
|
FailBoundsCheckFnLangItem as uint);
|
2013-09-30 02:34:23 +02:00
|
|
|
item_refs.insert("exchange_malloc", ExchangeMallocFnLangItem as uint);
|
|
|
|
item_refs.insert("closure_exchange_malloc", ClosureExchangeMallocFnLangItem as uint);
|
|
|
|
item_refs.insert("exchange_free", ExchangeFreeFnLangItem as uint);
|
|
|
|
item_refs.insert("malloc", MallocFnLangItem as uint);
|
|
|
|
item_refs.insert("free", FreeFnLangItem as uint);
|
|
|
|
item_refs.insert("borrow_as_imm", BorrowAsImmFnLangItem as uint);
|
|
|
|
item_refs.insert("borrow_as_mut", BorrowAsMutFnLangItem as uint);
|
|
|
|
item_refs.insert("return_to_mut", ReturnToMutFnLangItem as uint);
|
|
|
|
item_refs.insert("check_not_borrowed",
|
2013-06-27 15:04:22 +02:00
|
|
|
CheckNotBorrowedFnLangItem as uint);
|
2013-09-30 02:34:23 +02:00
|
|
|
item_refs.insert("strdup_uniq", StrDupUniqFnLangItem as uint);
|
|
|
|
item_refs.insert("record_borrow", RecordBorrowFnLangItem as uint);
|
|
|
|
item_refs.insert("unrecord_borrow", UnrecordBorrowFnLangItem as uint);
|
|
|
|
item_refs.insert("start", StartFnLangItem as uint);
|
|
|
|
item_refs.insert("ty_desc", TyDescStructLangItem as uint);
|
|
|
|
item_refs.insert("ty_visitor", TyVisitorTraitLangItem as uint);
|
|
|
|
item_refs.insert("opaque", OpaqueStructLangItem as uint);
|
2013-10-22 15:13:18 -07:00
|
|
|
item_refs.insert("event_loop_factory", EventLoopFactoryLangItem as uint);
|
2013-06-27 15:04:22 +02:00
|
|
|
|
|
|
|
LanguageItemCollector {
|
|
|
|
session: session,
|
|
|
|
items: LanguageItems::new(),
|
|
|
|
item_refs: item_refs
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-10-22 15:13:18 -07:00
|
|
|
pub fn collect_item(&mut self, item_index: uint, item_def_id: ast::DefId) {
|
2013-01-07 10:51:53 -08:00
|
|
|
// Check for duplicates.
|
|
|
|
match self.items.items[item_index] {
|
|
|
|
Some(original_def_id) if original_def_id != item_def_id => {
|
2013-09-27 22:38:08 -07:00
|
|
|
self.session.err(format!("duplicate entry for `{}`",
|
2013-01-07 10:51:53 -08:00
|
|
|
LanguageItems::item_name(item_index)));
|
|
|
|
}
|
|
|
|
Some(_) | None => {
|
|
|
|
// OK.
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Matched.
|
|
|
|
self.items.items[item_index] = Some(item_def_id);
|
|
|
|
}
|
|
|
|
|
2013-10-22 15:13:18 -07:00
|
|
|
pub fn collect_local_language_items(&mut self, crate: &ast::Crate) {
|
|
|
|
let mut v = LanguageItemVisitor { this: self };
|
|
|
|
visit::walk_crate(&mut v, crate, ());
|
2012-07-25 18:36:18 -07:00
|
|
|
}
|
|
|
|
|
2013-05-31 15:17:22 -07:00
|
|
|
pub fn collect_external_language_items(&mut self) {
|
2012-07-25 18:36:18 -07:00
|
|
|
let crate_store = self.session.cstore;
|
2013-11-21 15:42:55 -08:00
|
|
|
iter_crate_data(crate_store, |crate_number, _crate_metadata| {
|
|
|
|
each_lang_item(crate_store, crate_number, |node_id, item_index| {
|
2013-10-22 15:13:18 -07:00
|
|
|
let def_id = ast::DefId { crate: crate_number, node: node_id };
|
2013-01-07 10:51:53 -08:00
|
|
|
self.collect_item(item_index, def_id);
|
2013-08-02 02:17:20 -04:00
|
|
|
true
|
2013-11-21 15:42:55 -08:00
|
|
|
});
|
|
|
|
})
|
2012-07-25 18:36:18 -07:00
|
|
|
}
|
|
|
|
|
2013-10-22 15:13:18 -07:00
|
|
|
pub fn collect(&mut self, crate: &ast::Crate) {
|
|
|
|
self.collect_local_language_items(crate);
|
2012-07-25 18:36:18 -07:00
|
|
|
self.collect_external_language_items();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-10-22 15:13:18 -07:00
|
|
|
pub fn extract(attrs: &[ast::Attribute]) -> Option<@str> {
|
|
|
|
for attribute in attrs.iter() {
|
|
|
|
match attribute.name_str_pair() {
|
|
|
|
Some((key, value)) if "lang" == key => {
|
|
|
|
return Some(value);
|
|
|
|
}
|
|
|
|
Some(*) | None => {}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return None;
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn collect_language_items(crate: &ast::Crate,
|
2013-01-30 13:44:24 -08:00
|
|
|
session: Session)
|
|
|
|
-> LanguageItems {
|
2013-10-22 15:13:18 -07:00
|
|
|
let mut collector = LanguageItemCollector::new(session);
|
|
|
|
collector.collect(crate);
|
2013-03-15 15:24:24 -04:00
|
|
|
let LanguageItemCollector { items, _ } = collector;
|
2013-06-14 17:55:38 -07:00
|
|
|
session.abort_if_errors();
|
2013-03-15 15:24:24 -04:00
|
|
|
items
|
2012-07-25 18:36:18 -07:00
|
|
|
}
|