-
Notifications
You must be signed in to change notification settings - Fork 13.6k
Conditionally compile contracts instead of deciding at run-time #145229
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
base: master
Are you sure you want to change the base?
Conditionally compile contracts instead of deciding at run-time #145229
Conversation
This comment has been minimized.
This comment has been minimized.
Hunh? All I can see in that PR is contract checks changing the optimized MIR of functions which is expected and should be fine. Disabled contract check MIR should be cleaned out as part of lowering to LLVM, or it should be immediately cleaned up by LLVM because it is under the equivalent of an |
How does one evaluate whether disabled contract check code is indeed cleaned up? You mentioned in #136578 (comment):
Could you elaborate on this please? Indeed one of the codegen tests (tests/codegen/cross-crate-inlining/leaf-inlining.rs) fails on that branch, and I don't have the expertise to judge what the consequences of that test failure are. I feel that contracts should not interfere with optimisations of the code in any way, since to me they serve as an executable piece of documentation, comparable to rustdoc tests. Having contracts get in the way of normal code will likely slow down their adoption. |
The initial implementation of contracts, despite requiring a compiler flag to enable runtime checking of contracts, still compiled contracts into function definitions, even when the compiler flag was disabled. This meant that contracts could not be safely added to functions without breaking optimisations, or even without potentially changing the behaviour of the function. This change guards macro expansion of the built-in contract macros with the contract-checks compiler flag. Additionally, it removes the contract_checks compiler intrinsic that was used to determine whether contract checks should be executed at runtime. Now, when contracts checks are compiled into the body of a function, they will always be executed.
c6064ad
to
8600932
Compare
I haven't been involved in the contracts work, so it might make sense to reassign this to someone who is more involved in that effort. I would expect that long term, we definitely want contracts to be type checked even if we don't actually enforce them at runtime, is that correct? I would also expect that |
r? @celinval
Yes, I believe that is the case.
Agreed, it sounds desirable. I'm less sure however about relying on optimisations, when we can simply not compile the contracts when disabled. E.g. I'm not sure what the compile-time penalty might be of optimising out contracts compared to excluding them from compilation in the first place. |
Excluding them from compilation must only happen post MIR borrowck/after analysis as it shouldn't be user-facing, should it not? At this point removing them has to be a MIR optimization pass, so it can just be a more general "remove |
And the advantage of this approach is that contracts stay type-checked, or was there some other advantage(s) you had in mind? |
If I recall correctly, contract checks have the same problem as the UB checks; whether they are enabled is not always known until after monomorphization. So this needs a post-mono MIR optimization. |
Not really. It's less about this being an advantage rather than "it being a MIR opt is unavoidable if we want stuff to get typechecked". If we want them to be type checked, they need to be in the MIR until after MIR borrowck. The only thing:tm: happening after MIR borrowck are MIR optimizations and codegen.
In that case removing them syntactically also doesn't help however, does it? |
The point of enabling/disabling checks at mono-time is to permit use of those checks without recompiling the standard library. As far as I'm aware, this is a highly desirable property. I certainly found it quite effective with the runtime UB checks, and I thought the contract checks work was following the same strategy for the same reasons. This approach must regress This PR does fix the problems that OP says the other one has, but at the cost of making contract checks in the standard library only usable via nightly-only opt-in, and I'm just confused by that tradeoff because I don't believe the motivating problem is real. |
I suggested the removal of the I agree that for wider adoption of std library contracts we will need to implement an easier way for std users to enable them. |
@saethlin said:
Given #136578 (comment) - how come we get different outcomes for aarch64 vs x86_64 if the stages are working as you suggest? If we are getting MIR before mono-time then I'd expect it to be the same across architectures for the same contracts are used irrespective of the platform. If it is past mono-time then there shouldn't be any contracts left so, again, the test outcomes should not have changed. |
Reproduce that difference outside of the mir-opt test suite by using |
The initial implementation of contracts, despite requiring a compiler flag to enable runtime checking of contracts, still compiled contracts into function definitions, even when the compiler flag was disabled. This meant that contracts could not be safely added to functions without breaking optimisations, or even without potentially changing the behaviour of the function. This was blocking the addition of contracts to standard library functions in #136578.
This change guards macro expansion of the built-in contract macros with the
contract-checks
compiler flag. Additionally, it removes thecontract_checks
compiler intrinsic that was used to determine whether contract checks should be executed at runtime. Now, when contracts checks are compiled into the body of a function, they will always be executed.The change is motivated by the following discussion: #144438 (comment)
Contracts tracking issue: #128044
Known limitations:
contract-checks
are disabled, contracts will not be parsed or type checked, meaning that they are susceptible to becoming out of sync with the rest of the codebase.