2014-01-17 17:44:44 -06:00
|
|
|
// Copyright 2013-2014 The Rust Project Developers. See the COPYRIGHT
|
Add generation of static libraries to rustc
This commit implements the support necessary for generating both intermediate
and result static rust libraries. This is an implementation of my thoughts in
https://mail.mozilla.org/pipermail/rust-dev/2013-November/006686.html.
When compiling a library, we still retain the "lib" option, although now there
are "rlib", "staticlib", and "dylib" as options for crate_type (and these are
stackable). The idea of "lib" is to generate the "compiler default" instead of
having too choose (although all are interchangeable). For now I have left the
"complier default" to be a dynamic library for size reasons.
Of the rust libraries, lib{std,extra,rustuv} will bootstrap with an
rlib/dylib pair, but lib{rustc,syntax,rustdoc,rustpkg} will only be built as a
dynamic object. I chose this for size reasons, but also because you're probably
not going to be embedding the rustc compiler anywhere any time soon.
Other than the options outlined above, there are a few defaults/preferences that
are now opinionated in the compiler:
* If both a .dylib and .rlib are found for a rust library, the compiler will
prefer the .rlib variant. This is overridable via the -Z prefer-dynamic option
* If generating a "lib", the compiler will generate a dynamic library. This is
overridable by explicitly saying what flavor you'd like (rlib, staticlib,
dylib).
* If no options are passed to the command line, and no crate_type is found in
the destination crate, then an executable is generated
With this change, you can successfully build a rust program with 0 dynamic
dependencies on rust libraries. There is still a dynamic dependency on
librustrt, but I plan on removing that in a subsequent commit.
This change includes no tests just yet. Our current testing
infrastructure/harnesses aren't very amenable to doing flavorful things with
linking, so I'm planning on adding a new mode of testing which I believe belongs
as a separate commit.
Closes #552
2013-11-15 16:03:29 -06:00
|
|
|
// 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.
|
|
|
|
|
|
|
|
//! A helper class for dealing with static archives
|
|
|
|
|
2015-02-26 23:00:43 -06:00
|
|
|
use std::env;
|
2015-04-14 18:28:50 -05:00
|
|
|
use std::fs::{self, File};
|
2015-02-26 23:00:43 -06:00
|
|
|
use std::io::prelude::*;
|
|
|
|
use std::io;
|
|
|
|
use std::path::{Path, PathBuf};
|
|
|
|
use std::process::{Command, Output, Stdio};
|
core: Remove the cast module
This commit revisits the `cast` module in libcore and libstd, and scrutinizes
all functions inside of it. The result was to remove the `cast` module entirely,
folding all functionality into the `mem` module. Specifically, this is the fate
of each function in the `cast` module.
* transmute - This function was moved to `mem`, but it is now marked as
#[unstable]. This is due to planned changes to the `transmute`
function and how it can be invoked (see the #[unstable] comment).
For more information, see RFC 5 and #12898
* transmute_copy - This function was moved to `mem`, with clarification that is
is not an error to invoke it with T/U that are different
sizes, but rather that it is strongly discouraged. This
function is now #[stable]
* forget - This function was moved to `mem` and marked #[stable]
* bump_box_refcount - This function was removed due to the deprecation of
managed boxes as well as its questionable utility.
* transmute_mut - This function was previously deprecated, and removed as part
of this commit.
* transmute_mut_unsafe - This function doesn't serve much of a purpose when it
can be achieved with an `as` in safe code, so it was
removed.
* transmute_lifetime - This function was removed because it is likely a strong
indication that code is incorrect in the first place.
* transmute_mut_lifetime - This function was removed for the same reasons as
`transmute_lifetime`
* copy_lifetime - This function was moved to `mem`, but it is marked
`#[unstable]` now due to the likelihood of being removed in
the future if it is found to not be very useful.
* copy_mut_lifetime - This function was also moved to `mem`, but had the same
treatment as `copy_lifetime`.
* copy_lifetime_vec - This function was removed because it is not used today,
and its existence is not necessary with DST
(copy_lifetime will suffice).
In summary, the cast module was stripped down to these functions, and then the
functions were moved to the `mem` module.
transmute - #[unstable]
transmute_copy - #[stable]
forget - #[stable]
copy_lifetime - #[unstable]
copy_mut_lifetime - #[unstable]
[breaking-change]
2014-05-09 12:34:51 -05:00
|
|
|
use std::str;
|
2014-08-18 10:29:44 -05:00
|
|
|
use syntax::diagnostic::Handler as ErrorHandler;
|
2015-04-14 18:28:50 -05:00
|
|
|
use rustc_llvm::archive_ro::ArchiveRO;
|
Add generation of static libraries to rustc
This commit implements the support necessary for generating both intermediate
and result static rust libraries. This is an implementation of my thoughts in
https://mail.mozilla.org/pipermail/rust-dev/2013-November/006686.html.
When compiling a library, we still retain the "lib" option, although now there
are "rlib", "staticlib", and "dylib" as options for crate_type (and these are
stackable). The idea of "lib" is to generate the "compiler default" instead of
having too choose (although all are interchangeable). For now I have left the
"complier default" to be a dynamic library for size reasons.
Of the rust libraries, lib{std,extra,rustuv} will bootstrap with an
rlib/dylib pair, but lib{rustc,syntax,rustdoc,rustpkg} will only be built as a
dynamic object. I chose this for size reasons, but also because you're probably
not going to be embedding the rustc compiler anywhere any time soon.
Other than the options outlined above, there are a few defaults/preferences that
are now opinionated in the compiler:
* If both a .dylib and .rlib are found for a rust library, the compiler will
prefer the .rlib variant. This is overridable via the -Z prefer-dynamic option
* If generating a "lib", the compiler will generate a dynamic library. This is
overridable by explicitly saying what flavor you'd like (rlib, staticlib,
dylib).
* If no options are passed to the command line, and no crate_type is found in
the destination crate, then an executable is generated
With this change, you can successfully build a rust program with 0 dynamic
dependencies on rust libraries. There is still a dynamic dependency on
librustrt, but I plan on removing that in a subsequent commit.
This change includes no tests just yet. Our current testing
infrastructure/harnesses aren't very amenable to doing flavorful things with
linking, so I'm planning on adding a new mode of testing which I believe belongs
as a separate commit.
Closes #552
2013-11-15 16:03:29 -06:00
|
|
|
|
std: Stabilize the `fs` module
This commit performs a stabilization pass over the `std::fs` module now that
it's had some time to bake. The change was largely just adding `#[stable]` tags,
but there are a few APIs that remain `#[unstable]`.
The following apis are now marked `#[stable]`:
* `std::fs` (the name)
* `File`
* `Metadata`
* `ReadDir`
* `DirEntry`
* `OpenOptions`
* `Permissions`
* `File::{open, create}`
* `File::{sync_all, sync_data}`
* `File::set_len`
* `File::metadata`
* Trait implementations for `File` and `&File`
* `OpenOptions::new`
* `OpenOptions::{read, write, append, truncate, create}`
* `OpenOptions::open` - this function was modified, however, to not attempt to
reject cross-platform openings of directories. This means that some platforms
will succeed in opening a directory and others will fail.
* `Metadata::{is_dir, is_file, len, permissions}`
* `Permissions::{readonly, set_readonly}`
* `Iterator for ReadDir`
* `DirEntry::path`
* `remove_file` - like with `OpenOptions::open`, the extra windows code to
remove a readonly file has been removed. This means that removing a readonly
file will succeed on some platforms but fail on others.
* `metadata`
* `rename`
* `copy`
* `hard_link`
* `soft_link`
* `read_link`
* `create_dir`
* `create_dir_all`
* `remove_dir`
* `remove_dir_all`
* `read_dir`
The following apis remain `#[unstable]`.
* `WalkDir` and `walk` - there are many methods by which a directory walk can be
constructed, and it's unclear whether the current semantics are the right
ones. For example symlinks are not handled super well currently. This is now
behind a new `fs_walk` feature.
* `File::path` - this is an extra abstraction which the standard library
provides on top of what the system offers and it's unclear whether we should
be doing so. This is now behind a new `file_path` feature.
* `Metadata::{accessed, modified}` - we do not currently have a good
abstraction for a moment in time which is what these APIs should likely be
returning, so these remain `#[unstable]` for now. These are now behind a new
`fs_time` feature
* `set_file_times` - like with `Metadata::accessed`, we do not currently have
the appropriate abstraction for the arguments here so this API remains
unstable behind the `fs_time` feature gate.
* `PathExt` - the precise set of methods on this trait may change over time and
some methods may be removed. This API remains unstable behind the `path_ext`
feature gate.
* `set_permissions` - we may wish to expose a more granular ability to set the
permissions on a file instead of just a blanket "set all permissions" method.
This function remains behind the `fs` feature.
The following apis are now `#[deprecated]`
* The `TempDir` type is now entirely deprecated and is [located on
crates.io][tempdir] as the `tempdir` crate with [its source][github] at
rust-lang/tempdir.
[tempdir]: https://crates.io/crates/tempdir
[github]: https://github.com/rust-lang/tempdir
The stability of some of these APIs has been questioned over the past few weeks
in using these APIs, and it is intentional that the majority of APIs here are
marked `#[stable]`. The `std::fs` module has a lot of room to grow and the
material is [being tracked in a RFC issue][rfc-issue].
[rfc-issue]: https://github.com/rust-lang/rfcs/issues/939
[breaking-change]
2015-03-03 21:18:29 -06:00
|
|
|
use tempdir::TempDir;
|
|
|
|
|
2015-02-27 08:36:53 -06:00
|
|
|
pub const METADATA_FILENAME: &'static str = "rust.metadata.bin";
|
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 19:41:01 -06:00
|
|
|
|
2014-07-06 22:46:00 -05:00
|
|
|
pub struct ArchiveConfig<'a> {
|
|
|
|
pub handler: &'a ErrorHandler,
|
2015-02-26 23:00:43 -06:00
|
|
|
pub dst: PathBuf,
|
|
|
|
pub lib_search_paths: Vec<PathBuf>,
|
2014-07-23 13:56:36 -05:00
|
|
|
pub slib_prefix: String,
|
|
|
|
pub slib_suffix: String,
|
2015-05-11 16:32:53 -05:00
|
|
|
pub ar_prog: String
|
2014-07-06 22:46:00 -05:00
|
|
|
}
|
|
|
|
|
2014-03-05 08:36:01 -06:00
|
|
|
pub struct Archive<'a> {
|
2015-05-11 16:32:53 -05:00
|
|
|
config: ArchiveConfig<'a>,
|
Add generation of static libraries to rustc
This commit implements the support necessary for generating both intermediate
and result static rust libraries. This is an implementation of my thoughts in
https://mail.mozilla.org/pipermail/rust-dev/2013-November/006686.html.
When compiling a library, we still retain the "lib" option, although now there
are "rlib", "staticlib", and "dylib" as options for crate_type (and these are
stackable). The idea of "lib" is to generate the "compiler default" instead of
having too choose (although all are interchangeable). For now I have left the
"complier default" to be a dynamic library for size reasons.
Of the rust libraries, lib{std,extra,rustuv} will bootstrap with an
rlib/dylib pair, but lib{rustc,syntax,rustdoc,rustpkg} will only be built as a
dynamic object. I chose this for size reasons, but also because you're probably
not going to be embedding the rustc compiler anywhere any time soon.
Other than the options outlined above, there are a few defaults/preferences that
are now opinionated in the compiler:
* If both a .dylib and .rlib are found for a rust library, the compiler will
prefer the .rlib variant. This is overridable via the -Z prefer-dynamic option
* If generating a "lib", the compiler will generate a dynamic library. This is
overridable by explicitly saying what flavor you'd like (rlib, staticlib,
dylib).
* If no options are passed to the command line, and no crate_type is found in
the destination crate, then an executable is generated
With this change, you can successfully build a rust program with 0 dynamic
dependencies on rust libraries. There is still a dynamic dependency on
librustrt, but I plan on removing that in a subsequent commit.
This change includes no tests just yet. Our current testing
infrastructure/harnesses aren't very amenable to doing flavorful things with
linking, so I'm planning on adding a new mode of testing which I believe belongs
as a separate commit.
Closes #552
2013-11-15 16:03:29 -06:00
|
|
|
}
|
|
|
|
|
2014-07-03 18:03:26 -05:00
|
|
|
/// Helper for adding many files to an archive with a single invocation of
|
|
|
|
/// `ar`.
|
|
|
|
#[must_use = "must call build() to finish building the archive"]
|
|
|
|
pub struct ArchiveBuilder<'a> {
|
|
|
|
archive: Archive<'a>,
|
|
|
|
work_dir: TempDir,
|
|
|
|
/// Filename of each member that should be added to the archive.
|
2015-02-26 23:00:43 -06:00
|
|
|
members: Vec<PathBuf>,
|
2014-07-03 18:03:26 -05:00
|
|
|
should_update_symbols: bool,
|
|
|
|
}
|
|
|
|
|
2015-05-11 16:32:53 -05:00
|
|
|
enum Action<'a> {
|
|
|
|
Remove(&'a Path),
|
|
|
|
AddObjects(&'a [&'a PathBuf], bool),
|
|
|
|
UpdateSymbols,
|
Add generation of static libraries to rustc
This commit implements the support necessary for generating both intermediate
and result static rust libraries. This is an implementation of my thoughts in
https://mail.mozilla.org/pipermail/rust-dev/2013-November/006686.html.
When compiling a library, we still retain the "lib" option, although now there
are "rlib", "staticlib", and "dylib" as options for crate_type (and these are
stackable). The idea of "lib" is to generate the "compiler default" instead of
having too choose (although all are interchangeable). For now I have left the
"complier default" to be a dynamic library for size reasons.
Of the rust libraries, lib{std,extra,rustuv} will bootstrap with an
rlib/dylib pair, but lib{rustc,syntax,rustdoc,rustpkg} will only be built as a
dynamic object. I chose this for size reasons, but also because you're probably
not going to be embedding the rustc compiler anywhere any time soon.
Other than the options outlined above, there are a few defaults/preferences that
are now opinionated in the compiler:
* If both a .dylib and .rlib are found for a rust library, the compiler will
prefer the .rlib variant. This is overridable via the -Z prefer-dynamic option
* If generating a "lib", the compiler will generate a dynamic library. This is
overridable by explicitly saying what flavor you'd like (rlib, staticlib,
dylib).
* If no options are passed to the command line, and no crate_type is found in
the destination crate, then an executable is generated
With this change, you can successfully build a rust program with 0 dynamic
dependencies on rust libraries. There is still a dynamic dependency on
librustrt, but I plan on removing that in a subsequent commit.
This change includes no tests just yet. Our current testing
infrastructure/harnesses aren't very amenable to doing flavorful things with
linking, so I'm planning on adding a new mode of testing which I believe belongs
as a separate commit.
Closes #552
2013-11-15 16:03:29 -06:00
|
|
|
}
|
|
|
|
|
2014-07-23 13:56:36 -05:00
|
|
|
pub fn find_library(name: &str, osprefix: &str, ossuffix: &str,
|
2015-02-26 23:00:43 -06:00
|
|
|
search_paths: &[PathBuf],
|
|
|
|
handler: &ErrorHandler) -> PathBuf {
|
2014-07-30 09:44:20 -05:00
|
|
|
// On Windows, static libraries sometimes show up as libfoo.a and other
|
|
|
|
// times show up as foo.lib
|
2014-07-23 13:56:36 -05:00
|
|
|
let oslibname = format!("{}{}{}", osprefix, name, ossuffix);
|
2014-07-30 09:44:20 -05:00
|
|
|
let unixlibname = format!("lib{}.a", name);
|
|
|
|
|
2015-01-31 11:20:46 -06:00
|
|
|
for path in search_paths {
|
2015-02-26 23:00:43 -06:00
|
|
|
debug!("looking for {} inside {:?}", name, path);
|
2015-02-18 13:48:57 -06:00
|
|
|
let test = path.join(&oslibname[..]);
|
2014-07-30 09:44:20 -05:00
|
|
|
if test.exists() { return test }
|
|
|
|
if oslibname != unixlibname {
|
2015-02-18 13:48:57 -06:00
|
|
|
let test = path.join(&unixlibname[..]);
|
2014-07-30 09:44:20 -05:00
|
|
|
if test.exists() { return test }
|
|
|
|
}
|
|
|
|
}
|
2015-01-07 10:58:31 -06:00
|
|
|
handler.fatal(&format!("could not find native static library `{}`, \
|
2014-07-30 09:44:20 -05:00
|
|
|
perhaps an -L flag is missing?",
|
2015-02-20 13:08:14 -06:00
|
|
|
name));
|
2014-07-30 09:44:20 -05:00
|
|
|
}
|
|
|
|
|
2014-03-05 08:36:01 -06:00
|
|
|
impl<'a> Archive<'a> {
|
2014-07-03 18:03:26 -05:00
|
|
|
fn new(config: ArchiveConfig<'a>) -> Archive<'a> {
|
2015-05-11 16:32:53 -05:00
|
|
|
Archive { config: config }
|
Add generation of static libraries to rustc
This commit implements the support necessary for generating both intermediate
and result static rust libraries. This is an implementation of my thoughts in
https://mail.mozilla.org/pipermail/rust-dev/2013-November/006686.html.
When compiling a library, we still retain the "lib" option, although now there
are "rlib", "staticlib", and "dylib" as options for crate_type (and these are
stackable). The idea of "lib" is to generate the "compiler default" instead of
having too choose (although all are interchangeable). For now I have left the
"complier default" to be a dynamic library for size reasons.
Of the rust libraries, lib{std,extra,rustuv} will bootstrap with an
rlib/dylib pair, but lib{rustc,syntax,rustdoc,rustpkg} will only be built as a
dynamic object. I chose this for size reasons, but also because you're probably
not going to be embedding the rustc compiler anywhere any time soon.
Other than the options outlined above, there are a few defaults/preferences that
are now opinionated in the compiler:
* If both a .dylib and .rlib are found for a rust library, the compiler will
prefer the .rlib variant. This is overridable via the -Z prefer-dynamic option
* If generating a "lib", the compiler will generate a dynamic library. This is
overridable by explicitly saying what flavor you'd like (rlib, staticlib,
dylib).
* If no options are passed to the command line, and no crate_type is found in
the destination crate, then an executable is generated
With this change, you can successfully build a rust program with 0 dynamic
dependencies on rust libraries. There is still a dynamic dependency on
librustrt, but I plan on removing that in a subsequent commit.
This change includes no tests just yet. Our current testing
infrastructure/harnesses aren't very amenable to doing flavorful things with
linking, so I'm planning on adding a new mode of testing which I believe belongs
as a separate commit.
Closes #552
2013-11-15 16:03:29 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Opens an existing static archive
|
2014-07-06 22:46:00 -05:00
|
|
|
pub fn open(config: ArchiveConfig<'a>) -> Archive<'a> {
|
2014-07-03 18:03:26 -05:00
|
|
|
let archive = Archive::new(config);
|
2015-05-11 16:32:53 -05:00
|
|
|
assert!(archive.config.dst.exists());
|
2014-07-03 18:03:26 -05:00
|
|
|
archive
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Removes a file from this archive
|
|
|
|
pub fn remove_file(&mut self, file: &str) {
|
2015-05-11 16:32:53 -05:00
|
|
|
self.run(None, Action::Remove(Path::new(file)));
|
2014-07-03 18:03:26 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Lists all files in an archive
|
|
|
|
pub fn files(&self) -> Vec<String> {
|
2015-05-11 16:32:53 -05:00
|
|
|
let archive = match ArchiveRO::open(&self.config.dst) {
|
|
|
|
Some(ar) => ar,
|
|
|
|
None => return Vec::new(),
|
|
|
|
};
|
|
|
|
let ret = archive.iter().filter_map(|child| child.name())
|
|
|
|
.map(|name| name.to_string())
|
|
|
|
.collect();
|
|
|
|
return ret;
|
2014-07-03 18:03:26 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Creates an `ArchiveBuilder` for adding files to this archive.
|
|
|
|
pub fn extend(self) -> ArchiveBuilder<'a> {
|
|
|
|
ArchiveBuilder::new(self)
|
|
|
|
}
|
2015-05-11 16:32:53 -05:00
|
|
|
|
|
|
|
fn run(&self, cwd: Option<&Path>, action: Action) -> Output {
|
|
|
|
let abs_dst = env::current_dir().unwrap().join(&self.config.dst);
|
|
|
|
let ar = &self.config.ar_prog;
|
|
|
|
let mut cmd = Command::new(ar);
|
|
|
|
cmd.stdout(Stdio::piped()).stderr(Stdio::piped());
|
|
|
|
self.prepare_ar_action(&mut cmd, &abs_dst, action);
|
|
|
|
info!("{:?}", cmd);
|
|
|
|
|
|
|
|
if let Some(p) = cwd {
|
|
|
|
cmd.current_dir(p);
|
|
|
|
info!("inside {:?}", p.display());
|
|
|
|
}
|
|
|
|
|
|
|
|
let handler = &self.config.handler;
|
|
|
|
match cmd.spawn() {
|
|
|
|
Ok(prog) => {
|
|
|
|
let o = prog.wait_with_output().unwrap();
|
|
|
|
if !o.status.success() {
|
|
|
|
handler.err(&format!("{:?} failed with: {}", cmd, o.status));
|
|
|
|
handler.note(&format!("stdout ---\n{}",
|
|
|
|
str::from_utf8(&o.stdout).unwrap()));
|
|
|
|
handler.note(&format!("stderr ---\n{}",
|
|
|
|
str::from_utf8(&o.stderr).unwrap()));
|
|
|
|
handler.abort_if_errors();
|
|
|
|
}
|
|
|
|
o
|
|
|
|
},
|
|
|
|
Err(e) => {
|
|
|
|
handler.err(&format!("could not exec `{}`: {}",
|
|
|
|
self.config.ar_prog, e));
|
|
|
|
handler.abort_if_errors();
|
|
|
|
panic!("rustc::back::archive::run() should not reach this point");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn prepare_ar_action(&self, cmd: &mut Command, dst: &Path, action: Action) {
|
|
|
|
match action {
|
|
|
|
Action::Remove(file) => {
|
|
|
|
cmd.arg("d").arg(dst).arg(file);
|
|
|
|
}
|
|
|
|
Action::AddObjects(objs, update_symbols) => {
|
|
|
|
cmd.arg(if update_symbols {"crus"} else {"cruS"})
|
|
|
|
.arg(dst)
|
|
|
|
.args(objs);
|
|
|
|
}
|
|
|
|
Action::UpdateSymbols => {
|
|
|
|
cmd.arg("s").arg(dst);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2014-07-03 18:03:26 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a> ArchiveBuilder<'a> {
|
|
|
|
fn new(archive: Archive<'a>) -> ArchiveBuilder<'a> {
|
|
|
|
ArchiveBuilder {
|
|
|
|
archive: archive,
|
|
|
|
work_dir: TempDir::new("rsar").unwrap(),
|
|
|
|
members: vec![],
|
|
|
|
should_update_symbols: false,
|
2014-07-06 22:46:00 -05:00
|
|
|
}
|
Add generation of static libraries to rustc
This commit implements the support necessary for generating both intermediate
and result static rust libraries. This is an implementation of my thoughts in
https://mail.mozilla.org/pipermail/rust-dev/2013-November/006686.html.
When compiling a library, we still retain the "lib" option, although now there
are "rlib", "staticlib", and "dylib" as options for crate_type (and these are
stackable). The idea of "lib" is to generate the "compiler default" instead of
having too choose (although all are interchangeable). For now I have left the
"complier default" to be a dynamic library for size reasons.
Of the rust libraries, lib{std,extra,rustuv} will bootstrap with an
rlib/dylib pair, but lib{rustc,syntax,rustdoc,rustpkg} will only be built as a
dynamic object. I chose this for size reasons, but also because you're probably
not going to be embedding the rustc compiler anywhere any time soon.
Other than the options outlined above, there are a few defaults/preferences that
are now opinionated in the compiler:
* If both a .dylib and .rlib are found for a rust library, the compiler will
prefer the .rlib variant. This is overridable via the -Z prefer-dynamic option
* If generating a "lib", the compiler will generate a dynamic library. This is
overridable by explicitly saying what flavor you'd like (rlib, staticlib,
dylib).
* If no options are passed to the command line, and no crate_type is found in
the destination crate, then an executable is generated
With this change, you can successfully build a rust program with 0 dynamic
dependencies on rust libraries. There is still a dynamic dependency on
librustrt, but I plan on removing that in a subsequent commit.
This change includes no tests just yet. Our current testing
infrastructure/harnesses aren't very amenable to doing flavorful things with
linking, so I'm planning on adding a new mode of testing which I believe belongs
as a separate commit.
Closes #552
2013-11-15 16:03:29 -06:00
|
|
|
}
|
|
|
|
|
2014-07-03 18:03:26 -05:00
|
|
|
/// Create a new static archive, ready for adding files.
|
|
|
|
pub fn create(config: ArchiveConfig<'a>) -> ArchiveBuilder<'a> {
|
|
|
|
let archive = Archive::new(config);
|
|
|
|
ArchiveBuilder::new(archive)
|
|
|
|
}
|
|
|
|
|
Add generation of static libraries to rustc
This commit implements the support necessary for generating both intermediate
and result static rust libraries. This is an implementation of my thoughts in
https://mail.mozilla.org/pipermail/rust-dev/2013-November/006686.html.
When compiling a library, we still retain the "lib" option, although now there
are "rlib", "staticlib", and "dylib" as options for crate_type (and these are
stackable). The idea of "lib" is to generate the "compiler default" instead of
having too choose (although all are interchangeable). For now I have left the
"complier default" to be a dynamic library for size reasons.
Of the rust libraries, lib{std,extra,rustuv} will bootstrap with an
rlib/dylib pair, but lib{rustc,syntax,rustdoc,rustpkg} will only be built as a
dynamic object. I chose this for size reasons, but also because you're probably
not going to be embedding the rustc compiler anywhere any time soon.
Other than the options outlined above, there are a few defaults/preferences that
are now opinionated in the compiler:
* If both a .dylib and .rlib are found for a rust library, the compiler will
prefer the .rlib variant. This is overridable via the -Z prefer-dynamic option
* If generating a "lib", the compiler will generate a dynamic library. This is
overridable by explicitly saying what flavor you'd like (rlib, staticlib,
dylib).
* If no options are passed to the command line, and no crate_type is found in
the destination crate, then an executable is generated
With this change, you can successfully build a rust program with 0 dynamic
dependencies on rust libraries. There is still a dynamic dependency on
librustrt, but I plan on removing that in a subsequent commit.
This change includes no tests just yet. Our current testing
infrastructure/harnesses aren't very amenable to doing flavorful things with
linking, so I'm planning on adding a new mode of testing which I believe belongs
as a separate commit.
Closes #552
2013-11-15 16:03:29 -06:00
|
|
|
/// Adds all of the contents of a native library to this archive. This will
|
|
|
|
/// search in the relevant locations for a library named `name`.
|
2015-02-26 23:00:43 -06:00
|
|
|
pub fn add_native_library(&mut self, name: &str) -> io::Result<()> {
|
2014-07-23 13:56:36 -05:00
|
|
|
let location = find_library(name,
|
2015-05-11 16:32:53 -05:00
|
|
|
&self.archive.config.slib_prefix,
|
|
|
|
&self.archive.config.slib_suffix,
|
|
|
|
&self.archive.config.lib_search_paths,
|
|
|
|
self.archive.config.handler);
|
2014-11-07 11:12:43 -06:00
|
|
|
self.add_archive(&location, name, |_| false)
|
Add generation of static libraries to rustc
This commit implements the support necessary for generating both intermediate
and result static rust libraries. This is an implementation of my thoughts in
https://mail.mozilla.org/pipermail/rust-dev/2013-November/006686.html.
When compiling a library, we still retain the "lib" option, although now there
are "rlib", "staticlib", and "dylib" as options for crate_type (and these are
stackable). The idea of "lib" is to generate the "compiler default" instead of
having too choose (although all are interchangeable). For now I have left the
"complier default" to be a dynamic library for size reasons.
Of the rust libraries, lib{std,extra,rustuv} will bootstrap with an
rlib/dylib pair, but lib{rustc,syntax,rustdoc,rustpkg} will only be built as a
dynamic object. I chose this for size reasons, but also because you're probably
not going to be embedding the rustc compiler anywhere any time soon.
Other than the options outlined above, there are a few defaults/preferences that
are now opinionated in the compiler:
* If both a .dylib and .rlib are found for a rust library, the compiler will
prefer the .rlib variant. This is overridable via the -Z prefer-dynamic option
* If generating a "lib", the compiler will generate a dynamic library. This is
overridable by explicitly saying what flavor you'd like (rlib, staticlib,
dylib).
* If no options are passed to the command line, and no crate_type is found in
the destination crate, then an executable is generated
With this change, you can successfully build a rust program with 0 dynamic
dependencies on rust libraries. There is still a dynamic dependency on
librustrt, but I plan on removing that in a subsequent commit.
This change includes no tests just yet. Our current testing
infrastructure/harnesses aren't very amenable to doing flavorful things with
linking, so I'm planning on adding a new mode of testing which I believe belongs
as a separate commit.
Closes #552
2013-11-15 16:03:29 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Adds all of the contents of the rlib at the specified path to this
|
|
|
|
/// archive.
|
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-03 01:19:29 -06:00
|
|
|
///
|
|
|
|
/// This ignores adding the bytecode from the rlib, and if LTO is enabled
|
|
|
|
/// then the object file also isn't added.
|
2014-01-29 20:42:19 -06:00
|
|
|
pub fn add_rlib(&mut self, rlib: &Path, name: &str,
|
2015-02-26 23:00:43 -06:00
|
|
|
lto: bool) -> io::Result<()> {
|
2014-11-07 11:12:43 -06:00
|
|
|
// Ignoring obj file starting with the crate name
|
|
|
|
// as simple comparison is not enough - there
|
|
|
|
// might be also an extra name suffix
|
|
|
|
let obj_start = format!("{}", name);
|
2015-02-18 13:48:57 -06:00
|
|
|
let obj_start = &obj_start[..];
|
2014-11-07 11:12:43 -06:00
|
|
|
// Ignoring all bytecode files, no matter of
|
|
|
|
// name
|
|
|
|
let bc_ext = ".bytecode.deflate";
|
|
|
|
|
2015-02-18 13:48:57 -06:00
|
|
|
self.add_archive(rlib, &name[..], |fname: &str| {
|
2014-11-07 11:12:43 -06:00
|
|
|
let skip_obj = lto && fname.starts_with(obj_start)
|
|
|
|
&& fname.ends_with(".o");
|
|
|
|
skip_obj || fname.ends_with(bc_ext) || fname == METADATA_FILENAME
|
|
|
|
})
|
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 19:41:01 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Adds an arbitrary file to this archive
|
2015-02-26 23:00:43 -06:00
|
|
|
pub fn add_file(&mut self, file: &Path) -> io::Result<()> {
|
|
|
|
let filename = Path::new(file.file_name().unwrap());
|
2014-07-03 18:03:26 -05:00
|
|
|
let new_file = self.work_dir.path().join(&filename);
|
|
|
|
try!(fs::copy(file, &new_file));
|
2015-02-26 23:00:43 -06:00
|
|
|
self.members.push(filename.to_path_buf());
|
2014-07-03 18:03:26 -05:00
|
|
|
Ok(())
|
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-03 01:19:29 -06:00
|
|
|
}
|
|
|
|
|
2014-07-03 18:03:26 -05:00
|
|
|
/// Indicate that the next call to `build` should updates all symbols in
|
|
|
|
/// the archive (run 'ar s' over it).
|
2013-12-10 11:28:20 -06:00
|
|
|
pub fn update_symbols(&mut self) {
|
2014-07-03 18:03:26 -05:00
|
|
|
self.should_update_symbols = true;
|
2013-12-10 11:28:20 -06:00
|
|
|
}
|
|
|
|
|
2014-07-03 18:03:26 -05:00
|
|
|
/// Combine the provided files, rlibs, and native libraries into a single
|
|
|
|
/// `Archive`.
|
|
|
|
pub fn build(self) -> Archive<'a> {
|
|
|
|
// Get an absolute path to the destination, so `ar` will work even
|
|
|
|
// though we run it from `self.work_dir`.
|
2015-05-11 16:32:53 -05:00
|
|
|
let mut objects = Vec::new();
|
|
|
|
let mut total_len = self.archive.config.dst.to_string_lossy().len();
|
2014-07-03 18:03:26 -05:00
|
|
|
|
|
|
|
if self.members.is_empty() {
|
|
|
|
if self.should_update_symbols {
|
2015-05-11 16:32:53 -05:00
|
|
|
self.archive.run(Some(self.work_dir.path()),
|
|
|
|
Action::UpdateSymbols);
|
2014-07-03 18:03:26 -05:00
|
|
|
}
|
|
|
|
return self.archive;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Don't allow the total size of `args` to grow beyond 32,000 bytes.
|
|
|
|
// Windows will raise an error if the argument string is longer than
|
|
|
|
// 32,768, and we leave a bit of extra space for the program name.
|
2015-03-25 19:06:52 -05:00
|
|
|
const ARG_LENGTH_LIMIT: usize = 32_000;
|
2014-07-03 18:03:26 -05:00
|
|
|
|
2015-01-31 11:20:46 -06:00
|
|
|
for member_name in &self.members {
|
2015-02-26 23:00:43 -06:00
|
|
|
let len = member_name.to_string_lossy().len();
|
2014-07-03 18:03:26 -05:00
|
|
|
|
|
|
|
// `len + 1` to account for the space that's inserted before each
|
|
|
|
// argument. (Windows passes command-line arguments as a single
|
|
|
|
// string, not an array of strings.)
|
|
|
|
if total_len + len + 1 > ARG_LENGTH_LIMIT {
|
|
|
|
// Add the archive members seen so far, without updating the
|
2015-05-11 16:32:53 -05:00
|
|
|
// symbol table.
|
|
|
|
self.archive.run(Some(self.work_dir.path()),
|
|
|
|
Action::AddObjects(&objects, false));
|
2014-07-03 18:03:26 -05:00
|
|
|
|
2015-05-11 16:32:53 -05:00
|
|
|
objects.clear();
|
|
|
|
total_len = self.archive.config.dst.to_string_lossy().len();
|
2014-07-03 18:03:26 -05:00
|
|
|
}
|
|
|
|
|
2015-05-11 16:32:53 -05:00
|
|
|
objects.push(member_name);
|
2014-07-03 18:03:26 -05:00
|
|
|
total_len += len + 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add the remaining archive members, and update the symbol table if
|
|
|
|
// necessary.
|
2015-05-11 16:32:53 -05:00
|
|
|
self.archive.run(Some(self.work_dir.path()),
|
|
|
|
Action::AddObjects(&objects, self.should_update_symbols));
|
2014-07-03 18:03:26 -05:00
|
|
|
|
|
|
|
self.archive
|
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-03 01:19:29 -06:00
|
|
|
}
|
|
|
|
|
2015-01-22 18:31:00 -06:00
|
|
|
fn add_archive<F>(&mut self, archive: &Path, name: &str,
|
2015-02-26 23:00:43 -06:00
|
|
|
mut skip: F) -> io::Result<()>
|
2015-01-22 18:31:00 -06:00
|
|
|
where F: FnMut(&str) -> bool,
|
2014-12-08 17:35:22 -06:00
|
|
|
{
|
2015-04-14 18:28:50 -05:00
|
|
|
let archive = match ArchiveRO::open(archive) {
|
|
|
|
Some(ar) => ar,
|
|
|
|
None => return Err(io::Error::new(io::ErrorKind::Other,
|
|
|
|
"failed to open archive")),
|
|
|
|
};
|
Add generation of static libraries to rustc
This commit implements the support necessary for generating both intermediate
and result static rust libraries. This is an implementation of my thoughts in
https://mail.mozilla.org/pipermail/rust-dev/2013-November/006686.html.
When compiling a library, we still retain the "lib" option, although now there
are "rlib", "staticlib", and "dylib" as options for crate_type (and these are
stackable). The idea of "lib" is to generate the "compiler default" instead of
having too choose (although all are interchangeable). For now I have left the
"complier default" to be a dynamic library for size reasons.
Of the rust libraries, lib{std,extra,rustuv} will bootstrap with an
rlib/dylib pair, but lib{rustc,syntax,rustdoc,rustpkg} will only be built as a
dynamic object. I chose this for size reasons, but also because you're probably
not going to be embedding the rustc compiler anywhere any time soon.
Other than the options outlined above, there are a few defaults/preferences that
are now opinionated in the compiler:
* If both a .dylib and .rlib are found for a rust library, the compiler will
prefer the .rlib variant. This is overridable via the -Z prefer-dynamic option
* If generating a "lib", the compiler will generate a dynamic library. This is
overridable by explicitly saying what flavor you'd like (rlib, staticlib,
dylib).
* If no options are passed to the command line, and no crate_type is found in
the destination crate, then an executable is generated
With this change, you can successfully build a rust program with 0 dynamic
dependencies on rust libraries. There is still a dynamic dependency on
librustrt, but I plan on removing that in a subsequent commit.
This change includes no tests just yet. Our current testing
infrastructure/harnesses aren't very amenable to doing flavorful things with
linking, so I'm planning on adding a new mode of testing which I believe belongs
as a separate commit.
Closes #552
2013-11-15 16:03:29 -06:00
|
|
|
|
|
|
|
// Next, we must rename all of the inputs to "guaranteed unique names".
|
2015-04-14 18:28:50 -05:00
|
|
|
// We write each file into `self.work_dir` under its new unique name.
|
2014-07-03 18:03:26 -05:00
|
|
|
// The reason for this renaming is that archives are keyed off the name
|
|
|
|
// of the files, so if two files have the same name they will override
|
|
|
|
// one another in the archive (bad).
|
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-03 01:19:29 -06:00
|
|
|
//
|
|
|
|
// We skip any files explicitly desired for skipping, and we also skip
|
|
|
|
// all SYMDEF files as these are just magical placeholders which get
|
|
|
|
// re-created when we make a new archive anyway.
|
2015-04-14 18:28:50 -05:00
|
|
|
for file in archive.iter() {
|
|
|
|
let filename = match file.name() {
|
|
|
|
Some(s) => s,
|
|
|
|
None => continue,
|
|
|
|
};
|
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-03 01:19:29 -06:00
|
|
|
if filename.contains(".SYMDEF") { continue }
|
2015-04-14 18:28:50 -05:00
|
|
|
if skip(filename) { continue }
|
2015-05-11 16:32:53 -05:00
|
|
|
let filename = Path::new(filename).file_name().unwrap()
|
|
|
|
.to_str().unwrap();
|
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-03 01:19:29 -06:00
|
|
|
|
2015-05-09 00:37:49 -05:00
|
|
|
// Archives on unix systems typically do not have slashes in
|
|
|
|
// filenames as the `ar` utility generally only uses the last
|
|
|
|
// component of a path for the filename list in the archive. On
|
|
|
|
// Windows, however, archives assembled with `lib.exe` will preserve
|
|
|
|
// the full path to the file that was placed in the archive,
|
|
|
|
// including path separators.
|
|
|
|
//
|
|
|
|
// The code below is munging paths so it'll go wrong pretty quickly
|
|
|
|
// if there's some unexpected slashes in the filename, so here we
|
|
|
|
// just chop off everything but the filename component. Note that
|
|
|
|
// this can cause duplicate filenames, but that's also handled below
|
|
|
|
// as well.
|
|
|
|
let filename = Path::new(filename).file_name().unwrap()
|
|
|
|
.to_str().unwrap();
|
|
|
|
|
2015-04-14 18:28:50 -05:00
|
|
|
// An archive can contain files of the same name multiple times, so
|
|
|
|
// we need to be sure to not have them overwrite one another when we
|
|
|
|
// extract them. Consequently we need to find a truly unique file
|
|
|
|
// name for us!
|
|
|
|
let mut new_filename = String::new();
|
|
|
|
for n in 0.. {
|
|
|
|
let n = if n == 0 {String::new()} else {format!("-{}", n)};
|
|
|
|
new_filename = format!("r{}-{}-{}", n, name, filename);
|
|
|
|
|
|
|
|
// LLDB (as mentioned in back::link) crashes on filenames of
|
|
|
|
// exactly
|
|
|
|
// 16 bytes in length. If we're including an object file with
|
|
|
|
// exactly 16-bytes of characters, give it some prefix so
|
|
|
|
// that it's not 16 bytes.
|
|
|
|
new_filename = if new_filename.len() == 16 {
|
|
|
|
format!("lldb-fix-{}", new_filename)
|
|
|
|
} else {
|
|
|
|
new_filename
|
|
|
|
};
|
|
|
|
|
|
|
|
let present = self.members.iter().filter_map(|p| {
|
|
|
|
p.file_name().and_then(|f| f.to_str())
|
|
|
|
}).any(|s| s == new_filename);
|
|
|
|
if !present {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
let dst = self.work_dir.path().join(&new_filename);
|
|
|
|
try!(try!(File::create(&dst)).write_all(file.data()));
|
|
|
|
self.members.push(PathBuf::from(new_filename));
|
Add generation of static libraries to rustc
This commit implements the support necessary for generating both intermediate
and result static rust libraries. This is an implementation of my thoughts in
https://mail.mozilla.org/pipermail/rust-dev/2013-November/006686.html.
When compiling a library, we still retain the "lib" option, although now there
are "rlib", "staticlib", and "dylib" as options for crate_type (and these are
stackable). The idea of "lib" is to generate the "compiler default" instead of
having too choose (although all are interchangeable). For now I have left the
"complier default" to be a dynamic library for size reasons.
Of the rust libraries, lib{std,extra,rustuv} will bootstrap with an
rlib/dylib pair, but lib{rustc,syntax,rustdoc,rustpkg} will only be built as a
dynamic object. I chose this for size reasons, but also because you're probably
not going to be embedding the rustc compiler anywhere any time soon.
Other than the options outlined above, there are a few defaults/preferences that
are now opinionated in the compiler:
* If both a .dylib and .rlib are found for a rust library, the compiler will
prefer the .rlib variant. This is overridable via the -Z prefer-dynamic option
* If generating a "lib", the compiler will generate a dynamic library. This is
overridable by explicitly saying what flavor you'd like (rlib, staticlib,
dylib).
* If no options are passed to the command line, and no crate_type is found in
the destination crate, then an executable is generated
With this change, you can successfully build a rust program with 0 dynamic
dependencies on rust libraries. There is still a dynamic dependency on
librustrt, but I plan on removing that in a subsequent commit.
This change includes no tests just yet. Our current testing
infrastructure/harnesses aren't very amenable to doing flavorful things with
linking, so I'm planning on adding a new mode of testing which I believe belongs
as a separate commit.
Closes #552
2013-11-15 16:03:29 -06:00
|
|
|
}
|
2015-04-14 18:28:50 -05:00
|
|
|
|
2014-01-29 20:42:19 -06:00
|
|
|
Ok(())
|
Add generation of static libraries to rustc
This commit implements the support necessary for generating both intermediate
and result static rust libraries. This is an implementation of my thoughts in
https://mail.mozilla.org/pipermail/rust-dev/2013-November/006686.html.
When compiling a library, we still retain the "lib" option, although now there
are "rlib", "staticlib", and "dylib" as options for crate_type (and these are
stackable). The idea of "lib" is to generate the "compiler default" instead of
having too choose (although all are interchangeable). For now I have left the
"complier default" to be a dynamic library for size reasons.
Of the rust libraries, lib{std,extra,rustuv} will bootstrap with an
rlib/dylib pair, but lib{rustc,syntax,rustdoc,rustpkg} will only be built as a
dynamic object. I chose this for size reasons, but also because you're probably
not going to be embedding the rustc compiler anywhere any time soon.
Other than the options outlined above, there are a few defaults/preferences that
are now opinionated in the compiler:
* If both a .dylib and .rlib are found for a rust library, the compiler will
prefer the .rlib variant. This is overridable via the -Z prefer-dynamic option
* If generating a "lib", the compiler will generate a dynamic library. This is
overridable by explicitly saying what flavor you'd like (rlib, staticlib,
dylib).
* If no options are passed to the command line, and no crate_type is found in
the destination crate, then an executable is generated
With this change, you can successfully build a rust program with 0 dynamic
dependencies on rust libraries. There is still a dynamic dependency on
librustrt, but I plan on removing that in a subsequent commit.
This change includes no tests just yet. Our current testing
infrastructure/harnesses aren't very amenable to doing flavorful things with
linking, so I'm planning on adding a new mode of testing which I believe belongs
as a separate commit.
Closes #552
2013-11-15 16:03:29 -06:00
|
|
|
}
|
|
|
|
}
|