-
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
Possible to interleave &mut
and &
safely?
#562
Comments
Even in the absence of reads, it's exactly the "pinning problem," or well, the aliasing part of it, not the stable address part. The stack bytes will need to only ever be The other options are only using raw pointers, such that retags don't occur, or using a type with identical layout which does not cover the stack bytes, only the "header" fields that matter to the kernel, not the managed thread. (And in the models that only have type driven retagging, not field driven "precise" retagging, this practice can improve the optimization potential, at the expense of more typing effort.) |
If |
I considered this, but worried that I might run into provenance issues since, at some point in the future, I'd need to take a |
The " However, that potentially doesn't even matter for you here. Restricted provenance only ever applies once retags are involved; casting from Footnotes
|
Not quite; But also if that In the end this is similar to what happens in the global allocator: there need to be explicit transition points where the memory that stores the stack gets removed from the allocation that holds |
Does Rust expose operations that instruct it to perform these transitions? How do allocators accomplish this? |
Global allocators are registered specifically with If you want to do this entirely yourself, I think you need inline assembly "barriers" of sorts to clearly tell the compiler "there's some reorganization of the AM state going on here that I am not writing out in Rust code". We'd have to discuss the concrete case in more detail to say anything specific (and I don't currently have the time to really dig into the details, sorry). |
Mostly vibes, tbqh. However, the vibes mostly work out. The magic layer happens at But, as far as I'm aware, the way to formally justify a |
I'm working on a linked list implementation for an embedded kernel. We have the following
Thread
structure, which will be a member of a list:The
stack
is the thread's stack, which will be used to hold arbitrary Rust values, which might include&mut
references to other values in the stack (i.e., it might be self-referential). (While pinning is obviously a concern, that's not what this issue is about.)In addition to self-references in the stack, we also need to be able to inspect other fields in
Thread
from inside the scheduler.Here's an example - possibly problematic - sequence of events:
&mut
reference to the stack that also refers to the stack.&Node<Thread>
reference to the currently-running thread and reads itsid
fieldWhile, in (3), we haven't read data from
stack
, I presume that it's still considered UB to have a&Node<Thread>
while other&mut
references exist which point into thestack
, which overlaps with the referent of the&Node<Thread>
?My first thought was to just wrap the whole thing in
UnsafeCell
so that theNode
API doesn't provide&Thread
s, but instead provides&UnsafeCell<Thread>
s. But IIUC that doesn't solve the problem - it's still illegal to have&UnsafeCell<T>
and&mut T
overlap.My next thought was to just use raw pointers and never synthesize
&
references.Concretely, my questions are the following. All of them are in the context of
&mut
references existing which refer tostack
.NonNull<Node<Thread>>
, is it sound to perform pointer arithmetic to obtain aNonNull<u32>
and dereference it? Specifically, is it problematic that the originalNonNull<Node<Thread>>
refers to the wholeThread
, including thestack
?T
in aNode<T>
which would play nicely with this use case?The text was updated successfully, but these errors were encountered: