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-06 06:38:01 -05:00
|
|
|
use driver::config;
|
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-05-20 23:25:42 -05:00
|
|
|
use std::mem;
|
|
|
|
|
2014-05-12 17:30:24 -05:00
|
|
|
pub static VERSION: &'static str = "0.11.0-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-05-16 12:45:16 -05:00
|
|
|
VERSION).as_slice()),
|
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-05-20 23:25:42 -05:00
|
|
|
fn fold_crate(&mut self, mut 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-05-20 23:25:42 -05:00
|
|
|
attr::mk_attr_outer(attr::mk_list_item(
|
2014-01-21 12:08:10 -06:00
|
|
|
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| {
|
2014-05-06 06:38:01 -05:00
|
|
|
*ty == config::CrateTypeExecutable
|
2014-05-02 17:26:45 -05:00
|
|
|
});
|
|
|
|
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-04-16 00:32:35 -05:00
|
|
|
// `extern crate` must be precede `use` items
|
2014-05-20 23:25:42 -05:00
|
|
|
mem::swap(&mut vis, &mut krate.module.view_items);
|
|
|
|
krate.module.view_items.push_all_move(vis);
|
|
|
|
|
|
|
|
// don't add #![no_std] here, that will block the prelude injection later.
|
|
|
|
// Add it during the prelude injection instead.
|
|
|
|
|
|
|
|
// Add #![feature(phase)] here, because we use #[phase] on extern crate std.
|
|
|
|
let feat_phase_attr = attr::mk_attr_inner(attr::mk_list_item(
|
|
|
|
InternedString::new("feature"),
|
|
|
|
vec![attr::mk_word_item(InternedString::new("phase"))],
|
|
|
|
));
|
|
|
|
krate.attrs.push(feat_phase_attr);
|
|
|
|
|
|
|
|
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-05-20 23:25:42 -05:00
|
|
|
fn fold_crate(&mut self, mut krate: ast::Crate) -> ast::Crate {
|
|
|
|
// Add #![no_std] here, so we don't re-inject when compiling pretty-printed source.
|
|
|
|
// This must happen here and not in StandardLibraryInjector because this
|
|
|
|
// fold happens second.
|
|
|
|
|
|
|
|
let no_std_attr = attr::mk_attr_inner(attr::mk_word_item(InternedString::new("no_std")));
|
|
|
|
krate.attrs.push(no_std_attr);
|
|
|
|
|
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-04-16 00:32:35 -05:00
|
|
|
|
|
|
|
// fold_mod() will insert glob path.
|
2014-05-20 23:25:42 -05:00
|
|
|
let globs_attr = attr::mk_attr_inner(attr::mk_list_item(
|
2014-04-16 00:32:35 -05:00
|
|
|
InternedString::new("feature"),
|
|
|
|
vec!(
|
|
|
|
attr::mk_word_item(InternedString::new("globs")),
|
|
|
|
)));
|
2014-05-20 23:25:42 -05:00
|
|
|
krate.attrs.push(globs_attr);
|
2014-04-16 00:32:35 -05:00
|
|
|
|
2014-05-20 23:25:42 -05:00
|
|
|
krate.module = self.fold_mod(&krate.module);
|
2014-01-24 01:40:54 -06:00
|
|
|
}
|
2014-05-20 23:25:42 -05: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-04-16 00:32:35 -05:00
|
|
|
let (crates, uses) = module.view_items.partitioned(|x| {
|
|
|
|
match x.node {
|
|
|
|
ast::ViewItemExternCrate(..) => true,
|
|
|
|
_ => false,
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
// add vi2 after any `extern crate` but before any `use`
|
|
|
|
let mut view_items = crates;
|
|
|
|
view_items.push(vi2);
|
|
|
|
view_items.push_all_move(uses);
|
2013-08-29 14:10:02 -05:00
|
|
|
|
2014-01-09 07:05:33 -06:00
|
|
|
let new_module = ast::Mod {
|
2014-04-16 00:32:35 -05:00
|
|
|
view_items: view_items,
|
2013-08-29 14:10:02 -05:00
|
|
|
..(*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
|
|
|
}
|