Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Feature/847 allow caching multi arch builds #1467

Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
5 changes: 4 additions & 1 deletion docs/Caching.md
Original file line number Diff line number Diff line change
Expand Up @@ -23,7 +23,9 @@ In parallel, we also take into account in the hash:
### C/C++ compiler

For C/C++, the hash is generated with a blake3 digest of the preprocessed
file (-E with gcc/clang).
file (-E with gcc/clang). For compilations that specify multiple `-arch` flags,
these flags are rewritten to their corresponding preprocessor defines to allow
pre-processing the file (e.g `-arch x86_64` is rewritten to `-D__X86_64__=1`).

We also take into account in the hash:
* Hash of the compiler binary
Expand All @@ -32,6 +34,7 @@ We also take into account in the hash:
* File in which to generate dependencies.
* Commandline arguments for dependency generation
* Commandline arguments for the preprocessor
* Commandline arguments specifying the architecture to compile for
* Extra files that need to have their contents hashed
* Whether the compilation is generating profiling or coverage data
* Color mode
Expand Down
11 changes: 9 additions & 2 deletions src/compiler/c.rs
Original file line number Diff line number Diff line change
Expand Up @@ -98,6 +98,8 @@ pub struct ParsedArguments {
pub preprocessor_args: Vec<OsString>,
/// Commandline arguments for the preprocessor or the compiler.
pub common_args: Vec<OsString>,
/// Commandline arguments for the compiler that specify the architecture given
pub arch_args: Vec<OsString>,
/// Extra files that need to have their contents hashed.
pub extra_hash_files: Vec<PathBuf>,
/// Whether or not the `-showIncludes` argument is passed on MSVC
Expand Down Expand Up @@ -382,11 +384,16 @@ where
preprocessor_result.stdout.len()
);

// Create an argument vector containing both common and arch args, to
// use in creating a hash key
let mut common_and_arch_args = parsed_args.common_args.clone();
common_and_arch_args.extend(parsed_args.arch_args.to_vec());

let key = {
hash_key(
&executable_digest,
parsed_args.language,
&parsed_args.common_args,
&common_and_arch_args,
&extra_hashes,
&env_vars,
&preprocessor_result.stdout,
Expand Down Expand Up @@ -673,7 +680,7 @@ impl pkg::ToolchainPackager for CToolchainPackager {
}

/// The cache is versioned by the inputs to `hash_key`.
pub const CACHE_VERSION: &[u8] = b"10";
pub const CACHE_VERSION: &[u8] = b"11";

lazy_static! {
/// Environment variables that are factored into the cache key.
Expand Down
6 changes: 2 additions & 4 deletions src/compiler/clang.rs
Original file line number Diff line number Diff line change
Expand Up @@ -293,10 +293,8 @@ mod test {
)
);
assert_eq!(ovec!["-Iinclude", "-include", "file"], a.preprocessor_args);
assert_eq!(
ovec!["-arch", "xyz", "-fabc", "/winsysroot", "../some/dir"],
a.common_args
);
assert_eq!(ovec!["-fabc", "/winsysroot", "../some/dir"], a.common_args);
assert_eq!(ovec!["-arch", "xyz"], a.arch_args);
}

#[test]
Expand Down
2 changes: 2 additions & 0 deletions src/compiler/diab.rs
Original file line number Diff line number Diff line change
Expand Up @@ -291,6 +291,7 @@ where
dependency_args,
preprocessor_args,
common_args,
arch_args: vec![],
extra_hash_files: vec![],
msvc_show_includes: false,
profile_generate: false,
Expand Down Expand Up @@ -753,6 +754,7 @@ mod test {
dependency_args: vec![],
preprocessor_args: vec![],
common_args: vec![],
arch_args: vec![],
extra_hash_files: vec![],
msvc_show_includes: false,
profile_generate: false,
Expand Down
108 changes: 91 additions & 17 deletions src/compiler/gcc.rs
Original file line number Diff line number Diff line change
Expand Up @@ -142,6 +142,8 @@ ArgData! { pub

use self::ArgData::*;

const ARCH_FLAG: &str = "-arch";

// Mostly taken from https://github.com/ccache/ccache/blob/master/src/compopt.c#L32-L84
counted_array!(pub static ARGS: [ArgInfo<ArgData>; _] = [
flag!("-", TooHardFlag),
Expand Down Expand Up @@ -174,7 +176,7 @@ counted_array!(pub static ARGS: [ArgInfo<ArgData>; _] = [
take_arg!("-Xassembler", OsString, Separated, PassThrough),
take_arg!("-Xlinker", OsString, Separated, PassThrough),
take_arg!("-Xpreprocessor", OsString, Separated, PreprocessorArgument),
take_arg!("-arch", OsString, Separated, Arch),
take_arg!(ARCH_FLAG, OsString, Separated, Arch),
take_arg!("-aux-info", OsString, Separated, PassThrough),
take_arg!("-b", OsString, Separated, PassThrough),
flag!("-c", DoCompilation),
Expand Down Expand Up @@ -244,6 +246,7 @@ where
let mut dep_target = None;
let mut dep_flag = OsString::from("-MT");
let mut common_args = vec![];
let mut arch_args = vec![];
let mut preprocessor_args = vec![];
let mut dependency_args = vec![];
let mut extra_hash_files = vec![];
Expand All @@ -266,7 +269,6 @@ where
let mut outputs_gcno = false;
let mut xclangs: Vec<OsString> = vec![];
let mut color_mode = ColorMode::Auto;
let mut seen_arch = None;

// Custom iterator to expand `@` arguments which stand for reading a file
// and interpreting it as a list of more arguments.
Expand Down Expand Up @@ -355,13 +357,7 @@ where
_ => cannot_cache!("-x"),
};
}
Some(Arch(arch)) => {
match seen_arch {
Some(s) if &s != arch => cannot_cache!("multiple different -arch"),
_ => {}
};
seen_arch = Some(arch.clone());
}
Some(Arch(_)) => {}
Some(XClang(s)) => xclangs.push(s.clone()),
None => match arg {
Argument::Raw(ref val) => {
Expand All @@ -386,9 +382,9 @@ where
| Some(DiagnosticsColor(_))
| Some(DiagnosticsColorFlag)
| Some(NoDiagnosticsColorFlag)
| Some(Arch(_))
| Some(PassThrough(_))
| Some(PassThroughPath(_)) => &mut common_args,
Some(Arch(_)) => &mut arch_args,
Some(ExtraHashFile(path)) => {
extra_hash_files.push(cwd.join(path));
&mut common_args
Expand Down Expand Up @@ -567,6 +563,7 @@ where
dependency_args,
preprocessor_args,
common_args,
arch_args,
extra_hash_files,
msvc_show_includes: false,
profile_generate,
Expand Down Expand Up @@ -620,13 +617,30 @@ fn preprocess_cmd<T>(
}
}
}

// Explicitly rewrite the -arch args to be preprocessor defines of the form
// __arch__ so that they affect the preprocessor output but don't cause
// clang to error.
debug!("arch args before rewrite: {:?}", parsed_args.arch_args);
let rewritten_arch_args = parsed_args
.arch_args
.iter()
.filter(|&arg| arg.ne(ARCH_FLAG))
.filter_map(|arg| {
arg.to_str()
.map(|arg_string| format!("-D__{}__=1", arg_string).into())
})
.collect::<Vec<OsString>>();

cmd.arg(&parsed_args.input)
.args(&parsed_args.preprocessor_args)
.args(&parsed_args.dependency_args)
.args(&parsed_args.common_args)
.args(&rewritten_arch_args)
.env_clear()
.envs(env_vars.iter().map(|&(ref k, ref v)| (k, v)))
.current_dir(cwd);
debug!("cmd after -arch rewrite: {:?}", cmd);
}

#[allow(clippy::too_many_arguments)]
Expand Down Expand Up @@ -705,6 +719,7 @@ pub fn generate_compile_commands(
];
arguments.extend(parsed_args.preprocessor_args.clone());
arguments.extend(parsed_args.common_args.clone());
arguments.extend(parsed_args.arch_args.clone());
let command = CompileCommand {
executable: executable.to_owned(),
arguments,
Expand Down Expand Up @@ -1391,6 +1406,40 @@ mod test {
assert!(args.common_args.contains(&"-fdiagnostics-color".into()));
}

#[test]
fn test_preprocess_cmd_rewrites_archs() {
let args = stringvec!["-arch", "arm64", "-arch", "i386", "-c", "foo.cc"];
let parsed_args = match parse_arguments_(args, false) {
CompilerArguments::Ok(args) => args,
o => panic!("Got unexpected parse result: {:?}", o),
};
let mut cmd = MockCommand {
child: None,
args: vec![],
};
preprocess_cmd(
&mut cmd,
&parsed_args,
Path::new(""),
&[],
true,
CCompilerKind::Gcc,
true,
vec![],
);
// make sure the architectures were rewritten to prepocessor defines
let expected_args = ovec![
"-x",
"c++",
"-E",
"-fdirectives-only",
"foo.cc",
"-D__arm64__=1",
"-D__i386__=1"
];
assert_eq!(cmd.args, expected_args);
}

#[test]
fn pedantic_default() {
let args = stringvec!["-pedantic", "-c", "foo.cc"];
Expand Down Expand Up @@ -1630,15 +1679,39 @@ mod test {
o => panic!("Got unexpected parse result: {:?}", o),
}

assert_eq!(
CompilerArguments::CannotCache("multiple different -arch", None),
parse_arguments_(
stringvec![
"-fPIC", "-arch", "arm64", "-arch", "i386", "-o", "foo.o", "-c", "foo.cpp"
],
false
let args =
stringvec!["-fPIC", "-arch", "arm64", "-arch", "i386", "-o", "foo.o", "-c", "foo.cpp"];
let ParsedArguments {
input,
language,
compilation_flag,
outputs,
preprocessor_args,
msvc_show_includes,
common_args,
arch_args,
..
} = match parse_arguments_(args, false) {
CompilerArguments::Ok(args) => args,
o => panic!("Got unexpected parse result: {:?}", o),
};
assert_eq!(Some("foo.cpp"), input.to_str());
assert_eq!(Language::Cxx, language);
assert_eq!(Some("-c"), compilation_flag.to_str());
assert_map_contains!(
outputs,
(
"obj",
ArtifactDescriptor {
path: "foo.o".into(),
optional: false
}
)
);
assert!(preprocessor_args.is_empty());
assert_eq!(ovec!["-fPIC"], common_args);
assert_eq!(ovec!["-arch", "arm64", "-arch", "i386"], arch_args);
assert!(!msvc_show_includes);
}

#[test]
Expand Down Expand Up @@ -1706,6 +1779,7 @@ mod test {
dependency_args: vec![],
preprocessor_args: vec![],
common_args: vec![],
arch_args: vec![],
extra_hash_files: vec![],
msvc_show_includes: false,
profile_generate: false,
Expand Down
3 changes: 3 additions & 0 deletions src/compiler/msvc.rs
Original file line number Diff line number Diff line change
Expand Up @@ -818,6 +818,7 @@ pub fn parse_arguments(
dependency_args,
preprocessor_args,
common_args,
arch_args: vec![],
extra_hash_files,
msvc_show_includes: show_includes,
profile_generate,
Expand Down Expand Up @@ -1622,6 +1623,7 @@ mod test {
dependency_args: vec![],
preprocessor_args: vec![],
common_args: vec![],
arch_args: vec![],
extra_hash_files: vec![],
msvc_show_includes: false,
profile_generate: false,
Expand Down Expand Up @@ -1681,6 +1683,7 @@ mod test {
dependency_args: vec![],
preprocessor_args: vec![],
common_args: vec![],
arch_args: vec![],
extra_hash_files: vec![],
msvc_show_includes: false,
profile_generate: false,
Expand Down
2 changes: 2 additions & 0 deletions src/compiler/tasking_vx.rs
Original file line number Diff line number Diff line change
Expand Up @@ -272,6 +272,7 @@ where
dependency_args: vec![],
preprocessor_args,
common_args,
arch_args: vec![],
extra_hash_files: vec![],
msvc_show_includes: false,
profile_generate: false,
Expand Down Expand Up @@ -693,6 +694,7 @@ mod test {
dependency_args: vec![],
preprocessor_args: vec![],
common_args: vec![],
arch_args: vec![],
extra_hash_files: vec![],
msvc_show_includes: false,
profile_generate: false,
Expand Down