afonso360 opened issue #5900:
:wave: Hey,
Feature
After #5893 we now have access to
MemFlags
when loading and storing to memory. We should check thereadonly
flag, and always trap if there is a write to it.Technically the current behaviour of ignoring it and loading correctly anyway is allowed.
Loads with this flag have no memory dependencies. This results in undefined behavior if the dereferenced memory is mutated at any time between when the function is called and when it is exited.
There is a world where we keep track of these flags (in both loads and stores) and can trap at any time a store to a previously
readonly
'd memory location. But this is a good start.Benefit
This gets us a better interpretation of CLIF semantics and allows whoever is using the interpreter to catch these cases sooner.
Implementation
In
State::checked_store
check if the memory flags have the readonly flag set and return aMemoryError
if so.We can't build the usual tests for this.
- Our
runtests
don't support checking for traps (cc #4781).- What we can do instead is build a CLIF program and invoke the interpreter using the library API. See this example for a test case testing traps on
srem
, we can do something similar for load/store.Alternatives
The current implementation is fine, and we should technically never see a store with a
readonly
flag.
afonso360 labeled issue #5900:
:wave: Hey,
Feature
After #5893 we now have access to
MemFlags
when loading and storing to memory. We should check thereadonly
flag, and always trap if there is a write to it.Technically the current behaviour of ignoring it and loading correctly anyway is allowed.
Loads with this flag have no memory dependencies. This results in undefined behavior if the dereferenced memory is mutated at any time between when the function is called and when it is exited.
There is a world where we keep track of these flags (in both loads and stores) and can trap at any time a store to a previously
readonly
'd memory location. But this is a good start.Benefit
This gets us a better interpretation of CLIF semantics and allows whoever is using the interpreter to catch these cases sooner.
Implementation
In
State::checked_store
check if the memory flags have the readonly flag set and return aMemoryError
if so.We can't build the usual tests for this.
- Our
runtests
don't support checking for traps (cc #4781).- What we can do instead is build a CLIF program and invoke the interpreter using the library API. See this example for a test case testing traps on
srem
, we can do something similar for load/store.Alternatives
The current implementation is fine, and we should technically never see a store with a
readonly
flag.
afonso360 labeled issue #5900:
:wave: Hey,
Feature
After #5893 we now have access to
MemFlags
when loading and storing to memory. We should check thereadonly
flag, and always trap if there is a write to it.Technically the current behaviour of ignoring it and loading correctly anyway is allowed.
Loads with this flag have no memory dependencies. This results in undefined behavior if the dereferenced memory is mutated at any time between when the function is called and when it is exited.
There is a world where we keep track of these flags (in both loads and stores) and can trap at any time a store to a previously
readonly
'd memory location. But this is a good start.Benefit
This gets us a better interpretation of CLIF semantics and allows whoever is using the interpreter to catch these cases sooner.
Implementation
In
State::checked_store
check if the memory flags have the readonly flag set and return aMemoryError
if so.We can't build the usual tests for this.
- Our
runtests
don't support checking for traps (cc #4781).- What we can do instead is build a CLIF program and invoke the interpreter using the library API. See this example for a test case testing traps on
srem
, we can do something similar for load/store.Alternatives
The current implementation is fine, and we should technically never see a store with a
readonly
flag.
afonso360 labeled issue #5900:
:wave: Hey,
Feature
After #5893 we now have access to
MemFlags
when loading and storing to memory. We should check thereadonly
flag, and always trap if there is a write to it.Technically the current behaviour of ignoring it and loading correctly anyway is allowed.
Loads with this flag have no memory dependencies. This results in undefined behavior if the dereferenced memory is mutated at any time between when the function is called and when it is exited.
There is a world where we keep track of these flags (in both loads and stores) and can trap at any time a store to a previously
readonly
'd memory location. But this is a good start.Benefit
This gets us a better interpretation of CLIF semantics and allows whoever is using the interpreter to catch these cases sooner.
Implementation
In
State::checked_store
check if the memory flags have the readonly flag set and return aMemoryError
if so.We can't build the usual tests for this.
- Our
runtests
don't support checking for traps (cc #4781).- What we can do instead is build a CLIF program and invoke the interpreter using the library API. See this example for a test case testing traps on
srem
, we can do something similar for load/store.Alternatives
The current implementation is fine, and we should technically never see a store with a
readonly
flag.
bjorn3 commented on issue #5900:
Maybe disallow the readonly flag on stores in the clif ir verifier instead of the interpreter? Checking at runtime that no stores to readonly memory happen can't be done in the verifier, but checking that the readonly flag isn't set for stores can easily be checked in the verifier. I can't imagine any case why you would intentionally want the readonly flag on a store. It did unconditionally be UB, so you might as well insert the trap instruction instead.
afonso360 commented on issue #5900:
I think we had a similar situation here and I think a similar reasoning might apply? We can optimize other
loads
/stores
such that we now statically know this memory region isreadonly
and thus we are now in the same situation where we get invalid CLIF by optimizing valid CLIF.
afonso360 edited a comment on issue #5900:
I think we had a similar situation here and I think a similar reasoning might apply? We can optimize other
loads
/stores
such that we now statically know this memory region isreadonly
and thus we are now in the same situation where we get invalid CLIF by optimizing valid CLIF.Edit: Although replacing with a
trap
is somewhat different, since it keeps it as legal IR. I'm not sure about this one.
bjorn3 commented on issue #5900:
The readonly flag applies to a specific instruction. If an optimization propagates readonly to other memory accesses, it has to explicitly set it for store operations. I don't think the same reasoning as #5167 applies as an access being in bounds of a memory region is a dynamic property where optimizations may cause more knowledge being gained, and only being known in the general case at runtime, while a store being marked as readonly is a static property that is always known at compile time. While the accesses memory region being readonly is a dynamic property, the actual flag on the store is a static property.
littlebenlittle commented on issue #5900:
Just opened a draft PR for this. The implementation is straightforward. To write a test case I need a little more info.
In the cranelift IR, what are the semantics of a read only store? Is it just "error", as in a conformant compiler should never do this?
The next issue is how to invoke a read only store using CLIF. How are
MemFlags
set? Is that the responsibility of the compiler or the programmer? If its on the programmer, i.e. read only stores are supported by CLIF, then this should be easy. If it's on the compiler, then we might need to write a non-conformant compiler and that's quite a bit more complex, I would assume.
jameysharp commented on issue #5900:
I like @bjorn3's suggestion in this thread that we should reject read-only stores in the verifier (
cranelift/codegen/src/verifier/mod.rs
), rather than in the interpreter. TheMemFlags
are a static property of the store instruction so we don't need to wait until we're running the program to check this. Thealigned
flag (#5899) is different because although the flag is static, the address is not.I have to think some about @afonso360's comments, about whether optimizations could discover that a store is to a location which should be read-only. I'm not sure what optimization we might want to implement that would do that, and I think it's more likely that if there's a store after a read-only load we need to conclude that the load was wrong. But in any case this would still be an error that we should detect during or immediately after optimization, not at runtime.
As for how to test this:
MemFlags
are specified by whoever constructed the CLIF. That can be a frontend like Wasmtime or cg-clif, or it can be in the text representation; seecranelift/filetests/filetests/egraph/load-hoist.clif
for an example with several readonly loads.
littlebenlittle commented on issue #5900:
Is this already implemented in the verifier on the
main
branch? [1]There doesn't appear to be a test case, however.
littlebenlittle edited a comment on issue #5900:
Is this already implemented in the verifier on the
main
branch?There doesn't appear to be a test case, however.
jameysharp commented on issue #5900:
Hah, you're right!
Want to open a PR with a test case confirming that it works? There are other verifier tests in
cranelift/filetests/filetests/verifier/
.
jameysharp commented on issue #5900:
Closing this as @littlebenlittle just provided a test in #6593 showing that this already works the way we've decided it should. Thank you!
jameysharp closed issue #5900:
:wave: Hey,
Feature
After #5893 we now have access to
MemFlags
when loading and storing to memory. We should check thereadonly
flag, and always trap if there is a write to it.Technically the current behaviour of ignoring it and loading correctly anyway is allowed.
Loads with this flag have no memory dependencies. This results in undefined behavior if the dereferenced memory is mutated at any time between when the function is called and when it is exited.
There is a world where we keep track of these flags (in both loads and stores) and can trap at any time a store to a previously
readonly
'd memory location. But this is a good start.Benefit
This gets us a better interpretation of CLIF semantics and allows whoever is using the interpreter to catch these cases sooner.
Implementation
In
State::checked_store
check if the memory flags have the readonly flag set and return aMemoryError
if so.We can't build the usual tests for this.
- Our
runtests
don't support checking for traps (cc #4781).- What we can do instead is build a CLIF program and invoke the interpreter using the library API. See this example for a test case testing traps on
srem
, we can do something similar for load/store.Alternatives
The current implementation is fine, and we should technically never see a store with a
readonly
flag.
Last updated: Jan 24 2025 at 00:11 UTC