-
Notifications
You must be signed in to change notification settings - Fork 20
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
Expose Arc::is_unique
#560
Comments
Seems like your motivating example is already covered by |
|
Maybe I'm misunderstanding your example. Let me write out how I understand it in pseudocode:
But since we've already established that the reference points into the arc, the last line could just as well be:
And in that case, you can simply the
By the time you've checked the refcounts to check if it's unique, you've already done all of the expensive stuff, so whether you clone the reference or from the arc with |
The entire point of my motivating example is that one can’t swap around the
|
So I misunderstood, I didn't realize that
meant those checks in that order. I'm not sure if putting the checks in that specific order is sufficient to avoid UB, though. Do you have some working code using |
I’m confused what UB you mean. The current UB is precisely because of I realized I was slightly confused – my description was derived from a real case in which this function is necessary, but my attempts to simplify it have made it seem trivial. In particular, even if the |
Never mind |
To be clear, this function already exists – this ACP is just about making it public. |
I think this proposal is needlessly restrictive. There is also the use case which I've had where you own multiple impl<T, A> Arc<T, A> {
pub fn counts(&self) -> (usize, usize) {}
} Ensuring atomicity. |
Also on a side note, if we do go with |
impl<T, A> Arc<T, A> {
pub fn is_unique(&mut self) -> bool;
} as a public API, it is strange that it takes |
I’m not against this, but I think it would be good to have I updated the proposal to use |
We discussed this in today's @rust-lang/libs-api meeting, and we agreed to accept this. Code searches suggest that the primary use of Also, we discussed memory ordering, and felt that this probably warrants acquire ordering; we also felt that we should re-add the acquire ordering to strong_count as well, but that's a separate discussion. |
Proposal
Problem statement
There is currently no way to check whether an
Arc
has unique access to its data that doesn’t involve producing an&mut
to the inner data. However, this is undesirable for some use cases as it invalidates existing pointers to theArc
.Motivating examples or use cases
Suppose one has a self-referential data structure consisting of an
Arc<T>
and an&T
, though the&T
may also for instance point to static data and not theArc
, and the goal is to write a function that produces aT
from this data structure. One implementation would be to first check whether if theArc
is unique and then check if the&T
indeed points to theArc
, and if so, callArc::into_inner
, otherwise clone theT
from the reference. However, currently the step of checking if theArc
is unique will force the&T
to be invalidated, meaning one cannot later clone should the second check fail.In a simple case one could just reorder the two checks to solve this issue, but this is not always trivial to achieve depending on what abstractions are involved: for example, if there were multiple possible backing
Arc
s, one would have to check all of the pointers before attempting toArc::into_inner
the first, instead of being able to short-circuit should the first succeed. In a more complex case, if the dependent data was not&T
but had a destructor, there would be no way to correctly run it: if the destructor was run beforeget_mut
orinto_inner
, it would be impossible to recover the original data to clone it later, but if it was run after, it would already be invalidated.Solution sketch
One argument for the inclusion of this API directly is that, even when lifetime-extended references are not involved, some people might try and, instead of calling
Arc::get_mut(&mut arc).is_some()
, callArc::strong_count(&arc) == 1
– indeed, grep.app displays many results for this pattern – and not be aware of the race condition this may introduce should weak references exist, leading to code that is subtly wrong. Introducing this method makes it more likely that people will choose the correct way of implementing this function.Alternatives
This doesn’t come with the discoverability advantages of
is_unique
mentioned above. However, it does make it more explicit that one can obtain mutable access to the internal value, which might be convenient. It is currently not guaranteed that one can mutate through the pointer returned byArc::as_ptr
– there was one attempt to guarantee this that fizzled out – but it is unlikely that we will not end up guaranteeing this.Another alternative is having the method take
&mut self
instead of&self
. This can reduce footguns sinceis_unique
is rarely useful when used as an&self
method, but it is possibly outweighed by the flexibility.Links and related work
None I’m aware of.
What happens now?
This issue contains an API change proposal (or ACP) and is part of the libs-api team feature lifecycle. Once this issue is filed, the libs-api team will review open proposals as capability becomes available. Current response times do not have a clear estimate, but may be up to several months.
Possible responses
The libs team may respond in various different ways. First, the team will consider the problem (this doesn't require any concrete solution or alternatives to have been proposed):
Second, if there's a concrete solution:
The text was updated successfully, but these errors were encountered: