Skip to content

Commit 2f9fff2

Browse files
committed
Keep multiple files per work-product
In the older version, a `.o` and ` .bc` file were separate work-products. This newer version keeps, for each codegen-unit, a set of files of different kinds. We assume that if any kinds are available then all the kinds we need are available, since the precise set of switches will depend on attributes and command-line switches. Should probably test this: the effect of changing attributes in particular might not be successfully tracked?
1 parent ceeb158 commit 2f9fff2

File tree

12 files changed

+202
-92
lines changed

12 files changed

+202
-92
lines changed

src/librustc/dep_graph/dep_node.rs

+1-3
Original file line numberDiff line numberDiff line change
@@ -246,7 +246,5 @@ impl<D: Clone + Debug> DepNode<D> {
246246
/// the need to be mapped or unmapped. (This ensures we can serialize
247247
/// them even in the absence of a tcx.)
248248
#[derive(Clone, Debug, PartialEq, Eq, Hash, RustcEncodable, RustcDecodable)]
249-
pub enum WorkProductId {
250-
PartitionObjectFile(String), // see (*TransPartition) below
251-
}
249+
pub struct WorkProductId(pub String);
252250

src/librustc/dep_graph/graph.rs

+4-3
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,7 @@
1010

1111
use hir::def_id::DefId;
1212
use rustc_data_structures::fnv::FnvHashMap;
13+
use session::config::OutputType;
1314
use std::cell::{Ref, RefCell};
1415
use std::rc::Rc;
1516
use std::sync::Arc;
@@ -157,11 +158,11 @@ impl DepGraph {
157158
/// previous hash. If it matches up, we can reuse the object file.
158159
#[derive(Clone, Debug, RustcEncodable, RustcDecodable)]
159160
pub struct WorkProduct {
160-
/// extra hash used to decide if work-product is still suitable;
161+
/// Extra hash used to decide if work-product is still suitable;
161162
/// note that this is *not* a hash of the work-product itself.
162163
/// See documentation on `WorkProduct` type for an example.
163164
pub input_hash: u64,
164165

165-
/// filename storing this work-product (found in the incr. comp. directory)
166-
pub file_name: String,
166+
/// Saved files associated with this CGU
167+
pub saved_files: Vec<(OutputType, String)>,
167168
}

src/librustc/session/config.rs

+13-10
Original file line numberDiff line numberDiff line change
@@ -61,7 +61,7 @@ pub enum DebugInfoLevel {
6161
FullDebugInfo,
6262
}
6363

64-
#[derive(Clone, Copy, PartialEq, Eq, Hash)]
64+
#[derive(Clone, Copy, PartialEq, Eq, Hash, Debug, RustcEncodable, RustcDecodable)]
6565
pub enum OutputType {
6666
Bitcode,
6767
Assembly,
@@ -105,6 +105,17 @@ impl OutputType {
105105
OutputType::DepInfo => "dep-info",
106106
}
107107
}
108+
109+
pub fn extension(&self) -> &'static str {
110+
match *self {
111+
OutputType::Bitcode => "bc",
112+
OutputType::Assembly => "s",
113+
OutputType::LlvmAssembly => "ll",
114+
OutputType::Object => "o",
115+
OutputType::DepInfo => "d",
116+
OutputType::Exe => "",
117+
}
118+
}
108119
}
109120

110121
#[derive(Clone)]
@@ -215,15 +226,7 @@ impl OutputFilenames {
215226
flavor: OutputType,
216227
codegen_unit_name: Option<&str>)
217228
-> PathBuf {
218-
let extension = match flavor {
219-
OutputType::Bitcode => "bc",
220-
OutputType::Assembly => "s",
221-
OutputType::LlvmAssembly => "ll",
222-
OutputType::Object => "o",
223-
OutputType::DepInfo => "d",
224-
OutputType::Exe => "",
225-
};
226-
229+
let extension = flavor.extension();
227230
self.temp_path_ext(extension, codegen_unit_name)
228231
}
229232

src/librustc_incremental/persist/load.rs

+21-10
Original file line numberDiff line numberDiff line change
@@ -260,11 +260,20 @@ fn reconcile_work_products<'a, 'tcx>(tcx: TyCtxt<'a, 'tcx, 'tcx>,
260260
debug!("reconcile_work_products: dep-node for {:?} is dirty", swp);
261261
delete_dirty_work_product(tcx, swp);
262262
} else {
263-
let path = in_incr_comp_dir(tcx.sess, &swp.work_product.file_name).unwrap();
264-
if path.exists() {
263+
let all_files_exist =
264+
swp.work_product
265+
.saved_files
266+
.iter()
267+
.all(|&(_, ref file_name)| {
268+
let path = in_incr_comp_dir(tcx.sess, &file_name).unwrap();
269+
path.exists()
270+
});
271+
if all_files_exist {
272+
debug!("reconcile_work_products: all files for {:?} exist", swp);
265273
tcx.dep_graph.insert_previous_work_product(&swp.id, swp.work_product);
266274
} else {
267-
debug!("reconcile_work_products: file for {:?} does not exist", swp);
275+
debug!("reconcile_work_products: some file for {:?} does not exist", swp);
276+
delete_dirty_work_product(tcx, swp);
268277
}
269278
}
270279
}
@@ -273,13 +282,15 @@ fn reconcile_work_products<'a, 'tcx>(tcx: TyCtxt<'a, 'tcx, 'tcx>,
273282
fn delete_dirty_work_product(tcx: TyCtxt,
274283
swp: SerializedWorkProduct) {
275284
debug!("delete_dirty_work_product({:?})", swp);
276-
let path = in_incr_comp_dir(tcx.sess, &swp.work_product.file_name).unwrap();
277-
match fs::remove_file(&path) {
278-
Ok(()) => { }
279-
Err(err) => {
280-
tcx.sess.warn(
281-
&format!("file-system error deleting outdated file `{}`: {}",
282-
path.display(), err));
285+
for &(_, ref file_name) in &swp.work_product.saved_files {
286+
let path = in_incr_comp_dir(tcx.sess, file_name).unwrap();
287+
match fs::remove_file(&path) {
288+
Ok(()) => { }
289+
Err(err) => {
290+
tcx.sess.warn(
291+
&format!("file-system error deleting outdated file `{}`: {}",
292+
path.display(), err));
293+
}
283294
}
284295
}
285296
}

src/librustc_incremental/persist/work_product.rs

+36-32
Original file line numberDiff line numberDiff line change
@@ -13,47 +13,51 @@
1313
use persist::util::*;
1414
use rustc::dep_graph::{WorkProduct, WorkProductId};
1515
use rustc::session::Session;
16+
use rustc::session::config::OutputType;
1617
use rustc::util::fs::link_or_copy;
17-
use std::fs;
18-
use std::path::Path;
18+
use std::path::PathBuf;
1919
use std::sync::Arc;
2020

2121
pub fn save_trans_partition(sess: &Session,
22-
partition_name: &str,
22+
cgu_name: &str,
2323
partition_hash: u64,
24-
path_to_obj_file: &Path) {
25-
debug!("save_trans_partition({:?},{},{})",
26-
partition_name,
24+
files: &[(OutputType, PathBuf)]) {
25+
debug!("save_trans_partition({:?},{},{:?})",
26+
cgu_name,
2727
partition_hash,
28-
path_to_obj_file.display());
28+
files);
2929
if sess.opts.incremental.is_none() {
3030
return;
3131
}
32-
let id = Arc::new(WorkProductId::PartitionObjectFile(partition_name.to_string()));
33-
let file_name = format!("cgu-{}", partition_name);
34-
let path_in_incr_dir = in_incr_comp_dir(sess, &file_name).unwrap();
32+
let work_product_id = Arc::new(WorkProductId(cgu_name.to_string()));
3533

36-
// try to delete the file if it already exists
37-
//
38-
// FIXME(#34955) we can be smarter here -- if we are re-using, no need to do anything
39-
if path_in_incr_dir.exists() {
40-
let _ = fs::remove_file(&path_in_incr_dir);
41-
}
34+
let saved_files: Option<Vec<_>> =
35+
files.iter()
36+
.map(|&(kind, ref path)| {
37+
let file_name = format!("cgu-{}.{}", cgu_name, kind.extension());
38+
let path_in_incr_dir = in_incr_comp_dir(sess, &file_name).unwrap();
39+
match link_or_copy(path, &path_in_incr_dir) {
40+
Ok(_) => Some((kind, file_name)),
41+
Err(err) => {
42+
sess.warn(&format!("error copying object file `{}` \
43+
to incremental directory as `{}`: {}",
44+
path.display(),
45+
path_in_incr_dir.display(),
46+
err));
47+
None
48+
}
49+
}
50+
})
51+
.collect();
52+
let saved_files = match saved_files {
53+
Some(v) => v,
54+
None => return,
55+
};
4256

43-
match link_or_copy(path_to_obj_file, &path_in_incr_dir) {
44-
Ok(_) => {
45-
let work_product = WorkProduct {
46-
input_hash: partition_hash,
47-
file_name: file_name,
48-
};
49-
sess.dep_graph.insert_work_product(&id, work_product);
50-
}
51-
Err(err) => {
52-
sess.warn(&format!("error copying object file `{}` \
53-
to incremental directory as `{}`: {}",
54-
path_to_obj_file.display(),
55-
path_in_incr_dir.display(),
56-
err));
57-
}
58-
}
57+
let work_product = WorkProduct {
58+
input_hash: partition_hash,
59+
saved_files: saved_files,
60+
};
61+
62+
sess.dep_graph.insert_work_product(&work_product_id, work_product);
5963
}

src/librustc_trans/back/write.rs

+38-18
Original file line numberDiff line numberDiff line change
@@ -337,6 +337,8 @@ struct CodegenContext<'a> {
337337
remark: Passes,
338338
// Worker thread number
339339
worker: usize,
340+
// Directory where incremental data is stored (if any)
341+
incremental: Option<PathBuf>,
340342
}
341343

342344
impl<'a> CodegenContext<'a> {
@@ -347,6 +349,7 @@ impl<'a> CodegenContext<'a> {
347349
plugin_passes: sess.plugin_llvm_passes.borrow().clone(),
348350
remark: sess.opts.cg.remark.clone(),
349351
worker: 0,
352+
incremental: sess.opts.incremental.clone(),
350353
}
351354
}
352355
}
@@ -612,7 +615,7 @@ unsafe fn optimize_and_codegen(cgcx: &CodegenContext,
612615

613616
if copy_bc_to_obj {
614617
debug!("copying bitcode {:?} to obj {:?}", bc_out, obj_out);
615-
if let Err(e) = fs::copy(&bc_out, &obj_out) {
618+
if let Err(e) = link_or_copy(&bc_out, &obj_out) {
616619
cgcx.handler.err(&format!("failed to copy bitcode to object file: {}", e));
617620
}
618621
}
@@ -754,9 +757,19 @@ pub fn run_passes(sess: &Session,
754757

755758
// If in incr. comp. mode, preserve the `.o` files for potential re-use
756759
for mtrans in trans.modules.iter() {
757-
let path_to_obj = crate_output.temp_path(OutputType::Object, Some(&mtrans.name));
758-
debug!("wrote module {:?} to {:?}", mtrans.name, path_to_obj);
759-
save_trans_partition(sess, &mtrans.name, mtrans.symbol_name_hash, &path_to_obj);
760+
let mut files = vec![];
761+
762+
if modules_config.emit_obj {
763+
let path = crate_output.temp_path(OutputType::Object, Some(&mtrans.name));
764+
files.push((OutputType::Object, path));
765+
}
766+
767+
if modules_config.emit_bc {
768+
let path = crate_output.temp_path(OutputType::Bitcode, Some(&mtrans.name));
769+
files.push((OutputType::Bitcode, path));
770+
}
771+
772+
save_trans_partition(sess, &mtrans.name, mtrans.symbol_name_hash, &files);
760773
}
761774

762775
// All codegen is finished.
@@ -941,20 +954,24 @@ fn execute_work_item(cgcx: &CodegenContext,
941954
work_item.config,
942955
work_item.output_names);
943956
}
944-
ModuleSource::Preexisting(ref buf) => {
945-
let obj_out = work_item.output_names.temp_path(OutputType::Object,
946-
Some(&work_item.mtrans.name));
947-
debug!("copying pre-existing module `{}` from {} to {}",
948-
work_item.mtrans.name,
949-
buf.display(),
950-
obj_out.display());
951-
match link_or_copy(buf, &obj_out) {
952-
Ok(()) => { }
953-
Err(err) => {
954-
cgcx.handler.err(&format!("unable to copy {} to {}: {}",
955-
buf.display(),
956-
obj_out.display(),
957-
err));
957+
ModuleSource::Preexisting(wp) => {
958+
let incremental = cgcx.incremental.as_ref().unwrap();
959+
let name = &work_item.mtrans.name;
960+
for (kind, saved_file) in wp.saved_files {
961+
let obj_out = work_item.output_names.temp_path(kind, Some(name));
962+
let source_file = incremental.join(&saved_file);
963+
debug!("copying pre-existing module `{}` from {:?} to {}",
964+
work_item.mtrans.name,
965+
source_file,
966+
obj_out.display());
967+
match link_or_copy(&source_file, &obj_out) {
968+
Ok(()) => { }
969+
Err(err) => {
970+
cgcx.handler.err(&format!("unable to copy {} to {}: {}",
971+
source_file.display(),
972+
obj_out.display(),
973+
err));
974+
}
958975
}
959976
}
960977
}
@@ -994,6 +1011,8 @@ fn run_work_multithreaded(sess: &Session,
9941011
let mut tx = Some(tx);
9951012
futures.push(rx);
9961013

1014+
let incremental = sess.opts.incremental.clone();
1015+
9971016
thread::Builder::new().name(format!("codegen-{}", i)).spawn(move || {
9981017
let diag_handler = Handler::with_emitter(true, false, box diag_emitter);
9991018

@@ -1005,6 +1024,7 @@ fn run_work_multithreaded(sess: &Session,
10051024
plugin_passes: plugin_passes,
10061025
remark: remark,
10071026
worker: i,
1027+
incremental: incremental,
10081028
};
10091029

10101030
loop {

src/librustc_trans/base.rs

+3-5
Original file line numberDiff line numberDiff line change
@@ -43,10 +43,9 @@ use rustc::ty::subst::{self, Substs};
4343
use rustc::traits;
4444
use rustc::ty::{self, Ty, TyCtxt, TypeFoldable};
4545
use rustc::ty::adjustment::CustomCoerceUnsized;
46-
use rustc::dep_graph::DepNode;
46+
use rustc::dep_graph::{DepNode, WorkProduct};
4747
use rustc::hir::map as hir_map;
4848
use rustc::util::common::time;
49-
use rustc_incremental::in_incr_comp_dir;
5049
use rustc::mir::mir_map::MirMap;
5150
use rustc_data_structures::graph::OUTGOING;
5251
use session::config::{self, NoDebugInfo, FullDebugInfo};
@@ -103,7 +102,6 @@ use std::cell::{Cell, RefCell};
103102
use std::collections::HashMap;
104103
use std::ptr;
105104
use std::rc::Rc;
106-
use std::path::PathBuf;
107105
use std::str;
108106
use std::{i8, i16, i32, i64};
109107
use syntax_pos::{Span, DUMMY_SP};
@@ -2721,7 +2719,7 @@ pub fn trans_crate<'a, 'tcx>(tcx: TyCtxt<'a, 'tcx, 'tcx>,
27212719
fn trans_reuse_previous_work_products(tcx: TyCtxt,
27222720
codegen_units: &[CodegenUnit],
27232721
symbol_map: &SymbolMap)
2724-
-> Vec<Option<PathBuf>> {
2722+
-> Vec<Option<WorkProduct>> {
27252723
debug!("trans_reuse_previous_work_products()");
27262724
codegen_units
27272725
.iter()
@@ -2735,7 +2733,7 @@ fn trans_reuse_previous_work_products(tcx: TyCtxt,
27352733
if let Some(work_product) = tcx.dep_graph.previous_work_product(&id) {
27362734
if work_product.input_hash == hash {
27372735
debug!("trans_reuse_previous_work_products: reusing {:?}", work_product);
2738-
return Some(in_incr_comp_dir(tcx.sess, &work_product.file_name).unwrap());
2736+
return Some(work_product);
27392737
} else {
27402738
debug!("trans_reuse_previous_work_products: \
27412739
not reusing {:?} because hash changed to {:?}",

0 commit comments

Comments
 (0)