-
Notifications
You must be signed in to change notification settings - Fork 12.8k
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
Emit simpler code from format_args #91359
Conversation
(rust-highfive has picked a reviewer for you, use r? to override) |
// Right now there is a bug such that for the expression: | ||
// foo(bar(&1)) | ||
// the lifetime of `1` doesn't outlast the call to `bar`, so it's not | ||
// valid for the call to `foo`. To work around this all arguments to the |
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 am planning to make Clippy less sensitive to macro changes at some point in the near future (rust-lang/rust-clippy#7843). You might want to hold this until that lands. |
Labeling blocked on rust-lang/rust-clippy#7843. |
☔ The latest upstream changes (presumably #91945) made this pull request unmergeable. Please resolve the merge conflicts. |
|
Clippy should be out of your way after the next sync! |
@rustbot label -S-blocked +S-waiting-on-review |
Let's check if this has any implications for compile time. @bors try @rust-timer queue |
Awaiting bors try build completion. @rustbot label: +S-waiting-on-perf |
⌛ Trying commit 20bda20b5f66e30cc64ccb64b087f0ea3cb5edae with merge 3a6b09020f099db8b66b764a71e2ef5608e6c4ea... |
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.
Thanks for the PR, @dtolnay! Looks like a really nice simplification. I left a couple of suggestions below.
Is there anything that could go subtly wrong with this? Like hygiene or borrow checking? Should we ask someone with more experience in macro expansion to take a look too?
// we can generate simpler code. | ||
let nicely_ordered = fmt_arg_index_and_ty | ||
.clone() | ||
.is_sorted_by(|(i, _), (j, _)| (i < j).then_some(Ordering::Less)); |
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.
Is that different from .is_sorted_by_key(|(i, _)| i)
?
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.
Yeah, we need to catch permuted as well as repeated elements. Yours is true on 0 0
which would result in an incorrect expansion.
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.
That's very subtle. It would be great if you could add a comment. And maybe find a simpler (if more verbose) way to express this. I personally cannot decode what's going on here in a reasonable amount of time.
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.
array_windows
(#75027) may be clearer for this. The sequence has no indices out of order or repeated if: for every adjacent pair of elements, the first one's index is less than the second one's index.
// The following Iterator<Item = (usize, &ArgumentType)> has one item | ||
// per element of our output slice, identifying the index of which | ||
// element of original_args it's passing, and that argument's type. | ||
let fmt_arg_index_and_ty = self |
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 make sense to use a SmallVec for this?
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.
SmallVec<(usize, &ArgumentType)>
instead of Iterator<Item = (usize, &ArgumentType)>
? I'm not sure why that would be better. Do you think any of the code below would get clearer? We only ever need to iterate it forward.
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.
The sequence would be "cached" on the stack instead of evaluated twice. But I'm fine with keeping this as an iterator either.
From what I know about macros, I do not expect any difference in hygiene or borrow checking between the two expansions. The only other thing to be mindful of is const promotion. The kind of thing in the following snippet -- even though both macros nominally appear equivalent, only one of them works when a promotion is required. However in the PR we are only going in the direction of can't promote ⟶ can promote so nothing would break, and conversely in format_args nothing relies on promotion anyway because the relevant argument of macro_rules! a {
($e:expr) => {
match ($e,) {
(e,) => &[e],
}
};
}
macro_rules! b {
($e:expr) => {
&[$e]
};
}
fn main() {
let _: &'static [i32] = a!(1); // fail
let _: &'static [i32] = b!(1); // ok
} error[E0716]: temporary value dropped while borrowed
--> src/main.rs:4:22
|
4 | (e,) => &[e],
| ^^-
| | |
| | temporary value is freed at the end of this statement
| creates a temporary which is freed while still in use
...
16 | let _: &'static [i32] = a!(1); // fail
| -------------- ----- in this macro invocation
| |
| type annotation requires that borrow lasts for `'static` |
To elaborate slightly: if let _: core::fmt::Arguments<'static> = format_args!("{}", 1); where the formatted values are all consts. Today ArgumentV1::new is not const so the PR doesn't make any code newly compile. |
☀️ Try build successful - checks-actions |
Queued 3a6b09020f099db8b66b764a71e2ef5608e6c4ea with parent 181e915, future comparison URL. |
Finished benchmarking commit (3a6b09020f099db8b66b764a71e2ef5608e6c4ea): comparison url. Summary: This change led to very large relevant mixed results 🤷 in compiler performance.
If you disagree with this performance assessment, please file an issue in rust-lang/rustc-perf. Benchmarking this pull request likely means that it is perf-sensitive, so we're automatically marking it as not fit for rolling up. While you can manually mark this PR as fit for rollup, we strongly recommend not doing so since this PR led to changes in compiler perf. Next Steps: If you can justify the regressions found in this try perf run, please indicate this with @bors rollup=never |
⌛ Testing commit 7ee21e3 with merge 1513d8ff8709dc2258a37f7e184e4d4130a94db3... |
💥 Test timed out |
@bors retry |
The "x86_64-msvc-1" job timed out during "Testing rustc_macros stage1 (x86_64-pc-windows-msvc -> x86_64-pc-windows-msvc)". All other jobs succeeded. Gonna gamble that this is an infra glitch and not an msvc-specific issue with this PR. We'll find out after the next round. |
⌛ Testing commit 7ee21e3 with merge 77a8c17aa7dd1ffeadf8a39e118b35ead50ec940... |
@bors retry Yielding priority to 1.58.1 security point release. |
☀️ Test successful - checks-actions |
Finished benchmarking commit (0bcacb3): comparison url. Summary: This change led to large relevant mixed results 🤷 in compiler performance.
If you disagree with this performance assessment, please file an issue in rust-lang/rustc-perf. Next Steps: If you can justify the regressions found in this perf run, please indicate this with @rustbot label: +perf-regression |
the latest performance report nearly exactly matches the predicted performance and @Mark-Simulacrum had a great analysis for where that came from. @rustbot label: +perf-regression-triaged |
Emit simpler code from format_args I made this PR so that `cargo expand` dumps a less overwhelming amount of formatting-related code. <br> `println!("rust")` **Before:** ```rust { ::std::io::_print(::core::fmt::Arguments::new_v1(&["rust\n"], &match () { _args => [], })); }; ``` **After:** ```rust { ::std::io::_print(::core::fmt::Arguments::new_v1(&["rust\n"], &[])); }; ``` `println!("{}", x)` **Before:** ```rust { ::std::io::_print(::core::fmt::Arguments::new_v1( &["", "\n"], &match (&x,) { _args => [::core::fmt::ArgumentV1::new( _args.0, ::core::fmt::Display::fmt, )], }, )); }; ``` **After:** ```rust { ::std::io::_print(::core::fmt::Arguments::new_v1( &["", "\n"], &[::core::fmt::ArgumentV1::new(&x, ::core::fmt::Display::fmt)], )); }; ```
Rename _args -> args in format_args expansion As observed in rust-lang#91359 (comment), prior to that PR this variable was sometimes never used, such as in the case of: ```rust println!(""); // used to expand to: ::std::io::_print( ::core::fmt::Arguments::new_v1( &["\n"], &match () { _args => [], }, ), ); ``` so the leading underscore in `_args` was used to suppress an unused variable lint. However after rust-lang#91359 the variable is always used when present, as the unused case would instead expand to: ```rust ::std::io::_print(::core::fmt::Arguments::new_v1(&["\n"], &[])); ```
I made this PR so that
cargo expand
dumps a less overwhelming amount of formatting-related code.println!("rust")
Before:After:
println!("{}", x)
Before:After: