@@ -17,11 +17,11 @@ use std::path::{Path, PathBuf};
17
17
use std:: process:: Command ;
18
18
19
19
use context:: SharedCrateContext ;
20
- use monomorphize:: Instance ;
21
20
22
21
use back:: archive;
22
+ use back:: symbol_export:: { self , ExportedSymbols } ;
23
23
use middle:: dependency_format:: Linkage ;
24
- use rustc:: hir:: def_id:: CrateNum ;
24
+ use rustc:: hir:: def_id:: { LOCAL_CRATE , CrateNum } ;
25
25
use session:: Session ;
26
26
use session:: config:: CrateType ;
27
27
use session:: config;
@@ -34,10 +34,10 @@ pub struct LinkerInfo {
34
34
35
35
impl < ' a , ' tcx > LinkerInfo {
36
36
pub fn new ( scx : & SharedCrateContext < ' a , ' tcx > ,
37
- reachable : & [ String ] ) -> LinkerInfo {
37
+ exports : & ExportedSymbols ) -> LinkerInfo {
38
38
LinkerInfo {
39
39
exports : scx. sess ( ) . crate_types . borrow ( ) . iter ( ) . map ( |& c| {
40
- ( c, exported_symbols ( scx, reachable , c) )
40
+ ( c, exported_symbols ( scx, exports , c) )
41
41
} ) . collect ( ) ,
42
42
}
43
43
}
@@ -253,46 +253,47 @@ impl<'a> Linker for GnuLinker<'a> {
253
253
let mut arg = OsString :: new ( ) ;
254
254
let path = tmpdir. join ( "list" ) ;
255
255
256
- if self . sess . target . target . options . is_like_solaris {
256
+ debug ! ( "EXPORTED SYMBOLS:" ) ;
257
+
258
+ if self . sess . target . target . options . is_like_osx {
259
+ // Write a plain, newline-separated list of symbols
257
260
let res = ( || -> io:: Result < ( ) > {
258
261
let mut f = BufWriter :: new ( File :: create ( & path) ?) ;
259
- writeln ! ( f, "{{\n global:" ) ?;
260
262
for sym in self . info . exports [ & crate_type] . iter ( ) {
261
- writeln ! ( f, " {};" , sym) ?;
263
+ debug ! ( " _{}" , sym) ;
264
+ writeln ! ( f, "_{}" , sym) ?;
262
265
}
263
- writeln ! ( f, "\n local:\n *;\n }};" ) ?;
264
266
Ok ( ( ) )
265
267
} ) ( ) ;
266
268
if let Err ( e) = res {
267
- self . sess . fatal ( & format ! ( "failed to write version script : {}" , e) ) ;
269
+ self . sess . fatal ( & format ! ( "failed to write lib.def file : {}" , e) ) ;
268
270
}
269
-
270
- arg. push ( "-Wl,-M," ) ;
271
- arg. push ( & path) ;
272
271
} else {
273
- let prefix = if self . sess . target . target . options . is_like_osx {
274
- "_"
275
- } else {
276
- ""
277
- } ;
272
+ // Write an LD version script
278
273
let res = ( || -> io:: Result < ( ) > {
279
274
let mut f = BufWriter :: new ( File :: create ( & path) ?) ;
275
+ writeln ! ( f, "{{\n global:" ) ?;
280
276
for sym in self . info . exports [ & crate_type] . iter ( ) {
281
- writeln ! ( f, "{}{}" , prefix, sym) ?;
277
+ debug ! ( " {};" , sym) ;
278
+ writeln ! ( f, " {};" , sym) ?;
282
279
}
280
+ writeln ! ( f, "\n local:\n *;\n }};" ) ?;
283
281
Ok ( ( ) )
284
282
} ) ( ) ;
285
283
if let Err ( e) = res {
286
- self . sess . fatal ( & format ! ( "failed to write lib.def file: {}" , e) ) ;
287
- }
288
- if self . sess . target . target . options . is_like_osx {
289
- arg. push ( "-Wl,-exported_symbols_list," ) ;
290
- } else {
291
- arg. push ( "-Wl,--retain-symbols-file=" ) ;
284
+ self . sess . fatal ( & format ! ( "failed to write version script: {}" , e) ) ;
292
285
}
293
- arg. push ( & path) ;
294
286
}
295
287
288
+ if self . sess . target . target . options . is_like_osx {
289
+ arg. push ( "-Wl,-exported_symbols_list," ) ;
290
+ } else if self . sess . target . target . options . is_like_solaris {
291
+ arg. push ( "-Wl,-M," ) ;
292
+ } else {
293
+ arg. push ( "-Wl,--version-script=" ) ;
294
+ }
295
+
296
+ arg. push ( & path) ;
296
297
self . cmd . arg ( arg) ;
297
298
}
298
299
@@ -473,43 +474,29 @@ impl<'a> Linker for MsvcLinker<'a> {
473
474
}
474
475
475
476
fn exported_symbols ( scx : & SharedCrateContext ,
476
- reachable : & [ String ] ,
477
+ exported_symbols : & ExportedSymbols ,
477
478
crate_type : CrateType )
478
479
-> Vec < String > {
479
- // See explanation in GnuLinker::export_symbols, for
480
- // why we don't ever need dylib symbols on non-MSVC.
481
- if crate_type == CrateType :: CrateTypeDylib ||
482
- crate_type == CrateType :: CrateTypeProcMacro {
483
- if !scx. sess ( ) . target . target . options . is_like_msvc {
484
- return vec ! [ ] ;
485
- }
486
- }
480
+ let export_threshold = symbol_export:: crate_export_threshold ( crate_type) ;
487
481
488
- let mut symbols = reachable. to_vec ( ) ;
482
+ let mut symbols = Vec :: new ( ) ;
483
+ exported_symbols. for_each_exported_symbol ( LOCAL_CRATE , export_threshold, |name, _| {
484
+ symbols. push ( name. to_owned ( ) ) ;
485
+ } ) ;
489
486
490
- // If we're producing anything other than a dylib then the `reachable` array
491
- // above is the exhaustive set of symbols we should be exporting.
492
- //
493
- // For dylibs, however, we need to take a look at how all upstream crates
494
- // are linked into this dynamic library. For all statically linked
495
- // libraries we take all their reachable symbols and emit them as well.
496
- if crate_type != CrateType :: CrateTypeDylib {
497
- return symbols
498
- }
499
-
500
- let cstore = & scx. sess ( ) . cstore ;
501
487
let formats = scx. sess ( ) . dependency_formats . borrow ( ) ;
502
488
let deps = formats[ & crate_type] . iter ( ) ;
503
- symbols. extend ( deps. enumerate ( ) . filter_map ( |( i, f) | {
504
- if * f == Linkage :: Static {
505
- Some ( CrateNum :: new ( i + 1 ) )
506
- } else {
507
- None
489
+
490
+ for ( index, dep_format) in deps. enumerate ( ) {
491
+ let cnum = CrateNum :: new ( index + 1 ) ;
492
+ // For each dependency that we are linking to statically ...
493
+ if * dep_format == Linkage :: Static {
494
+ // ... we add its symbol list to our export list.
495
+ exported_symbols. for_each_exported_symbol ( cnum, export_threshold, |name, _| {
496
+ symbols. push ( name. to_owned ( ) ) ;
497
+ } )
508
498
}
509
- } ) . flat_map ( |cnum| {
510
- cstore. reachable_ids ( cnum)
511
- } ) . map ( |did| -> String {
512
- Instance :: mono ( scx, did) . symbol_name ( scx)
513
- } ) ) ;
499
+ }
500
+
514
501
symbols
515
502
}
0 commit comments