-
Notifications
You must be signed in to change notification settings - Fork 2.5k
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
Profile Overrides (RFC #2282 Part 1) #5384
Conversation
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Holy cow, this looks awesome @ehuss !
src/doc/src/reference/unstable.md
Outdated
opt-level = 3 | ||
|
||
# All dependencies (but not this crate itself) will be compiled | ||
# with -Copt-level=2 . This includes build dependencies. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This probably should mention workspaces?
tests/testsuite/profiles.rs
Outdated
[RUNNING] `rustc --crate-name foo [..]` | ||
[FINISHED] dev [unoptimized + debuginfo] target(s) in [..]", | ||
) | ||
// TODO: does_not_contain does not support patterns! |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think it does support [..]
? Looks like [COMPILING]
is missing?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I don't think it does, see:
cargo/tests/testsuite/cargotest/support/mod.rs
Lines 689 to 700 in 5ac4ab1
MatchKind::NotPresent => { | |
if !actual.contains(out) { | |
Ok(()) | |
} else { | |
Err(format!( | |
"expected not to find:\n\ | |
{}\n\n\ | |
but found in output:\n\ | |
{}", | |
out, actual | |
)) | |
} |
I mentioned it on IRC, I'll take a look at fixing it.
tests/testsuite/test.rs
Outdated
@@ -1649,6 +1649,7 @@ fn test_run_implicit_test_target() { | |||
) | |||
.build(); | |||
|
|||
// TODO FIXME: This needs to better verify that examples are not built. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This can be done by putting a compile-time error in the example.
tests/testsuite/test.rs
Outdated
@@ -1742,13 +1742,13 @@ fn test_run_implicit_example_target() { | |||
) | |||
.build(); | |||
|
|||
// TODO FIXME - verify example does NOT get run. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Example could panic with runtime error to detect this.
@@ -347,20 +369,65 @@ impl<'de> de::Deserialize<'de> for U32OrBool { | |||
} | |||
|
|||
#[derive(Deserialize, Serialize, Clone, Debug, Default)] | |||
#[serde(rename_all = "kebab-case")] |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
❤️
src/cargo/util/toml/mod.rs
Outdated
bail!("Profile overrides cannot be nested."); | ||
} | ||
if self.panic.is_some() { | ||
bail!("`panic` may not be specified in a build override.") |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
s/build/profile/ ?
src/cargo/ops/cargo_compile.rs
Outdated
Check { test: bool }, | ||
/// A target being built for a benchmark. | ||
Bench, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I wonder if we can merge Test
and Bench
, or perhaps move Bench
to ProfileFor
? From compiler pov, I think test & bench are indistinguishable?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
An interesting idea. I'll think about it some more, but it may complicate cargo_compile
since it needs to know the overall mode we're in (like selecting default targets, etc.).
src/cargo/core/profiles.rs
Outdated
// ancestor's profile. However `cargo clean -p` can hit this | ||
// path. | ||
// TODO: I think `cargo clean -p xxx` is not cleaning out | ||
// the "OUT_DIR" directory. This is not a new bug. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
OUT_DIR
here is the --out-dir
? It can't be cleaned by clean
at all I think.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I don't think so. I was thinking of the build_script_out_dir
which is something like target/debug/build/pkgname-hash/out
. But I was mistaken, I was actually just seeing the output
, stderr
, and root-output
directories left behind. Probably not too important, and unrelated to these changes.
src/cargo/core/profiles.rs
Outdated
cx: &Context<'a, 'cfg>, | ||
map: &mut HashMap<Unit<'a>, Profile>, | ||
) { | ||
if !map.contains_key(unit) { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
It's slightly better to use early returns here, to save indentation and keep mental stack shallow:
if map.contains_key(unit) {
return;
}
Ideally, I think we should just top-sort the units once and avoid writing recursive functions all over the place, but this definitely shouldn't be done in this PR.
let unit_deps = compute_deps(unit, cx, deps, profile_for)?; | ||
let to_insert: Vec<_> = unit_deps.iter().map(|&(unit, _)| unit).collect(); | ||
deps.insert(*unit, to_insert); | ||
for (unit, profile_for) in unit_deps { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
If I add assert_eq!(unit.profile_for, profile_for);
it is not triggered, so, perhaps, we don't need to return pairs after all?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
There was a bug. Since units no longer include ProfileFor, this is no longer the case.
👍 👍 👍
I think current implementation is OK: linear scan is perfectly fine until you have hundreds of members.
I have no opinion here :)
It still works correctly when no overrides are present, and otherwise it is a minor detail, so I suggest to punt on this for know: current solution seems at least OK to me!
Yeah, that seems generally wrong, but OK for now, because it more or less matches the current behavior. I would say that this is probably an instance where we do need a separate, not dev/not release profile, but let's deal with that later.
Hm, is this the current behavior? It definitely is a bug!
Let's give at least a warning if it's not too hard: typoes are likely in the names of packages, especially with
It is allowed now, right? Then I think error is probably not feasible, but the warning seems useful! |
src/cargo/core/profiles.rs
Outdated
} | ||
|
||
impl ProfileFor { | ||
pub fn all_values() -> Vec<ProfileFor> { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Mincooptimization, but here and in CompileMode::all_modes
, we could return &'static [Stuff]
, to save some allocations.
/// command-line for `cargo rustc` and `cargo rustdoc`. These commands | ||
/// only support one target, but we don't want the args passed to any | ||
/// dependencies. | ||
pub extra_compiler_args: HashMap<Unit<'a>, Vec<String>>, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Perhaps we can use Option<(Unit, Vec<String>)>
instead? Let's also document that this is overloaded in the sense that this can be arguments for different tools, rustc
or rustdoc
, depending on the command.
src/cargo/ops/cargo_compile.rs
Outdated
// Helper for creating a Unit struct. | ||
let new_unit = | ||
|pkg: &'a Package, target: &'a Target, mode: CompileMode, profile_for: ProfileFor| { | ||
let actual_profile_for = if profile_for != ProfileFor::Any { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Just a personal taste, but I would name this just profile_for
to shadow the input parameter, so that, down the line, we can't use the wrong profile_for
.
Thanks @matklad for doing such a detailed review so quickly! Unfortunately, I've discovered a pretty fundamental flaw in how it works. Units are not being properly de-duplicated. Example: test→lib and bin→lib, the lib gets compiled twice (once as a regular dependency, and again as a test dependency). But if you don't have I think Profile really must be a part of Unit. I had a previous attempt that had it part of the Unit, but I was concerned about the huge amount of copying. However, I was using Strings before (instead of InternedString), and it seemed much worse at the time (require lots of Sorry for the churn, I'll try to fix this today and address all your comments. |
Wow, that's interesting! I quite like the current approach actually, but yeah, looks like we'd better to store raw flags to avoid duplication 😿 |
I've pushed the update that now embeds the profile inside |
☔ The latest upstream changes (presumably #5392) made this pull request unmergeable. Please resolve the merge conflicts. |
// cleared, and avoid building the lib thrice (once with `panic`, once | ||
// without, once for --test). In particular, the lib included for | ||
// doctests and examples are `Build` mode here. | ||
let profile_for = if unit.mode.is_any_test() || cx.build_config.test { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Hm, I am genuinely not sure what the correct behavior would be here? Suppose we are compiling the integration test, which only executes the binary in the same package. It is reasonable to assume that the binary might be compiled with panic=abort
, but the test itself should be compiled with panic=unwind
, so common dependencies should be compiled twice.
Perhaps the best path forward is to preserve the current, buggy, behavior, land this PR, and then fix the behavior in a follow-up PR?
src/cargo/ops/cargo_compile.rs
Outdated
// `panic` set. | ||
// | ||
// As a consequence, Examples and Binaries get compiled | ||
// without `panic` set. This probably isn't a bad deal. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ah yeah, this is exactly what I was thinking about... Hm, compiling binaries with different panic setting actually seems like a big deal to me...
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yea, it's a tricky issue, I was going to ask about it. The immediate problem is that doc tests attempt to link with everything in Context.libraries
. I'm uncertain what the easiest way to fix that would be. One thought I had would be to change run_doc_tests
to only grab the dependencies for the CompileMode::Doctest
unit and remove Context.libraries
. There would be some other minor things to accommodate that, but I think it should work. I can do it now or later, let me know. It would remove this weird special-case.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Would it be difficult to preserve exactly the behavior (and bugs :) ) we have today? I think this would be the best way forward for now, because changes which both add unstable features and fix/modify stable behavior are tricky...
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yea, it should be preserving the old behavior. I have a few changes I haven't pushed, yet, that ensure that. I'm keeping a list of behavior changes, and I'll post it soon. A few are clearly bug fixes, but a couple are questionable. We can review them when I post it.
The CI failures are due to the introduction of rust-lang/rust#49289 (starting with nightly 2018-04-15). |
(I've updated the first comment with some other Misc Fixes that probably aren't too controversial, but please review them. Sorry for the long comment, I just wanted to be thorough.) The latest push includes some tests ( Questions
Behavior Changes to DiscussThe following are new changes in behavior that I'm uncertain if we should keep or change.
Panic Changes and IssuesGetting
Known Bugs
|
Yeah, that seems reasonable. Let's add |
I think we can add
|
Yep, I believe we can safely deprecate this profile. That is, leave the syntax in Cargo.toml, give a warning about it, and remove traces of doc profile from the code. Eventually I think we'll do the same for
I personally think about it as just "package profile". We need
I think it is important to remove this duplication. Could we get rid of the
Hm, so currently build_scripts are compiled only with
Long term, I believe the current behavior of using both I don't think there should be any sharing between the build script and the main code. Especially for cross-compilation, it seems unreasonable to use the same profile for both. |
This is the current behavior, right? I believe this is very wrong, but fixing it indeed adds even more compilations. So let's leave it as is for know.
Could you elaborate why do you think so? At least to me, they seem very similar indeed! And my hypothecical
I think it's fine to leave this duplication as is, but, if we are to fix it, then perhaps " Internally it thinks one is "dev without panic" (for tests) and the other is regular "dev" (for build.rs)." is the bit that needs adjustments? |
Would you like that now, or in a separate PR (since this one is getting a little long)?
Hm, that is a little clearer. Not sure if anyone else wants to weigh in?
It would be nice, because once the
This is how it works (before and after this change) for compiling the
Isn't that what This brings up a issue that was nagging at me. I didn't fully understand the reasoning for the override precedence in the RFC. Since other rules take precedence over
The only reason this exception is made is to ensure the
Correct, it's a small, trivial change to say "panic = None" for build scripts in the profile builder. |
A separate one. I'd prefer to merge this sooner, and not to fix all profile bugs at once :)
I personally don't care about syntax at this stage. There's already
Another option is to split
Oh, right! I've completely misunderstood your original comment then! The new behavior indeed seems more correct to me as well!
Yep, it's almost like build-override, with two important differences: you don't need to override it twice, and, by default, build profile is the same for
Hm, I think we though that dependencies are shared between the library and the build script. Because they are independent, priority for build_overrides indeed does not make sense. |
OK, let me know if you want to iterate on this now or later. It's pretty easy to change. Let me know if there's anything else. I think I've gone through the issues I can think of. |
@@ -38,27 +50,33 @@ fn deps_of<'a, 'b, 'cfg>( | |||
unit: &Unit<'a>, | |||
cx: &Context<'a, 'cfg>, | |||
deps: &'b mut HashMap<Unit<'a>, Vec<Unit<'a>>>, | |||
profile_for: ProfileFor, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I feel uneasy about this profile_for
argument. This function is memoized, so the sequence of calls deps_of(u, cx, deps, ProfileFor::Any); deps_of(u, cx, deps, ProfileFor::TestDependency)
will return the same result each time (and the result might be different if we swap the order of calls).
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I understand what you mean. In practice, that's never a problem for TestDependency
since it only exists for clearing panic
, and you'll never ask for a unit with panic set as a TestDependency
.
I'm trying to think of a scenario where it would matter for CustomBuild
, but I can't think of any. It might matter if we later add a "build-script"-specific profile, and you have an override deep in a dependency chain.
Adding it to the map would require a separate de-duplication phase, which I think would add ~15 lines of code.
Let me know if you think it's important. I'd lean towards not changing it unless there's a testable case. Or to make it easier to add a build-script profile later if there's a strong chance of that.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think I'm fine with it: I would love to refactor code in such a way that this problem just not arises, or at least to put some sort of run-time assertion, but I have no ideas of how to actually do this. Storing pairs in the HashMap
would be worse than the status quo.
Let's perhaps just add a small comment here though?
@@ -273,28 +283,65 @@ fn maybe_lib<'a, 'cfg>(unit: &Unit<'a>, cx: &Context<'a, 'cfg>) -> Option<Unit<' | |||
/// script itself doesn't have any dependencies, so even in that case a unit | |||
/// of work is still returned. `None` is only returned if the package has no | |||
/// build script. | |||
fn dep_build_script<'a, 'cfg>(unit: &Unit<'a>, cx: &Context<'a, 'cfg>) -> Option<Unit<'a>> { | |||
fn dep_build_script<'a>(cx: &Context, unit: &Unit<'a>) -> Option<(Unit<'a>, ProfileFor)> { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Some function in this module now use cx, unit arg order, and some use (unit, cx). Let's unify them to one order? I don't have any preference for a specific order, as long as it is an order :-)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yea, I derped. I had an earlier version that I had removed Context, but when I added it back I should have kept it the same.
foo custom build PROFILE=release DEBUG=false OPT_LEVEL=3 | ||
[RUNNING] `rustc --crate-name foo src[/]lib.rs --crate-type lib --emit=dep-info,link -C opt-level=3 -C panic=abort -C codegen-units=2 [..] | ||
[RUNNING] `rustc --crate-name foo src[/]main.rs --crate-type bin --emit=dep-info,link -C opt-level=3 -C panic=abort -C codegen-units=2 [..] | ||
[FINISHED] release [optimized] [..] |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I guess I don't understand something about panic handling... How this even works? We link bar
, which is compiled with panic=unwind
into foo
, which is compiled with panic=abort
and rustc
does not complain. Is it the expected behavior? Testing locally, I see that unwind -> abort
direction works, and abort -> unwind
fails. Does this mean that it is always safe to compile dependencies with panic = unwind
?
This probably has nothing to do with the PR though, because the behavior has not changed here.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Hm, yea, I don't know much how panic handling is implemented. I did a test where I attempt to panic in a dependency that was compiled with unwind, and it still aborts. It looks like it just affects whether libpanic_abort
or libpanic_unwind
is linked. I'm guessing since the API is the same, the dependencies don't know the difference.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
It looks like it just affects whether libpanic_abort or libpanic_unwind is linked. I
I would guess that it also affects generation of landing pads? So we might be getting a bunch of dead code here?
@alexcrichton could you explain how mixing panic modes is supposed to work?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ah yes the observed behavior here is expected, we , for example, ship panic=unwind libstd to everyone but still allow compiling with panic=abort
I've reviewed everything except the tests (which are super awesome!), and I don't have any more comments here! I would probably want to inrdocude a separate feature flag to build override, and maybe switch to a separate build profile, but this don't need to happen in this PR. I'd like to hear what @alexcrichton thinks as well! :) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Everything is looking amazing here. I'm blown away at the thoroughness and cleanliness of the patch, thanks so much for taking this on @ehuss!
I've got basically a few tiny nits below, but overall the strategy here sounds great. I don't think I've fully grok'd everything yet, especially the ProfileFor
threw me particularly for a spin around the unit_dependencies.rs
files. Nothing here looks wrong though, tests are all passing, and the caveats you've mentioned along the way all sound great to me.
tl;dr; I'd be happy to land this at any time!
src/cargo/core/profiles.rs
Outdated
} | ||
|
||
fn validate_packages(&self, shell: &mut Shell, packages: &HashSet<&str>) -> CargoResult<()> { | ||
if let Some(ref toml) = self.toml { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Stylistically I'm always personally a fan of reducing indent here:
let toml = match self.toml {
Some(ref toml) => toml,
None => return Ok(()),
};
// ...
src/cargo/core/profiles.rs
Outdated
if let Some(ref toml) = self.toml { | ||
if let Some(ref overrides) = toml.overrides { | ||
for key in overrides.keys().filter(|k| k.as_str() != "*") { | ||
if !packages.contains(key.as_str()) { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think this is something we'll want to note on the tracking issue, I'd expect the packages
set here to be more like a HashSet<PackageId>
(or a Resolve
) and the keys of overrides
here would be a PackageIdSpec
in theory
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I'm not sure I follow. Are you saying that overrides could support full spec strings? That looks like it'd be pretty easy to support.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Basically yeah. We've got the idea of a "package id specification" which is intended to describe a particular crate in a crate graph unbambiguously, and this would allow you to, for example, customize the compilation of foo
from crates.io but not customize foo
from github.com/example/foo (or something like that)
…lease`. This changes it so that `Build` mode is not set in the Unit, since there is no difference between Bench and Test once the profile has been selected.
📋 Looks like this PR is still in progress, ignoring approval |
@bors r+ My undestanding is that this is not really a WIP, but, anyway, we could land additional fixes in separate PR's just fine! |
📌 Commit e82e9c1 has been approved by |
Profile Overrides (RFC #2282 Part 1) Profile Overrides (RFC #2282 Part 1) WIP: Putting this up before I dig into writing tests, but should be mostly complete. I also have a variety of questions below. This implements the ability to override profiles for dependencies and build scripts. This includes a general rework of how profiles work internally. Closes #5298. Profile overrides are available with `profile-overrides` set in `cargo-features` in the manifest. Part 2 is to implement profiles in config files (to be in a separate PR). General overview of changes: - `Profiles` moved to `core/profiles.rs`. All profile selection is centralized there. - Removed Profile flags `test`, `doc`, `run_custom_build`, and `check`. - Removed `Profile` from `Unit` and replaced it with two enums: `CompileMode` and `ProfileFor`. This is the minimum information needed to compute profiles at a later stage. - Also removed `rustc_args`/`rustdoc_args` from `Profile` and place them in `Context`. This is currently not very elegant because it is a special case, but it works. An alternate solution I considered was to leave them in the `Profile` and add a special uber-override layer. Let me know if you think it should change. - Did some general cleanup in `generate_targets`. ## Misc Fixes - `cargo check` now honors the `--release` flag. Fixes #5218. - `cargo build --test` will set `panic` correctly for dependences. Fixes #5369. - `cargo check --tests` will no longer include bins twice (once as a normal check, once as a `--test` check). It only does `--test` check now. - Similarly, `cargo check --test name` no longer implicitly checks bins. - Examples are no longer considered a "test". (See #5397). Consequences: - `cargo test` will continue to build examples as a regular build (no change). - `cargo test --tests` will no longer build examples at all. - `cargo test --all-targets` will no longer build examples as tests, but instead build them as a regular build (now matches `cargo test` behavior). - `cargo check --all-targets` will no longer check examples twice (once as normal, once as `--test`). It now only checks it once as a normal target. ## Questions - Thumbs up/down on the general approach? - The method to detect if a package is a member of a workspace should probably be redone. I'm uncertain of the best approach. Maybe `Workspace.members` could be a set? - `Hash` and `PartialEq` are implemented manually for `Profile` only to avoid matching on the `name` field. The `name` field is only there for debug purposes. Is it worth it to keep `name`? Maybe useful for future use (like #4140)? - I'm unhappy with the `Finished` line summary that displays `[unoptimized + debuginfo]`. It doesn't actually show what was compiled. Currently it just picks the base "dev" or "release" profile. I'm not sure what a good solution is (to be accurate it would need to potentially display a list of different options). Is it ok? (See also #4140 for the wrong profile name being printed.) - Build-dependencies use different profiles based on whether or not `--release` flag is given. This means that if you want build-dependencies to always use a specific set of settings, you have to specify both `[profile.dev.build_override]` and `[profile.release.build_override]`. Is that reasonable (for now)? I've noticed some issues (like #1774, #2234, #2424) discussing having more control over how build-dependencies are handled. - `build --bench xxx` or `--benches` builds dependencies with dev profile, which may be surprising. `--release` does the correct thing. Perhaps print a warning when using `cargo build` that builds benchmark deps in dev mode? - Should it warn/error if you have an override for a package that does not exist? - Should it warn/error if you attempt to set `panic` on the `test` or `bench` profile? ## TODO - I have a long list of tests to add. - Address a few "TODO" comments left behind.
☀️ Test successful - status-appveyor, status-travis |
Stabilize profile-overrides. This stabilizes the profile-overrides feature. This was proposed in [RFC 2282](rust-lang/rfcs#2282) and implemented in #5384. Tracking issue is rust-lang/rust#48683. This is intended to land in 1.41 which will reach the stable channel on Jan 30th. This includes a new documentation chapter on profiles. See the ["Overrides" section](/~https://github.com/rust-lang/cargo/blob/9c993a92ce33f219aaaed963bef51fc0f6a7677a/src/doc/src/reference/profiles.md#overrides) in `profiles.md` file for details on what is being stabilized. Note: The `config-profile` and `named-profiles` features are still unstable. Closes #6214 **Concerns** - There is some risk that `build-override` may be confusing with the [proposed future dedicated `build` profile](#6577). There is some more discussion about the differences at rust-lang/rust#48683 (comment). I don't expect it to be a significant drawback. If we proceed later with a dedicated `build` profile, I think we can handle explaining the differences in the documentation. (The linked PR is designed to work with profile-overrides.) - I don't anticipate any unexpected interactions with `config-profiles` or `named-profiles`. - Some of the syntax like `"*"` or `build-override` may not be immediately obvious what it means without reading the documentation. Nobody suggested any alternatives, though. - Configuring overrides for multiple packages is currently a pain, as you have to repeat the settings separately for each package. I propose that we can extend the syntax in the future to allow a comma-separated list of package names to alleviate this concern if it is deemed worthwhile. - The user may not know which packages to override, particularly for some dependencies that are split across multiple crates. I think, since this is an advanced feature, the user will likely be comfortable with using things like `cargo tree` to understand what needs to be overridden. There is [some discussion](rust-lang/rust#48683 (comment)) in the tracking issue about automatically including a package's dependencies, but this is somewhat complex. - There is some possibly confusing interaction with the test/bench profile. Because dependencies are always built with the dev/release profiles, overridding test/bench usually does not have an effect (unless specifying a workspace member that is being tested/benched). Overriding test/bench was previously prohibited, but was relaxed when named profiles were added. - We may want to allow overriding the `panic`, `lto`, and `rpath` settings in the future. I can imagine a case where someone has a large workspace, and wants to override those settings for just one package in the workspace. They are currently not allowed because it doesn't make sense to change those settings for rlibs, and `panic` usually needs to be in sync for the whole profile. - There are some strange interactions with `dylib` crates detailed in rust-lang/rust#64319. A fix was attempted, but later reverted. Since `dylib` crates are rare (this mostly applied to `libstd`), and a workaround was implemented for `build-std` (it no longer builds a dylib), I'm not too worried about this. - The interaction with `share-generics` can be quite confusing (see rust-lang/rust#63484). I added a section in the docs that tries to address this concern. It's also possible future versions of `rustc` may handle this better. - The new documentation duplicates some of the information in the rustc book. I think it's fine, as there are subtle differences, and it avoids needing to flip back and forth between the two books to learn what the settings do.
Profile Overrides (RFC #2282 Part 1)
WIP: Putting this up before I dig into writing tests, but should be mostly complete. I also have a variety of questions below.
This implements the ability to override profiles for dependencies and build scripts. This includes a general rework of how profiles work internally. Closes #5298.
Profile overrides are available with
profile-overrides
set incargo-features
in the manifest.Part 2 is to implement profiles in config files (to be in a separate PR).
General overview of changes:
Profiles
moved tocore/profiles.rs
. All profile selection is centralized there.test
,doc
,run_custom_build
, andcheck
. AddedCompileMode
toUnit
to replace them.Profile
is no longer a reference inUnit
.rustc_args
/rustdoc_args
fromProfile
and place them inContext
. This is currently not very elegant because it is a special case, but it works. An alternate solution I considered was to leave them in theProfile
and add a special uber-override layer. Let me know if you think it should change.generate_targets
.Misc Fixes
cargo check
now honors the--release
flag. Fixes Cargo check does not respect the --release flag #5218.cargo build --test
will setpanic
correctly for dependences. Fixescargo build
for tests builds dependencies withpanic
#5369.cargo check --tests
will no longer include bins twice (once as a normal check, once as a--test
check). It only does--test
check now.cargo check --test name
no longer implicitly checks bins.cargo test
will continue to build examples as a regular build (no change).cargo test --tests
will no longer build examples at all.cargo test --all-targets
will no longer build examples as tests, but instead build them as a regular build (now matchescargo test
behavior).cargo check --all-targets
will no longer check examples twice (once asnormal, once as
--test
). It now only checks it once as a normaltarget.
Questions
Workspace.members
could be a set?Hash
andPartialEq
are implemented manually forProfile
only to avoid matching on thename
field. Thename
field is only there for debug purposes. Is it worth it to keepname
? Maybe useful for future use (like Internally storing profiles as special-cased booleans #4140)?Finished
line summary that displays[unoptimized + debuginfo]
. It doesn't actually show what was compiled. Currently it just picks the base "dev" or "release" profile. I'm not sure what a good solution is (to be accurate it would need to potentially display a list of different options). Is it ok? (See also Internally storing profiles as special-cased booleans #4140 for the wrong profile name being printed.)--release
flag is given. This means that if you want build-dependencies to always use a specific set of settings, you have to specify both[profile.dev.build_override]
and[profile.release.build_override]
. Is that reasonable (for now)? I've noticed some issues (like Single profile for build dependencies #1774,cargo bench
builds build dependencies in release mode #2234, Feature request: compile build.rs scripts in release mode even when doing a debug build #2424) discussing having more control over how build-dependencies are handled.build --bench xxx
or--benches
builds dependencies with dev profile, which may be surprising.--release
does the correct thing. Perhaps print a warning when usingcargo build
that builds benchmark deps in dev mode?panic
on thetest
orbench
profile?TODO