-
Notifications
You must be signed in to change notification settings - Fork 13k
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
[tracking issue] union
field access inside const fn
#51909
Comments
Now that we don't permit any new functions from getting promoted, I guess we can lift this restriction? In acutal const contexts we check the final value. Although it does mean that users can transmute arbitrary input to safe references and thus dereference them, so via projection we should allow dereferencing raw pointers at the same time. |
It may now be used in constants e.g. for static assertions (but not inside constant functions, yet, see rust-lang/rust#51909)
There's still the pesky run-time issue in that we get the ptr-to-int operation if we stabilize this now. So we get unconst operations without having settled whether it is UB for However, we might reasonably use this only in the standard library for now proviso that we can at least agree on the determinism angle in lib{core,alloc,std} and review carefully (which didn't work super well re. promotability before, but at least now we know better...). |
we will get this problem whenever we stabilize. If we want to prevent it we'd need to introduce "blessed transmutes", but that just moves the complexity elsewhere and lock us into layouts. We could set it up in a manner where the feature is still unstable but doesn't block stabilizing |
Granted, but timing is key here imo -- we'll need to work towards having the "education" available.
Yep, and it has served us well but perhaps the pragmatic choice is to make an exception now.
It'll need to be followed by a comment justifying each use but otherwise it seems reasonable. |
ptr-to-int is not itself non-deterministic though. Allocation and pointer comparison to my knowledge are the only non-deterministic operations (and for ptr comparison I have not been able to observe this non-determinism in a single execution). However, ptr-to-int makes the allocator non-determinism observable, maybe that's what you mean? I suppose you are worried about functions like (maybe using hacks to avoid unstable features): const fn mk_int() -> usize {
let v = 42;
&v as *const _ as usize
} I think we cannot reasonably make such functions UB -- undefined behavior should be decidable dynamically (e.g. by an interpreter like Miri), but "this function is deterministic" is not decidable. So this can be a safety invariant at best. Good enough for unsafe code to rely on it when interfacting with safe code, but not good enough for the compiler to exploit it. |
Yep.
We intentionally tried to prevent that; would be very unfortunate to find out we failed.
Good points; although it would be unfortunate not to take advantage of CSE for calls to |
It would be unfortunate indeed, but non-decidable UB would also be unfortunate.
What do you mean by this? |
For example: const fn foo(x: usize) -> usize { /* stuff */ }
fn bar() {
let n = /* stuff */;
let x: [u8; dyn foo(n)] = /* stuff */;
let y: [u8; dyn foo(n)] = x; // OK.
} |
Ah. Well for executing |
This comment has been minimized.
This comment has been minimized.
This comment has been minimized.
This comment has been minimized.
Since this issue is holding back a few other issues, what needs to be done to fix it? |
Quoting from a Zulip discussion:
IOW, the biggest concern here is how tot each people to use There's some more background on this at https://github.com/rust-lang/const-eval/blob/master/const_safety.md. |
Slightly stupid idea: lint about all unconst operations (with the lint specifically explaining each case individually) and require users to write This may lead to ppl just adding It could also be an unconditional warning that can be silenced by using |
Seems strange to me to have a completely different system for unsafe and unconst, when they are conceptually basically the same thing (just for two different languages: full Rust vs const Rust). |
Yea, I just noticed we could go even more fine-grained:
But this made me realize that any unsafe operation can likely end up doing unconst things in some cases... so we're back to the root problem that we'd just end up marking everything, so there's no useful distinction between |
Ok. How do unions act differently in compile time and run time? |
They don't? Not sure where you got the idea. I linked to a discussion about this above. I think the main concern is that "well-behaved" const code simply has different rules than to be well-behaved "normal" code. Also see this document. Accordingly, I am somewhat confused by @oli-obk's focus on differences between CTFE and runtime. Our CTFE checks should already make sure that does not happen. (Smells like an XY problem to me -- some of the reasons const has different rules is that during CTFE we cannot do certain operations, and the reason for that has to do with not knowing the actual base address of an allocation. But that's already two layers down the rabbit hole from the key point, which is that the rules are different.) |
They do right now, but if we ever want to allow And yes, the root problem for this is that we don't know the base address of an allocation, so for anything CTFE related, the base addresses must be irrelevant. An operation that would observe the base addresses is const-unsafe because you can't let safe code ever observe that base address. "Observing a base address" can e.g. be the fact that we'd return |
I don't think so. Slice equality itself does not actually differ. The only issue is the pointer equality test fast-path, and I've seen a PR that proposed removing it (it seems to be a perf win in some cases and a perf loss in others). If needed we could have a perma-unstable hack to skip the ptr check in CTFE. Also, I see no connection of this problem to transmutes/unions. The problem you mention is (I think) related to "how can we implement ptr equality binops in CTFE without breaking anything"; that question is fully orthogonal to "how do we teach people to not misuse transmutes in CTFE". Transmutes cause problems even with our current non-implementation of ptr binops, and ptr binops cause problems even without transmute. The kind of misuse I have in mind is something like const fn foo(x: usize, y: &i32) -> usize {
let y: usize = unsafe { transmute(y) };
x * y
} This is a sound function as far as runtime Rust is concerned, but it is const-unsound ("unconst") because it can cause a const-failure with const-safe inputs. |
Ok. Let's stop focussing on hypothetical differences and just address the part where a function is unsound if it causes an error at compile time while being sound at runtime. I'll adjust the text accordingly |
Yeah, basically. Of course compile-time functions also can make stronger assumptions about their inputs (e.g.,
|
We have dataflow analysis already, why not to use it to permit what is already known to be sound? For example we can check usage of alocations to forbid pointer to numbers casts which change the output. So we actually want much better check of function purity. I don't think we want to forbid use of entire language primitive because it could cause UB. |
Also we may want to make the ops const someday, but i'm not sure which ones. |
That's the whole point! We want to permit things that are sound, but there's no way to make it safe in general, just like we can't make All we're trying to say is that the rules are stricter (or there are more rules) at compile time than at rumtime, and devs need to be aware of that. |
@tema3210 this entire discussion is about unsafe code, which the compiler cannot check for correctness. If it could, it would be safe code. ;) So no static analysis is going to help. |
The difference is that A constant item is always interpreted at compile-time, so we do not have this duality. Only its final value is used at runtime, and we have checks ensuring that said final value is const sound. |
@powerboat9 those are easy to check as it's a single constant that we can just evaluate. But to do the same with a function, we'd have to run it with all possible inputs, which clearly is not possible. |
I don't really see how unions would behave differently during runtime and during compile time. If a union throws a miri evaluation error, then the code never compiles and thus the function can never be called at runtime. |
And if a function could error at compile time but doesn't, then it will be impossible to observe running at compile time and no difference between runtime and compile time behavior can be observed. |
TLDR: the examples shown have unspecified behaviour depending on compiler versions and optimization levels, which is the exact same behaviour that the already known set of runtime UB exhibits at compile-time. It's not about causing UB at runtime due to UB within a constant evaluation (we prevent that via static checks on constants). Let's just consider the regular UB things that the rustonomicon defines. If you have a function like const fn foo() -> u32 {
unsafe {
let x: usize = mem::uninitialized();
[5, 6, 7][x]
}
} and you evaluate it at compile-time, there's no clear behaviour that can come of this. Either the function
The function will do the same thing no matter how many times and in what circumstances you call it during CTFE. That is, for the same compiler with the same compiler flags. Any change in optimizations or compiler versions can and will change what the function returns at CTFE. Calling it at runtime is UB anyway, so we can ignore that, that has been covered thoroughly by lots of articles. Now, let's consider the additional UB proposed in this issue: const fn foo() -> u32 {
let x = 42;
let y = &x as *const i32 as usize;
50000 / y
} The above is considered UB in CTFE (as suggested in this issue), because you are inspecting a pointer beyond your ability to dereference or offset it. So, what can happen here? The function can either
The argument was that we can make a guaranteed detection of the UB, so it's not a problem. But that's the thing about UB: once bring optimizations into the mix, even deterministic UB detection will become nondeterministic depending on whatever the optimizations are up to. In the above example the optimizer detecting UB can resonably replace the entire function by |
Somewhat. It is much more constrained in that the effects of UB are "returns a weird result from this one const-eval query", and there is no chance of it deleting your files on its own, but that's about it.
One possibility here is also "returns an uninitialized value", which is different from picking any fixed value non-deterministically. |
Citing what I said on Zulip:
@oli-obk basically extended on what that would mean, I think. |
Perhaps have Scalar::Uninit for currently uninitialized values? |
@powerboat9 I don't know what your comment refers to and what you are suggesting where such a |
We have |
union
field accesses allow arbitrary transmutations insideconst fn
and are thus marked asunstable
for now to allow us more time to think about the semantics of this without blocking the stabilization ofconst fn
on it.This is behind the
const_fn_union
feature gate.Blocked on rust-lang/const-eval#14.
The text was updated successfully, but these errors were encountered: