-
Notifications
You must be signed in to change notification settings - Fork 13.2k
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
Added Box::take()
method
#93653
Added Box::take()
method
#93653
Conversation
(rust-highfive has picked a reviewer for you, use r? to override) |
This comment has been minimized.
This comment has been minimized.
aea701b
to
b54257f
Compare
This comment has been minimized.
This comment has been minimized.
This comment was marked as resolved.
This comment was marked as resolved.
I think you might want to add it here, between these two: Lines 103 to 104 in cb18e83
|
Looks legit, thanks! |
This comment has been minimized.
This comment has been minimized.
The new `Box::take()` method allows to read from `Box` and reuse the allocation safely. This commit also changes some `unsafe` code in the compiler to use this method instead and becomes safe.
I wonder how this would affect learning rust. As a beginner, I was looking for a |
@wwylele interesting point. I was also considering the names |
#[unstable(feature = "new_uninit", issue = "63291")] | ||
#[rustc_const_unstable(feature = "const_box", issue = "92521")] | ||
#[inline] | ||
pub const fn take(this: Self) -> (T, Box<mem::MaybeUninit<T>, A>) { |
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.
take
to me has a very strong implication of mem::take
(or Option::take
or ...) to me, with &mut self
and such, so I would encourage a different name here.
👍 to the functionality, though! Makes good sense to me.
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 actually picked the name because it is similar to mem::take
in the sense of "taking the value out and replacing it with something", in this case replacing it with nothing and tracking it in the state. However if people feel it's strongly associated with &mut self
I'm open to a better name, I'd just like to hear some suggestions on alternatives or a feedback on my own ides for alternatives (see above).
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 do see the parallel, though to me this API doesn't do the "and replacing it with something" part that distinguishes take
from into_inner
kinds of things.
But more importantly to me, mem::take(&mut my_box)
works today, so I think Box::take
doing something pretty different would be surprising. If anything, I might expect it to be mem::take(&mut *my_box)
-- note the deref -- like how RefCell::take
works. (And note that mem::take(&mut my_option)
does exactly the same thing as Option::take(&mut my_option)
, as another example.)
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.
Yeah, what do you think bout into_parts()
? That one feels like the next best to me now.
I would like to see a somewhat stronger use case for this API, ideally in the wild. I don't think the compiler code pointed to is necessarily a good case -- that code is more closely representative not of reusing an underlying allocation, but rather expressing roughly a I'm somewhat weakly concerned by the continued growth of API surface area around the notion of containers and their 'untracked empty' states (i.e., MaybeUninit) -- it feels like we've added a lot of surface area here (cc #63291) and it's not clear to me that we want all of it. In general, most of these APIs are relatively simple under the hood (3-4 lines of unsafe), and while providing abstractions over unsafe code is generally positive, I'm not convinced it warrants the prominent surface area in std's documentation this PR and those APIs are placing. This need not block this PR, necessarily, but I would like to bring this up with T-libs-api and see if there's thoughts on the surface area being added over time here. I would be particularly interested in seeing if we can design some better interfaces -- having every container in std contain functions for construction (fallible allocation * uninit/zeroed * with/without a custom allocator * with/without pin * with [T] or not) adds a ton of surface area -- that's at least 32 functions if we fill out the matrix completely. On the deconstruction side, there's less, but APIs like Box::write which 'complement' the Box::take added by this PR also add surface area. I'm not sure it's quite the right fit. I would also like to consider whether there's language support that may be warranted here -- this particular PR, as #93653 (comment) sort of mentions, feels like it could be more 'native' -- e.g., if you move out of a Box, it naturally becomes a |
@Mark-Simulacrum fair points. When approaching If there are better ways to deal with this then great. Your suggestion to extend the language rather than the library sounds interesting. OTOH I'm unsure if it's really better to extend the language or library. A possible obstacle is that |
In general I agree with expressing fundamental operations as building blocks, but I think it depends on whether you consider this particular API a building block or not -- the use cases I asked for may help push me to see this as more of a building block. For this particular API, I could see us supporting a |
I've opened a discussion that'll hopefully help gather some feedback. Generalized API sounds good. |
I think an API like If I know I'm going to move out of the box and reuse it then I'd probably use We could always turn this into a function or extension trait and iterate in the ecosystem if you want. trait BoxTake<T> {
fn take_(this: Box<T>) -> (T, Box<MaybeUninit<T>>);
fn initialize(this: Box<MaybeUninit<T>>, value: T) -> Box<T>;
}
impl<T> BoxTake<T> for Box<T> {
fn take(this: Box<T>) -> (T, Box<MaybeUninit<T>>) {
unsafe {
let ptr = Box::into_raw(this);
let value = ptr.read();
(value, Box::from_raw(ptr.cast()))
}
}
fn replace(mut this: Box<MaybeUninit<T>>, value: T) -> Box<T> {
unsafe {
this.write(value);
Box::from_raw(Box::into_raw(this).cast())
}
}
}
fn main() {
let boxed = Box::new(42);
let (fourty_two, allocation) = Box::take(boxed);
assert_eq!(fourty_two, 42);
let initialized = Box::replace(allocation, 7);
assert_eq!(initialized, 7);
}
Rust doesn't have any mechanism where calling a method or executing an operator on a value will implicitly change its type so implementing a general "move out of pointer changes the original variable's type to Do we really want to introduce another subtle |
@Michael-F-Bryan I think that |
The libs-api team discussed this today, and particularly in the context of concerns around the design of the standard library surface area around MaybeUninit (cc #63291, zulip, and rust-lang/wg-allocators#90), we don't want to move ahead with adding this API at this time. @m-ou-se is going to leave a comment elsewhere elaborating on that discussion a little. It's also the case that this particular API likely does not pull its weight in terms of providing a sufficient abstraction to merit inclusion even without those larger concerns; if there are more extensive use cases (and in particular, ones that move the needle from completely avoiding unsafe code to reducing it a little bit), we might reconsider. There was also some discussion around the fact that the API does not leverage any std-special bits, so any crate in the ecosystem could provide a reusable API layer here to experiment with the options for exposing good APIs. I'm going to go ahead and close this PR as such, but if the users.rust-lang.org thread results in good use cases, feel free to poke on Zulip (or reopen a PR) to see if they move the needle on accepting this. |
OK, thanks! |
The new
Box::take()
method allows to read fromBox
and reuse theallocation safely. This commit also changes some
unsafe
code in thecompiler to use this method instead and becomes safe.
This was suggested in #63291 (comment) where it got two thumbs ups and no objections and is pretty simple, so should be OK without RFC. It reuses the feature
new_uninit
, hope it's OK.Related: I noticed number of
unsafe
casts ofBox<T>, A
toBox<U, A>
, maybe we should have apub
commonunsafe
methodcast_unchecked
for these? It'd be clearer and less boilerplate-y. Also cast (From
)Box<T, A>
->Box<MaybeUninit<T>, A>
could be safe (though leaking). Can open a new PR for this if there's interest.