-
Notifications
You must be signed in to change notification settings - Fork 13.2k
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
support config extensions #138934
base: master
Are you sure you want to change the base?
support config extensions #138934
Conversation
The rustc-dev-guide subtree was changed. If this PR only touches the dev guide consider submitting a PR directly to rust-lang/rustc-dev-guide otherwise thank you for updating the dev guide with your changes. cc @BoxyUwU, @jieyouxu, @Kobzol This PR modifies If appropriate, please update |
Some thoughts (aside from the fact that I like this feature):
|
Rerolling review due to bandwidth because I'm still slowly testing #119899. |
r? kobzol Will take a look tomorrow. |
I haven't looked too closely, but based on description
can this infinite loop? I guess even if it does, just don't do that 😆 |
I'm planning to rework on this next week. @rustbot author |
That's not actually true. I keep my config file outside the rust repository and invoke |
Ah, sorry, I didn't consider that case (I didn't know that was even an option). |
6504079
to
c7b6606
Compare
Signed-off-by: onur-ozkan <[email protected]>
Signed-off-by: onur-ozkan <[email protected]>
c7b6606
to
13fcf1c
Compare
This PR modifies If appropriate, please update |
@rustbot ready |
dd1536d
to
586efa1
Compare
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.
Looks simple enough. I think that we should also mark profile
as deprecated and just replace it with include = ["src/bootstrap/defaults/<profile>"]
, and then remove it sometime in the future. Both to avoid having two ways of achieving a similar thing, and also to make the contents of the profiles more discoverable, before it was not at all obvious what does profile = "compiler"
mean and from where it is taken.
Signed-off-by: onur-ozkan <[email protected]>
Signed-off-by: onur-ozkan <[email protected]>
585778f
to
ad2e4a4
Compare
If we do it now, I have to implement the |
Sure, that can wait :) |
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.
Looks good! Left a few nits.
9a17719
to
5d105b1
Compare
5d105b1
to
0111b9d
Compare
exit!(2); | ||
}); | ||
toml.merge( | ||
Some(config.src.join(include_path)), |
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 should be resolved relative to the top-level config path. Otherwise it would be inconsistent with how we resolve relative paths in the nested configs. Plus the top-level config might not be in the src directory, you can pass an arbitrary path to the config.
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.
Oops, I thought config.src
was the top-level path of config.toml
for a second (confused with config.config
), sorry!
Signed-off-by: onur-ozkan <[email protected]>
0111b9d
to
9369518
Compare
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 tried it with a few complicated-ish situations locally, and I'm afraid that the approach with ignoring duplicates doesn't work 🙈
Consider this:
# configs/root.toml
include = ["nested.toml"]
# configs/nested.toml
include = ["lld-disable.toml"]
[rust]
lld = true
# configs/lld-disable.toml
[rust]
lld = false
When you load configs/root.toml
, the lld
value will be false
instead of true
. Because when you load the config, this happens:
- You load
root.toml
, and find thenested.toml
include - You merge
nested.toml
into yourself withIgnoreDuplicate
- You load
nested.toml
and find thelld-disable.toml
include - You merge
lld-disable.toml
into yourself withIgnoreDuplicate
. This setslld = false
. - The implementation of the merge from step 2) continues, where we try to merge
lld = true
intoself.rust
. However, since we already haverust.lld = false
and we use theIgnoreDuplicate
mode, this is skipped, and thuslld = true
never overrideslld = false
, as it should.
The overriding works for the top-level config, but not for the nested ones.
I think that in order to fix this, it should be enough to reorder code in the merge
implementation. Essentially, we have to proceed in reverse order - the most important things should be merged first. So first we merge the actual config, then the includes, and then the profile.
I tried this locally and it seems to work, but I don't know if it's the right solution, because essentially it only works if we use IgnoreDuplicate
. If we used Override
instead, then the order should be reversed again. So maybe we should decide the order based on the replace mode? 🤷♂️
Here is my diff:
diff --git a/src/bootstrap/src/core/config/config.rs b/src/bootstrap/src/core/config/config.rs
index 9a53ab71567..5357202b52e 100644
--- a/src/bootstrap/src/core/config/config.rs
+++ b/src/bootstrap/src/core/config/config.rs
@@ -686,7 +686,7 @@ pub fn from_triple(triple: &str) -> Self {
/// This structure uses `Decodable` to automatically decode a TOML configuration
/// file into this format, and then this is traversed and written into the above
/// `Config` structure.
-#[derive(Deserialize, Default)]
+#[derive(Deserialize, Debug, Default)]
#[serde(deny_unknown_fields, rename_all = "kebab-case")]
pub(crate) struct TomlConfig {
#[serde(flatten)]
@@ -713,7 +713,7 @@ pub enum ChangeId {
/// for the "change-id" field to parse it even if other fields are invalid. This ensures
/// that if deserialization fails due to other fields, we can still provide the changelogs
/// to allow developers to potentially find the reason for the failure in the logs..
-#[derive(Deserialize, Default)]
+#[derive(Deserialize, Debug, Default)]
pub(crate) struct ChangeIdWrapper {
#[serde(alias = "change-id", default, deserialize_with = "deserialize_change_id")]
pub(crate) inner: Option<ChangeId>,
@@ -773,6 +773,31 @@ fn do_merge<T: Merge>(x: &mut Option<T>, y: Option<T>, replace: ReplaceOpt) {
}
}
+ // We need to merge in reverse order of priotity - first the config, then includes, then
+ // the profile.
+ do_merge(&mut self.build, build, replace);
+ do_merge(&mut self.install, install, replace);
+ do_merge(&mut self.llvm, llvm, replace);
+ do_merge(&mut self.gcc, gcc, replace);
+ do_merge(&mut self.rust, rust, replace);
+ do_merge(&mut self.dist, dist, replace);
+
+ match (self.target.as_mut(), target) {
+ (_, None) => {}
+ (None, Some(target)) => self.target = Some(target),
+ (Some(original_target), Some(new_target)) => {
+ for (triple, new) in new_target {
+ if let Some(original) = original_target.get_mut(&triple) {
+ original.merge(None, &mut Default::default(), new, replace);
+ } else {
+ original_target.insert(triple, new);
+ }
+ }
+ }
+ }
+
+ self.change_id.inner.merge(None, &mut Default::default(), change_id.inner, replace);
+
let parent_dir = parent_config_path
.as_ref()
.and_then(|p| p.parent().map(ToOwned::to_owned))
@@ -808,29 +833,7 @@ fn do_merge<T: Merge>(x: &mut Option<T>, y: Option<T>, replace: ReplaceOpt) {
included_extensions.remove(&include_path);
}
- self.change_id.inner.merge(None, &mut Default::default(), change_id.inner, replace);
self.profile.merge(None, &mut Default::default(), profile, replace);
-
- do_merge(&mut self.build, build, replace);
- do_merge(&mut self.install, install, replace);
- do_merge(&mut self.llvm, llvm, replace);
- do_merge(&mut self.gcc, gcc, replace);
- do_merge(&mut self.rust, rust, replace);
- do_merge(&mut self.dist, dist, replace);
-
- match (self.target.as_mut(), target) {
- (_, None) => {}
- (None, Some(target)) => self.target = Some(target),
- (Some(original_target), Some(new_target)) => {
- for (triple, new) in new_target {
- if let Some(original) = original_target.get_mut(&triple) {
- original.merge(None, &mut Default::default(), new, replace);
- } else {
- original_target.insert(triple, new);
- }
- }
- }
- }
}
}
@@ -840,6 +843,7 @@ macro_rules! define_config {
$($field:ident: Option<$field_ty:ty> = $field_key:literal,)*
}) => {
$(#[$attr])*
+ #[derive(Debug)]
struct $name {
$($field: Option<$field_ty>,)*
}
@@ -1639,12 +1643,13 @@ pub(crate) fn parse_inner(
// This must be handled before applying the `profile` since `include`s should always take
// precedence over `profile`s.
for include_path in toml.include.clone().unwrap_or_default().iter().rev() {
- let included_toml = get_toml(include_path).unwrap_or_else(|e| {
+ let include_path = toml_path.parent().unwrap().join(include_path);
+ let included_toml = get_toml(&include_path).unwrap_or_else(|e| {
eprintln!("ERROR: Failed to parse '{}': {e}", include_path.display());
exit!(2);
});
toml.merge(
- Some(toml_path.join(include_path)),
+ Some(include_path),
&mut Default::default(),
included_toml,
ReplaceOpt::IgnoreDuplicate,
@@ -1721,6 +1726,7 @@ fn get_table(option: &str) -> Result<TomlConfig, toml::de::Error> {
}
toml.merge(None, &mut Default::default(), override_toml, ReplaceOpt::Override);
+ eprintln!("{toml:#?}");
config.change_id = toml.change_id.inner;
let Build {
// This must be handled before applying the `profile` since `include`s should always take | ||
// precedence over `profile`s. | ||
for include_path in toml.include.clone().unwrap_or_default().iter().rev() { | ||
let included_toml = get_toml(include_path).unwrap_or_else(|e| { |
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.
let included_toml = get_toml(include_path).unwrap_or_else(|e| { | |
let included_toml = get_toml(toml_path.parent().unwrap().join(include_path)).unwrap_or_else(|e| { |
Otherwise the file isn't resolved correctly. (Better do something like let include_path = toml_path.parent().unwrap().join(include_path);
at the beginning of the loop).
I feel like there might be some incorrect cases with your approach. I need to re-check/test everything again. The implementation becomes flaky with each change. I will add test coverage (next week or so) for each case to ensure everything works properly now and in the future. @rustbot author |
Reminder, once the PR becomes ready for a review, use |
Copied from the
rustc-dev-guide
addition: