2013-06-13 19:19:50 +12:00
|
|
|
// Copyright 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 <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.
|
|
|
|
|
2013-06-13 14:02:33 +12:00
|
|
|
|
|
|
|
use driver::session;
|
2013-06-16 22:52:44 +12:00
|
|
|
use lib::llvm::{ContextRef, ModuleRef, ValueRef};
|
2013-06-13 14:49:01 +12:00
|
|
|
use lib::llvm::{llvm, TargetData, TypeNames};
|
2013-08-11 21:12:57 -04:00
|
|
|
use lib::llvm::mk_target_data;
|
2013-06-13 14:02:33 +12:00
|
|
|
use metadata::common::LinkMeta;
|
|
|
|
use middle::astencode;
|
|
|
|
use middle::resolve;
|
|
|
|
use middle::trans::adt;
|
|
|
|
use middle::trans::base;
|
2013-07-21 15:33:40 +02:00
|
|
|
use middle::trans::builder::Builder;
|
2013-06-13 14:02:33 +12:00
|
|
|
use middle::trans::debuginfo;
|
2013-08-11 13:42:26 -04:00
|
|
|
use middle::trans::common::{C_i32, C_null};
|
2013-06-13 14:02:33 +12:00
|
|
|
use middle::ty;
|
|
|
|
|
2013-06-15 22:16:47 +12:00
|
|
|
use middle::trans::type_::Type;
|
|
|
|
|
2013-12-09 14:56:53 -07:00
|
|
|
use util::sha2::Sha256;
|
|
|
|
|
2013-12-18 18:35:33 -08:00
|
|
|
use std::cell::{Cell, RefCell};
|
2013-08-03 17:13:14 -07:00
|
|
|
use std::c_str::ToCStr;
|
2013-06-28 18:32:26 -04:00
|
|
|
use std::hashmap::{HashMap, HashSet};
|
|
|
|
use std::local_data;
|
2013-08-11 13:42:26 -04:00
|
|
|
use std::libc::c_uint;
|
2013-06-13 14:49:01 +12:00
|
|
|
use syntax::ast;
|
2013-06-13 14:02:33 +12:00
|
|
|
|
2013-06-21 20:28:33 +12:00
|
|
|
use middle::trans::common::{mono_id,ExternMap,tydesc_info,BuilderRef_res,Stats};
|
2013-06-13 14:49:01 +12:00
|
|
|
|
|
|
|
use middle::trans::base::{decl_crate_map};
|
|
|
|
|
2013-06-13 14:02:33 +12:00
|
|
|
pub struct CrateContext {
|
|
|
|
sess: session::Session,
|
|
|
|
llmod: ModuleRef,
|
|
|
|
llcx: ContextRef,
|
Store metadata separately in rlib files
Right now whenever an rlib file is linked against, all of the metadata from the
rlib is pulled in to the final staticlib or binary. The reason for this is that
the metadata is currently stored in a section of the object file. Note that this
is intentional for dynamic libraries in order to distribute metadata bundled
with static libraries.
This commit alters the situation for rlib libraries to instead store the
metadata in a separate file in the archive. In doing so, when the archive is
passed to the linker, none of the metadata will get pulled into the result
executable. Furthermore, the metadata file is skipped when assembling rlibs into
an archive.
The snag in this implementation comes with multiple output formats. When
generating a dylib, the metadata needs to be in the object file, but when
generating an rlib this needs to be separate. In order to accomplish this, the
metadata variable is inserted into an entirely separate LLVM Module which is
then codegen'd into a different location (foo.metadata.o). This is then linked
into dynamic libraries and silently ignored for rlib files.
While changing how metadata is inserted into archives, I have also stopped
compressing metadata when inserted into rlib files. We have wanted to stop
compressing metadata, but the sections it creates in object file sections are
apparently too large. Thankfully if it's just an arbitrary file it doesn't
matter how large it is.
I have seen massive reductions in executable sizes, as well as staticlib output
sizes (to confirm that this is all working).
2013-12-03 17:41:01 -08:00
|
|
|
metadata_llmod: ModuleRef,
|
2013-06-13 14:02:33 +12:00
|
|
|
td: TargetData,
|
2013-06-15 14:31:52 +12:00
|
|
|
tn: TypeNames,
|
2013-12-18 18:24:34 -08:00
|
|
|
externs: RefCell<ExternMap>,
|
2013-06-13 14:02:33 +12:00
|
|
|
intrinsics: HashMap<&'static str, ValueRef>,
|
2013-12-18 16:41:15 -08:00
|
|
|
item_vals: RefCell<HashMap<ast::NodeId, ValueRef>>,
|
2013-06-13 14:02:33 +12:00
|
|
|
exp_map2: resolve::ExportMap2,
|
2013-12-20 20:14:51 -08:00
|
|
|
reachable: @RefCell<HashSet<ast::NodeId>>,
|
2013-12-18 16:53:23 -08:00
|
|
|
item_symbols: RefCell<HashMap<ast::NodeId, ~str>>,
|
2013-06-13 14:02:33 +12:00
|
|
|
link_meta: LinkMeta,
|
2013-12-21 16:43:43 -08:00
|
|
|
tydescs: RefCell<HashMap<ty::t, @tydesc_info>>,
|
2013-06-13 14:02:33 +12:00
|
|
|
// Set when running emit_tydescs to enforce that no more tydescs are
|
|
|
|
// created.
|
2013-12-18 18:35:33 -08:00
|
|
|
finished_tydescs: Cell<bool>,
|
2013-06-13 14:02:33 +12:00
|
|
|
// Track mapping of external ids to local items imported for inlining
|
2013-12-18 18:32:52 -08:00
|
|
|
external: RefCell<HashMap<ast::DefId, Option<ast::NodeId>>>,
|
2013-09-11 10:06:16 -07:00
|
|
|
// Backwards version of the `external` map (inlined items to where they
|
|
|
|
// came from)
|
2013-12-18 18:32:52 -08:00
|
|
|
external_srcs: RefCell<HashMap<ast::NodeId, ast::DefId>>,
|
2013-09-11 10:06:16 -07:00
|
|
|
// A set of static items which cannot be inlined into other crates. This
|
|
|
|
// will pevent in ii_item() structures from being encoded into the metadata
|
|
|
|
// that is generated
|
2013-12-18 18:11:41 -08:00
|
|
|
non_inlineable_statics: RefCell<HashSet<ast::NodeId>>,
|
2013-06-13 14:02:33 +12:00
|
|
|
// Cache instances of monomorphized functions
|
2013-12-18 16:35:29 -08:00
|
|
|
monomorphized: RefCell<HashMap<mono_id, ValueRef>>,
|
2013-12-18 16:38:25 -08:00
|
|
|
monomorphizing: RefCell<HashMap<ast::DefId, uint>>,
|
2013-06-13 14:02:33 +12:00
|
|
|
// Cache generated vtables
|
2013-12-18 17:00:56 -08:00
|
|
|
vtables: RefCell<HashMap<(ty::t, mono_id), ValueRef>>,
|
2013-06-13 14:02:33 +12:00
|
|
|
// Cache of constant strings,
|
2013-12-18 17:02:30 -08:00
|
|
|
const_cstr_cache: RefCell<HashMap<@str, ValueRef>>,
|
2013-06-13 14:02:33 +12:00
|
|
|
|
|
|
|
// Reverse-direction for const ptrs cast from globals.
|
|
|
|
// Key is an int, cast from a ValueRef holding a *T,
|
|
|
|
// Val is a ValueRef holding a *[T].
|
|
|
|
//
|
|
|
|
// Needed because LLVM loses pointer->pointee association
|
|
|
|
// when we ptrcast, and we have to ptrcast during translation
|
|
|
|
// of a [T] const because we form a slice, a [*T,int] pair, not
|
|
|
|
// a pointer to an LLVM array type.
|
2013-12-18 17:03:55 -08:00
|
|
|
const_globals: RefCell<HashMap<int, ValueRef>>,
|
2013-06-13 14:02:33 +12:00
|
|
|
|
|
|
|
// Cache of emitted const values
|
2013-12-18 17:08:56 -08:00
|
|
|
const_values: RefCell<HashMap<ast::NodeId, ValueRef>>,
|
2013-06-13 14:02:33 +12:00
|
|
|
|
|
|
|
// Cache of external const values
|
2013-12-18 17:11:22 -08:00
|
|
|
extern_const_values: RefCell<HashMap<ast::DefId, ValueRef>>,
|
2013-06-13 14:02:33 +12:00
|
|
|
|
2013-12-18 17:14:25 -08:00
|
|
|
impl_method_cache: RefCell<HashMap<(ast::DefId, ast::Name), ast::DefId>>,
|
2013-06-14 17:38:17 +12:00
|
|
|
|
2013-12-18 17:44:25 -08:00
|
|
|
module_data: RefCell<HashMap<~str, ValueRef>>,
|
2013-12-18 17:49:17 -08:00
|
|
|
lltypes: RefCell<HashMap<ty::t, Type>>,
|
2013-12-18 17:53:54 -08:00
|
|
|
llsizingtypes: RefCell<HashMap<ty::t, Type>>,
|
2013-12-18 17:58:24 -08:00
|
|
|
adt_reprs: RefCell<HashMap<ty::t, @adt::Repr>>,
|
2013-12-19 15:43:13 -08:00
|
|
|
symbol_hasher: RefCell<Sha256>,
|
2013-12-18 18:03:59 -08:00
|
|
|
type_hashcodes: RefCell<HashMap<ty::t, @str>>,
|
2013-12-18 18:07:16 -08:00
|
|
|
all_llvm_symbols: RefCell<HashSet<@str>>,
|
2013-06-13 14:02:33 +12:00
|
|
|
tcx: ty::ctxt,
|
|
|
|
maps: astencode::Maps,
|
2013-06-22 18:38:02 -04:00
|
|
|
stats: @mut Stats,
|
2013-06-15 22:16:47 +12:00
|
|
|
tydesc_type: Type,
|
|
|
|
int_type: Type,
|
|
|
|
opaque_vec_type: Type,
|
2013-06-13 14:02:33 +12:00
|
|
|
builder: BuilderRef_res,
|
|
|
|
crate_map: ValueRef,
|
Implement LTO
This commit implements LTO for rust leveraging LLVM's passes. What this means
is:
* When compiling an rlib, in addition to insdering foo.o into the archive, also
insert foo.bc (the LLVM bytecode) of the optimized module.
* When the compiler detects the -Z lto option, it will attempt to perform LTO on
a staticlib or binary output. The compiler will emit an error if a dylib or
rlib output is being generated.
* The actual act of performing LTO is as follows:
1. Force all upstream libraries to have an rlib version available.
2. Load the bytecode of each upstream library from the rlib.
3. Link all this bytecode into the current LLVM module (just using llvm
apis)
4. Run an internalization pass which internalizes all symbols except those
found reachable for the local crate of compilation.
5. Run the LLVM LTO pass manager over this entire module
6a. If assembling an archive, then add all upstream rlibs into the output
archive. This ignores all of the object/bitcode/metadata files rust
generated and placed inside the rlibs.
6b. If linking a binary, create copies of all upstream rlibs, remove the
rust-generated object-file, and then link everything as usual.
As I have explained in #10741, this process is excruciatingly slow, so this is
*not* turned on by default, and it is also why I have decided to hide it behind
a -Z flag for now. The good news is that the binary sizes are about as small as
they can be as a result of LTO, so it's definitely working.
Closes #10741
Closes #10740
2013-12-02 23:19:29 -08:00
|
|
|
crate_map_name: ~str,
|
2013-06-13 14:02:33 +12:00
|
|
|
// Set when at least one function uses GC. Needed so that
|
|
|
|
// decl_gc_metadata knows whether to link to the module metadata, which
|
|
|
|
// is not emitted by LLVM's GC pass when no functions use GC.
|
2013-06-13 19:19:50 +12:00
|
|
|
uses_gc: bool,
|
2013-08-29 11:44:11 +02:00
|
|
|
dbg_cx: Option<debuginfo::CrateDebugContext>,
|
2013-12-19 16:27:58 -08:00
|
|
|
do_not_commit_warning_issued: Cell<bool>,
|
2013-06-13 14:02:33 +12:00
|
|
|
}
|
2013-06-13 14:49:01 +12:00
|
|
|
|
|
|
|
impl CrateContext {
|
2013-06-18 09:39:16 -07:00
|
|
|
pub fn new(sess: session::Session,
|
|
|
|
name: &str,
|
|
|
|
tcx: ty::ctxt,
|
|
|
|
emap2: resolve::ExportMap2,
|
|
|
|
maps: astencode::Maps,
|
2013-12-09 14:56:53 -07:00
|
|
|
symbol_hasher: Sha256,
|
2013-06-18 09:39:16 -07:00
|
|
|
link_meta: LinkMeta,
|
2013-12-20 20:14:51 -08:00
|
|
|
reachable: @RefCell<HashSet<ast::NodeId>>)
|
2013-06-18 09:39:16 -07:00
|
|
|
-> CrateContext {
|
2013-06-13 14:49:01 +12:00
|
|
|
unsafe {
|
|
|
|
let llcx = llvm::LLVMContextCreate();
|
|
|
|
set_task_llcx(llcx);
|
2013-11-21 15:42:55 -08:00
|
|
|
let llmod = name.with_c_str(|buf| {
|
2013-08-03 17:13:14 -07:00
|
|
|
llvm::LLVMModuleCreateWithNameInContext(buf, llcx)
|
2013-11-21 15:42:55 -08:00
|
|
|
});
|
Store metadata separately in rlib files
Right now whenever an rlib file is linked against, all of the metadata from the
rlib is pulled in to the final staticlib or binary. The reason for this is that
the metadata is currently stored in a section of the object file. Note that this
is intentional for dynamic libraries in order to distribute metadata bundled
with static libraries.
This commit alters the situation for rlib libraries to instead store the
metadata in a separate file in the archive. In doing so, when the archive is
passed to the linker, none of the metadata will get pulled into the result
executable. Furthermore, the metadata file is skipped when assembling rlibs into
an archive.
The snag in this implementation comes with multiple output formats. When
generating a dylib, the metadata needs to be in the object file, but when
generating an rlib this needs to be separate. In order to accomplish this, the
metadata variable is inserted into an entirely separate LLVM Module which is
then codegen'd into a different location (foo.metadata.o). This is then linked
into dynamic libraries and silently ignored for rlib files.
While changing how metadata is inserted into archives, I have also stopped
compressing metadata when inserted into rlib files. We have wanted to stop
compressing metadata, but the sections it creates in object file sections are
apparently too large. Thankfully if it's just an arbitrary file it doesn't
matter how large it is.
I have seen massive reductions in executable sizes, as well as staticlib output
sizes (to confirm that this is all working).
2013-12-03 17:41:01 -08:00
|
|
|
let metadata_llmod = format!("{}_metadata", name).with_c_str(|buf| {
|
|
|
|
llvm::LLVMModuleCreateWithNameInContext(buf, llcx)
|
|
|
|
});
|
2013-06-13 14:49:01 +12:00
|
|
|
let data_layout: &str = sess.targ_cfg.target_strs.data_layout;
|
|
|
|
let targ_triple: &str = sess.targ_cfg.target_strs.target_triple;
|
Store metadata separately in rlib files
Right now whenever an rlib file is linked against, all of the metadata from the
rlib is pulled in to the final staticlib or binary. The reason for this is that
the metadata is currently stored in a section of the object file. Note that this
is intentional for dynamic libraries in order to distribute metadata bundled
with static libraries.
This commit alters the situation for rlib libraries to instead store the
metadata in a separate file in the archive. In doing so, when the archive is
passed to the linker, none of the metadata will get pulled into the result
executable. Furthermore, the metadata file is skipped when assembling rlibs into
an archive.
The snag in this implementation comes with multiple output formats. When
generating a dylib, the metadata needs to be in the object file, but when
generating an rlib this needs to be separate. In order to accomplish this, the
metadata variable is inserted into an entirely separate LLVM Module which is
then codegen'd into a different location (foo.metadata.o). This is then linked
into dynamic libraries and silently ignored for rlib files.
While changing how metadata is inserted into archives, I have also stopped
compressing metadata when inserted into rlib files. We have wanted to stop
compressing metadata, but the sections it creates in object file sections are
apparently too large. Thankfully if it's just an arbitrary file it doesn't
matter how large it is.
I have seen massive reductions in executable sizes, as well as staticlib output
sizes (to confirm that this is all working).
2013-12-03 17:41:01 -08:00
|
|
|
data_layout.with_c_str(|buf| {
|
|
|
|
llvm::LLVMSetDataLayout(llmod, buf);
|
|
|
|
llvm::LLVMSetDataLayout(metadata_llmod, buf);
|
|
|
|
});
|
2013-11-21 15:42:55 -08:00
|
|
|
targ_triple.with_c_str(|buf| {
|
Store metadata separately in rlib files
Right now whenever an rlib file is linked against, all of the metadata from the
rlib is pulled in to the final staticlib or binary. The reason for this is that
the metadata is currently stored in a section of the object file. Note that this
is intentional for dynamic libraries in order to distribute metadata bundled
with static libraries.
This commit alters the situation for rlib libraries to instead store the
metadata in a separate file in the archive. In doing so, when the archive is
passed to the linker, none of the metadata will get pulled into the result
executable. Furthermore, the metadata file is skipped when assembling rlibs into
an archive.
The snag in this implementation comes with multiple output formats. When
generating a dylib, the metadata needs to be in the object file, but when
generating an rlib this needs to be separate. In order to accomplish this, the
metadata variable is inserted into an entirely separate LLVM Module which is
then codegen'd into a different location (foo.metadata.o). This is then linked
into dynamic libraries and silently ignored for rlib files.
While changing how metadata is inserted into archives, I have also stopped
compressing metadata when inserted into rlib files. We have wanted to stop
compressing metadata, but the sections it creates in object file sections are
apparently too large. Thankfully if it's just an arbitrary file it doesn't
matter how large it is.
I have seen massive reductions in executable sizes, as well as staticlib output
sizes (to confirm that this is all working).
2013-12-03 17:41:01 -08:00
|
|
|
llvm::LLVMRustSetNormalizedTarget(llmod, buf);
|
|
|
|
llvm::LLVMRustSetNormalizedTarget(metadata_llmod, buf);
|
2013-11-21 15:42:55 -08:00
|
|
|
});
|
2013-06-13 14:49:01 +12:00
|
|
|
let targ_cfg = sess.targ_cfg;
|
2013-06-15 14:31:52 +12:00
|
|
|
|
2013-06-13 14:49:01 +12:00
|
|
|
let td = mk_target_data(sess.targ_cfg.target_strs.data_layout);
|
2013-12-19 16:22:26 -08:00
|
|
|
let tn = TypeNames::new();
|
2013-06-15 14:31:52 +12:00
|
|
|
|
2013-06-13 14:49:01 +12:00
|
|
|
let mut intrinsics = base::declare_intrinsics(llmod);
|
|
|
|
if sess.opts.extra_debuginfo {
|
|
|
|
base::declare_dbg_intrinsics(llmod, &mut intrinsics);
|
|
|
|
}
|
2013-06-15 22:16:47 +12:00
|
|
|
let int_type = Type::int(targ_cfg.arch);
|
|
|
|
let tydesc_type = Type::tydesc(targ_cfg.arch);
|
|
|
|
let opaque_vec_type = Type::opaque_vec(targ_cfg.arch);
|
|
|
|
|
2013-06-16 22:52:44 +12:00
|
|
|
let mut str_slice_ty = Type::named_struct("str_slice");
|
|
|
|
str_slice_ty.set_struct_body([Type::i8p(), int_type], false);
|
2013-06-16 15:45:48 +12:00
|
|
|
|
2013-06-15 22:16:47 +12:00
|
|
|
tn.associate_type("tydesc", &tydesc_type);
|
2013-06-16 15:45:48 +12:00
|
|
|
tn.associate_type("str_slice", &str_slice_ty);
|
2013-06-15 22:16:47 +12:00
|
|
|
|
2013-12-09 14:56:53 -07:00
|
|
|
let (crate_map_name, crate_map) = decl_crate_map(sess, link_meta.clone(), llmod);
|
2013-06-13 14:49:01 +12:00
|
|
|
let dbg_cx = if sess.opts.debuginfo {
|
2013-08-29 11:44:11 +02:00
|
|
|
Some(debuginfo::CrateDebugContext::new(llmod, name.to_owned()))
|
2013-06-13 14:49:01 +12:00
|
|
|
} else {
|
|
|
|
None
|
|
|
|
};
|
|
|
|
|
2013-06-17 16:23:24 +12:00
|
|
|
if sess.count_llvm_insns() {
|
|
|
|
base::init_insn_ctxt()
|
|
|
|
}
|
|
|
|
|
2013-06-13 14:49:01 +12:00
|
|
|
CrateContext {
|
|
|
|
sess: sess,
|
|
|
|
llmod: llmod,
|
|
|
|
llcx: llcx,
|
Store metadata separately in rlib files
Right now whenever an rlib file is linked against, all of the metadata from the
rlib is pulled in to the final staticlib or binary. The reason for this is that
the metadata is currently stored in a section of the object file. Note that this
is intentional for dynamic libraries in order to distribute metadata bundled
with static libraries.
This commit alters the situation for rlib libraries to instead store the
metadata in a separate file in the archive. In doing so, when the archive is
passed to the linker, none of the metadata will get pulled into the result
executable. Furthermore, the metadata file is skipped when assembling rlibs into
an archive.
The snag in this implementation comes with multiple output formats. When
generating a dylib, the metadata needs to be in the object file, but when
generating an rlib this needs to be separate. In order to accomplish this, the
metadata variable is inserted into an entirely separate LLVM Module which is
then codegen'd into a different location (foo.metadata.o). This is then linked
into dynamic libraries and silently ignored for rlib files.
While changing how metadata is inserted into archives, I have also stopped
compressing metadata when inserted into rlib files. We have wanted to stop
compressing metadata, but the sections it creates in object file sections are
apparently too large. Thankfully if it's just an arbitrary file it doesn't
matter how large it is.
I have seen massive reductions in executable sizes, as well as staticlib output
sizes (to confirm that this is all working).
2013-12-03 17:41:01 -08:00
|
|
|
metadata_llmod: metadata_llmod,
|
2013-06-13 14:49:01 +12:00
|
|
|
td: td,
|
|
|
|
tn: tn,
|
2013-12-18 18:24:34 -08:00
|
|
|
externs: RefCell::new(HashMap::new()),
|
2013-06-13 14:49:01 +12:00
|
|
|
intrinsics: intrinsics,
|
2013-12-18 16:41:15 -08:00
|
|
|
item_vals: RefCell::new(HashMap::new()),
|
2013-06-13 14:49:01 +12:00
|
|
|
exp_map2: emap2,
|
|
|
|
reachable: reachable,
|
2013-12-18 16:53:23 -08:00
|
|
|
item_symbols: RefCell::new(HashMap::new()),
|
2013-06-13 14:49:01 +12:00
|
|
|
link_meta: link_meta,
|
2013-12-18 18:15:27 -08:00
|
|
|
tydescs: RefCell::new(HashMap::new()),
|
2013-12-18 18:35:33 -08:00
|
|
|
finished_tydescs: Cell::new(false),
|
2013-12-18 18:32:52 -08:00
|
|
|
external: RefCell::new(HashMap::new()),
|
|
|
|
external_srcs: RefCell::new(HashMap::new()),
|
2013-12-18 18:11:41 -08:00
|
|
|
non_inlineable_statics: RefCell::new(HashSet::new()),
|
2013-12-18 16:35:29 -08:00
|
|
|
monomorphized: RefCell::new(HashMap::new()),
|
2013-12-18 16:38:25 -08:00
|
|
|
monomorphizing: RefCell::new(HashMap::new()),
|
2013-12-18 17:00:56 -08:00
|
|
|
vtables: RefCell::new(HashMap::new()),
|
2013-12-18 17:02:30 -08:00
|
|
|
const_cstr_cache: RefCell::new(HashMap::new()),
|
2013-12-18 17:03:55 -08:00
|
|
|
const_globals: RefCell::new(HashMap::new()),
|
2013-12-18 17:08:56 -08:00
|
|
|
const_values: RefCell::new(HashMap::new()),
|
2013-12-18 17:11:22 -08:00
|
|
|
extern_const_values: RefCell::new(HashMap::new()),
|
2013-12-18 17:14:25 -08:00
|
|
|
impl_method_cache: RefCell::new(HashMap::new()),
|
2013-12-18 17:44:25 -08:00
|
|
|
module_data: RefCell::new(HashMap::new()),
|
2013-12-18 17:49:17 -08:00
|
|
|
lltypes: RefCell::new(HashMap::new()),
|
2013-12-18 17:53:54 -08:00
|
|
|
llsizingtypes: RefCell::new(HashMap::new()),
|
2013-12-18 17:58:24 -08:00
|
|
|
adt_reprs: RefCell::new(HashMap::new()),
|
2013-12-19 15:43:13 -08:00
|
|
|
symbol_hasher: RefCell::new(symbol_hasher),
|
2013-12-18 18:03:59 -08:00
|
|
|
type_hashcodes: RefCell::new(HashMap::new()),
|
2013-12-18 18:07:16 -08:00
|
|
|
all_llvm_symbols: RefCell::new(HashSet::new()),
|
2013-06-13 14:49:01 +12:00
|
|
|
tcx: tcx,
|
|
|
|
maps: maps,
|
2013-06-22 18:38:02 -04:00
|
|
|
stats: @mut Stats {
|
2013-12-22 12:31:10 -08:00
|
|
|
n_static_tydescs: Cell::new(0u),
|
2013-12-22 12:32:37 -08:00
|
|
|
n_glues_created: Cell::new(0u),
|
2013-12-22 13:39:02 -08:00
|
|
|
n_null_glues: Cell::new(0u),
|
2013-12-22 13:41:16 -08:00
|
|
|
n_real_glues: Cell::new(0u),
|
2013-12-22 13:44:42 -08:00
|
|
|
n_fns: Cell::new(0u),
|
2013-12-22 13:46:59 -08:00
|
|
|
n_monos: Cell::new(0u),
|
2013-12-22 13:48:37 -08:00
|
|
|
n_inlines: Cell::new(0u),
|
2013-12-22 13:50:04 -08:00
|
|
|
n_closures: Cell::new(0u),
|
2013-12-22 13:52:05 -08:00
|
|
|
n_llvm_insns: Cell::new(0u),
|
2013-12-22 13:56:22 -08:00
|
|
|
llvm_insns: RefCell::new(HashMap::new()),
|
2013-06-28 11:15:34 -07:00
|
|
|
fn_stats: ~[]
|
2013-06-13 14:49:01 +12:00
|
|
|
},
|
|
|
|
tydesc_type: tydesc_type,
|
|
|
|
int_type: int_type,
|
2013-06-15 22:16:47 +12:00
|
|
|
opaque_vec_type: opaque_vec_type,
|
2013-06-20 17:52:02 +12:00
|
|
|
builder: BuilderRef_res(llvm::LLVMCreateBuilderInContext(llcx)),
|
2013-06-13 14:49:01 +12:00
|
|
|
crate_map: crate_map,
|
Implement LTO
This commit implements LTO for rust leveraging LLVM's passes. What this means
is:
* When compiling an rlib, in addition to insdering foo.o into the archive, also
insert foo.bc (the LLVM bytecode) of the optimized module.
* When the compiler detects the -Z lto option, it will attempt to perform LTO on
a staticlib or binary output. The compiler will emit an error if a dylib or
rlib output is being generated.
* The actual act of performing LTO is as follows:
1. Force all upstream libraries to have an rlib version available.
2. Load the bytecode of each upstream library from the rlib.
3. Link all this bytecode into the current LLVM module (just using llvm
apis)
4. Run an internalization pass which internalizes all symbols except those
found reachable for the local crate of compilation.
5. Run the LLVM LTO pass manager over this entire module
6a. If assembling an archive, then add all upstream rlibs into the output
archive. This ignores all of the object/bitcode/metadata files rust
generated and placed inside the rlibs.
6b. If linking a binary, create copies of all upstream rlibs, remove the
rust-generated object-file, and then link everything as usual.
As I have explained in #10741, this process is excruciatingly slow, so this is
*not* turned on by default, and it is also why I have decided to hide it behind
a -Z flag for now. The good news is that the binary sizes are about as small as
they can be as a result of LTO, so it's definitely working.
Closes #10741
Closes #10740
2013-12-02 23:19:29 -08:00
|
|
|
crate_map_name: crate_map_name,
|
2013-06-13 19:19:50 +12:00
|
|
|
uses_gc: false,
|
2013-06-13 14:49:01 +12:00
|
|
|
dbg_cx: dbg_cx,
|
2013-12-19 16:27:58 -08:00
|
|
|
do_not_commit_warning_issued: Cell::new(false),
|
2013-06-13 14:49:01 +12:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2013-07-21 15:33:40 +02:00
|
|
|
|
2013-12-19 16:47:15 -08:00
|
|
|
pub fn builder(@self) -> Builder {
|
2013-07-21 15:33:40 +02:00
|
|
|
Builder::new(self)
|
|
|
|
}
|
2013-08-11 13:29:14 -04:00
|
|
|
|
|
|
|
pub fn const_inbounds_gepi(&self,
|
|
|
|
pointer: ValueRef,
|
|
|
|
indices: &[uint]) -> ValueRef {
|
2013-10-21 13:08:31 -07:00
|
|
|
debug!("const_inbounds_gepi: pointer={} indices={:?}",
|
2013-08-11 13:29:14 -04:00
|
|
|
self.tn.val_to_str(pointer), indices);
|
|
|
|
let v: ~[ValueRef] =
|
2013-08-11 14:56:43 -04:00
|
|
|
indices.iter().map(|i| C_i32(*i as i32)).collect();
|
2013-08-11 13:29:14 -04:00
|
|
|
unsafe {
|
|
|
|
llvm::LLVMConstInBoundsGEP(pointer,
|
2013-12-15 23:35:12 +11:00
|
|
|
v.as_ptr(),
|
2013-08-11 13:29:14 -04:00
|
|
|
indices.len() as c_uint)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn offsetof_gep(&self,
|
|
|
|
llptr_ty: Type,
|
|
|
|
indices: &[uint]) -> ValueRef {
|
|
|
|
/*!
|
|
|
|
* Returns the offset of applying the given GEP indices
|
|
|
|
* to an instance of `llptr_ty`. Similar to `offsetof` in C,
|
|
|
|
* except that `llptr_ty` must be a pointer type.
|
|
|
|
*/
|
|
|
|
|
|
|
|
unsafe {
|
|
|
|
let null = C_null(llptr_ty);
|
|
|
|
llvm::LLVMConstPtrToInt(self.const_inbounds_gepi(null, indices),
|
|
|
|
self.int_type.to_ref())
|
|
|
|
}
|
|
|
|
}
|
2013-06-13 14:49:01 +12:00
|
|
|
}
|
|
|
|
|
|
|
|
#[unsafe_destructor]
|
|
|
|
impl Drop for CrateContext {
|
2013-09-16 21:18:07 -04:00
|
|
|
fn drop(&mut self) {
|
2013-07-12 01:38:44 -07:00
|
|
|
unset_task_llcx();
|
2013-06-13 14:49:01 +12:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-09-16 23:34:40 -07:00
|
|
|
local_data_key!(task_local_llcx_key: @ContextRef)
|
2013-07-12 01:38:44 -07:00
|
|
|
|
2013-06-13 14:49:01 +12:00
|
|
|
pub fn task_llcx() -> ContextRef {
|
2013-09-20 02:08:47 -04:00
|
|
|
let opt = local_data::get(task_local_llcx_key, |k| k.map(|k| *k));
|
2013-06-13 14:49:01 +12:00
|
|
|
*opt.expect("task-local LLVMContextRef wasn't ever set!")
|
|
|
|
}
|
|
|
|
|
2013-07-12 01:38:44 -07:00
|
|
|
fn set_task_llcx(c: ContextRef) {
|
2013-07-09 17:25:28 -07:00
|
|
|
local_data::set(task_local_llcx_key, @c);
|
2013-06-13 14:49:01 +12:00
|
|
|
}
|
|
|
|
|
2013-07-12 01:38:44 -07:00
|
|
|
fn unset_task_llcx() {
|
2013-07-09 17:25:28 -07:00
|
|
|
local_data::pop(task_local_llcx_key);
|
2013-06-13 14:49:01 +12:00
|
|
|
}
|