-
-
Notifications
You must be signed in to change notification settings - Fork 220
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
Block GdCell::borrow
and GdCell::borrow_mut
on other threads
#736
Block GdCell::borrow
and GdCell::borrow_mut
on other threads
#736
Conversation
API docs are being generated and will be shortly available at: https://godot-rust.github.io/docs/gdext/pr-736 |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
As far as I see, the other two issues are both solved by enabling the |
2a90b98
to
09850a8
Compare
I now had the time to test #556, and it no longer panics during a reimport. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thanks a lot!
If ever needed in the future, we might also consider a separate cell with the threading logic, but for now I'd keep complexity low. As long as people don't use experimental-threads
, they still have the full performance.
There's one part where I'm not completely following -- in claim_mut_thread
, the caller intends to have mutable access. But why is a read lock (not a write one) returned, and why is the downgrade needed? If you could elaborate this in code, would be perfect.
id like to have a look at this later before it's merged, since i made the original gd-cell I'd just wanna be sure this seems correct |
So the The constraints on how many shared or mutable references can exist at the same time continues to be enforced by the existing implementation of I will try to better describe this in the code. |
I'm not sure this is the right approach. It's a bit difficult to tell whether this is gonna be sound or not, it'd be a lot better if you could somehow separate the blocking logic entirely from the gd-cell logic. Since i dont think any If you could somehow make a pub struct GdCellBlocking<T> {
cell: Pin<BoxGdCell<T>>>,
// Other stuff you need to check blocking
} Which has the same API as Additionally your usage of |
c4d0d15
to
a002864
Compare
This PR does not add any unsafe code or change the way the Yes, the
If you'd like, I can move the new parts of
That is certainly possible but would just add more overhead to everything. I would also have to wrap the two guard types to track when they are dropped. The guard types would also need an
What is the problem with inspecting the internal state of the cell? The internal state is not being mutated by these changes. By only using the current public interface, I would have to re-track the internal state that is inspected to make decisions on whether to block a thread or not, adding more overhead.
It's primarily a lock, and we are locking the access to the exclusive mutable reference. Do you see an actual issue with using it?
The same behavior could be replicated with a |
Yes, because the implementation relies on these invariants to be upheld for safety elsewhere. By adding more logic and arguments to these function you are increasing the surface area of what needs to be verified, even if it's all correct it's still more that needs to be kept in mind when ensuring things are safe.
When dealing with
I dont see why? The outer functions can just call
You can add new read-only public (or crate-public ideally, avoid leaking internal state where possible) methods that track the internal state. But if you have direct access to the internal state then there's nothing stopping you from mutating it. Which means that when verifying that the implementation is correct and doesn't break anything i need to also ensure that the implementation doesn't mutate any of the internal state it has access to. We're dealing with
It's confusing is my main point. Usually the
These are both non-standard uses of
I dont think it's that much more straight-forward. You end up needing this Whereas a
All of which happen much more implicitly with the Overall, having a "dumber" implementation which has more boilerplate/overhead, is preferable imo over a "clever" implementation with minimal boilerplate/overhead. Entirely because we're in a safety-critical situation, and it's very important that the unsafe code is correct. Keep in mind that rust can often optimize some amount of boilerplate and trivial overhead. At least until it's clear that this is something that needs to be optimized for performance, at which point there are likely several other places to optimize that are bigger gains than this. If we're confident the current unsafe code is correct, then i think we should touch it as little as possible when adding functionality. And i do think that in this case it is possible to implement blocking functionality while minimally changing the unsafe code (perhaps not at all), and just adding some extra functions which inspects the internal state without modifying it. |
I'm a bit split on this. I think @lilizoey is making a good point that it's safer to not touch existing logic as the surface area increases. However, that also means:
All in all, it's hard for me to estimate how complex an external solution (wrapper cell) would be, so maybe we should try to find that out? |
yeah, if an external solution is a lot worse/more complicated in some ways then it's preferable to do something internal to GdCell. i just dont think it is gonna be much worse here, it may even be simpler to understand. and if that is the case then it'd be better to avoid touching the current unsafe code if we can. |
@lilizoey yes, but we also have to track (in the case of
@Bromeon I don't think it would be overly complex and primarily just introduce more overhead and boilerplate. I will give it a shot and report back if I think it's getting out of hand. |
Thank you! Maybe makes sense to do it in a new commit, so we can compare side by side? |
a002864
to
1ac3e44
Compare
This PR now has the new version, which is implemented as a wrapper type. I just did it as quickly as possible, and we can streamline it and remove unnecessary stuff. I left some new crate internal functions on The previous version is here: 747a372 |
It is a bit unfortunate to have to duplicate some of the reference counting logic, and having to add a new set of guards. but in my opinion this new version is easier to reason through and simpler to understand. The fact that no |
This is now our 5th guard type/pair in the library 😀 But generally I agree, it's nice that this can be done non-intrusively and safely, and I think this is a typical case of "the complexity has to live somewhere": as much as I'd like to make things simpler, I think we can't deny reality that interfacing with Godot is inherently difficult, especially when it comes to re-entrancy and threads. As long as we can shield users from all that complexity, I'm OK with a bit of internal duplication, if it in turn makes 2 separate concepts (blocking access in this PR, and safe re-entrancy in the existing Currently the library would enable either the blocking implementations (in Looking at the implementation, there's possibly some code that can be deduplicated between Also, you currently use |
I can't think of a reason why someone would want to use the existing
There shouldn't be any reason for a panic as far as I can see. The mutex lock is always very short-lived when doing the reference counting and acquiring the inner
Maybe we want to consider hiding the |
i can imagine in the future allowing people to choose whichever user-data wrapper they want. (like in gdnative i believe?) in that case having both available would be nice. But for that we'd probably want to introduce a trait to abstract over userdata wrappers. the existing implementation should be equivalent to the blocking one for single-threaded uses, but the blocking impl might be slightly less performant. |
I think the current naming is OK, maybe I'd use Regarding conditional compilation, wouldn't it be possible to provide the same API for both towards the outside and hide any differences behind use statements or type aliases? #[cfg(feature = "threads")]
mod types {
pub use crate::BlockingGdCell as Cell;
pub use ... as RefGuard;
pub use ... as MutGuard;
}
#[cfg(not(feature = "threads"))]
mod types {
pub use crate::GdCell as Cell;
pub use ... as RefGuard;
pub use ... as MutGuard;
}
// The public API is then this:
pub use types::*; To avoid exposing the basic |
@Bromeon sure we can do this, it's just that All this means that the usage of the two types is slightly different. (this is something visible in the changes I made to the mock tests). If everyone is fine with these differences, then I'm happy to just use a type alias. |
But it doesn't seem like the For example, https://github.com/godot-rust/gdext/blob/master/godot-core/src/obj/script.rs mentiones both owned and borrowed types: Pin<Box<GdCell<T>>>
Pin<&'a GdCell<T>> Or maybe you have a suggestion? Would you wrap those separately? |
Everything is still WIP, and I'm still working on cleaning things up and documenting the new types and methods. |
d0d0ab2
to
ce50e82
Compare
Documentation is currently messed up due to everything being private except the type aliases. I haven't looked into solving it yet, but if you have suggestions on what you would prefer, please let me know. The |
ce50e82
to
146fce3
Compare
Which documentation are you referring to?
That should be easy to solve with a |
I think there's another way to solve that, just make both the blocking and non-blocking version public to gd-cell. then pick the appropriate one in godot-core. like // in gd-cell/src/lib.rs
pub mod blocking {
pub use some::path::to::GdCellBlocking as GdCell;
pub use some::path::to::RefGuardBlocking as RefGuard;
...
}
pub mod panicking {
pub use some::other::path::to::GdCell;
pub use some::other::path::to::RefGuard;
...
}
// in godot-core somewhere
#[cfg(feature = experimental_threads)]
use gd_cell::blocking::*;
#[cfg(not(feature = experimental_threads))]
use gd_cell::panicking::*; Since both the blocking and non-blocking version are public here, and thus accessible outside of gd-cell, this shouldn't trigger unused code warnings. |
455db8d
to
ddd331c
Compare
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thanks again for the update!
ddd331c
to
f4fcdc7
Compare
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think this looks pretty good, some nitpicks here and there and also just noticed that i didn't include a Send
bound in the Sync
impl for GdCell
which should be there.
f4fcdc7
to
eec138d
Compare
8859d01
to
efa52b7
Compare
I think this looks good now! Might cause a minor conflict with #741 but i can just fix that on my end. |
Should fix: #709 and #556.
This introduces a new
threads
feature to godot-cell. When the feature is enabled,GdCell::borrow
will block the current thread if any other thread is currently holding a mutable reference.GdCell::borrow_mut
on the other hand, will block if any other thread is currently holding a shared (or mutable reference), but the current thread is not. In any other case, the methods panic as before.I named the feature
threads
and notexperimental-threads
as this should not be considered experimental insidegodot-cell
. The feature is then enabled bygodot-core
sexperimental-threads
feature.This also adds
parking_lot
as a dependency ofgodot-cell
as the standard libraryRwLock
does not support write lock downgrading.