Skip to content

Commit 3d55891

Browse files
committed
Avoid no-op unlink+link dances in incr comp
1 parent 27b93da commit 3d55891

File tree

6 files changed

+52
-21
lines changed

6 files changed

+52
-21
lines changed

compiler/rustc_codegen_cranelift/src/driver/aot.rs

+7
Original file line numberDiff line numberDiff line change
@@ -96,12 +96,14 @@ impl OngoingCodegen {
9696
("o", &module_regular.object.as_ref().unwrap()),
9797
("asm.o", &module_global_asm.object.as_ref().unwrap()),
9898
],
99+
&[],
99100
)
100101
} else {
101102
rustc_incremental::copy_cgu_workproduct_to_incr_comp_cache_dir(
102103
sess,
103104
&module_regular.name,
104105
&[("o", &module_regular.object.as_ref().unwrap())],
106+
&[],
105107
)
106108
};
107109
if let Some((work_product_id, work_product)) = work_product {
@@ -371,6 +373,7 @@ fn emit_cgu(
371373
bytecode: None,
372374
assembly: None,
373375
llvm_ir: None,
376+
links_from_incr_cache: Vec::new(),
374377
}),
375378
existing_work_product: None,
376379
})
@@ -416,6 +419,7 @@ fn emit_module(
416419
bytecode: None,
417420
assembly: None,
418421
llvm_ir: None,
422+
links_from_incr_cache: Vec::new(),
419423
})
420424
}
421425

@@ -466,6 +470,7 @@ fn reuse_workproduct_for_cgu(
466470
bytecode: None,
467471
assembly: None,
468472
llvm_ir: None,
473+
links_from_incr_cache: Vec::new(),
469474
},
470475
module_global_asm: has_global_asm.then(|| CompiledModule {
471476
name: cgu.name().to_string(),
@@ -475,6 +480,7 @@ fn reuse_workproduct_for_cgu(
475480
bytecode: None,
476481
assembly: None,
477482
llvm_ir: None,
483+
links_from_incr_cache: Vec::new(),
478484
}),
479485
existing_work_product: Some((cgu.work_product_id(), work_product)),
480486
})
@@ -712,6 +718,7 @@ pub(crate) fn run_aot(
712718
bytecode: None,
713719
assembly: None,
714720
llvm_ir: None,
721+
links_from_incr_cache: Vec::new(),
715722
})
716723
} else {
717724
None

compiler/rustc_codegen_ssa/src/back/write.rs

+17-7
Original file line numberDiff line numberDiff line change
@@ -18,7 +18,7 @@ use rustc_errors::{
1818
Diag, DiagArgMap, DiagCtxt, DiagMessage, ErrCode, FatalError, FluentBundle, Level, MultiSpan,
1919
Style,
2020
};
21-
use rustc_fs_util::link_or_copy;
21+
use rustc_fs_util::{link_or_copy, LinkOrCopy};
2222
use rustc_hir::def_id::{CrateNum, LOCAL_CRATE};
2323
use rustc_incremental::{
2424
copy_cgu_workproduct_to_incr_comp_cache_dir, in_incr_comp_dir, in_incr_comp_dir_sess,
@@ -546,9 +546,12 @@ fn copy_all_cgu_workproducts_to_incr_comp_cache_dir(
546546
if let Some(path) = &module.bytecode {
547547
files.push((OutputType::Bitcode.extension(), path.as_path()));
548548
}
549-
if let Some((id, product)) =
550-
copy_cgu_workproduct_to_incr_comp_cache_dir(sess, &module.name, files.as_slice())
551-
{
549+
if let Some((id, product)) = copy_cgu_workproduct_to_incr_comp_cache_dir(
550+
sess,
551+
&module.name,
552+
files.as_slice(),
553+
&module.links_from_incr_cache,
554+
) {
552555
work_products.insert(id, product);
553556
}
554557
}
@@ -940,7 +943,9 @@ fn execute_copy_from_cache_work_item<B: ExtraBackendMethods>(
940943
) -> WorkItemResult<B> {
941944
let incr_comp_session_dir = cgcx.incr_comp_session_dir.as_ref().unwrap();
942945

943-
let load_from_incr_comp_dir = |output_path: PathBuf, saved_path: &str| {
946+
let mut links_from_incr_cache = Vec::new();
947+
948+
let mut load_from_incr_comp_dir = |output_path: PathBuf, saved_path: &str| {
944949
let source_file = in_incr_comp_dir(incr_comp_session_dir, saved_path);
945950
debug!(
946951
"copying preexisting module `{}` from {:?} to {}",
@@ -949,7 +954,11 @@ fn execute_copy_from_cache_work_item<B: ExtraBackendMethods>(
949954
output_path.display()
950955
);
951956
match link_or_copy(&source_file, &output_path) {
952-
Ok(_) => Some(output_path),
957+
Ok(LinkOrCopy::Copy) => Some(output_path),
958+
Ok(LinkOrCopy::Link) => {
959+
links_from_incr_cache.push(source_file);
960+
Some(output_path)
961+
}
953962
Err(error) => {
954963
cgcx.create_dcx().handle().emit_err(errors::CopyPathBuf {
955964
source_file,
@@ -972,7 +981,7 @@ fn execute_copy_from_cache_work_item<B: ExtraBackendMethods>(
972981
load_from_incr_comp_dir(dwarf_obj_out, saved_dwarf_object_file)
973982
});
974983

975-
let load_from_incr_cache = |perform, output_type: OutputType| {
984+
let mut load_from_incr_cache = |perform, output_type: OutputType| {
976985
if perform {
977986
let saved_file = module.source.saved_files.get(output_type.extension())?;
978987
let output_path = cgcx.output_filenames.temp_path(output_type, Some(&module.name));
@@ -992,6 +1001,7 @@ fn execute_copy_from_cache_work_item<B: ExtraBackendMethods>(
9921001
}
9931002

9941003
WorkItemResult::Finished(CompiledModule {
1004+
links_from_incr_cache,
9951005
name: module.name,
9961006
kind: ModuleKind::Regular,
9971007
object,

compiler/rustc_codegen_ssa/src/base.rs

+1
Original file line numberDiff line numberDiff line change
@@ -626,6 +626,7 @@ pub fn codegen_crate<B: ExtraBackendMethods>(
626626
bytecode: None,
627627
assembly: None,
628628
llvm_ir: None,
629+
links_from_incr_cache: Vec::new(),
629630
}
630631
})
631632
});

compiler/rustc_codegen_ssa/src/lib.rs

+2
Original file line numberDiff line numberDiff line change
@@ -94,6 +94,7 @@ impl<M> ModuleCodegen<M> {
9494
bytecode,
9595
assembly,
9696
llvm_ir,
97+
links_from_incr_cache: Vec::new(),
9798
}
9899
}
99100
}
@@ -107,6 +108,7 @@ pub struct CompiledModule {
107108
pub bytecode: Option<PathBuf>,
108109
pub assembly: Option<PathBuf>, // --emit=asm
109110
pub llvm_ir: Option<PathBuf>, // --emit=llvm-ir, llvm-bc is in bytecode
111+
pub links_from_incr_cache: Vec<PathBuf>,
110112
}
111113

112114
impl CompiledModule {

compiler/rustc_fs_util/src/lib.rs

+19-13
Original file line numberDiff line numberDiff line change
@@ -55,25 +55,31 @@ pub enum LinkOrCopy {
5555
Copy,
5656
}
5757

58-
/// Copies `p` into `q`, preferring to use hard-linking if possible. If
59-
/// `q` already exists, it is removed first.
58+
/// Copies `p` into `q`, preferring to use hard-linking if possible.
6059
/// The result indicates which of the two operations has been performed.
6160
pub fn link_or_copy<P: AsRef<Path>, Q: AsRef<Path>>(p: P, q: Q) -> io::Result<LinkOrCopy> {
61+
// Creating a hard-link will fail if the destination path already exists. We could defensively
62+
// call remove_file in this function, but that pessimizes callers who can avoid such calls.
63+
// Incremental compilation calls this function a lot, and is able to avoid calls that
64+
// would fail the first hard_link attempt.
65+
6266
let p = p.as_ref();
6367
let q = q.as_ref();
64-
match fs::remove_file(q) {
65-
Ok(()) => (),
66-
Err(err) if err.kind() == io::ErrorKind::NotFound => (),
67-
Err(err) => return Err(err),
68-
}
6968

70-
match fs::hard_link(p, q) {
71-
Ok(()) => Ok(LinkOrCopy::Link),
72-
Err(_) => match fs::copy(p, q) {
73-
Ok(_) => Ok(LinkOrCopy::Copy),
74-
Err(e) => Err(e),
75-
},
69+
let err = match fs::hard_link(p, q) {
70+
Ok(()) => return Ok(LinkOrCopy::Link),
71+
Err(err) => err,
72+
};
73+
74+
if err.kind() == io::ErrorKind::AlreadyExists {
75+
fs::remove_file(q)?;
76+
if fs::hard_link(p, q).is_ok() {
77+
return Ok(LinkOrCopy::Link);
78+
}
7679
}
80+
81+
// Hard linking failed, fall back to copying.
82+
fs::copy(p, q).map(|_| LinkOrCopy::Copy)
7783
}
7884

7985
#[cfg(unix)]

compiler/rustc_incremental/src/persist/work_product.rs

+6-1
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,7 @@
33
//! [work products]: WorkProduct
44
55
use std::fs as std_fs;
6-
use std::path::Path;
6+
use std::path::{Path, PathBuf};
77

88
use rustc_data_structures::unord::UnordMap;
99
use rustc_fs_util::link_or_copy;
@@ -20,6 +20,7 @@ pub fn copy_cgu_workproduct_to_incr_comp_cache_dir(
2020
sess: &Session,
2121
cgu_name: &str,
2222
files: &[(&'static str, &Path)],
23+
known_links: &[PathBuf],
2324
) -> Option<(WorkProductId, WorkProduct)> {
2425
debug!(?cgu_name, ?files);
2526
sess.opts.incremental.as_ref()?;
@@ -28,6 +29,10 @@ pub fn copy_cgu_workproduct_to_incr_comp_cache_dir(
2829
for (ext, path) in files {
2930
let file_name = format!("{cgu_name}.{ext}");
3031
let path_in_incr_dir = in_incr_comp_dir_sess(sess, &file_name);
32+
if known_links.contains(&path_in_incr_dir) {
33+
let _ = saved_files.insert(ext.to_string(), file_name);
34+
continue;
35+
}
3136
match link_or_copy(path, &path_in_incr_dir) {
3237
Ok(_) => {
3338
let _ = saved_files.insert(ext.to_string(), file_name);

0 commit comments

Comments
 (0)