-
Notifications
You must be signed in to change notification settings - Fork 59
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
Glossary unspecified/implementation-defined behavior #201
Comments
Adapting the C++ standard, it could look like this:
These refer to well-formed programs. We might want to define that term as well:
|
This sounds strange to me. Shouldn't it be “that does not depend on concrete implementation”? |
Yeah that should probably say "that does not depend on" instead of "that depends on" |
Thanks, fixed. |
I would prefer if we could make those definitions directly refer to behavior of the Rust Abstract Machine. That machine is the "core artifact" defined by the spec, after all. For example, implementation-defined behavior would be aspects of the Rust Abstract Machine that are left open by the standard, but have to be specified precisely by an implementation. You say "the reference often delineates the range of possible behaviors", but isn't that "always"? Because without bounding the range of possible behavior, implementations could say "this is UB", and that does not seem very helpful? Also, I think this should touch on the matter of stability. Like, I assume unspecified behavior is allowed to change from version to version of an implementation, but changing implementation-defined behavior is a breaking change? That doesn't really apply to he "well-formed" part -- TBH that's nitpicking I care little about, other than one question: you are still implicitly using the term "a Rust program" without defining it. If you go down this route, please go all the way. ;) |
You say "the reference often delineates the range of possible behaviors",
but isn't that "always"?
I think we should always delineate the behavior, at least to say that it
cannot be defined to be undefined - we could have optional behavior, eg, an
implementation can offer atomic, but it does not have to.
I also think we should try to “delineate” as much behavior as possible. For
example, the size of usize isn’t just an implementation defined integer -
it also matches the size of a pointer, the size of uintptr_t in C, etc. All
of these guarantees we provide constrain the valid sizes an implementation
could pick for usize.
Also, I think this should touch on the matter of stability. Like, I
assume unspecified behavior is allowed to change from version to version of
an implementation, but changing implementation-defined behavior is a
breaking change?
Yes. For a stable Rust target, eg, changing the size of usize would be a
breaking change. Changing some detail of the Rust ABI isn’t (eg how scalar
pairs are passed). Implementation defined behavior must be documented and
is stable. Unspecified behavior must be something, but that something
doesn’t even have to be deterministic. Eg
let x = 3;
Let y = 3;
assert_eq(&x as *, &y as *);
The behavior of that program is defined. But whether the assert passes or
fails is unspecified. Can change with the opt level, etc.
|
Note that behavior could then still be "it resets the state of the Rust Abstract Machine to be empty". So, it seems we need a more specific parameterization, that well depend on the individual case. In the context of what I discussed with @rkruppe in the other thread, it seems like we might also have "unspecified behavior" where we can't really say what happens though, such as messing with page tables. Not sure where to fit that in. Some of this related to whether it is even possible to think in terms of the Rust Abstract Machine -- which leads us into a cycle, because it is the Rust Abstract Machine that defines whether it is even possible to think in terms of hardware or page tables... |
The main purpose of unspec as I see it is to facilitate future language design by giving it options to work with. As such, what I really want from unspec is:
(...other than to document it for internal purposes.) |
@RalfJung agreed. FWIW I think it would be ok to start with a loose definitions of these terms, and making them precise over time. Initially it would be enough for me if the definitions were not contradictory. E.g. if we define undefined behavior as "behavior for which the abstract machine provides no guarantees", then the "unspecified behavior" definition proposed by @Centril might be undefined behavior. We don't want, e.g., the optimizer reasoning that if your program exhibits unspecified behavior that state is unreachable (as opposed to UB, where that would be fine). |
A PR to define these terms (unspecified vs implementation-defined) would be very welcome. |
We should probably document these terms in the glossary somewhere.
The text was updated successfully, but these errors were encountered: