169 lines
5.1 KiB
Rust
169 lines
5.1 KiB
Rust
// Copyright 2017 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.
|
|
|
|
//! # Note
|
|
//!
|
|
//! This API is completely unstable and subject to change.
|
|
|
|
#![doc(html_logo_url = "https://www.rust-lang.org/logos/rust-logo-128x128-blk-v2.png",
|
|
html_favicon_url = "https://doc.rust-lang.org/favicon.ico",
|
|
html_root_url = "https://doc.rust-lang.org/nightly/")]
|
|
|
|
#![feature(box_patterns)]
|
|
#![feature(box_syntax)]
|
|
#![feature(custom_attribute)]
|
|
#![feature(nll)]
|
|
#![allow(unused_attributes)]
|
|
#![allow(dead_code)]
|
|
#![feature(quote)]
|
|
#![feature(rustc_diagnostic_macros)]
|
|
|
|
#![recursion_limit="256"]
|
|
|
|
extern crate flate2;
|
|
#[macro_use]
|
|
extern crate log;
|
|
|
|
extern crate serialize;
|
|
#[macro_use]
|
|
extern crate rustc;
|
|
extern crate rustc_allocator;
|
|
extern crate rustc_target;
|
|
extern crate rustc_metadata;
|
|
extern crate rustc_mir;
|
|
extern crate rustc_incremental;
|
|
extern crate syntax;
|
|
extern crate syntax_pos;
|
|
#[macro_use] extern crate rustc_data_structures;
|
|
|
|
use std::path::PathBuf;
|
|
|
|
use rustc::session::Session;
|
|
use rustc::ty::TyCtxt;
|
|
use rustc::dep_graph::WorkProduct;
|
|
use rustc::session::config::{OutputFilenames, OutputType};
|
|
|
|
pub mod command;
|
|
pub mod interfaces;
|
|
pub mod link;
|
|
pub mod linker;
|
|
pub mod codegen_backend;
|
|
pub mod symbol_export;
|
|
pub mod symbol_names;
|
|
pub mod symbol_names_test;
|
|
pub mod common;
|
|
|
|
pub struct ModuleCodegen<M> {
|
|
/// The name of the module. When the crate may be saved between
|
|
/// compilations, incremental compilation requires that name be
|
|
/// unique amongst **all** crates. Therefore, it should contain
|
|
/// something unique to this crate (e.g., a module path) as well
|
|
/// as the crate name and disambiguator.
|
|
/// We currently generate these names via CodegenUnit::build_cgu_name().
|
|
pub name: String,
|
|
pub module_llvm: M,
|
|
pub kind: ModuleKind,
|
|
}
|
|
|
|
pub const RLIB_BYTECODE_EXTENSION: &str = "bc.z";
|
|
|
|
impl<M> ModuleCodegen<M> {
|
|
pub fn into_compiled_module(self,
|
|
emit_obj: bool,
|
|
emit_bc: bool,
|
|
emit_bc_compressed: bool,
|
|
outputs: &OutputFilenames) -> CompiledModule {
|
|
let object = if emit_obj {
|
|
Some(outputs.temp_path(OutputType::Object, Some(&self.name)))
|
|
} else {
|
|
None
|
|
};
|
|
let bytecode = if emit_bc {
|
|
Some(outputs.temp_path(OutputType::Bitcode, Some(&self.name)))
|
|
} else {
|
|
None
|
|
};
|
|
let bytecode_compressed = if emit_bc_compressed {
|
|
Some(outputs.temp_path(OutputType::Bitcode, Some(&self.name))
|
|
.with_extension(RLIB_BYTECODE_EXTENSION))
|
|
} else {
|
|
None
|
|
};
|
|
|
|
CompiledModule {
|
|
name: self.name.clone(),
|
|
kind: self.kind,
|
|
object,
|
|
bytecode,
|
|
bytecode_compressed,
|
|
}
|
|
}
|
|
}
|
|
|
|
#[derive(Debug)]
|
|
pub struct CompiledModule {
|
|
pub name: String,
|
|
pub kind: ModuleKind,
|
|
pub object: Option<PathBuf>,
|
|
pub bytecode: Option<PathBuf>,
|
|
pub bytecode_compressed: Option<PathBuf>,
|
|
}
|
|
|
|
pub struct CachedModuleCodegen {
|
|
pub name: String,
|
|
pub source: WorkProduct,
|
|
}
|
|
|
|
#[derive(Copy, Clone, Debug, PartialEq)]
|
|
pub enum ModuleKind {
|
|
Regular,
|
|
Metadata,
|
|
Allocator,
|
|
}
|
|
|
|
/// check for the #[rustc_error] annotation, which forces an
|
|
/// error in codegen. This is used to write compile-fail tests
|
|
/// that actually test that compilation succeeds without
|
|
/// reporting an error.
|
|
pub fn check_for_rustc_errors_attr(tcx: TyCtxt) {
|
|
if let Some((id, span, _)) = *tcx.sess.entry_fn.borrow() {
|
|
let main_def_id = tcx.hir.local_def_id(id);
|
|
|
|
if tcx.has_attr(main_def_id, "rustc_error") {
|
|
tcx.sess.span_fatal(span, "compilation successful");
|
|
}
|
|
}
|
|
}
|
|
|
|
pub fn find_library(name: &str, search_paths: &[PathBuf], sess: &Session)
|
|
-> PathBuf {
|
|
// On Windows, static libraries sometimes show up as libfoo.a and other
|
|
// times show up as foo.lib
|
|
let oslibname = format!("{}{}{}",
|
|
sess.target.target.options.staticlib_prefix,
|
|
name,
|
|
sess.target.target.options.staticlib_suffix);
|
|
let unixlibname = format!("lib{}.a", name);
|
|
|
|
for path in search_paths {
|
|
debug!("looking for {} inside {:?}", name, path);
|
|
let test = path.join(&oslibname);
|
|
if test.exists() { return test }
|
|
if oslibname != unixlibname {
|
|
let test = path.join(&unixlibname);
|
|
if test.exists() { return test }
|
|
}
|
|
}
|
|
sess.fatal(&format!("could not find native static library `{}`, \
|
|
perhaps an -L flag is missing?", name));
|
|
}
|
|
|
|
__build_diagnostic_array! { librustc_codegen_utils, DIAGNOSTICS }
|