// Copyright 2012-2013 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 or the MIT license // , at your // option. This file may not be copied, modified, or distributed // except according to those terms. use back::link; use back::{arm, x86, x86_64, mips}; use driver::session::{Aggressive, CrateTypeExecutable, CrateType, FullDebugInfo, LimitedDebugInfo, NoDebugInfo}; use driver::session::{Session, No, Less, Default}; use driver::session; use front; use lib::llvm::llvm; use lib::llvm::{ContextRef, ModuleRef}; use metadata::common::LinkMeta; use metadata::{creader, filesearch}; use metadata::cstore::CStore; use metadata::creader::Loader; use metadata; use middle::{trans, freevars, kind, ty, typeck, lint, astencode, reachable}; use middle; use util::common::time; use util::ppaux; use util::nodemap::{NodeMap, NodeSet}; use serialize::{json, Encodable}; use std::cell::{Cell, RefCell}; use std::io; use std::io::fs; use std::io::MemReader; use std::mem::drop; use std::os; use getopts::{optopt, optmulti, optflag, optflagopt}; use getopts; use syntax::ast; use syntax::abi; use syntax::attr; use syntax::attr::{AttrMetaMethods}; use syntax::codemap; use syntax::crateid::CrateId; use syntax::diagnostic; use syntax::diagnostic::Emitter; use syntax::ext::base::CrateLoader; use syntax::parse; use syntax::parse::token::InternedString; use syntax::parse::token; use syntax::print::{pp, pprust}; use syntax; pub enum PpMode { PpmNormal, PpmExpanded, PpmTyped, PpmIdentified, PpmExpandedIdentified } /** * The name used for source code that doesn't originate in a file * (e.g. source from stdin or a string) */ pub fn anon_src() -> ~str { "".to_str() } pub fn source_name(input: &Input) -> ~str { match *input { // FIXME (#9639): This needs to handle non-utf8 paths FileInput(ref ifile) => ifile.as_str().unwrap().to_str(), StrInput(_) => anon_src() } } pub fn default_configuration(sess: &Session) -> ast::CrateConfig { let tos = match sess.targ_cfg.os { abi::OsWin32 => InternedString::new("win32"), abi::OsMacos => InternedString::new("macos"), abi::OsLinux => InternedString::new("linux"), abi::OsAndroid => InternedString::new("android"), abi::OsFreebsd => InternedString::new("freebsd"), }; // ARM is bi-endian, however using NDK seems to default // to little-endian unless a flag is provided. let (end,arch,wordsz) = match sess.targ_cfg.arch { abi::X86 => ("little", "x86", "32"), abi::X86_64 => ("little", "x86_64", "64"), abi::Arm => ("little", "arm", "32"), abi::Mips => ("big", "mips", "32") }; let fam = match sess.targ_cfg.os { abi::OsWin32 => InternedString::new("windows"), _ => InternedString::new("unix") }; let mk = attr::mk_name_value_item_str; return vec!(// Target bindings. attr::mk_word_item(fam.clone()), mk(InternedString::new("target_os"), tos), mk(InternedString::new("target_family"), fam), mk(InternedString::new("target_arch"), InternedString::new(arch)), mk(InternedString::new("target_endian"), InternedString::new(end)), mk(InternedString::new("target_word_size"), InternedString::new(wordsz)) ); } pub fn append_configuration(cfg: &mut ast::CrateConfig, name: InternedString) { if !cfg.iter().any(|mi| mi.name() == name) { cfg.push(attr::mk_word_item(name)) } } pub fn build_configuration(sess: &Session) -> ast::CrateConfig { // Combine the configuration requested by the session (command line) with // some default and generated configuration items let default_cfg = default_configuration(sess); let mut user_cfg = sess.opts.cfg.clone(); // If the user wants a test runner, then add the test cfg if sess.opts.test { append_configuration(&mut user_cfg, InternedString::new("test")) } // If the user requested GC, then add the GC cfg append_configuration(&mut user_cfg, if sess.opts.gc { InternedString::new("gc") } else { InternedString::new("nogc") }); user_cfg.move_iter().collect::>().append(default_cfg.as_slice()) } // Convert strings provided as --cfg [cfgspec] into a crate_cfg fn parse_cfgspecs(cfgspecs: Vec<~str> ) -> ast::CrateConfig { cfgspecs.move_iter().map(|s| { parse::parse_meta_from_source_str("cfgspec".to_str(), s, Vec::new(), &parse::new_parse_sess()) }).collect::() } pub enum Input { /// Load source from file FileInput(Path), /// The string is the source StrInput(~str) } impl Input { fn filestem(&self) -> ~str { match *self { FileInput(ref ifile) => ifile.filestem_str().unwrap().to_str(), StrInput(_) => "rust_out".to_owned(), } } } pub fn phase_1_parse_input(sess: &Session, cfg: ast::CrateConfig, input: &Input) -> ast::Crate { let krate = time(sess.time_passes(), "parsing", (), |_| { match *input { FileInput(ref file) => { parse::parse_crate_from_file(&(*file), cfg.clone(), &sess.parse_sess) } StrInput(ref src) => { parse::parse_crate_from_source_str(anon_src(), (*src).clone(), cfg.clone(), &sess.parse_sess) } } }); if sess.opts.debugging_opts & session::AST_JSON_NOEXPAND != 0 { let mut stdout = io::BufferedWriter::new(io::stdout()); let mut json = json::PrettyEncoder::new(&mut stdout); // unwrapping so IoError isn't ignored krate.encode(&mut json).unwrap(); } if sess.show_span() { front::show_span::run(sess, &krate); } krate } // For continuing compilation after a parsed crate has been // modified /// Run the "early phases" of the compiler: initial `cfg` processing, /// syntax expansion, secondary `cfg` expansion, synthesis of a test /// harness if one is to be provided and injection of a dependency on the /// standard library and prelude. pub fn phase_2_configure_and_expand(sess: &Session, loader: &mut CrateLoader, mut krate: ast::Crate, crate_id: &CrateId) -> (ast::Crate, syntax::ast_map::Map) { let time_passes = sess.time_passes(); sess.building_library.set(session::building_library(&sess.opts, &krate)); *sess.crate_types.borrow_mut() = session::collect_crate_types(sess, krate.attrs.as_slice()); time(time_passes, "gated feature checking", (), |_| front::feature_gate::check_crate(sess, &krate)); krate = time(time_passes, "crate injection", krate, |krate| front::std_inject::maybe_inject_crates_ref(sess, krate)); // strip before expansion to allow macros to depend on // configuration variables e.g/ in // // #[macro_escape] #[cfg(foo)] // mod bar { macro_rules! baz!(() => {{}}) } // // baz! should not use this definition unless foo is enabled. krate = time(time_passes, "configuration 1", krate, |krate| front::config::strip_unconfigured_items(krate)); krate = time(time_passes, "expansion", krate, |krate| { let cfg = syntax::ext::expand::ExpansionConfig { loader: loader, deriving_hash_type_parameter: sess.features.default_type_params.get(), crate_id: crate_id.clone(), }; syntax::ext::expand::expand_crate(&sess.parse_sess, cfg, krate) }); // strip again, in case expansion added anything with a #[cfg]. krate = time(time_passes, "configuration 2", krate, |krate| front::config::strip_unconfigured_items(krate)); krate = time(time_passes, "maybe building test harness", krate, |krate| front::test::modify_for_testing(sess, krate)); krate = time(time_passes, "prelude injection", krate, |krate| front::std_inject::maybe_inject_prelude(sess, krate)); let (krate, map) = time(time_passes, "assinging node ids and indexing ast", krate, |krate| front::assign_node_ids_and_map::assign_node_ids_and_map(sess, krate)); if sess.opts.debugging_opts & session::AST_JSON != 0 { let mut stdout = io::BufferedWriter::new(io::stdout()); let mut json = json::PrettyEncoder::new(&mut stdout); // unwrapping so IoError isn't ignored krate.encode(&mut json).unwrap(); } (krate, map) } pub struct CrateAnalysis { pub exp_map2: middle::resolve::ExportMap2, pub exported_items: middle::privacy::ExportedItems, pub public_items: middle::privacy::PublicItems, pub ty_cx: ty::ctxt, pub maps: astencode::Maps, pub reachable: NodeSet, } /// Run the resolution, typechecking, region checking and other /// miscellaneous analysis passes on the crate. Return various /// structures carrying the results of the analysis. pub fn phase_3_run_analysis_passes(sess: Session, krate: &ast::Crate, ast_map: syntax::ast_map::Map) -> CrateAnalysis { let time_passes = sess.time_passes(); time(time_passes, "external crate/lib resolution", (), |_| creader::read_crates(&sess, krate, session::sess_os_to_meta_os(sess.targ_cfg.os), token::get_ident_interner())); let lang_items = time(time_passes, "language item collection", (), |_| middle::lang_items::collect_language_items(krate, &sess)); let middle::resolve::CrateMap { def_map: def_map, exp_map2: exp_map2, trait_map: trait_map, external_exports: external_exports, last_private_map: last_private_map } = time(time_passes, "resolution", (), |_| middle::resolve::resolve_crate(&sess, lang_items, krate)); // Discard MTWT tables that aren't required past resolution. syntax::ext::mtwt::clear_tables(); let named_region_map = time(time_passes, "lifetime resolution", (), |_| middle::resolve_lifetime::krate(&sess, krate)); time(time_passes, "looking for entry point", (), |_| middle::entry::find_entry_point(&sess, krate, &ast_map)); sess.macro_registrar_fn.set( time(time_passes, "looking for macro registrar", (), |_| syntax::ext::registrar::find_macro_registrar( sess.diagnostic(), krate))); let freevars = time(time_passes, "freevar finding", (), |_| freevars::annotate_freevars(def_map, krate)); let region_map = time(time_passes, "region resolution", (), |_| middle::region::resolve_crate(&sess, krate)); time(time_passes, "loop checking", (), |_| middle::check_loop::check_crate(&sess, krate)); let ty_cx = ty::mk_ctxt(sess, def_map, named_region_map, ast_map, freevars, region_map, lang_items); // passes are timed inside typeck typeck::check_crate(&ty_cx, trait_map, krate); time(time_passes, "check static items", (), |_| middle::check_static::check_crate(&ty_cx, krate)); // These next two const passes can probably be merged time(time_passes, "const marking", (), |_| middle::const_eval::process_crate(krate, &ty_cx)); time(time_passes, "const checking", (), |_| middle::check_const::check_crate(krate, &ty_cx)); let maps = (external_exports, last_private_map); let (exported_items, public_items) = time(time_passes, "privacy checking", maps, |(a, b)| middle::privacy::check_crate(&ty_cx, &exp_map2, a, b, krate)); time(time_passes, "effect checking", (), |_| middle::effect::check_crate(&ty_cx, krate)); let middle::moves::MoveMaps {moves_map, moved_variables_set, capture_map} = time(time_passes, "compute moves", (), |_| middle::moves::compute_moves(&ty_cx, krate)); time(time_passes, "match checking", (), |_| middle::check_match::check_crate(&ty_cx, &moves_map, krate)); time(time_passes, "liveness checking", (), |_| middle::liveness::check_crate(&ty_cx, &capture_map, krate)); let root_map = time(time_passes, "borrow checking", (), |_| middle::borrowck::check_crate(&ty_cx, &moves_map, &moved_variables_set, &capture_map, krate)); drop(moves_map); drop(moved_variables_set); time(time_passes, "kind checking", (), |_| kind::check_crate(&ty_cx, krate)); let reachable_map = time(time_passes, "reachability checking", (), |_| reachable::find_reachable(&ty_cx, &exported_items)); time(time_passes, "death checking", (), |_| { middle::dead::check_crate(&ty_cx, &exported_items, &reachable_map, krate) }); time(time_passes, "lint checking", (), |_| lint::check_crate(&ty_cx, &exported_items, krate)); CrateAnalysis { exp_map2: exp_map2, ty_cx: ty_cx, exported_items: exported_items, public_items: public_items, maps: astencode::Maps { root_map: root_map, capture_map: RefCell::new(capture_map) }, reachable: reachable_map } } pub struct CrateTranslation { pub context: ContextRef, pub module: ModuleRef, pub metadata_module: ModuleRef, pub link: LinkMeta, pub metadata: Vec, pub reachable: Vec<~str>, } /// Run the translation phase to LLVM, after which the AST and analysis can /// be discarded. pub fn phase_4_translate_to_llvm(krate: ast::Crate, analysis: CrateAnalysis, outputs: &OutputFilenames) -> (ty::ctxt, CrateTranslation) { // Option dance to work around the lack of stack once closures. let time_passes = analysis.ty_cx.sess.time_passes(); let mut analysis = Some(analysis); time(time_passes, "translation", krate, |krate| trans::base::trans_crate(krate, analysis.take_unwrap(), outputs)) } /// Run LLVM itself, producing a bitcode file, assembly file or object file /// as a side effect. pub fn phase_5_run_llvm_passes(sess: &Session, trans: &CrateTranslation, outputs: &OutputFilenames) { if sess.opts.cg.no_integrated_as { let output_type = link::OutputTypeAssembly; time(sess.time_passes(), "LLVM passes", (), |_| link::write::run_passes(sess, trans, [output_type], outputs)); link::write::run_assembler(sess, outputs); // Remove assembly source, unless --save-temps was specified if !sess.opts.cg.save_temps { fs::unlink(&outputs.temp_path(link::OutputTypeAssembly)).unwrap(); } } else { time(sess.time_passes(), "LLVM passes", (), |_| link::write::run_passes(sess, trans, sess.opts.output_types.as_slice(), outputs)); } } /// Run the linker on any artifacts that resulted from the LLVM run. /// This should produce either a finished executable or library. pub fn phase_6_link_output(sess: &Session, trans: &CrateTranslation, outputs: &OutputFilenames) { time(sess.time_passes(), "linking", (), |_| link::link_binary(sess, trans, outputs, &trans.link.crateid)); } pub fn stop_after_phase_3(sess: &Session) -> bool { if sess.opts.no_trans { debug!("invoked with --no-trans, returning early from compile_input"); return true; } return false; } pub fn stop_after_phase_1(sess: &Session) -> bool { if sess.opts.parse_only { debug!("invoked with --parse-only, returning early from compile_input"); return true; } if sess.show_span() { return true; } return sess.opts.debugging_opts & session::AST_JSON_NOEXPAND != 0; } pub fn stop_after_phase_2(sess: &Session) -> bool { if sess.opts.no_analysis { debug!("invoked with --no-analysis, returning early from compile_input"); return true; } return sess.opts.debugging_opts & session::AST_JSON != 0; } pub fn stop_after_phase_5(sess: &Session) -> bool { if !sess.opts.output_types.iter().any(|&i| i == link::OutputTypeExe) { debug!("not building executable, returning early from compile_input"); return true; } return false; } fn write_out_deps(sess: &Session, input: &Input, outputs: &OutputFilenames, krate: &ast::Crate) -> io::IoResult<()> { let id = link::find_crate_id(krate.attrs.as_slice(), outputs.out_filestem); let mut out_filenames = Vec::new(); for output_type in sess.opts.output_types.iter() { let file = outputs.path(*output_type); match *output_type { link::OutputTypeExe => { for output in sess.crate_types.borrow().iter() { let p = link::filename_for_input(sess, *output, &id, &file); out_filenames.push(p); } } _ => { out_filenames.push(file); } } } // Write out dependency rules to the dep-info file if requested with // --dep-info let deps_filename = match sess.opts.write_dependency_info { // Use filename from --dep-file argument if given (true, Some(ref filename)) => filename.clone(), // Use default filename: crate source filename with extension replaced // by ".d" (true, None) => match *input { FileInput(..) => outputs.with_extension("d"), StrInput(..) => { sess.warn("can not write --dep-info without a filename \ when compiling stdin."); return Ok(()); }, }, _ => return Ok(()), }; // Build a list of files used to compile the output and // write Makefile-compatible dependency rules let files: Vec<~str> = sess.codemap().files.borrow() .iter().filter_map(|fmap| { if fmap.is_real_file() { Some(fmap.name.clone()) } else { None } }).collect(); let mut file = try!(io::File::create(&deps_filename)); for path in out_filenames.iter() { try!(write!(&mut file as &mut Writer, "{}: {}\n\n", path.display(), files.connect(" "))); } Ok(()) } pub fn compile_input(sess: Session, cfg: ast::CrateConfig, input: &Input, outdir: &Option, output: &Option) { // We need nested scopes here, because the intermediate results can keep // large chunks of memory alive and we want to free them as soon as // possible to keep the peak memory usage low let (outputs, trans, sess) = { let (outputs, expanded_crate, ast_map) = { let krate = phase_1_parse_input(&sess, cfg, input); if stop_after_phase_1(&sess) { return; } let outputs = build_output_filenames(input, outdir, output, krate.attrs.as_slice(), &sess); let loader = &mut Loader::new(&sess); let id = link::find_crate_id(krate.attrs.as_slice(), outputs.out_filestem); let (expanded_crate, ast_map) = phase_2_configure_and_expand(&sess, loader, krate, &id); (outputs, expanded_crate, ast_map) }; write_out_deps(&sess, input, &outputs, &expanded_crate).unwrap(); if stop_after_phase_2(&sess) { return; } let analysis = phase_3_run_analysis_passes(sess, &expanded_crate, ast_map); if stop_after_phase_3(&analysis.ty_cx.sess) { return; } let (tcx, trans) = phase_4_translate_to_llvm(expanded_crate, analysis, &outputs); // Discard interned strings as they are no longer required. token::get_ident_interner().clear(); (outputs, trans, tcx.sess) }; phase_5_run_llvm_passes(&sess, &trans, &outputs); if stop_after_phase_5(&sess) { return; } phase_6_link_output(&sess, &trans, &outputs); } struct IdentifiedAnnotation; impl pprust::PpAnn for IdentifiedAnnotation { fn pre(&self, s: &mut pprust::State, node: pprust::AnnNode) -> io::IoResult<()> { match node { pprust::NodeExpr(_) => s.popen(), _ => Ok(()) } } fn post(&self, s: &mut pprust::State, node: pprust::AnnNode) -> io::IoResult<()> { match node { pprust::NodeItem(item) => { try!(pp::space(&mut s.s)); s.synth_comment(item.id.to_str()) } pprust::NodeBlock(blk) => { try!(pp::space(&mut s.s)); s.synth_comment("block ".to_owned() + blk.id.to_str()) } pprust::NodeExpr(expr) => { try!(pp::space(&mut s.s)); try!(s.synth_comment(expr.id.to_str())); s.pclose() } pprust::NodePat(pat) => { try!(pp::space(&mut s.s)); s.synth_comment("pat ".to_owned() + pat.id.to_str()) } } } } struct TypedAnnotation { analysis: CrateAnalysis, } impl pprust::PpAnn for TypedAnnotation { fn pre(&self, s: &mut pprust::State, node: pprust::AnnNode) -> io::IoResult<()> { match node { pprust::NodeExpr(_) => s.popen(), _ => Ok(()) } } fn post(&self, s: &mut pprust::State, node: pprust::AnnNode) -> io::IoResult<()> { let tcx = &self.analysis.ty_cx; match node { pprust::NodeExpr(expr) => { try!(pp::space(&mut s.s)); try!(pp::word(&mut s.s, "as")); try!(pp::space(&mut s.s)); try!(pp::word(&mut s.s, ppaux::ty_to_str(tcx, ty::expr_ty(tcx, expr)))); s.pclose() } _ => Ok(()) } } } pub fn pretty_print_input(sess: Session, cfg: ast::CrateConfig, input: &Input, ppm: PpMode, ofile: Option) { let krate = phase_1_parse_input(&sess, cfg, input); let id = link::find_crate_id(krate.attrs.as_slice(), input.filestem()); let (krate, ast_map, is_expanded) = match ppm { PpmExpanded | PpmExpandedIdentified | PpmTyped => { let loader = &mut Loader::new(&sess); let (krate, ast_map) = phase_2_configure_and_expand(&sess, loader, krate, &id); (krate, Some(ast_map), true) } _ => (krate, None, false) }; let src_name = source_name(input); let src = Vec::from_slice(sess.codemap().get_filemap(src_name).src.as_bytes()); let mut rdr = MemReader::new(src); let out = match ofile { None => ~io::stdout() as ~Writer, Some(p) => { let r = io::File::create(&p); match r { Ok(w) => ~w as ~Writer, Err(e) => fail!("print-print failed to open {} due to {}", p.display(), e), } } }; match ppm { PpmIdentified | PpmExpandedIdentified => { pprust::print_crate(sess.codemap(), sess.diagnostic(), &krate, src_name, &mut rdr, out, &IdentifiedAnnotation, is_expanded) } PpmTyped => { let ast_map = ast_map.expect("--pretty=typed missing ast_map"); let analysis = phase_3_run_analysis_passes(sess, &krate, ast_map); let annotation = TypedAnnotation { analysis: analysis }; pprust::print_crate(annotation.analysis.ty_cx.sess.codemap(), annotation.analysis.ty_cx.sess.diagnostic(), &krate, src_name, &mut rdr, out, &annotation, is_expanded) } _ => { pprust::print_crate(sess.codemap(), sess.diagnostic(), &krate, src_name, &mut rdr, out, &pprust::NoAnn, is_expanded) } }.unwrap() } pub fn get_os(triple: &str) -> Option { for &(name, os) in os_names.iter() { if triple.contains(name) { return Some(os) } } None } static os_names : &'static [(&'static str, abi::Os)] = &'static [ ("mingw32", abi::OsWin32), ("win32", abi::OsWin32), ("darwin", abi::OsMacos), ("android", abi::OsAndroid), ("linux", abi::OsLinux), ("freebsd", abi::OsFreebsd)]; pub fn get_arch(triple: &str) -> Option { for &(arch, abi) in architecture_abis.iter() { if triple.contains(arch) { return Some(abi) } } None } static architecture_abis : &'static [(&'static str, abi::Architecture)] = &'static [ ("i386", abi::X86), ("i486", abi::X86), ("i586", abi::X86), ("i686", abi::X86), ("i786", abi::X86), ("x86_64", abi::X86_64), ("arm", abi::Arm), ("xscale", abi::Arm), ("thumb", abi::Arm), ("mips", abi::Mips)]; pub fn build_target_config(sopts: &session::Options) -> session::Config { let os = match get_os(sopts.target_triple) { Some(os) => os, None => early_error("unknown operating system") }; let arch = match get_arch(sopts.target_triple) { Some(arch) => arch, None => early_error("unknown architecture: " + sopts.target_triple) }; let (int_type, uint_type) = match arch { abi::X86 => (ast::TyI32, ast::TyU32), abi::X86_64 => (ast::TyI64, ast::TyU64), abi::Arm => (ast::TyI32, ast::TyU32), abi::Mips => (ast::TyI32, ast::TyU32) }; let target_triple = sopts.target_triple.clone(); let target_strs = match arch { abi::X86 => x86::get_target_strs(target_triple, os), abi::X86_64 => x86_64::get_target_strs(target_triple, os), abi::Arm => arm::get_target_strs(target_triple, os), abi::Mips => mips::get_target_strs(target_triple, os) }; session::Config { os: os, arch: arch, target_strs: target_strs, int_type: int_type, uint_type: uint_type, } } pub fn host_triple() -> ~str { // Get the host triple out of the build environment. This ensures that our // idea of the host triple is the same as for the set of libraries we've // actually built. We can't just take LLVM's host triple because they // normalize all ix86 architectures to i386. // // Instead of grabbing the host triple (for the current host), we grab (at // compile time) the target triple that this rustc is built with and // calling that (at runtime) the host triple. (env!("CFG_COMPILER_HOST_TRIPLE")).to_owned() } pub fn build_session_options(matches: &getopts::Matches) -> session::Options { let mut crate_types: Vec = Vec::new(); let unparsed_crate_types = matches.opt_strs("crate-type"); for unparsed_crate_type in unparsed_crate_types.iter() { for part in unparsed_crate_type.split(',') { let new_part = match part { "lib" => session::default_lib_output(), "rlib" => session::CrateTypeRlib, "staticlib" => session::CrateTypeStaticlib, "dylib" => session::CrateTypeDylib, "bin" => session::CrateTypeExecutable, _ => early_error(format!("unknown crate type: `{}`", part)) }; crate_types.push(new_part) } } let parse_only = matches.opt_present("parse-only"); let no_trans = matches.opt_present("no-trans"); let no_analysis = matches.opt_present("no-analysis"); let lint_levels = [lint::allow, lint::warn, lint::deny, lint::forbid]; let mut lint_opts = Vec::new(); let lint_dict = lint::get_lint_dict(); for level in lint_levels.iter() { let level_name = lint::level_to_str(*level); let level_short = level_name.slice_chars(0, 1); let level_short = level_short.to_ascii().to_upper().into_str(); let flags = matches.opt_strs(level_short).move_iter().collect::>().append( matches.opt_strs(level_name).as_slice()); for lint_name in flags.iter() { let lint_name = lint_name.replace("-", "_"); match lint_dict.find_equiv(&lint_name) { None => { early_error(format!("unknown {} flag: {}", level_name, lint_name)); } Some(lint) => { lint_opts.push((lint.lint, *level)); } } } } let mut debugging_opts = 0; let debug_flags = matches.opt_strs("Z"); let debug_map = session::debugging_opts_map(); for debug_flag in debug_flags.iter() { let mut this_bit = 0; for tuple in debug_map.iter() { let (name, bit) = match *tuple { (ref a, _, b) => (a, b) }; if *name == *debug_flag { this_bit = bit; break; } } if this_bit == 0 { early_error(format!("unknown debug flag: {}", *debug_flag)) } debugging_opts |= this_bit; } if debugging_opts & session::DEBUG_LLVM != 0 { unsafe { llvm::LLVMSetDebug(1); } } let mut output_types = Vec::new(); if !parse_only && !no_trans { let unparsed_output_types = matches.opt_strs("emit"); for unparsed_output_type in unparsed_output_types.iter() { for part in unparsed_output_type.split(',') { let output_type = match part.as_slice() { "asm" => link::OutputTypeAssembly, "ir" => link::OutputTypeLlvmAssembly, "bc" => link::OutputTypeBitcode, "obj" => link::OutputTypeObject, "link" => link::OutputTypeExe, _ => early_error(format!("unknown emission type: `{}`", part)) }; output_types.push(output_type) } } }; output_types.as_mut_slice().sort(); output_types.dedup(); if output_types.len() == 0 { output_types.push(link::OutputTypeExe); } let sysroot_opt = matches.opt_str("sysroot").map(|m| Path::new(m)); let target = matches.opt_str("target").unwrap_or(host_triple()); let opt_level = { if (debugging_opts & session::NO_OPT) != 0 { No } else if matches.opt_present("O") { if matches.opt_present("opt-level") { early_error("-O and --opt-level both provided"); } Default } else if matches.opt_present("opt-level") { match matches.opt_str("opt-level").as_ref().map(|s| s.as_slice()) { None | Some("0") => No, Some("1") => Less, Some("2") => Default, Some("3") => Aggressive, Some(arg) => { early_error(format!("optimization level needs to be between 0-3 \ (instead was `{}`)", arg)); } } } else { No } }; let gc = debugging_opts & session::GC != 0; let debuginfo = if matches.opt_present("g") { if matches.opt_present("debuginfo") { early_error("-g and --debuginfo both provided"); } FullDebugInfo } else if matches.opt_present("debuginfo") { match matches.opt_str("debuginfo").as_ref().map(|s| s.as_slice()) { Some("0") => NoDebugInfo, Some("1") => LimitedDebugInfo, None | Some("2") => FullDebugInfo, Some(arg) => { early_error(format!("optimization level needs to be between 0-3 \ (instead was `{}`)", arg)); } } } else { NoDebugInfo }; let addl_lib_search_paths = matches.opt_strs("L").iter().map(|s| { Path::new(s.as_slice()) }).collect(); let cfg = parse_cfgspecs(matches.opt_strs("cfg").move_iter().collect()); let test = matches.opt_present("test"); let write_dependency_info = (matches.opt_present("dep-info"), matches.opt_str("dep-info").map(|p| Path::new(p))); let print_metas = (matches.opt_present("crate-id"), matches.opt_present("crate-name"), matches.opt_present("crate-file-name")); let cg = build_codegen_options(matches); session::Options { crate_types: crate_types, gc: gc, optimize: opt_level, debuginfo: debuginfo, lint_opts: lint_opts, output_types: output_types, addl_lib_search_paths: RefCell::new(addl_lib_search_paths), maybe_sysroot: sysroot_opt, target_triple: target, cfg: cfg, test: test, parse_only: parse_only, no_trans: no_trans, no_analysis: no_analysis, debugging_opts: debugging_opts, write_dependency_info: write_dependency_info, print_metas: print_metas, cg: cg, } } pub fn build_codegen_options(matches: &getopts::Matches) -> session::CodegenOptions { let mut cg = session::basic_codegen_options(); for option in matches.opt_strs("C").move_iter() { let mut iter = option.splitn('=', 1); let key = iter.next().unwrap(); let value = iter.next(); let option_to_lookup = key.replace("-", "_"); let mut found = false; for &(candidate, setter, _) in session::CG_OPTIONS.iter() { if option_to_lookup.as_slice() != candidate { continue } if !setter(&mut cg, value) { match value { Some(..) => early_error(format!("codegen option `{}` takes \ no value", key)), None => early_error(format!("codegen option `{0}` requires \ a value (-C {0}=)", key)) } } found = true; break; } if !found { early_error(format!("unknown codegen option: `{}`", key)); } } return cg; } pub fn build_session(sopts: session::Options, local_crate_source_file: Option) -> Session { let codemap = codemap::CodeMap::new(); let diagnostic_handler = diagnostic::default_handler(); let span_diagnostic_handler = diagnostic::mk_span_handler(diagnostic_handler, codemap); build_session_(sopts, local_crate_source_file, span_diagnostic_handler) } pub fn build_session_(sopts: session::Options, local_crate_source_file: Option, span_diagnostic: diagnostic::SpanHandler) -> Session { let target_cfg = build_target_config(&sopts); let p_s = parse::new_parse_sess_special_handler(span_diagnostic); let default_sysroot = match sopts.maybe_sysroot { Some(_) => None, None => Some(filesearch::get_or_default_sysroot()) }; // Make the path absolute, if necessary let local_crate_source_file = local_crate_source_file.map(|path| if path.is_absolute() { path.clone() } else { os::getcwd().join(path.clone()) } ); Session { targ_cfg: target_cfg, opts: sopts, cstore: CStore::new(token::get_ident_interner()), parse_sess: p_s, // For a library crate, this is always none entry_fn: RefCell::new(None), entry_type: Cell::new(None), macro_registrar_fn: Cell::new(None), default_sysroot: default_sysroot, building_library: Cell::new(false), local_crate_source_file: local_crate_source_file, working_dir: os::getcwd(), lints: RefCell::new(NodeMap::new()), node_id: Cell::new(1), crate_types: RefCell::new(Vec::new()), features: front::feature_gate::Features::new(), recursion_limit: Cell::new(64), } } pub fn parse_pretty(sess: &Session, name: &str) -> PpMode { match name { &"normal" => PpmNormal, &"expanded" => PpmExpanded, &"typed" => PpmTyped, &"expanded,identified" => PpmExpandedIdentified, &"identified" => PpmIdentified, _ => { sess.fatal("argument to `pretty` must be one of `normal`, \ `expanded`, `typed`, `identified`, \ or `expanded,identified`"); } } } // rustc command line options pub fn optgroups() -> Vec { vec!( optflag("h", "help", "Display this message"), optmulti("", "cfg", "Configure the compilation environment", "SPEC"), optmulti("L", "", "Add a directory to the library search path", "PATH"), optmulti("", "crate-type", "Comma separated list of types of crates for the compiler to emit", "[bin|lib|rlib|dylib|staticlib]"), optmulti("", "emit", "Comma separated list of types of output for the compiler to emit", "[asm|bc|ir|obj|link]"), optflag("", "crate-id", "Output the crate id and exit"), optflag("", "crate-name", "Output the crate name and exit"), optflag("", "crate-file-name", "Output the file(s) that would be written if compilation \ continued and exit"), optflag("g", "", "Equivalent to --debuginfo=2"), optopt("", "debuginfo", "Emit DWARF debug info to the objects created: 0 = no debug info, 1 = line-tables only (for stacktraces and breakpoints), 2 = full debug info with variable and type information (same as -g)", "LEVEL"), optflag("", "no-trans", "Run all passes except translation; no output"), optflag("", "no-analysis", "Parse and expand the source, but run no analysis and produce no output"), optflag("O", "", "Equivalent to --opt-level=2"), optopt("o", "", "Write output to ", "FILENAME"), optopt("", "opt-level", "Optimize with possible levels 0-3", "LEVEL"), optopt( "", "out-dir", "Write output to compiler-chosen filename in ", "DIR"), optflag("", "parse-only", "Parse only; do not compile, assemble, or link"), optflagopt("", "pretty", "Pretty-print the input instead of compiling; valid types are: normal (un-annotated source), expanded (crates expanded), typed (crates expanded, with type annotations), or identified (fully parenthesized, AST nodes and blocks with IDs)", "TYPE"), optflagopt("", "dep-info", "Output dependency info to after compiling, \ in a format suitable for use by Makefiles", "FILENAME"), optopt("", "sysroot", "Override the system root", "PATH"), optflag("", "test", "Build a test harness"), optopt("", "target", "Target triple cpu-manufacturer-kernel[-os] to compile for (see chapter 3.4 of http://www.sourceware.org/autobook/ for details)", "TRIPLE"), optmulti("W", "warn", "Set lint warnings", "OPT"), optmulti("A", "allow", "Set lint allowed", "OPT"), optmulti("D", "deny", "Set lint denied", "OPT"), optmulti("F", "forbid", "Set lint forbidden", "OPT"), optmulti("C", "codegen", "Set a codegen option", "OPT[=VALUE]"), optmulti("Z", "", "Set internal debugging options", "FLAG"), optflag( "v", "version", "Print version info and exit")) } pub struct OutputFilenames { pub out_directory: Path, pub out_filestem: ~str, pub single_output_file: Option, } impl OutputFilenames { pub fn path(&self, flavor: link::OutputType) -> Path { match self.single_output_file { Some(ref path) => return path.clone(), None => {} } self.temp_path(flavor) } pub fn temp_path(&self, flavor: link::OutputType) -> Path { let base = self.out_directory.join(self.out_filestem.as_slice()); match flavor { link::OutputTypeBitcode => base.with_extension("bc"), link::OutputTypeAssembly => base.with_extension("s"), link::OutputTypeLlvmAssembly => base.with_extension("ll"), link::OutputTypeObject => base.with_extension("o"), link::OutputTypeExe => base, } } pub fn with_extension(&self, extension: &str) -> Path { let stem = self.out_filestem.as_slice(); self.out_directory.join(stem).with_extension(extension) } } pub fn build_output_filenames(input: &Input, odir: &Option, ofile: &Option, attrs: &[ast::Attribute], sess: &Session) -> OutputFilenames { match *ofile { None => { // "-" as input file will cause the parser to read from stdin so we // have to make up a name // We want to toss everything after the final '.' let dirpath = match *odir { Some(ref d) => d.clone(), None => Path::new(".") }; let mut stem = input.filestem(); // If a crateid is present, we use it as the link name let crateid = attr::find_crateid(attrs); match crateid { None => {} Some(crateid) => stem = crateid.name.to_str(), } OutputFilenames { out_directory: dirpath, out_filestem: stem, single_output_file: None, } } Some(ref out_file) => { let ofile = if sess.opts.output_types.len() > 1 { sess.warn("ignoring specified output filename because multiple \ outputs were requested"); None } else { Some(out_file.clone()) }; if *odir != None { sess.warn("ignoring --out-dir flag due to -o flag."); } OutputFilenames { out_directory: out_file.dir_path(), out_filestem: out_file.filestem_str().unwrap().to_str(), single_output_file: ofile, } } } } pub fn early_error(msg: &str) -> ! { let mut emitter = diagnostic::EmitterWriter::stderr(); emitter.emit(None, msg, diagnostic::Fatal); fail!(diagnostic::FatalError); } pub fn list_metadata(sess: &Session, path: &Path, out: &mut io::Writer) -> io::IoResult<()> { metadata::loader::list_file_metadata( session::sess_os_to_meta_os(sess.targ_cfg.os), path, out) } #[cfg(test)] mod test { use driver::driver::{build_configuration, build_session}; use driver::driver::{build_session_options, optgroups}; use getopts::getopts; use syntax::attr; use syntax::attr::AttrMetaMethods; // When the user supplies --test we should implicitly supply --cfg test #[test] fn test_switch_implies_cfg_test() { let matches = &match getopts(["--test".to_owned()], optgroups().as_slice()) { Ok(m) => m, Err(f) => fail!("test_switch_implies_cfg_test: {}", f.to_err_msg()) }; let sessopts = build_session_options(matches); let sess = build_session(sessopts, None); let cfg = build_configuration(&sess); assert!((attr::contains_name(cfg.as_slice(), "test"))); } // When the user supplies --test and --cfg test, don't implicitly add // another --cfg test #[test] fn test_switch_implies_cfg_test_unless_cfg_test() { let matches = &match getopts(["--test".to_owned(), "--cfg=test".to_owned()], optgroups().as_slice()) { Ok(m) => m, Err(f) => { fail!("test_switch_implies_cfg_test_unless_cfg_test: {}", f.to_err_msg()); } }; let sessopts = build_session_options(matches); let sess = build_session(sessopts, None); let cfg = build_configuration(&sess); let mut test_items = cfg.iter().filter(|m| m.name().equiv(&("test"))); assert!(test_items.next().is_some()); assert!(test_items.next().is_none()); } }