Skip to content
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

Drop type destructuring #3738

Open
wants to merge 4 commits into
base: master
Choose a base branch
from
Open
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
392 changes: 392 additions & 0 deletions text/3738-drop-type-destructuring.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,392 @@
- Feature Name: `drop_type_destructuring`
- Start Date: 2024-12-08
- RFC PR: [rust-lang/rfcs#3738](https://github.com/rust-lang/rfcs/pull/3738)
- Rust Issue: [rust-lang/rust#0000](https://github.com/rust-lang/rust/issues/0000)

# Summary
[summary]: #summary

Rust does not allow destructuring types which implement the `Drop` trait. This means that moving data out of such types is hard and error prone. The rationale is that once fields are moved out, the type's `Drop` implementation cannot run, which can be undesired. This RFC proposes to allow destructuring anyway, in certain situations.
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Could you add more detail here? The reasons for this language quirk are crucial to this RFC, what exactly is it trying to prevent?

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Consider this example:

let x = X { y: Y }; // X: Drop, Y: !Copy
let y = x.y;
// implicitly this line contains `drop(x)`

The implicit drop(x) would be an error, because x.y is moved out. So the "you can't move fields out of Drop types" is a specialization of "use of partially moved value".

We could say that let y = x.y disables the drop, but that is a footgun, see #3738 (comment).

Do you think we should write this more clearly in the RFC?

Copy link
Member

@Nadrieril Nadrieril Dec 8, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Do you think we should write this more clearly in the RFC?

I would quite like that. My mental model makes me expect let y = x.y to disable the drop, and it's not obvious to me that this is a footgun, so I'd appreciate a motivation + example for why we don't allow this.

Copy link

@Aloso Aloso Dec 10, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

So the "you can't move fields out of Drop types" is a specialization of "use of partially moved value".

Except it is more strict, because moving fields out of Drop types is forbidden even if you put something back afterwards:

// only allowed if `Foo` does NOT implement Drop
let mut foo = Foo {
    string: "old value".to_string(),
};
drop(foo.string);
foo.string = "new value".to_string();
drop(foo);

EDIT: I realized why this is necessary. If a panic occurred between moving the field out and moving a new value in, the struct would be in an invalid state while unwinding.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Consider this example:

let x = X { y: Y }; // X: Drop, Y: !Copy
let y = x.y;
// implicitly this line contains `drop(x)`

The implicit drop(x) would be an error, because x.y is moved out. So the "you can't move fields out of Drop types" is a specialization of "use of partially moved value".

We could say that let y = x.y disables the drop, but that is a footgun, see #3738 (comment).

Do you think we should write this more clearly in the RFC?

Is let X { y } = x is disallowed when X: Drop, Y: !Copy simply because it would be weird and seemingly inconsistent for let X { y } = x to be legal while let y = x.y is illegal? Or is there a deeper reason?

The reason I ask is that, as discussed in this thread, an alternative design would just be to permit let X { y } = x, and have it semantically move all of the fields rather than dropping the original x. That option would still permit let y = x.y to remain illegal.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@programmerjake I'm not sure how what you showed is a footgun. S {} = trivially doesn't do anything, it's basically equivalent to (_: S) (if we had type ascription)

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@Nadrieril the main footgun with both "allow destructuring to just work" and "make moving a field out disable drop" is that an innocuous at first glance change may lead to drop being disabled out of nowhere:

struct D { a: u32, b: String, c: C }

-let D { a, .. } = d;
+let D { a, b, .. } = d;

There is nothing in the code to suggest that the drop is now disabled, where it wasn't before. This would be very hard to catch in code review even if you know that D has a drop impl. (example for field access is basically the same) (this is doubly bad in matches, since there some arms could disable the drop while others wouldn't)

Additionally allowing destructuring doesn't solve the case of "all fields are Copy". This can be done with ManuallyDrop already, but this RFC still provides a nicer alternative:

// Current:
let v = ManuallyDrop::new(v);
(v.a, v.b, v.c)

// With this rfc:
destructure!(V { a, b, c } = v);
(a, b, c)

(I will admit the difference is not that big, but I think the latter is still esier to understand/harder to misuse)

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@programmerjake is illustrating that if you have two non-zst structs and use an irrefutable pattern to destructure, then you get a compile error; but zsts can effectively be "moved". That is, we cannot reliably count "instances" of them. I did not actually know this myself.

The thing is though that you have to be in the very weird place of having a zst with no fields to make that happen. I can't actually think of a practical case where you would want to let a public API consumer (or a private one, for that matter) materialize stuff "out of thin air" and so the private field would prevent it. Drop as RAII usually "undoes" logic, but if you can make a zst without running that logic then the API is already incorrect. So you practically need a case where there is a Drop impl that does not require any state, does not do anything based on instance identity, and does not "undo" previous logic. Literally the only thing I can think of is counters, except that nothing counted up--maybe you could use it to instrument Drop of other types that don't have Drop but now you just made them not Copy for the sake of your metrics API which isn't counting them up in the first place...

I think it's safe to ignore this in other words. If not I'd love to know about an actual use case for Drop on ZSTs without any fields. I cannot think of one and I'm usually at least passable of thinking up weird scenarios.

Copy link
Member

@Nadrieril Nadrieril Jan 23, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

an innocuous at first glance change may lead to drop being disabled out of nowhere

Tbh I'm more surprised that the first case doesn't disable drop, but anyway that's how it is. I'm convinced. I think this is a good argument to include in the Rationale section.

Copy link
Member

@WaffleLapkin WaffleLapkin Jan 24, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@ahicks92

@programmerjake is illustrating that if you have two non-zst structs and use an irrefutable pattern to destructure, then you get a compile error;

In the code linked by @programmerjake I do not see any non-zsts, neither are there compilation errors.

but zsts can effectively be "moved". That is, we cannot reliably count "instances" of them. I did not actually know this myself.

I don't really get what you mean by that. All types can be moved. And you can reliably count the instances of a ZST, if it is not Copy (same as with non-zsts).

Also I just want to highlight that this compiles:

let v: S; // not initialized!
let S {} = v;

A struct pattern is equivalent to accessing the fields, and since S {} mentions no fields, it's always a noop (and requires initialization of all 0 fields it mentions).


# Motivation
[motivation]: #motivation

There are real use-cases in which you do want to move out of types that implement `Drop`. Two main types of patterns that often need it are:

- implementing methods like `into_raw_parts` which explicitly hand over their internals and move the burden of dropping those
- "guard" types which might need to be "disarmed"

In both cases, you want to avoid the regular `Drop` implementation from running at all and get ownership of the fields.

Right now, code that wants to do this needs to use `unsafe`. For example, [`std::io::BufWriter::into_parts()`](https://doc.rust-lang.org/stable/std/io/struct.BufWriter.html#method.into_parts) has to perform the following gymastics using `ManuallyDrop`:

```rust
pub fn into_parts(self) -> (W, Result<Vec<u8>, WriterPanicked>) {
let mut this = ManuallyDrop::new(self);
let buf = mem::take(&mut this.buf);
let buf = if !this.panicked { Ok(buf) } else { Err(WriterPanicked { buf }) };

// SAFETY: double-drops are prevented by putting `this` in a ManuallyDrop that is never dropped
let inner = unsafe { ptr::read(&this.inner) };

(inner, buf)
}
Comment on lines +24 to +33
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

A safer workaround is to use Option, Option::take, and decomposing a struct further into components, so that you can "hide" the pieces that need to run the Drop impl from a destructuring pattern. But "Option + decomposition just for the sake of workarounds" is also a really bad negative pattern if you didn't actually need to decompose anything or make the value, er, optional, since it makes the code open to incorrectness, as you may actually want to always have the value there and may want to always have certain values together... except in the code that wants to perform this trick.

I can't produce a good example right now, unfortunately, because I lost the logs of the conversation where I explained the (goofy) solution to this question in one case.

```

When writing this RFC, we even spent some time to make sure this example from `std` was even correct. By exposing `drop_type_destructuring`, we can reduce the complexity of such use cases.

Through this, we can avoid `Drop`ping types for which running `Drop` is very important. However, this is already the case because you could use `mem::forget`.

# Guide-level explanation
[guide-level-explanation]: #guide-level-explanation

`drop_type_destructuring` adds a new built-in macro, `destructure`. You can use this macro to destructure types that implement `Drop`. For example, assuming `Foo` is a type that implements `Drop` and has fields `a` and `b`, you could write the following:

```rust
fn example(x: Foo) {
destructure!(let Foo { a, b: _ } = x);
// yay we own `a` now, we can even drop it!
drop(a);
}
```

This moves the field `a` out of `x`, and immediately drops `b`.
Instead of creating a new binding, you can also assign to an existing one by leaving out the `let`, similarly to destructuring assignment:

```rust
fn example(x: Foo) {
let mut a = vec![1, 2, 3];

destructure!(Foo { a, b: _ } = x);
// yay we own `a` now, we can even drop it!
drop(a);
}
```

When a type implements `Drop`, this can be because the order in which fields are dropped is very important. It could be unsound to do it in the wrong order. When you write a destructuring, soundly dropping the fields is now your responsibility.

This can be a heavy burden, so if you are the author of a module or crate, you might want to limit other people destructuring your type. The rule for this is that you can only use `destructure!()` on types in a location where you could also construct that type. This means that in that location, all fields must be visible.

Importantly, this means that this code is not valid:

```rust
mod foo {
pub struct Bar {
pub a: Vec<u8>,
b: Vec<u8>,
}
}

fn example(x: foo::Bar) {
destructure!(let Foo{ a, .. } = x)
}
```

By using `..`, we could ignore the fact that `b` was not visible and move out `a` anyway. This is undesirable, as it would implicitly run `drop(b)` even though `b` was not accessible here. In fact, if there was a requirement that `a` was dropped before `b`, it would never be sound to destructure a type `Bar` in a location where `b` is not accessible.

Finally, you might ask why we need a macro for this at all. This is because for some types, the behavior is slightly different.
Let's use the following example. In Rust right now, this does not compile:

```rust
struct Foo {
a: Vec<u64>
}

impl Drop for Foo {
fn drop(&mut self) {}
}

fn example(x: Foo) {
// error: cannot move out of type `Foo`, which implements the `Drop` trait
let Foo { a } = x;
}
```

This is because we move the field `a` out, and `Foo` implements `Drop`.
The following does compile:

```rust
struct Foo {
a: Vec<u64>
}

impl Drop for Foo {
fn drop(&mut self) {}
}

fn example(x: Foo) {
// fine, because we don't actually move any fields out
let Foo { .. } = x;
}
```

Similarly, the following also works because we don't have to move out any fields, we can just copy `a`, because `u64` implements `Copy`.

```rust
struct Foo {
a: u64
}

impl Drop for Foo {
fn drop(&mut self) {}
}

fn example(x: Foo) {
// Totally fine, we just copy a
let Foo { a } = x;
}
```

In the two examples above, `Drop` still runs for `x`.
`destructure!()` represents a different operation. When we use it, we actually move the fields out of the type, and prevent `Drop` from running.

```rust
struct Foo {
a: u64
}

impl Drop for Foo {
fn drop(&mut self) {}
}

fn example(x: Foo) {
// `a` is moved out of `x`, Drop never runs for `x`
destructure!(let Foo { a } = x);
}
Comment on lines +152 to +155
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I've definitely written code that would have been much-improved... or at least written faster... by this being available! Thank you!

```

Preventing `Drop` from running is not inherently problematic in Rust. You can already do this for any type through `mem::forget` (and other means), which is safe.


Finally, the following is an example of how one could implement the example from [Motivation](#motivation):

```rust
pub fn into_parts(self) -> (W, Result<Vec<u8>, WriterPanicked>) {
let mut this = ManuallyDrop::new(self);
let buf = mem::take(&mut this.buf);
let buf = if !this.panicked { Ok(buf) } else { Err(WriterPanicked { buf }) };

// SAFETY: double-drops are prevented by putting `this` in a ManuallyDrop that is never dropped
let inner = unsafe { ptr::read(&this.inner) };

(inner, buf)
}
```

Now with `destructure!`:

```rust
pub fn into_parts(self) -> (W, Result<Vec<u8>, WriterPanicked>) {
destructure!(let Self { buf, inner, panicked } = self);
let buf = if !panicked { Ok(buf) } else { Err(WriterPanicked { buf }) };

(inner, buf)
}
```

We don't even need unsafe anymore :tada:

# Reference-level explanation
[reference-level-explanation]: #reference-level-explanation
Comment on lines +189 to +190
Copy link
Member

@Nadrieril Nadrieril Dec 8, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Something important is missing: this is adding a new capability to safe code. Unsafe code writers could until today rely on the fact that safe code cannot take ownership of the value of fields in a struct with a Drop impl. As such, this RFC is technically a breaking change with soundness implications.

It's probably a rare case that a struct would have important drop-related invariants and be publicly constructible, so this may be acceptable, but this should be noted nonetheless.

Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I believe reliance on &mut _ not permitting ownership stealing is unsound in presence of take_mut, which has proven to be far too useful and natural to disallow.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yea, I'm not sure what this allows in addition to what take_mut already allows (which I guess technically isn't blessed, but I would not assume it's cursed either).

In addition to the fact that if all your fields are public, doing dangerous unsafe in Drop sounds extremely sketchy. In basically all scenarios Drop types have all fields private.

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

You can only take_mut() if you’re able to produce a replacement value. If there’s no way to get a replacement value for the fields, I think you can’t “steal” a field using take_mut(). E.g.

// crate foo {
pub struct Singleton(());

impl Singleton {
    pub fn claim() -> Self {
        static ONCE: AtomicBool = AtomicBool::new(true);
        if ONCE.swap(false, Relaxed) { Self(()) }
        else { panic!() }
    }
}

pub struct Complex {
    pub field: Singleton,
}

impl Complex {
    pub fn claim() -> Self {
        Self { field: Singleton::claim() }
    }
}

impl Drop for Complex { ... }

// crate bar
let mut complex = Complex::claim();
// not possible with `take_mut`:
// no way to provide replacement value
destruct!(Complex { field } = complex);

I’m not sure how important that is though.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

You can though!

take_mut(&mut complex.field, |f: Singleton | {
    // temporary ownership
    identity(f)
});

take_mut(&mut complex.field, |f: Singleton | {
    drop(f);
    panic!() // terminate the program
});

So at most this breaks something like

if a Drop type T with all public fields is passed to a closure, one of the following happens:

  • T::drop is called and all its fields are dropped (modulo panic in drop)
  • T is forgotten and none of its fields are dropped
  • T's fields are dropped (/owned) and the program terminates before reaching the end of the closure

And while I assume it's possible to craft a program that requires exactly this assumption, I don't think it's reasonable.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yea, I'm not sure what this allows in addition to what take_mut already allows

Dropping the fields in a different order I think, though maybe you can do that with nested take_muts?

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Nested take_mut do allow you to drop fields in whichever order you'd like.


A built-in macro with the following "signature" is added to the standard library:

```rust
#[macro_export]
macro_rules! destructure {
((let)? $pat:pat = $e:expr) => { /* built-in */ }
}
```

The result of the expansion checks the following requirements and produces a compilation error if any of them are not met:
- `$pat` must be an irrefutable pattern
- `$e` must be an owned expression that can be matched by pattern `$pat`
- `$e`'s type must be an ADT (`struct` or `enum`)
- The type of `$e` must be constructable in the current context, i.e.
- All fields must be visible to the current module wrt privacy
- If the type is marked with `#[non_exhaustive]`, it must be defined in the current crate

The semantic of this macro is equivalent to the following:
1. Wrapping `$e` in a `ManuallyDrop`
2. Copying **all** fields of `$e` (for enums all fields of the active variant)
3. If the `let` token is present in the macro invocation, fields that are listed in `$pat` are binded to their respective bindings or patterns. Otherwise, they are assigned similarly to destructuring assignment
4. If `..` is present in `$pat`, all fields that are not mentioned in it are dropped in the order of definition (simialrly to how [normal `let`/destructuring assignment works](https://play.rust-lang.org/?version=stable&mode=debug&edition=2021&gist=3abd3aebd3378690ff3d2006e12d4120))

# Drawbacks
[drawbacks]: #drawbacks

When implementing this change as a macro expanding to a statement, we do not think there are many drawbacks. However, in the coming section with alternatives we discuss another possible expansion to a pattern which may have a few advantages, but also has many drawbacks which we discuss there.

# Rationale and alternatives

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Another alternative that I haven't seen mentioned is having a macro that behaves like a match. Something like:

match_and_forget!(foo { Foo { .. } => something)

The main benefit, is that it would be straightforward to get that to work for enums:

match_and_forget!( thing {
    Variant1(x) => something(x),
    Variant2{ y, z } => something_else(y, z),
});

And it also would allow using it as an expression.

The downside is that it is a lot more verbose for simply binding fields to variables for a struct.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think this is basically just equivalent to the destructure! as a pattern, but more verbose for simple cases. It also suffers from similar inability to do nested matches.

Also:

The one scenario we could imagine is that might be useful for destructure!() on enums, but we also think that even using destructure!() on enums will be very uncommon (as types implementing Drop are almost always structs) and might not outweigh the increased implementation complexity and the fact that we need to communicate this one level matching rule to users.

[rationale-and-alternatives]: #rationale-and-alternatives
Comment on lines +220 to +221
Copy link
Member

@Nadrieril Nadrieril Dec 8, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

What about the really obvious alternative: "allow a plain let Struct { .. } = foo; without a macro". Specifically, we could allow this in exactly the cases where you propose the macro to work, i.e. when the type is constructible in the current context. I'd like to see why this alternative is not considered viable.

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think it is important to syntactically mark the fact that destructure!() is a different operation than this. We talk about this in the guide level explanation. For types implementing Copy, destructure!() moves anyway and doesn't run the original value's destructor at all

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Hah apologies, I skipped over the guide-level explanation because I asumed the technical details would be in the reference-level section.

Here's what I was missing, which imo should be highlighted in the motivation section: let Struct { <something> } = foo does not reliably prevent drop from running, because only a non-Copy binding would force a move out of foo. Hence I seem to understand that the problem statement of this RFC includes "find a reliable way to disable drop".

It seems like allowing let Struct { <something> } = foo is pretty orthogonal in fact: we could allow it and this RFC would still make sense. The argument against is that users might think this disables Drop when it doesn't always. I would really like to see that spelled out in the "Alternatives" section.

Copy link
Author

@jdonszelmann jdonszelmann Dec 8, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

fair enough, I'll talk to @WaffleLapkin (when it wakes up :P) and we'll see what modifications to make. I'll let you know when we have so you can take another look if you'd like :)

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Here's what I was missing, which imo should be highlighted in the motivation section: let Struct { <something> } = foo does not reliably prevent drop from running,

I do not understand the use of "reliably" here. This never disables foo's drop because it's one of these situations:

  • Struct implements Drop and <something> causes a move, which is currently error
  • <something> doesn't cause a move, so drops are not affected at all
  • <something> causes a move, but Struct does not implement Drop, which means that only the drop for the moved field is disabled (other fields still dropped)...

Hence I seem to understand that the problem statement of this RFC includes "find a reliable way to disable drop".

Not really? The problem statement of this RFC is very specifically to allow moving fields out of Drop types. Disabling drop is a requirement for that -- you have to disable the drop, because it gets access to the fields and so can't work when a field is moved out.

The rest of the RFC is "how do we make this operation not confusing and least footguny". Simply allowing normal destructuring of types which implement Drop is a footgun, since the syntax doesn't clearly differentiate between disabling drop or copying fields out.

We might want to spell this out more explicitly though.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Simply allowing normal destructuring of types which implement Drop is a footgun, since the syntax doesn't clearly differentiate between disabling drop or copying fields out.

Yes, that's precisely the point I was trying to make! By "not reliable", I mean "allowing the obvious thing is footgunny because nothing would syntactically indicate whether you're moving out or not". Hence why I would like to see this spelled out, because that was not obvious to me on first read of the RFC.

Copy link
Contributor

@oli-obk oli-obk Dec 17, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

That could be remedied by just turning the hard error into a lint. Then you #[expect] the lint wherever you want the destructuring to happen and subsequently the drop getting avoided. Though that has the downside of adding a drop impl not causing existing destructuring sites to error in other crates

This comment was marked as duplicate.


## Do not support destructuring assignment

Instead of allowing optional `let` token, we could always ommit it, and consider `destructure!` to always introduce new bindings.

It's not evident that destructure assignment is particularly useful in situations where `destructure!` would be used.

## Expand to a pattern

Instead of making this macro expand to a statement, containing a pattern and a right hand side, one could imagine a macro that expands *just* to a pattern:

```rust
let destructure!(Foo { a, b }) = x;
```

On its own this could be considered, it also means that whether to use `let` or not (the previous alternative) isn't a concern for the macro. However, by doing this, we might imply that you can also use `destructure!` in other places you might see a pattern:

```rust
let destructure!(Foo::A { a }) = x else {
panic!("not Foo::A")
};

match x {
destructure!(Foo::A { a }) => /* ... */,
Foo::B { .. } => panic!("unexpected Foo::B")
}
```

Now, it makes sense that patterns in `destructure!` can be refutable. Even this can be okay, as long as only the top-level pattern is refutable. If we allow nested patterns to be refutable as well, we can construct the following scenario:

```rust
match y {
destructure!(Bar { a, b: 1 }) => /* ... */,
_ => panic!(".b != 1")
}
```

Here, if we were to move fields out of `Bar` from left to right, we might move `a`, then check whether `b` matches `1`, and if it doesn't we already moved `a` out.
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I’m not sure this disadvantage is real. Why does the move have to happen before fully matching? As far as I know, the existing semantics of patterns are compatible with the steps being:

  1. Test the scrutinee against the pattern.
  2. Execute match guard expressions (using temporary shared borrows).
  3. If the pattern and the guard succeed, perform the moves and borrows specified by the pattern.

Copy link
Member

@Nadrieril Nadrieril Dec 27, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is indeed how rust works today and while it's not yet guaranteed behavior I expect this is what we'll guarantee once we get around to it.

Copy link
Member

@Nadrieril Nadrieril Jan 23, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actually this is somewhat guaranteed, moving fields before all the checks are done is plain wrong: if one of the checks fails now you've moved out a value that you shouldn't have moved.


In theory this is not impossible to implement. By making this expand to code that uses `ManuallyDrop`, and when a pattern is rejected semantically move values back. However, one can construct even more complicated scenarios than just this one, and actually defining the semantics of this operations quickly becomes incredibly complicated.

Just as a small example of that, consider this:

```rust
match z {
destructure!(Baz { a, b: destructure!(Bar { a, b: 1 }) }) => /* ... */,
_ => panic!(".b.b != 1")
}
```

Where both `Baz` and `Bar` implement `Drop`.

As a final note on this, making `destructure!()` expand to a pattern on its own is possible, and even refutability is possible if we only allow "one level" of refutability (`Some(_)`, but not `Some(1)`). However, we simply do not think there are many good uses for this.
The one scenario we could imagine is that might be useful for `destructure!()` on `enum`s, but we also think that even using `destructure!()` on `enum`s will be very uncommon (as types implementing `Drop` are almost always `struct`s) and might not outweigh the increased implementation complexity and the fact that we need to communicate this one level matching rule to users.

## Macro expression and Magic type

Instead of having a macro be a statement or pattern, we could make it an expression returning a type (say `Destructure<T>`) with magic similar to that of a `Box` and `ManuallyDrop`. Specifically, it would:
1. Allow moving `T`'s fields out of `Destructure<T>` even if `T: Drop`
2. Not drop `T` when `Destructure<T>` is dropped
3. But drop all fields of `T` when `Destructure<T>` is dropped (except the moved-out ones)
4. Be able to be matched directly with a pattern that would match `T`

Note that having a macro produce `Destructure<T>`, as opposed to a function is still required, since we still want to check that `T` is constructable in the current context.

This would allow writing the `BufWriter::into_parts` example from [Motivation](#motivation) like this:

```rust
pub fn into_parts(self) -> (W, Result<Vec<u8>, WriterPanicked>) {
let Self { buf, inner, panicked } = destructure!(self);
let buf = if !panicked { Ok(buf) } else { Err(WriterPanicked { buf }) };

(inner, buf)
}
```

Or closer to the original code:

```rust
pub fn into_parts(self) -> (W, Result<Vec<u8>, WriterPanicked>) {
let this = destructure!(self);
let buf = this.buf;
let buf = if !this.panicked { Ok(buf) } else { Err(WriterPanicked { buf }) };

(this.inner, buf)
}
```

The upside of this is that `Destructure` type can be used as input into functions which are expected to not drop `T`, such as potential `DropOwned`:
Copy link
Contributor

@kpreid kpreid Dec 24, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

A future version of Rust that had &move references (a kind of pointer which transfers ownership of the value, but doesn't own the memory it points to) could simply use the statement or pattern destructure!:

impl DropOwned for X {
    fn drop(self: &move Self) {
        destructure!(let Self(inner) = *self);
        drop(inner);
    }
}

This would have the advantage, compared to Destructure<Self>, of not forcing a move of the X value itself (only the fields that the author chooses to move out of it), and thus doesn’t change or magically fudge the ABI — in particular, the above DropOwned::drop() does exactly what drop_in_place() does (drop the entire value), whether or not the author of the code does so explicitly.

This would also align neatly with “pinned drop” (executing drop glue on a value that is statically known to be pinned), by offering a Pin<&move Self> instead. At that point, we have a drop with all the properties missing from today's Drop::drop(): it can move fields out, and it is unable to accidentally break pinning.

Copy link
Member

@workingjubilee workingjubilee Dec 26, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

..."a kind of pointer which transfers ownership of the value, but doesn't own the memory it points to"? ...que? That feels like it is not quite the thing you are trying to say.

Copy link
Contributor

@kpreid kpreid Dec 26, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm not sure what error you intend to point out. My understanding of the &move idea (which I could not find an existing writeup of to link to, or I would have) is that:

  • like Box<T>, a T can be moved out of &move T,
  • like Box<T>, if the T is not moved out, then it is dropped when &move T is dropped,
  • unlike Box<T>, no heap allocation is freed (because the allocation might belong to e.g. a parent stack frame, or some other data structure).

Thus, &move T owns the value but not the allocation. Does that clarify?

…that said, I just realized that this is incompatible with Pin without also having unleakable/linear types (you mustn't deallocate a pinned value without moving it). So, my last paragraph was a bit incomplete.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Here's my go-to ~writeup on the topic.


```rust
struct X(Vec<u8>);

// This is explicitly *not* proposed as part of this RFC,
// but is a kind of pattern that could be allowed by `Destructure`.
impl DropOwned for X {
fn drop(self: Destructure<Self>) {
drop(self.0); // ownership!
}
}
```

The downside of this approach is that it is significantly harder to specify, implement, and teach.

# Prior art
[prior-art]: #prior-art

@WaffleLapkin wrote a crude polyfill for this using a macro and `ManuallyDrop`:

```rust
/// Destructures `$e` using a provided pattern.
///
/// Importantly, this works with types which implement `Drop` (ofc, this doesn't run the destructor).
#[macro_export]
macro_rules! destructure {
($Type:ident { $($f:tt $(: $rename:pat)? ),+ $(,)? } = $e:expr) => (
// FIXME: use $crate:: paths
let tmp = $crate::unstd::_macro_reexport::core::mem::ManuallyDrop::new($e);

// assert that `$e` is an owned expression, rather than `&Type`
if false {
#[allow(unreachable_code)]
let _assert_owned_expr = [&tmp, &$crate::unstd::_macro_reexport::core::mem::ManuallyDrop::new($Type { $($f: todo!()),* })];
};

$(
let $crate::destructure!(@_internal_pat_helper $f $($rename)?)
// safety: `$e` is of type `$Type<..>` (as asserted above),
// so we have ownership over it's fields.
// `$f` is a field of `$Type<..>` (as asserted above).
// `$e` is moved into a `ManuallyDrop`, which means its `drop` won't be run,
// so we can safely move out its
// the pointer is created from a reference and is thus valid
= unsafe { $crate::unstd::_macro_reexport::core::ptr::read(&tmp.$f) };
)+

// remove the temporary we don't need anymore.
// doesn't actually drop, since `ManuallyDrop`.
_ = {tmp};
);
(@_internal_pat_helper $f:tt) => ($f);
(@_internal_pat_helper $f:tt $rename:pat) => ($rename);
}
```
Example usage:
```rust
// `destructure` does not run the destructor, so this **doesn't** unlock the lock.
destructure!(Lock { file, mode: _ } = self);
```
This polyfill has multiple downsides, such as:
- It does not support ommiting fields with `..` (this is not possible to support in the polyfill, as you are unable to prove that an expression is owned, without specifying its full type)
- It doesn't work with enums with multiple variants, even if you could specify an irrefutable pattern (this theoretically can be supported, but is quite complicated)
- It does not work with tuple struct syntax (impossible to support arbitrary number of fields, hard to support a set number)

# Unresolved questions
[unresolved-questions]: #unresolved-questions

These mainly relate to the above-mentioned alternatives:

- Do we support `let` inside?
- Should `destructure!()` expand to a statement, or a pattern?

# Future possibilities
[future-possibilities]: #future-possibilities

If we were to choose for the version that expands to a pattern, we could try to define the semantics of refutable patterns with more than one level, assuming this is even possible in a satisfactory and importantly sound manner.

In general, we could add some lints in relation to this feature.
General usage of `destructure!` is often okay, especially in places where there is visibility of all fields, but it could be used in unintentional ways so a lint in `clippy::pedantic` for using `destructure` might be considered.
A lint for using `destructure!` on types that don't implement `Drop` at all makes sense regardless. The suggestion there can be to remove `destructure!` completely as it is not necessary.