@@ -463,15 +463,18 @@ fn thin_lto(
463
463
// If previous imports have been deleted, or we get an IO error
464
464
// reading the file storing them, then we'll just use `None` as the
465
465
// prev_import_map, which will force the code to be recompiled.
466
- let prev =
467
- if path. exists ( ) { ThinLTOImports :: load_from_file ( & path) . ok ( ) } else { None } ;
468
- let curr = ThinLTOImports :: from_thin_lto_data ( data) ;
466
+ let prev = if path. exists ( ) {
467
+ ThinLTOImportMaps :: load_from_file ( & path) . ok ( )
468
+ } else {
469
+ None
470
+ } ;
471
+ let curr = ThinLTOImportMaps :: from_thin_lto_data ( data) ;
469
472
( Some ( path) , prev, curr)
470
473
} else {
471
474
// If we don't compile incrementally, we don't need to load the
472
475
// import data from LLVM.
473
476
assert ! ( green_modules. is_empty( ) ) ;
474
- let curr = ThinLTOImports :: default ( ) ;
477
+ let curr = ThinLTOImportMaps :: default ( ) ;
475
478
( None , None , curr)
476
479
} ;
477
480
info ! ( "thin LTO import map loaded" ) ;
@@ -497,10 +500,14 @@ fn thin_lto(
497
500
let module_name = module_name_to_str ( module_name) ;
498
501
499
502
// If (1.) the module hasn't changed, and (2.) none of the modules
500
- // it imports from has changed, *and* (3.) the import-set itself has
501
- // not changed from the previous compile when it was last
502
- // ThinLTO'ed, then we can re-use the post-ThinLTO version of the
503
- // module. Otherwise, freshly perform LTO optimization.
503
+ // it imports from nor exports to have changed, *and* (3.) the
504
+ // import and export sets themselves have not changed from the
505
+ // previous compile when it was last ThinLTO'ed, then we can re-use
506
+ // the post-ThinLTO version of the module. Otherwise, freshly
507
+ // perform LTO optimization.
508
+ //
509
+ // (Note that globally, the export set is just the inverse of the
510
+ // import set.)
504
511
//
505
512
// This strategy means we can always save the computed imports as
506
513
// canon: when we reuse the post-ThinLTO version, condition (3.)
@@ -509,19 +516,30 @@ fn thin_lto(
509
516
// version, the current import set *is* the correct one, since we
510
517
// are doing the ThinLTO in this current compilation cycle.)
511
518
//
512
- // See rust-lang/rust#59535.
519
+ // For more discussion, see rust-lang/rust#59535 (where the import
520
+ // issue was discovered) and rust-lang/rust#69798 (where the
521
+ // analogous export issue was discovered).
513
522
if let ( Some ( prev_import_map) , true ) =
514
523
( prev_import_map. as_ref ( ) , green_modules. contains_key ( module_name) )
515
524
{
516
525
assert ! ( cgcx. incr_comp_session_dir. is_some( ) ) ;
517
526
518
- let prev_imports = prev_import_map. modules_imported_by ( module_name) ;
519
- let curr_imports = curr_import_map. modules_imported_by ( module_name) ;
527
+ let prev_imports = prev_import_map. imports_of ( module_name) ;
528
+ let curr_imports = curr_import_map. imports_of ( module_name) ;
529
+ let prev_exports = prev_import_map. exports_of ( module_name) ;
530
+ let curr_exports = curr_import_map. exports_of ( module_name) ;
520
531
let imports_all_green = curr_imports
521
532
. iter ( )
522
533
. all ( |imported_module| green_modules. contains_key ( imported_module) ) ;
534
+ let exports_all_green = curr_exports
535
+ . iter ( )
536
+ . all ( |exported_module| green_modules. contains_key ( exported_module) ) ;
523
537
524
- if imports_all_green && equivalent_as_sets ( prev_imports, curr_imports) {
538
+ if imports_all_green
539
+ && equivalent_as_sets ( prev_imports, curr_imports)
540
+ && exports_all_green
541
+ && equivalent_as_sets ( prev_exports, curr_exports)
542
+ {
525
543
let work_product = green_modules[ module_name] . clone ( ) ;
526
544
copy_jobs. push ( work_product) ;
527
545
info ! ( " - {}: re-used" , module_name) ;
@@ -881,17 +899,32 @@ pub unsafe fn optimize_thin_module(
881
899
Ok ( module)
882
900
}
883
901
902
+ /// Summarizes module import/export relationships used by LLVM's ThinLTO pass.
903
+ ///
904
+ /// Note that we tend to have two such instances of `ThinLTOImportMaps` in use:
905
+ /// one loaded from a file that represents the relationships used during the
906
+ /// compilation associated with the incremetnal build artifacts we are
907
+ /// attempting to reuse, and another constructed via `from_thin_lto_data`, which
908
+ /// captures the relationships of ThinLTO in the current compilation.
884
909
#[ derive( Debug , Default ) ]
885
- pub struct ThinLTOImports {
910
+ pub struct ThinLTOImportMaps {
886
911
// key = llvm name of importing module, value = list of modules it imports from
887
912
imports : FxHashMap < String , Vec < String > > ,
913
+ // key = llvm name of exporting module, value = list of modules it exports to
914
+ exports : FxHashMap < String , Vec < String > > ,
888
915
}
889
916
890
- impl ThinLTOImports {
891
- fn modules_imported_by ( & self , llvm_module_name : & str ) -> & [ String ] {
917
+ impl ThinLTOImportMaps {
918
+ /// Returns modules imported by `llvm_module_name` during some ThinLTO pass.
919
+ fn imports_of ( & self , llvm_module_name : & str ) -> & [ String ] {
892
920
self . imports . get ( llvm_module_name) . map ( |v| & v[ ..] ) . unwrap_or ( & [ ] )
893
921
}
894
922
923
+ /// Returns modules exported by `llvm_module_name` during some ThinLTO pass.
924
+ fn exports_of ( & self , llvm_module_name : & str ) -> & [ String ] {
925
+ self . exports . get ( llvm_module_name) . map ( |v| & v[ ..] ) . unwrap_or ( & [ ] )
926
+ }
927
+
895
928
fn save_to_file ( & self , path : & Path ) -> io:: Result < ( ) > {
896
929
use std:: io:: Write ;
897
930
let file = File :: create ( path) ?;
@@ -906,16 +939,20 @@ impl ThinLTOImports {
906
939
Ok ( ( ) )
907
940
}
908
941
909
- fn load_from_file ( path : & Path ) -> io:: Result < ThinLTOImports > {
942
+ fn load_from_file ( path : & Path ) -> io:: Result < ThinLTOImportMaps > {
910
943
use std:: io:: BufRead ;
911
944
let mut imports = FxHashMap :: default ( ) ;
912
- let mut current_module = None ;
913
- let mut current_imports = vec ! [ ] ;
945
+ let mut exports: FxHashMap < _ , Vec < _ > > = FxHashMap :: default ( ) ;
946
+ let mut current_module: Option < String > = None ;
947
+ let mut current_imports: Vec < String > = vec ! [ ] ;
914
948
let file = File :: open ( path) ?;
915
949
for line in io:: BufReader :: new ( file) . lines ( ) {
916
950
let line = line?;
917
951
if line. is_empty ( ) {
918
952
let importing_module = current_module. take ( ) . expect ( "Importing module not set" ) ;
953
+ for imported in & current_imports {
954
+ exports. entry ( imported. clone ( ) ) . or_default ( ) . push ( importing_module. clone ( ) ) ;
955
+ }
919
956
imports. insert ( importing_module, mem:: replace ( & mut current_imports, vec ! [ ] ) ) ;
920
957
} else if line. starts_with ( ' ' ) {
921
958
// Space marks an imported module
@@ -927,17 +964,17 @@ impl ThinLTOImports {
927
964
current_module = Some ( line. trim ( ) . to_string ( ) ) ;
928
965
}
929
966
}
930
- Ok ( ThinLTOImports { imports } )
967
+ Ok ( ThinLTOImportMaps { imports, exports } )
931
968
}
932
969
933
970
/// Loads the ThinLTO import map from ThinLTOData.
934
- unsafe fn from_thin_lto_data ( data : * const llvm:: ThinLTOData ) -> ThinLTOImports {
971
+ unsafe fn from_thin_lto_data ( data : * const llvm:: ThinLTOData ) -> ThinLTOImportMaps {
935
972
unsafe extern "C" fn imported_module_callback (
936
973
payload : * mut libc:: c_void ,
937
974
importing_module_name : * const libc:: c_char ,
938
975
imported_module_name : * const libc:: c_char ,
939
976
) {
940
- let map = & mut * ( payload as * mut ThinLTOImports ) ;
977
+ let map = & mut * ( payload as * mut ThinLTOImportMaps ) ;
941
978
let importing_module_name = CStr :: from_ptr ( importing_module_name) ;
942
979
let importing_module_name = module_name_to_str ( & importing_module_name) ;
943
980
let imported_module_name = CStr :: from_ptr ( imported_module_name) ;
@@ -951,8 +988,18 @@ impl ThinLTOImports {
951
988
. get_mut ( importing_module_name)
952
989
. unwrap ( )
953
990
. push ( imported_module_name. to_owned ( ) ) ;
991
+
992
+ if !map. exports . contains_key ( imported_module_name) {
993
+ map. exports . insert ( imported_module_name. to_owned ( ) , vec ! [ ] ) ;
994
+ }
995
+
996
+ map. exports
997
+ . get_mut ( imported_module_name)
998
+ . unwrap ( )
999
+ . push ( importing_module_name. to_owned ( ) ) ;
954
1000
}
955
- let mut map = ThinLTOImports :: default ( ) ;
1001
+
1002
+ let mut map = ThinLTOImportMaps :: default ( ) ;
956
1003
llvm:: LLVMRustGetThinLTOModuleImports (
957
1004
data,
958
1005
imported_module_callback,
0 commit comments