-
Notifications
You must be signed in to change notification settings - Fork 1.6k
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
Cargo Check T-lang Policy #3477
Conversation
@rfcbot label +I-lang-nominated |
@scottmcm the bot didn't seem to do the thing |
@rustbot label +I-lang-nominated |
Slightly confused by the wording here, but taking away from this RFC, there are effectively three "tiers" to checking for compile errors in Rust:
I'm adding clippy here as an example because before reading this, I had the misconception that it was actually this:
Effectively, I thought that Based upon this RFC, Under this system, a CI system which lets you merge changes based upon the output of |
@clarfonthey Does the possibility of post-monomorphization errors impact your mental model? (Sorry for the broad and pithy question, but it's something that sprang to mind for me after reading your comment.) |
I'm not entirely sure which teams this RFC would fall under (though I labeled T-lang for now). This doesn't exactly directly relate to Rust-the-language, but more-so the tooling that exists. For example, would T-lang care which errors rust-analyzer emits? Why would that decision be different from what |
A small clarification, |
@ehuss this RFC was written at the request of members of T-lang. The main thing here is the stability promises we want to uphold. The RFC makes the case: If you do not pass This helps unblock things like the inline const development where there's been concern about exactly how much can be caught by a |
I think it might help the discussion to have some background information, definitions, and examples to clarify what this RFC means. This doesn't necessarily need to be codified in the RFC, but if people don't have the context of the particulars of how things work, it could be difficult to understand. I think this RFC is intentionally trying avoid defining what How it works today
What kinds of things aren't checked today?Some of the kinds of errors that are only generated with a full build are:
I'm sure @oli-obk or others could explain it better than I can. There are a variety of issues where you can see examples of what these mean: rust-lang/rust#49292 You can also search the compiler test suite for |
I mean, I have argued in favour of those in the past, but bringing them up in this case makes it reasonable to argue against them. But, I dunno, I still think that there are a large number of steps in building (linking by itself is a massive step, not to mention all the optimisations on the compiled code) that I would expect Arguably, monomorphisation is part of building. But if it can error, I would consider it necessary for Honestly, link-time errors are a better example here, since those effectively are the last possible errors you can encounter, and you can't easily detect them without building everything. It makes me question me approach more, but I personally am willing to accept them being excluded on account of most Rust code not needing to worry about them. |
I'm surprised by this stance. Similar to @clarfonthey, my mental model has been that So, to change this around: how much of a performance penalty would there be in the typical case if Or another way to put it is to suggest that this RFC needs a "how do we teach this?" section. |
That seems like something we could easily offer under a flag in general. A fail-fast/stop-on-first-error scheme |
I am confused. Doesn't Cargo already behave like that — stop on the first error from rustc? |
Yes, but rustc continues its current compilation for as long as possible, reporting as many things as possible. |
The rationale leading to this RFC was a discussion on rust-lang/rust#112879 , and the question of whether it'd be permitted to not do all of that work on That distinction is important, and we should establish which of those two scenarios we want, rather than leaving it ambiguous or deciding on a case-by-case basis. |
I don't think we have numbers, but this requires full monomorphization, whereas currently |
Co-authored-by: Ralf Jung <[email protected]>
From a compiler perspective (tho I am biased here as someone working on this logic) it would allow us more freedom in refactorings and keep the compiler implementation simpler if we allowed |
I think any error before LLVM should be caught by |
Sure, the compiler and cargo teams have the option of introducing such a flag. The RFC talks about |
Co-authored-by: Ralf Jung <[email protected]>
Co-authored-by: Ralf Jung <[email protected]>
Co-authored-by: Ralf Jung <[email protected]>
For me, I used a mix of One example is I run |
Ok, then what about Then, an option in [profile.dev]
check-level=1 |
This was discussed in today's lang triage meeting (notes). The consensus was that we should do it, after the updates in 309c26f. In particular, we can't guarantee that no optimization-dependent errors will ever exist, but we want the default to be that they shouldn't exist and are considered a bug (and to document the exceptions). @rfcbot fcp merge |
Team member @tmandry has proposed to merge this. The next step is review by the rest of the tagged team members: No concerns currently listed. Once a majority of reviewers approve (and at most 2 approvals are outstanding), this will enter its final comment period. If you spot a major issue that hasn't been raised at any point in this process, please speak up! cc @rust-lang/lang-advisors: FCP proposed for lang, please feel free to register concerns. |
text/0000-cargo-check-lang-policy.md
Outdated
Monomorphization is expensive: instead of having to check each function only once, each function now has to be checked once for all choices of generic parameters that the crate needs. | ||
Given this performance cost and the fact that errors during monomorphization are fairly rare, `cargo check` favors speed over completeness. | ||
|
||
Examples where the optimization level can affect if a program passes `cargo check` and/or `cargo build` are considered bugs unless there is a documented policy exception, approved by T-lang. One example of such an exception is [RFC #3016](https://rust-lang.github.io/rfcs/3016-const-ub.html), which indicated that undefined behavior in const functions cannot always be detected statically (and in particular, optimizations may cause the UB to be undetectable). |
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.
FWIW, this is currently not optimization-dependent. CTFE always runs on unoptimized MIR, exactly to avoid that situation.
But I understand the point of this paragraph is that it would be allowed to be optimization-dependent.
Co-authored-by: scottmcm <[email protected]>
I do think that it would be nice to have more non-normative references and examples (like #3477 (comment) brought up), but I agree with the actual policy. @rfcbot reviewed |
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.
TL;DR:
- Good outcomes of this policy:
- A program passing
cargo check
but failingcargo build
is an invalid program and not covered under stability promises. - Optimization dependent errors in
cargo build
are considered bugs barring specific policy.
- A program passing
- Bad outcomes of this policy:
cargo check
being "fast" is more important than being "complete," without any limitation on when sacrificing the latter for the former is considered acceptable.- Continued conflation of codegen errors and const panics as equally hard/bad post-monomorphization errors.
I want to note that I've found it less than useful to use the term "post-monomorphization errors" here; it's not a useful term to anyone who isn't the compiler. It's basically just tautologically the set of errors which are found by cargo build
but not cargo check
, as the difference between the two is that check
stops before performing monomorphization. The set of what errors are post-mono isn't even a fully consistent set with respect to surface language features.
I find a much more useful categorization is "codegen" and "generic instantiation" errors. The "codegen" category is those things that are intrinsically tied to the build process, so LLVM, inline asm, and linker errors, but also MIR lowering errors and (vendor) intrinsic usage errors. "Generic instantiation" errors are those that would be diagnosed pre-mono if the exact same code were written in a different function.
The latter category may even be solely populated by const evaluation panics. I still hold that such are a special case compared to every other post-mono error.
I have no qualms with cargo check
missing errors caught by cargo build
in general, nor even with errors reported by cargo build
being optimization dependent. Certain classes of errors ("codegen") are clearly build/optimization dependent. It's the specific case of const panics which I take issue with.
An example and more complaining about this specific case
impl Trait for () {}
trait Trait {
const ASSOC: () = panic!();
}
fn f_inner<T: Trait>() {
T::ASSOC // const { panic!() }
}
pub fn f() {
// cargo check: okay
// cargo clippy: error
// cargo build: error
<()>::ASSOC;
}
// cargo check: error
const _: () = <()>::ASSOC;
The reason this bothers me so much I think has to do with legacy — in the beginning, it used to be that a panic in a (promoted) constant evaluation was deferred until runtime, being the guaranteed panic lint. With const panic!
, though, this changed; even when the usage of a const
item is at runtime, the error is promised to occur at build time... if it happens at all.
If we made check
catch less and made the above into a deny-by-default guaranteed panic lint instead of a hard error, I'd care less. It's the inconsistency which I take issue with; const panic being a hard error says that const evaluation is a comptime thing, but deferring evaluation to mono time instead of obligation checking is based on comptime const eval being just an optimization over runtime eval. The two can't both be true at the same time.
While the policy is general and doesn't need to have a final say on what classes of errors check
does or doesn't find, I do think the policy should have some examples of what classes of errors check
is still expected to find, and what classes it's expected to ignore. I know it's not the actual position of any team, but as written, the stated policy as written of check
being a fast approximation of build
being more important than catching errors means I could PR massive wins by bypassing borrowck and name resolution independent trait obligation evaluation in check
, and accepting that would be allowed by this policy.
Much less drastic, I'll make it so that all unused const
items never get evaluated by check
. Unused associated const
items are something currently skipped by build, so there's clear precedent for skipping this kind of const evaluation. That should offer a consistent small win on real world crates. And likely make those crates unsound under check, since const _: () = assert!(...);
is a fairly standard way to do a compile-time const assertion.
I'm representing a deliberately hostile interpretation of the policy because without clarification, an expected outcome should be an erosion of the trust in cargo check
. So far it's been that sometimes check
won't catch some errors that build
will, but if you aren't doing anything unsafe
(e.g. linker errors) and/or weird and advanced (e.g. associated const
eval errors), check
will always be sufficient. This policy is explicitly saying that this statement is far too strong.
At absurdity, consider if borrowck were extremely expensive, and took on average 20% of the runtime of check
. Would removing borrowck from check
be permissable? Is there a vibe check available for when a completeness/speed tradeoff is considered desirable other than just legacy of what's pre- or post-mono in the current compiler architecture? How do we make justifiable calls when our largest test corpuses are primarily of correct code, and this is about diagnosing incorrect code?
At a minimum, I'd hope that adopting this as policy would come with a blog post, and for that blog post to illustrate an example of an error in each category of "caught by check, will always be caught by check;" "only caught by build, likely never caught by check;" and "might be caught by check, if doing so can be made fast."
text/0000-cargo-check-lang-policy.md
Outdated
# Unresolved questions | ||
[unresolved-questions]: #unresolved-questions | ||
|
||
* Is there any situation when we would *want* to allow optimization level to affect if a program passes or fails a build? This seems unlikely. |
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's one very notable case for linker errors: the "no panic trick" relies on making an unwind call an undefined linker symbol and optimization removing any references to the symbol, thus allowing the code to compile iff the compiler proves that no unwinding can occur.
Additionally, essentially the entirety of the average 5% regressions caused by rust-lang/rust#112879 come from not permitting dead code optimizations to skip middle-end MIR monomorphization of statically known to be unreachable code.
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.
Additionally, essentially the entirety of the average 5% regressions caused by rust-lang/rust#112879 come from not permitting dead code optimizations to skip middle-end MIR monomorphization of statically known to be unreachable code.
That's not clear at all, given that that PR reports regressions in check
as well.
I would say that while T-lang is in charge of defining "Rust the Language", it's still separately up to T-compiler to manage "Rust the Compiler Implementation". I don't think that T-compiler would carelessly approve any big PRs which make the user's experience worse, even if the bounds of the language technically allow it. |
That's not correct. We currently have some "errors skipped by There's a very clear definition of post-mono errors and it has nothing directly to do with check vs build. It's simply about whether the error can be detected on generic MIR or not. So I still find the term quite useful. That's not to say we can't have sub-categories of post-mono errors, but the overarching category has its use as well.
It's definitely not just const-eval panics, it's at least const-eval failure in general (which can be due to panic, UB, or doing something unsupported at compile-time). |
🔔 This is now entering its final comment period, as per the review above. 🔔 |
@rfcbot fcp reviewed I am happy with this being the lang policy. In short, the "official" semantics of Rust come with |
The final comment period, with a disposition to merge, as per the review above, is now complete. As the automated representative of the governance process, I would like to thank the author for their work and everyone else who contributed. This will be merged soon. |
Since I've been loudly against I still disagree with an unqualified |
The @rust-lang/lang team has decided to accept this RFC. There is no tracking issue for this RFC because it is a policy decision, and there is no associated implementation to stabilize. To report issues with the compiler, please open an issue in the rust-lang/rust repository. |
Rendered