2012-12-03 16:48:01 -08:00
|
|
|
// 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 <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.
|
|
|
|
|
2012-07-04 22:53:12 +01:00
|
|
|
//! Finds crate binaries and loads their metadata
|
2012-05-15 20:23:06 -07:00
|
|
|
|
2013-05-17 15:28:44 -07:00
|
|
|
|
2012-09-04 11:54:36 -07:00
|
|
|
use lib::llvm::{False, llvm, mk_object_file, mk_section_iter};
|
2012-12-23 17:41:37 -05:00
|
|
|
use metadata::decoder;
|
|
|
|
use metadata::encoder;
|
2012-12-13 13:05:22 -08:00
|
|
|
use metadata::filesearch::FileSearch;
|
2012-12-23 17:41:37 -05:00
|
|
|
use metadata::filesearch;
|
|
|
|
use syntax::codemap::span;
|
|
|
|
use syntax::diagnostic::span_handler;
|
2013-06-04 12:34:25 -07:00
|
|
|
use syntax::parse::token;
|
2012-09-04 11:54:36 -07:00
|
|
|
use syntax::parse::token::ident_interner;
|
2012-12-23 17:41:37 -05:00
|
|
|
use syntax::print::pprust;
|
|
|
|
use syntax::{ast, attr};
|
|
|
|
|
2013-06-28 18:32:26 -04:00
|
|
|
use std::cast;
|
|
|
|
use std::io;
|
Replaces the free-standing functions in f32, &c.
The free-standing functions in f32, f64, i8, i16, i32, i64, u8, u16,
u32, u64, float, int, and uint are replaced with generic functions in
num instead.
If you were previously using any of those functions, just replace them
with the corresponding function with the same name in num.
Note: If you were using a function that corresponds to an operator, use
the operator instead.
2013-07-08 18:05:17 +02:00
|
|
|
use std::num;
|
2013-06-28 18:32:26 -04:00
|
|
|
use std::option;
|
|
|
|
use std::os::consts::{macos, freebsd, linux, android, win32};
|
|
|
|
use std::ptr;
|
|
|
|
use std::str;
|
|
|
|
use std::vec;
|
2013-05-24 19:35:29 -07:00
|
|
|
use extra::flate;
|
2012-05-15 20:23:06 -07:00
|
|
|
|
2013-01-29 16:51:16 -08:00
|
|
|
pub enum os {
|
2012-05-22 17:16:26 -07:00
|
|
|
os_macos,
|
|
|
|
os_win32,
|
|
|
|
os_linux,
|
2012-11-30 09:21:49 +09:00
|
|
|
os_android,
|
2012-05-22 17:16:26 -07:00
|
|
|
os_freebsd
|
|
|
|
}
|
2012-05-15 20:23:06 -07:00
|
|
|
|
2013-02-19 02:40:42 -05:00
|
|
|
pub struct Context {
|
2013-03-12 13:00:50 -07:00
|
|
|
diag: @span_handler,
|
|
|
|
filesearch: @FileSearch,
|
2012-05-22 17:16:26 -07:00
|
|
|
span: span,
|
|
|
|
ident: ast::ident,
|
2012-09-20 18:15:39 -07:00
|
|
|
metas: ~[@ast::meta_item],
|
2013-06-13 03:02:55 +10:00
|
|
|
hash: @str,
|
2012-05-22 17:16:26 -07:00
|
|
|
os: os,
|
2013-02-19 02:40:42 -05:00
|
|
|
is_static: bool,
|
2012-09-19 18:50:24 -07:00
|
|
|
intr: @ident_interner
|
2013-02-19 02:40:42 -05:00
|
|
|
}
|
2012-05-22 17:16:26 -07:00
|
|
|
|
2013-04-17 12:15:37 -04:00
|
|
|
pub fn load_library_crate(cx: &Context) -> (~str, @~[u8]) {
|
2012-08-06 12:34:08 -07:00
|
|
|
match find_library_crate(cx) {
|
2013-07-02 12:47:32 -07:00
|
|
|
Some(ref t) => return (/*bad*/(*t).clone()),
|
2012-08-20 12:23:37 -07:00
|
|
|
None => {
|
2013-07-02 12:47:32 -07:00
|
|
|
cx.diag.span_fatal(cx.span,
|
|
|
|
fmt!("can't find crate for `%s`",
|
|
|
|
token::ident_to_str(&cx.ident)));
|
2012-05-15 20:23:06 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-04-17 12:15:37 -04:00
|
|
|
fn find_library_crate(cx: &Context) -> Option<(~str, @~[u8])> {
|
2013-01-11 15:07:48 -08:00
|
|
|
attr::require_unique_names(cx.diag, cx.metas);
|
2012-05-22 17:48:04 -07:00
|
|
|
find_library_crate_aux(cx, libname(cx), cx.filesearch)
|
2012-05-15 20:23:06 -07:00
|
|
|
}
|
|
|
|
|
2013-04-17 12:15:37 -04:00
|
|
|
fn libname(cx: &Context) -> (~str, ~str) {
|
2013-02-19 02:40:42 -05:00
|
|
|
if cx.is_static { return (~"lib", ~".rlib"); }
|
2012-12-20 18:26:27 +09:00
|
|
|
let (dll_prefix, dll_suffix) = match cx.os {
|
|
|
|
os_win32 => (win32::DLL_PREFIX, win32::DLL_SUFFIX),
|
|
|
|
os_macos => (macos::DLL_PREFIX, macos::DLL_SUFFIX),
|
|
|
|
os_linux => (linux::DLL_PREFIX, linux::DLL_SUFFIX),
|
2012-11-30 09:21:49 +09:00
|
|
|
os_android => (android::DLL_PREFIX, android::DLL_SUFFIX),
|
2012-12-20 18:26:27 +09:00
|
|
|
os_freebsd => (freebsd::DLL_PREFIX, freebsd::DLL_SUFFIX),
|
|
|
|
};
|
2013-02-19 02:40:42 -05:00
|
|
|
|
2013-06-13 19:06:47 -07:00
|
|
|
(str::to_owned(dll_prefix), str::to_owned(dll_suffix))
|
2012-05-15 20:23:06 -07:00
|
|
|
}
|
|
|
|
|
2013-03-03 08:50:20 -08:00
|
|
|
fn find_library_crate_aux(
|
2013-04-17 12:15:37 -04:00
|
|
|
cx: &Context,
|
2013-03-03 08:50:20 -08:00
|
|
|
(prefix, suffix): (~str, ~str),
|
2013-03-12 13:00:50 -07:00
|
|
|
filesearch: @filesearch::FileSearch
|
2013-03-03 08:50:20 -08:00
|
|
|
) -> Option<(~str, @~[u8])> {
|
|
|
|
let crate_name = crate_name_from_metas(cx.metas);
|
2013-06-24 12:38:14 -07:00
|
|
|
let prefix = prefix + crate_name + "-";
|
2012-05-15 20:23:06 -07:00
|
|
|
|
2012-06-29 16:26:56 -07:00
|
|
|
let mut matches = ~[];
|
2013-06-24 12:38:14 -07:00
|
|
|
filesearch::search(filesearch, |path| -> Option<()> {
|
2012-08-24 15:28:43 -07:00
|
|
|
debug!("inspecting file %s", path.to_str());
|
2013-06-24 12:38:14 -07:00
|
|
|
match path.filename() {
|
|
|
|
Some(ref f) if f.starts_with(prefix) && f.ends_with(suffix) => {
|
|
|
|
debug!("%s is a candidate", path.to_str());
|
|
|
|
match get_metadata_section(cx.os, path) {
|
|
|
|
Some(cvec) =>
|
|
|
|
if !crate_matches(cvec, cx.metas, cx.hash) {
|
|
|
|
debug!("skipping %s, metadata doesn't match",
|
|
|
|
path.to_str());
|
|
|
|
None
|
|
|
|
} else {
|
|
|
|
debug!("found %s with matching metadata", path.to_str());
|
|
|
|
matches.push((path.to_str(), cvec));
|
|
|
|
None
|
|
|
|
},
|
|
|
|
_ => {
|
|
|
|
debug!("could not load metadata for %s", path.to_str());
|
|
|
|
None
|
|
|
|
}
|
2012-05-15 20:23:06 -07:00
|
|
|
}
|
|
|
|
}
|
2013-06-24 12:38:14 -07:00
|
|
|
_ => {
|
|
|
|
debug!("skipping %s, doesn't look like %s*%s", path.to_str(),
|
|
|
|
prefix, suffix);
|
|
|
|
None
|
|
|
|
}
|
|
|
|
}});
|
2012-05-15 20:23:06 -07:00
|
|
|
|
2013-06-24 12:38:14 -07:00
|
|
|
match matches.len() {
|
|
|
|
0 => None,
|
|
|
|
1 => Some(matches[0]),
|
|
|
|
_ => {
|
|
|
|
cx.diag.span_err(
|
|
|
|
cx.span, fmt!("multiple matching crates for `%s`", crate_name));
|
|
|
|
cx.diag.handler().note("candidates:");
|
2013-06-20 15:26:54 -04:00
|
|
|
for matches.iter().advance |pair| {
|
|
|
|
let ident = pair.first();
|
|
|
|
let data = pair.second();
|
2013-06-24 12:38:14 -07:00
|
|
|
cx.diag.handler().note(fmt!("path: %s", ident));
|
|
|
|
let attrs = decoder::get_crate_attributes(data);
|
|
|
|
note_linkage_attrs(cx.intr, cx.diag, attrs);
|
|
|
|
}
|
|
|
|
cx.diag.handler().abort_if_errors();
|
|
|
|
None
|
|
|
|
}
|
2012-05-15 20:23:06 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-06-13 03:02:55 +10:00
|
|
|
pub fn crate_name_from_metas(metas: &[@ast::meta_item]) -> @str {
|
2013-06-26 18:22:15 -04:00
|
|
|
for metas.iter().advance |m| {
|
2013-06-24 12:38:14 -07:00
|
|
|
match m.node {
|
|
|
|
ast::meta_name_value(s, ref l) if s == @"name" =>
|
|
|
|
match l.node {
|
|
|
|
ast::lit_str(s) => return s,
|
|
|
|
_ => ()
|
|
|
|
},
|
|
|
|
_ => ()
|
2012-05-15 20:23:06 -07:00
|
|
|
}
|
|
|
|
}
|
2013-06-24 12:38:14 -07:00
|
|
|
fail!("expected to find the crate name")
|
2012-05-15 20:23:06 -07:00
|
|
|
}
|
|
|
|
|
2013-03-12 13:00:50 -07:00
|
|
|
pub fn note_linkage_attrs(intr: @ident_interner,
|
|
|
|
diag: @span_handler,
|
2013-01-29 16:51:16 -08:00
|
|
|
attrs: ~[ast::attribute]) {
|
2013-06-21 08:29:53 -04:00
|
|
|
let r = attr::find_linkage_metas(attrs);
|
|
|
|
for r.iter().advance |mi| {
|
|
|
|
diag.handler().note(fmt!("meta: %s", pprust::meta_item_to_str(*mi,intr)));
|
2012-05-15 20:23:06 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-01-10 10:59:58 -08:00
|
|
|
fn crate_matches(crate_data: @~[u8],
|
|
|
|
metas: &[@ast::meta_item],
|
2013-06-13 03:02:55 +10:00
|
|
|
hash: @str) -> bool {
|
2012-05-15 20:23:06 -07:00
|
|
|
let attrs = decoder::get_crate_attributes(crate_data);
|
|
|
|
let linkage_metas = attr::find_linkage_metas(attrs);
|
2013-01-24 23:24:57 -05:00
|
|
|
if !hash.is_empty() {
|
2012-05-15 20:23:06 -07:00
|
|
|
let chash = decoder::get_crate_hash(crate_data);
|
2012-07-18 16:18:02 -07:00
|
|
|
if chash != hash { return false; }
|
2012-05-15 20:23:06 -07:00
|
|
|
}
|
|
|
|
metadata_matches(linkage_metas, metas)
|
|
|
|
}
|
|
|
|
|
2013-03-10 11:02:16 -04:00
|
|
|
pub fn metadata_matches(extern_metas: &[@ast::meta_item],
|
2013-01-10 10:59:58 -08:00
|
|
|
local_metas: &[@ast::meta_item]) -> bool {
|
2012-05-15 20:23:06 -07:00
|
|
|
|
2012-08-22 17:24:52 -07:00
|
|
|
debug!("matching %u metadata requirements against %u items",
|
2013-05-14 18:52:12 +09:00
|
|
|
local_metas.len(), extern_metas.len());
|
2012-05-15 20:23:06 -07:00
|
|
|
|
2013-06-21 08:29:53 -04:00
|
|
|
for local_metas.iter().advance |needed| {
|
2012-09-20 18:15:39 -07:00
|
|
|
if !attr::contains(extern_metas, *needed) {
|
2012-08-01 17:30:05 -07:00
|
|
|
return false;
|
2012-05-15 20:23:06 -07:00
|
|
|
}
|
|
|
|
}
|
2012-08-01 17:30:05 -07:00
|
|
|
return true;
|
2012-05-15 20:23:06 -07:00
|
|
|
}
|
|
|
|
|
2012-05-22 17:16:26 -07:00
|
|
|
fn get_metadata_section(os: os,
|
2013-01-23 11:43:58 -08:00
|
|
|
filename: &Path) -> Option<@~[u8]> {
|
|
|
|
unsafe {
|
|
|
|
let mb = str::as_c_str(filename.to_str(), |buf| {
|
|
|
|
llvm::LLVMRustCreateMemoryBufferWithContentsOfFile(buf)
|
|
|
|
});
|
|
|
|
if mb as int == 0 { return option::None::<@~[u8]>; }
|
|
|
|
let of = match mk_object_file(mb) {
|
|
|
|
option::Some(of) => of,
|
|
|
|
_ => return option::None::<@~[u8]>
|
|
|
|
};
|
|
|
|
let si = mk_section_iter(of.llof);
|
|
|
|
while llvm::LLVMIsSectionIteratorAtEnd(of.llof, si.llsi) == False {
|
|
|
|
let name_buf = llvm::LLVMGetSectionName(si.llsi);
|
2013-06-20 17:52:02 +12:00
|
|
|
let name = str::raw::from_c_str(name_buf);
|
2013-05-03 16:20:29 -07:00
|
|
|
debug!("get_metadata_section: name %s", name);
|
2013-03-13 17:22:01 +09:00
|
|
|
if name == read_meta_section_name(os) {
|
2013-01-23 11:43:58 -08:00
|
|
|
let cbuf = llvm::LLVMGetSectionContents(si.llsi);
|
|
|
|
let csz = llvm::LLVMGetSectionSize(si.llsi) as uint;
|
|
|
|
let mut found = None;
|
2013-06-20 17:52:02 +12:00
|
|
|
let cvbuf: *u8 = cast::transmute(cbuf);
|
|
|
|
let vlen = encoder::metadata_encoding_version.len();
|
|
|
|
debug!("checking %u bytes of metadata-version stamp",
|
|
|
|
vlen);
|
Replaces the free-standing functions in f32, &c.
The free-standing functions in f32, f64, i8, i16, i32, i64, u8, u16,
u32, u64, float, int, and uint are replaced with generic functions in
num instead.
If you were previously using any of those functions, just replace them
with the corresponding function with the same name in num.
Note: If you were using a function that corresponds to an operator, use
the operator instead.
2013-07-08 18:05:17 +02:00
|
|
|
let minsz = num::min(vlen, csz);
|
2013-06-20 17:52:02 +12:00
|
|
|
let mut version_ok = false;
|
|
|
|
do vec::raw::buf_as_slice(cvbuf, minsz) |buf0| {
|
|
|
|
version_ok = (buf0 ==
|
|
|
|
encoder::metadata_encoding_version);
|
|
|
|
}
|
|
|
|
if !version_ok { return None; }
|
|
|
|
|
|
|
|
let cvbuf1 = ptr::offset(cvbuf, vlen);
|
|
|
|
debug!("inflating %u bytes of compressed metadata",
|
|
|
|
csz - vlen);
|
|
|
|
do vec::raw::buf_as_slice(cvbuf1, csz-vlen) |bytes| {
|
|
|
|
let inflated = flate::inflate_bytes(bytes);
|
|
|
|
found = Some(@(inflated));
|
|
|
|
}
|
|
|
|
if found != None {
|
|
|
|
return found;
|
2012-09-05 15:27:22 -07:00
|
|
|
}
|
2012-05-15 20:23:06 -07:00
|
|
|
}
|
2013-01-23 11:43:58 -08:00
|
|
|
llvm::LLVMMoveToNextSection(si.llsi);
|
2012-05-15 20:23:06 -07:00
|
|
|
}
|
2013-01-23 11:43:58 -08:00
|
|
|
return option::None::<@~[u8]>;
|
2012-05-15 20:23:06 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-01-29 16:51:16 -08:00
|
|
|
pub fn meta_section_name(os: os) -> ~str {
|
2012-08-06 12:34:08 -07:00
|
|
|
match os {
|
2012-08-03 19:59:04 -07:00
|
|
|
os_macos => ~"__DATA,__note.rustc",
|
|
|
|
os_win32 => ~".note.rustc",
|
|
|
|
os_linux => ~".note.rustc",
|
2012-11-30 09:21:49 +09:00
|
|
|
os_android => ~".note.rustc",
|
2012-08-03 19:59:04 -07:00
|
|
|
os_freebsd => ~".note.rustc"
|
2012-05-22 17:16:26 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-03-13 17:22:01 +09:00
|
|
|
pub fn read_meta_section_name(os: os) -> ~str {
|
|
|
|
match os {
|
|
|
|
os_macos => ~"__note.rustc",
|
|
|
|
os_win32 => ~".note.rustc",
|
|
|
|
os_linux => ~".note.rustc",
|
|
|
|
os_android => ~".note.rustc",
|
|
|
|
os_freebsd => ~".note.rustc"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-05-15 20:23:06 -07:00
|
|
|
// A diagnostic function for dumping crate metadata to an output stream
|
2013-01-29 16:51:16 -08:00
|
|
|
pub fn list_file_metadata(intr: @ident_interner,
|
2013-03-12 13:00:50 -07:00
|
|
|
os: os,
|
|
|
|
path: &Path,
|
|
|
|
out: @io::Writer) {
|
2012-08-06 12:34:08 -07:00
|
|
|
match get_metadata_section(os, path) {
|
2012-08-20 12:23:37 -07:00
|
|
|
option::Some(bytes) => decoder::list_crate_metadata(intr, bytes, out),
|
|
|
|
option::None => {
|
2013-05-29 20:10:16 +02:00
|
|
|
out.write_str(fmt!("could not find metadata in %s.\n", path.to_str()))
|
2012-05-15 20:23:06 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|