2012-12-03 18:48:01 -06: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.
|
|
|
|
|
2014-05-02 17:26:45 -05:00
|
|
|
use driver::session;
|
2012-10-15 16:56:42 -05:00
|
|
|
use driver::session::Session;
|
2012-12-23 16:41:37 -06:00
|
|
|
|
2012-09-04 13:54:36 -05:00
|
|
|
use syntax::ast;
|
|
|
|
use syntax::attr;
|
2014-01-01 00:53:22 -06:00
|
|
|
use syntax::codemap::DUMMY_SP;
|
2013-05-24 21:35:29 -05:00
|
|
|
use syntax::codemap;
|
2014-01-09 07:05:33 -06:00
|
|
|
use syntax::fold::Folder;
|
2012-12-23 16:41:37 -06:00
|
|
|
use syntax::fold;
|
2014-03-19 09:52:37 -05:00
|
|
|
use syntax::owned_slice::OwnedSlice;
|
2014-01-21 12:08:10 -06:00
|
|
|
use syntax::parse::token::InternedString;
|
|
|
|
use syntax::parse::token;
|
2013-11-25 01:08:53 -06:00
|
|
|
use syntax::util::small_vector::SmallVector;
|
2012-01-26 17:20:29 -06:00
|
|
|
|
2014-04-03 18:28:46 -05:00
|
|
|
pub static VERSION: &'static str = "0.11-pre";
|
2014-01-13 19:29:50 -06:00
|
|
|
|
2014-03-05 08:36:01 -06:00
|
|
|
pub fn maybe_inject_crates_ref(sess: &Session, krate: ast::Crate)
|
2013-09-27 21:46:09 -05:00
|
|
|
-> ast::Crate {
|
2014-02-05 15:15:24 -06:00
|
|
|
if use_std(&krate) {
|
|
|
|
inject_crates_ref(sess, krate)
|
2014-01-24 01:40:54 -06:00
|
|
|
} else {
|
2014-02-05 15:15:24 -06:00
|
|
|
krate
|
2014-01-24 01:40:54 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-03-05 08:36:01 -06:00
|
|
|
pub fn maybe_inject_prelude(sess: &Session, krate: ast::Crate) -> ast::Crate {
|
2014-02-05 15:15:24 -06:00
|
|
|
if use_std(&krate) {
|
|
|
|
inject_prelude(sess, krate)
|
2012-01-26 17:20:29 -06:00
|
|
|
} else {
|
2014-02-05 15:15:24 -06:00
|
|
|
krate
|
2012-01-26 17:20:29 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-02-05 15:15:24 -06:00
|
|
|
fn use_std(krate: &ast::Crate) -> bool {
|
2014-02-28 17:25:15 -06:00
|
|
|
!attr::contains_name(krate.attrs.as_slice(), "no_std")
|
2012-01-26 18:23:34 -06:00
|
|
|
}
|
2013-07-19 00:38:55 -05:00
|
|
|
|
2014-03-11 15:38:36 -05:00
|
|
|
fn use_start(krate: &ast::Crate) -> bool {
|
|
|
|
!attr::contains_name(krate.attrs.as_slice(), "no_start")
|
2013-10-22 17:13:18 -05:00
|
|
|
}
|
|
|
|
|
2013-07-19 06:51:37 -05:00
|
|
|
fn no_prelude(attrs: &[ast::Attribute]) -> bool {
|
|
|
|
attr::contains_name(attrs, "no_implicit_prelude")
|
2013-07-16 23:23:17 -05:00
|
|
|
}
|
2012-01-26 18:23:34 -06:00
|
|
|
|
2014-03-05 08:36:01 -06:00
|
|
|
struct StandardLibraryInjector<'a> {
|
|
|
|
sess: &'a Session,
|
2013-08-29 14:10:02 -05:00
|
|
|
}
|
2012-12-23 16:41:37 -06:00
|
|
|
|
2014-02-05 15:15:24 -06:00
|
|
|
pub fn with_version(krate: &str) -> Option<(InternedString, ast::StrStyle)> {
|
2014-01-13 19:29:50 -06:00
|
|
|
match option_env!("CFG_DISABLE_INJECT_STD_VERSION") {
|
|
|
|
Some("1") => None,
|
|
|
|
_ => {
|
2014-01-21 12:08:10 -06:00
|
|
|
Some((token::intern_and_get_ident(format!("{}\\#{}",
|
2014-02-05 15:15:24 -06:00
|
|
|
krate,
|
2014-01-21 12:08:10 -06:00
|
|
|
VERSION)),
|
2014-01-13 19:29:50 -06:00
|
|
|
ast::CookedStr))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-03-05 08:36:01 -06:00
|
|
|
impl<'a> fold::Folder for StandardLibraryInjector<'a> {
|
2014-02-05 15:15:24 -06:00
|
|
|
fn fold_crate(&mut self, krate: ast::Crate) -> ast::Crate {
|
2014-02-28 17:25:15 -06:00
|
|
|
let mut vis = vec!(ast::ViewItem {
|
2014-03-07 01:57:45 -06:00
|
|
|
node: ast::ViewItemExternCrate(token::str_to_ident("std"),
|
2014-01-13 19:29:50 -06:00
|
|
|
with_version("std"),
|
2014-01-09 07:05:33 -06:00
|
|
|
ast::DUMMY_NODE_ID),
|
2014-02-28 17:25:15 -06:00
|
|
|
attrs: vec!(
|
2014-01-21 12:08:10 -06:00
|
|
|
attr::mk_attr(attr::mk_list_item(
|
|
|
|
InternedString::new("phase"),
|
2014-02-28 17:25:15 -06:00
|
|
|
vec!(
|
2014-01-21 12:08:10 -06:00
|
|
|
attr::mk_word_item(InternedString::new("syntax")),
|
|
|
|
attr::mk_word_item(InternedString::new("link")
|
2014-02-28 17:25:15 -06:00
|
|
|
))))),
|
2014-01-16 15:27:27 -06:00
|
|
|
vis: ast::Inherited,
|
2014-01-01 00:53:22 -06:00
|
|
|
span: DUMMY_SP
|
2014-02-28 17:25:15 -06:00
|
|
|
});
|
2013-10-22 17:13:18 -05:00
|
|
|
|
2014-05-02 17:26:45 -05:00
|
|
|
let any_exe = self.sess.crate_types.borrow().iter().any(|ty| {
|
|
|
|
*ty == session::CrateTypeExecutable
|
|
|
|
});
|
|
|
|
if use_start(&krate) && any_exe {
|
2014-01-09 07:05:33 -06:00
|
|
|
vis.push(ast::ViewItem {
|
2014-03-11 15:38:36 -05:00
|
|
|
node: ast::ViewItemExternCrate(token::str_to_ident("native"),
|
|
|
|
with_version("native"),
|
2014-01-09 07:05:33 -06:00
|
|
|
ast::DUMMY_NODE_ID),
|
2014-02-28 17:25:15 -06:00
|
|
|
attrs: Vec::new(),
|
2014-01-16 15:27:27 -06:00
|
|
|
vis: ast::Inherited,
|
2014-01-01 00:53:22 -06:00
|
|
|
span: DUMMY_SP
|
2013-10-22 17:13:18 -05:00
|
|
|
});
|
|
|
|
}
|
2013-08-29 14:10:02 -05:00
|
|
|
|
2014-02-28 17:25:15 -06:00
|
|
|
vis.push_all_move(krate.module.view_items.clone());
|
2014-01-24 01:40:54 -06:00
|
|
|
let new_module = ast::Mod {
|
2013-08-29 14:10:02 -05:00
|
|
|
view_items: vis,
|
2014-02-05 15:15:24 -06:00
|
|
|
..krate.module.clone()
|
2013-08-29 14:10:02 -05:00
|
|
|
};
|
|
|
|
|
|
|
|
ast::Crate {
|
|
|
|
module: new_module,
|
2014-02-05 15:15:24 -06:00
|
|
|
..krate
|
2013-08-29 14:10:02 -05:00
|
|
|
}
|
|
|
|
}
|
2014-01-24 01:40:54 -06:00
|
|
|
}
|
|
|
|
|
2014-03-05 08:36:01 -06:00
|
|
|
fn inject_crates_ref(sess: &Session, krate: ast::Crate) -> ast::Crate {
|
2014-01-24 01:40:54 -06:00
|
|
|
let mut fold = StandardLibraryInjector {
|
|
|
|
sess: sess,
|
|
|
|
};
|
2014-02-05 15:15:24 -06:00
|
|
|
fold.fold_crate(krate)
|
2014-01-24 01:40:54 -06:00
|
|
|
}
|
|
|
|
|
2014-03-05 08:36:01 -06:00
|
|
|
struct PreludeInjector<'a> {
|
|
|
|
sess: &'a Session,
|
2014-01-24 01:40:54 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-03-05 08:36:01 -06:00
|
|
|
impl<'a> fold::Folder for PreludeInjector<'a> {
|
2014-02-05 15:15:24 -06:00
|
|
|
fn fold_crate(&mut self, krate: ast::Crate) -> ast::Crate {
|
2014-02-28 17:25:15 -06:00
|
|
|
if !no_prelude(krate.attrs.as_slice()) {
|
2014-01-24 01:40:54 -06:00
|
|
|
// only add `use std::prelude::*;` if there wasn't a
|
2014-04-14 10:30:31 -05:00
|
|
|
// `#![no_implicit_prelude]` at the crate level.
|
2014-01-24 01:40:54 -06:00
|
|
|
ast::Crate {
|
2014-02-05 15:15:24 -06:00
|
|
|
module: self.fold_mod(&krate.module),
|
|
|
|
..krate
|
2014-01-24 01:40:54 -06:00
|
|
|
}
|
|
|
|
} else {
|
2014-02-05 15:15:24 -06:00
|
|
|
krate
|
2014-01-24 01:40:54 -06:00
|
|
|
}
|
|
|
|
}
|
2013-08-29 14:10:02 -05:00
|
|
|
|
2014-01-09 07:05:33 -06:00
|
|
|
fn fold_item(&mut self, item: @ast::Item) -> SmallVector<@ast::Item> {
|
2014-02-28 17:25:15 -06:00
|
|
|
if !no_prelude(item.attrs.as_slice()) {
|
2014-04-14 10:30:31 -05:00
|
|
|
// only recur if there wasn't `#![no_implicit_prelude]`
|
2013-08-29 14:10:02 -05:00
|
|
|
// on this item, i.e. this means that the prelude is not
|
|
|
|
// implicitly imported though the whole subtree
|
|
|
|
fold::noop_fold_item(item, self)
|
|
|
|
} else {
|
2013-11-25 01:08:53 -06:00
|
|
|
SmallVector::one(item)
|
2013-08-29 14:10:02 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-01-09 07:05:33 -06:00
|
|
|
fn fold_mod(&mut self, module: &ast::Mod) -> ast::Mod {
|
2013-08-29 14:10:02 -05:00
|
|
|
let prelude_path = ast::Path {
|
2014-01-01 00:53:22 -06:00
|
|
|
span: DUMMY_SP,
|
2013-08-29 14:10:02 -05:00
|
|
|
global: false,
|
2014-02-28 17:25:15 -06:00
|
|
|
segments: vec!(
|
2013-08-29 14:10:02 -05:00
|
|
|
ast::PathSegment {
|
2014-02-13 23:07:09 -06:00
|
|
|
identifier: token::str_to_ident("std"),
|
2014-03-07 09:50:40 -06:00
|
|
|
lifetimes: Vec::new(),
|
2014-03-19 09:52:37 -05:00
|
|
|
types: OwnedSlice::empty(),
|
2013-08-29 14:10:02 -05:00
|
|
|
},
|
|
|
|
ast::PathSegment {
|
2014-02-13 23:07:09 -06:00
|
|
|
identifier: token::str_to_ident("prelude"),
|
2014-03-07 09:50:40 -06:00
|
|
|
lifetimes: Vec::new(),
|
2014-03-19 09:52:37 -05:00
|
|
|
types: OwnedSlice::empty(),
|
2014-02-28 17:25:15 -06:00
|
|
|
}),
|
2013-08-29 14:10:02 -05:00
|
|
|
};
|
|
|
|
|
2014-01-24 01:40:54 -06:00
|
|
|
let vp = @codemap::dummy_spanned(ast::ViewPathGlob(prelude_path, ast::DUMMY_NODE_ID));
|
2014-01-09 07:05:33 -06:00
|
|
|
let vi2 = ast::ViewItem {
|
2014-04-26 08:33:45 -05:00
|
|
|
node: ast::ViewItemUse(vp),
|
2014-02-28 17:25:15 -06:00
|
|
|
attrs: Vec::new(),
|
2014-01-16 15:27:27 -06:00
|
|
|
vis: ast::Inherited,
|
2014-01-01 00:53:22 -06:00
|
|
|
span: DUMMY_SP,
|
2013-08-29 14:10:02 -05:00
|
|
|
};
|
|
|
|
|
2014-03-30 22:53:26 -05:00
|
|
|
let vis = (vec!(vi2)).append(module.view_items.as_slice());
|
2013-08-29 14:10:02 -05:00
|
|
|
|
|
|
|
// FIXME #2543: Bad copy.
|
2014-01-09 07:05:33 -06:00
|
|
|
let new_module = ast::Mod {
|
2013-08-29 14:10:02 -05:00
|
|
|
view_items: vis,
|
|
|
|
..(*module).clone()
|
|
|
|
};
|
|
|
|
fold::noop_fold_mod(&new_module, self)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-03-05 08:36:01 -06:00
|
|
|
fn inject_prelude(sess: &Session, krate: ast::Crate) -> ast::Crate {
|
2014-01-24 01:40:54 -06:00
|
|
|
let mut fold = PreludeInjector {
|
2013-08-29 14:10:02 -05:00
|
|
|
sess: sess,
|
|
|
|
};
|
2014-02-05 15:15:24 -06:00
|
|
|
fold.fold_crate(krate)
|
2012-01-26 17:20:29 -06:00
|
|
|
}
|