-
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
Improve the documentation for std::hint::black_box. #62891
Conversation
r? @cramertj (rust_highfive has picked a reviewer for you, use r? to override) |
IIRC on the RFC we were specifically discussing not making these kinds of promises for this function. cc @RalfJung |
Indeed, see rust-lang/rfcs#2360 for the long and detailed discussion and rust-lang/rfcs#2360 (comment) for a kind of summary with links to some key posts. If anything the documentation should be changed to be more like the text of that RFC, and less about being "opaque to the optimizer". |
So something like:
? I guess I'm OK with it, even if it's a little vague. |
Yes, something like that. Maybe better with an explicit statement that this is done on a "best effort" basis. Quoting from the RFC:
|
OK, so here's a preliminary proposal:
Questions:
|
We don't want to/know how to properly specify that any more than for other restrictions on optimizations, so it certainly wouldn't be guaranteed. Plus, it's not even true of the current implementation. As the simplest possible counter-example, in this snippet: black_box(...); // doesn't mention x
let y = x + 1;
black_box(...); even if the (If you mean "sandwiched" in terms of data-flow, i.e. in the above example if the first |
OK, that wasn't my understanding. I thought that an inline asm block must be assumed to possibly modify x, or any variable for that matter, and thus it's position must not change. Let's just not mention it? |
I feel like you removed all the words that indicated that this is an entirely best-effort hint, and no guarantee. I propose something like:
|
Reads OK to me, but ultimately I think the Rust devs should decide. (In light of our conversation, I'm not certain I fully understand the intention of black_box) |
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.
Might be better to just say that black_box
is "A hint that might be treated as an unknown function by the optimizer."
And just leave it at that.
src/libcore/hint.rs
Outdated
/// A function that is opaque to the optimizer, to allow benchmarks to | ||
/// pretend to use outputs to assist in avoiding dead-code | ||
/// elimination. | ||
/// An identity function whose argument is opaque to the optimizer. This is useful for (e.g.) |
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 believe this is incorrect, the argument isn't opaque to the optimizer, and optimizations on the argument are possible, e.g., black_box(2 * 5)
is currently optimized to black_box(10)
.
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.
Also, "is opaque" is also incorrect - this function is a hint
, there are no guarantees that it will do anything at all, and in some platforms, it does more than in others (e.g. using volatile reads vs inline assembly).
src/libcore/hint.rs
Outdated
/// pretend to use outputs to assist in avoiding dead-code | ||
/// elimination. | ||
/// An identity function whose argument is opaque to the optimizer. This is useful for (e.g.) | ||
/// benchmarking, where you want to avoid things like dead-code elimination and constant |
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 probably be part of the body of the documentation and not of the "brief" string.
I would leave this as "This is useful for, e.g., benchmarking, to prevent benchmarks from being optimized away". There are no guarantees whatsoever about which optimizations this intends to inhibit.
I think that what @RalfJung proposes here (#62891 (comment)) is the best we can do, and from the RFC there is consensus that these are the semantics that the language team wants. The only open issue in the RFC there is the one raised by @cramertj about giving this intrinsic a clearer name like |
Other than what has been said already, I think we should point out various scenarios people would think to use I think we should really emphasize the bit about "identity function" being the only guarantee a bit more. |
I think we should do this as well, but not necessarily in this PR. I'd leave it up to @vext01 to decide if they feel like doing that. If they don't, we should definitely open an issue to track doing that.
We could add a sentence after the last bit that says "but too unreliable when disabling these optimizations is required for correctness.", e.g., "The only guaranteed semantics of That should make it clear that replacing |
Yeah entirely fair. At any point before stabilization should do from my POV. :) |
I don't mind making the change if we can come to a consensus. I'm not going to suggest anything myself, as I'm not sure I fully understand the intention of |
Second reviewer ping from triage, pinging reviewer from |
I agree that @RalfJung's suggestion is the most accurate description, possibly with @gnzlbg's addition. I'd be happy with an updated comment along those lines. |
Ping from triage, @vext01 as per varkor's comment can you please update the description as such. Thanks |
I'm happy to make the edit with one clarification: I don't understand the last part of @RalfJung's suggestion. If we elide the bracketed section for a moment, it reads:
Seems to need a re-wording? |
Something like:
|
The revised comment makes sense, but if I understand correctly, you are saying "you can't rely on black_box to definitely do anything". If that's the case, I question why it exists it at all. (I don't see why an opaque asm block wouldn't always block Rust from seeing through it). Regardless, if you are all happy with that wording I'll make the change. |
You can definitely rely on |
I've pushed the change, but honestly I'm not sure how much of an improvement it is. I think the wording will confuse the average Rust user. If you folks think this makes sense, I can squash. |
Because there are many use cases, like writing benchmarks, for which this guarantee is not necessary.
We can't use asm blocks on all supported targets because not all of them allow inline assembly, but even for the targets in which we do, nothing prevents LLVM from starting doing optimization on inline assembly blocks, and optimizing the block we are using to nothing would be trivial. |
Thanks @gnzlbg -- that was the missing info. In light of this, let me propose a revision to the wording. Watch this space. |
src/libcore/hint.rs
Outdated
/// that Rust code is allowed to without introducing undefined behavior in the calling code. This | ||
/// property makes `black_box` useful for writing code in which certain optimizations are not | ||
/// desired. However, this also makes `black_box` too unreliable when disabling these optimizations | ||
/// is required for correctness. |
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 would probably be good to include an example using #[bench]
specifically.
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'd rather punt that to some follow up PR. If anything I'd require opening a tracking issue.
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 agree. I'm running out of steam for black_box
:)
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.
Sure :)
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.
LGTM. Let's see what CI says.
CI says yes. Shall I squash? |
Yes please do.
|
abe9791
to
a4b3dbe
Compare
Splat. |
@RalfJung Good improvement; looks great. I think we can extend with examples of dos and don'ts in follow up PRs, but this is a good start. |
LGTM |
@bors r=RalfJung,Centril,gnzlbg |
📌 Commit a4b3dbe has been approved by |
⌛ Testing commit a4b3dbe with merge b3145e9b935ace63f0e3adb416f14f60b8bf4db3... |
💥 Test timed out |
@bors retry |
…ril,gnzlbg Improve the documentation for std::hint::black_box. The other day a colleague was reviewing some of my code which was using `black_box` to block constant propogation. There was a little confusion because the documentation kind of implies that `black_box` is only useful for dead code elimination, and only in benchmarking scenarios. The docs currently say: > A function that is opaque to the optimizer, to allow benchmarks to pretend to use outputs to assist in avoiding dead-code elimination. Here is our discussion, in which I show (using godbolt) that a black box can also block constant propagation: ykjit/yk#21 (comment) This change makes the docstring for `black_box` a little more general, and while we are here, I've added an example (the same one from our discussion).  OK to go in?
☀️ Test successful - checks-azure |
Thanks all. I've raised an issue for the rustdoc bug: |
The other day a colleague was reviewing some of my code which was using
black_box
to block constant propogation. There was a little confusion because the documentation kind of implies thatblack_box
is only useful for dead code elimination, and only in benchmarking scenarios.The docs currently say:
Here is our discussion, in which I show (using godbolt) that a black box can also block constant propagation:
ykjit/yk#21 (comment)
This change makes the docstring for
black_box
a little more general, and while we are here, I've added an example (the same one from our discussion).OK to go in?