// 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 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::target_strs; use back; use driver::driver::host_triple; use driver::session; use metadata::filesearch; use metadata; use middle::lint; use syntax::ast::node_id; use syntax::ast::{int_ty, uint_ty, float_ty}; use syntax::codemap::span; use syntax::diagnostic; use syntax::parse::ParseSess; use syntax::{ast, codemap}; use syntax::abi; use syntax; use core::hashmap::HashMap; #[deriving(Eq)] pub enum os { os_win32, os_macos, os_linux, os_android, os_freebsd, } pub enum crate_type { bin_crate, lib_crate, unknown_crate, } pub struct config { os: os, arch: abi::Architecture, target_strs: target_strs::t, int_type: int_ty, uint_type: uint_ty, float_type: float_ty } pub static verbose: uint = 1 << 0; pub static time_passes: uint = 1 << 1; pub static count_llvm_insns: uint = 1 << 2; pub static time_llvm_passes: uint = 1 << 3; pub static trans_stats: uint = 1 << 4; pub static asm_comments: uint = 1 << 5; pub static no_verify: uint = 1 << 6; pub static trace: uint = 1 << 7; pub static coherence: uint = 1 << 8; pub static borrowck_stats: uint = 1 << 9; pub static borrowck_note_pure: uint = 1 << 10; pub static borrowck_note_loan: uint = 1 << 11; pub static no_landing_pads: uint = 1 << 12; pub static debug_llvm: uint = 1 << 13; pub static count_type_sizes: uint = 1 << 14; pub static meta_stats: uint = 1 << 15; pub static no_opt: uint = 1 << 16; pub static no_monomorphic_collapse: uint = 1 << 17; pub static gc: uint = 1 << 18; pub static jit: uint = 1 << 19; pub static debug_info: uint = 1 << 20; pub static extra_debug_info: uint = 1 << 21; pub static statik: uint = 1 << 22; pub static print_link_args: uint = 1 << 23; pub fn debugging_opts_map() -> ~[(~str, ~str, uint)] { ~[(~"verbose", ~"in general, enable more debug printouts", verbose), (~"time-passes", ~"measure time of each rustc pass", time_passes), (~"count-llvm-insns", ~"count where LLVM \ instrs originate", count_llvm_insns), (~"time-llvm-passes", ~"measure time of each LLVM pass", time_llvm_passes), (~"trans-stats", ~"gather trans statistics", trans_stats), (~"asm-comments", ~"generate comments into the assembly (may change behavior)", asm_comments), (~"no-verify", ~"skip LLVM verification", no_verify), (~"trace", ~"emit trace logs", trace), (~"coherence", ~"perform coherence checking", coherence), (~"borrowck-stats", ~"gather borrowck statistics", borrowck_stats), (~"borrowck-note-pure", ~"note where purity is req'd", borrowck_note_pure), (~"borrowck-note-loan", ~"note where loans are req'd", borrowck_note_loan), (~"no-landing-pads", ~"omit landing pads for unwinding", no_landing_pads), (~"debug-llvm", ~"enable debug output from LLVM", debug_llvm), (~"count-type-sizes", ~"count the sizes of aggregate types", count_type_sizes), (~"meta-stats", ~"gather metadata statistics", meta_stats), (~"no-opt", ~"do not optimize, even if -O is passed", no_opt), (~"no-monomorphic-collapse", ~"do not collapse template instantiations", no_monomorphic_collapse), (~"print-link-args", ~"Print the arguments passed to the linker", print_link_args), (~"gc", ~"Garbage collect shared data (experimental)", gc), (~"jit", ~"Execute using JIT (experimental)", jit), (~"extra-debug-info", ~"Extra debugging info (experimental)", extra_debug_info), (~"debug-info", ~"Produce debug info (experimental)", debug_info), (~"static", ~"Use or produce static libraries or binaries " + "(experimental)", statik) ] } #[deriving(Eq)] pub enum OptLevel { No, // -O0 Less, // -O1 Default, // -O2 Aggressive // -O3 } pub struct options { // The crate config requested for the session, which may be combined // with additional crate configurations during the compile process crate_type: crate_type, is_static: bool, gc: bool, optimize: OptLevel, debuginfo: bool, extra_debuginfo: bool, lint_opts: ~[(lint::lint, lint::level)], save_temps: bool, jit: bool, output_type: back::link::output_type, addl_lib_search_paths: ~[Path], linker: Option<~str>, linker_args: ~[~str], maybe_sysroot: Option<@Path>, target_triple: ~str, target_feature: ~str, // User-specified cfg meta items. The compiler itself will add additional // items to the crate config, and during parsing the entire crate config // will be added to the crate AST node. This should not be used for // anything except building the full crate config prior to parsing. cfg: ast::crate_cfg, binary: @~str, test: bool, parse_only: bool, no_trans: bool, debugging_opts: uint, android_cross_path: Option<~str> } pub struct crate_metadata { name: ~str, data: ~[u8] } // The type of entry function, so // users can have their own entry // functions that don't start a // scheduler #[deriving(Eq)] pub enum EntryFnType { EntryMain, EntryStart } pub struct Session_ { targ_cfg: @config, opts: @options, cstore: @mut metadata::cstore::CStore, parse_sess: @mut ParseSess, codemap: @codemap::CodeMap, // For a library crate, this is always none entry_fn: @mut Option<(node_id, codemap::span)>, entry_type: @mut Option, span_diagnostic: @diagnostic::span_handler, filesearch: @filesearch::FileSearch, building_library: @mut bool, working_dir: Path, lints: @mut HashMap, } pub type Session = @Session_; pub impl Session_ { fn span_fatal(@self, sp: span, msg: &str) -> ! { self.span_diagnostic.span_fatal(sp, msg) } fn fatal(@self, msg: &str) -> ! { self.span_diagnostic.handler().fatal(msg) } fn span_err(@self, sp: span, msg: &str) { self.span_diagnostic.span_err(sp, msg) } fn err(@self, msg: &str) { self.span_diagnostic.handler().err(msg) } fn err_count(@self) -> uint { self.span_diagnostic.handler().err_count() } fn has_errors(@self) -> bool { self.span_diagnostic.handler().has_errors() } fn abort_if_errors(@self) { self.span_diagnostic.handler().abort_if_errors() } fn span_warn(@self, sp: span, msg: &str) { self.span_diagnostic.span_warn(sp, msg) } fn warn(@self, msg: &str) { self.span_diagnostic.handler().warn(msg) } fn span_note(@self, sp: span, msg: &str) { self.span_diagnostic.span_note(sp, msg) } fn note(@self, msg: &str) { self.span_diagnostic.handler().note(msg) } fn span_bug(@self, sp: span, msg: &str) -> ! { self.span_diagnostic.span_bug(sp, msg) } fn bug(@self, msg: &str) -> ! { self.span_diagnostic.handler().bug(msg) } fn span_unimpl(@self, sp: span, msg: &str) -> ! { self.span_diagnostic.span_unimpl(sp, msg) } fn unimpl(@self, msg: &str) -> ! { self.span_diagnostic.handler().unimpl(msg) } fn add_lint(@self, lint: lint::lint, id: ast::node_id, sp: span, msg: ~str) { match self.lints.find_mut(&id) { Some(arr) => { arr.push((lint, sp, msg)); return; } None => {} } self.lints.insert(id, ~[(lint, sp, msg)]); } fn next_node_id(@self) -> ast::node_id { return syntax::parse::next_node_id(self.parse_sess); } fn diagnostic(@self) -> @diagnostic::span_handler { self.span_diagnostic } fn debugging_opt(@self, opt: uint) -> bool { (self.opts.debugging_opts & opt) != 0u } // This exists to help with refactoring to eliminate impossible // cases later on fn impossible_case(@self, sp: span, msg: &str) -> ! { self.span_bug(sp, fmt!("Impossible case reached: %s", msg)); } fn verbose(@self) -> bool { self.debugging_opt(verbose) } fn time_passes(@self) -> bool { self.debugging_opt(time_passes) } fn count_llvm_insns(@self) -> bool { self.debugging_opt(count_llvm_insns) } fn count_type_sizes(@self) -> bool { self.debugging_opt(count_type_sizes) } fn time_llvm_passes(@self) -> bool { self.debugging_opt(time_llvm_passes) } fn trans_stats(@self) -> bool { self.debugging_opt(trans_stats) } fn meta_stats(@self) -> bool { self.debugging_opt(meta_stats) } fn asm_comments(@self) -> bool { self.debugging_opt(asm_comments) } fn no_verify(@self) -> bool { self.debugging_opt(no_verify) } fn trace(@self) -> bool { self.debugging_opt(trace) } fn coherence(@self) -> bool { self.debugging_opt(coherence) } fn borrowck_stats(@self) -> bool { self.debugging_opt(borrowck_stats) } fn borrowck_note_pure(@self) -> bool { self.debugging_opt(borrowck_note_pure) } fn borrowck_note_loan(@self) -> bool { self.debugging_opt(borrowck_note_loan) } fn no_monomorphic_collapse(@self) -> bool { self.debugging_opt(no_monomorphic_collapse) } fn str_of(@self, id: ast::ident) -> @~str { self.parse_sess.interner.get(id) } fn ident_of(@self, st: &str) -> ast::ident { self.parse_sess.interner.intern(st) } fn intr(@self) -> @syntax::parse::token::ident_interner { self.parse_sess.interner } } /// Some reasonable defaults pub fn basic_options() -> @options { @options { crate_type: session::lib_crate, is_static: false, gc: false, optimize: No, debuginfo: false, extra_debuginfo: false, lint_opts: ~[], save_temps: false, jit: false, output_type: link::output_type_exe, addl_lib_search_paths: ~[], linker: None, linker_args: ~[], maybe_sysroot: None, target_triple: host_triple(), target_feature: ~"", cfg: ~[], binary: @~"rustc", test: false, parse_only: false, no_trans: false, debugging_opts: 0u, android_cross_path: None } } // Seems out of place, but it uses session, so I'm putting it here pub fn expect(sess: Session, opt: Option, msg: &fn() -> ~str) -> T { diagnostic::expect(sess.diagnostic(), opt, msg) } pub fn building_library(req_crate_type: crate_type, crate: @ast::crate, testing: bool) -> bool { match req_crate_type { bin_crate => false, lib_crate => true, unknown_crate => { if testing { false } else { match syntax::attr::first_attr_value_str_by_name( crate.node.attrs, ~"crate_type") { Some(@~"lib") => true, _ => false } } } } } pub fn sess_os_to_meta_os(os: os) -> metadata::loader::os { use metadata::loader; match os { os_win32 => loader::os_win32, os_linux => loader::os_linux, os_android => loader::os_android, os_macos => loader::os_macos, os_freebsd => loader::os_freebsd } } #[cfg(test)] mod test { use driver::session::{bin_crate, building_library, lib_crate}; use driver::session::{unknown_crate}; use syntax::ast; use syntax::codemap; fn make_crate_type_attr(t: ~str) -> ast::attribute { codemap::respan(codemap::dummy_sp(), ast::attribute_ { style: ast::attr_outer, value: @codemap::respan(codemap::dummy_sp(), ast::meta_name_value( @~"crate_type", codemap::respan(codemap::dummy_sp(), ast::lit_str(@t)))), is_sugared_doc: false }) } fn make_crate(with_bin: bool, with_lib: bool) -> @ast::crate { let mut attrs = ~[]; if with_bin { attrs += ~[make_crate_type_attr(~"bin")]; } if with_lib { attrs += ~[make_crate_type_attr(~"lib")]; } @codemap::respan(codemap::dummy_sp(), ast::crate_ { module: ast::_mod { view_items: ~[], items: ~[] }, attrs: attrs, config: ~[] }) } #[test] fn bin_crate_type_attr_results_in_bin_output() { let crate = make_crate(true, false); assert!(!building_library(unknown_crate, crate, false)); } #[test] fn lib_crate_type_attr_results_in_lib_output() { let crate = make_crate(false, true); assert!(building_library(unknown_crate, crate, false)); } #[test] fn bin_option_overrides_lib_crate_type() { let crate = make_crate(false, true); assert!(!building_library(bin_crate, crate, false)); } #[test] fn lib_option_overrides_bin_crate_type() { let crate = make_crate(true, false); assert!(building_library(lib_crate, crate, false)); } #[test] fn bin_crate_type_is_default() { let crate = make_crate(false, false); assert!(!building_library(unknown_crate, crate, false)); } #[test] fn test_option_overrides_lib_crate_type() { let crate = make_crate(false, true); assert!(!building_library(unknown_crate, crate, true)); } #[test] fn test_option_does_not_override_requested_lib_type() { let crate = make_crate(false, false); assert!(building_library(lib_crate, crate, true)); } }