Skip to content

Allow shortening lifetime in CoerceUnsized for &mut#149219

Merged
rust-bors[bot] merged 2 commits into
rust-lang:mainfrom
theemathas:coerce-unsized-shorten
Jun 17, 2026
Merged

Allow shortening lifetime in CoerceUnsized for &mut#149219
rust-bors[bot] merged 2 commits into
rust-lang:mainfrom
theemathas:coerce-unsized-shorten

Conversation

@theemathas

@theemathas theemathas commented Nov 22, 2025

Copy link
Copy Markdown
Contributor

View all comments

This modifies the &mut -> &mut CoerceUnsized impl so that, it can shorten the lifetime.

Note that there are already two impls that allow shortening the lifetime like this (the &mut T -> &U and the &T -> &U impls). So this change makes the impls consistent with each other.

I initially tried to also do the same to the CoerceUnsized impl for core::cell::{Ref, RefMut}. However, this can't be done because Ref and RefMut "store" the lifetime and the data in different fields, and CoerceUnsized can only coerce one field.

This change has an effect on stable code, since it allows shortening lifetimes inside invariant types via a &mut -> &mut unsize coercion.

@theemathas theemathas added T-libs-api Relevant to the library API team, which will review and decide on the PR/issue. A-coercions Area: implicit and explicit `expr as Type` coercions F-coerce_unsized The `CoerceUnsized` trait T-types Relevant to the types team, which will review and decide on the PR/issue. labels Nov 22, 2025
@rustbot rustbot added the S-waiting-on-review Status: Awaiting review from the assignee but also interested parties. label Nov 22, 2025
@rustbot

rustbot commented Nov 22, 2025

Copy link
Copy Markdown
Collaborator

r? @scottmcm

rustbot has assigned @scottmcm.
They will have a look at your PR within the next two weeks and either review your PR or reassign to another reviewer.

Use r? to explicitly pick a reviewer

@rust-log-analyzer

This comment has been minimized.

@theemathas

This comment was marked as resolved.

@zachs18

This comment was marked as resolved.

@theemathas theemathas force-pushed the coerce-unsized-shorten branch from 6bfef63 to cfcd9bf Compare November 28, 2025 14:25
@theemathas theemathas changed the title Allow shortening lifetimes in CoerceUnsized impls Allow shortening lifetime in CoerceUnsized for &mut Nov 28, 2025
@theemathas

Copy link
Copy Markdown
Contributor Author

I changed the PR to just edit the impl for &mut, and leave the impls for Ref and RefMut alone. I don't know if this is the right way to go though...

@scottmcm scottmcm added the I-types-nominated Nominated for discussion during a types team meeting. label Dec 6, 2025
@scottmcm

scottmcm commented Dec 6, 2025

Copy link
Copy Markdown
Member

TBH, I feel quite unqualified to review the soundness of this. Maybe someone from types feels more confident?

@scottmcm

Copy link
Copy Markdown
Member

r? types

@rustbot rustbot assigned jackh726 and unassigned scottmcm Jan 27, 2026
@jackh726

jackh726 commented Feb 8, 2026

Copy link
Copy Markdown
Member

So, there should be some test that we can add to observe this behavior change (stable or unstable). Without it, we definitely shouldn't be making it.

I can think more about this and try to come up with such a test later this week, but @theemathas perhaps you can try before I get to it, since you're proposing the change there must be some reason.

@theemathas

This comment has been minimized.

@jackh726

jackh726 commented Feb 8, 2026

Copy link
Copy Markdown
Member

One thing that you could try is to remove the lifetime shortening on the other impl(s) and see if you can find tests that require that - and use that as a starting point for a similar test.

@theemathas

This comment has been minimized.

@theemathas

theemathas commented Feb 8, 2026

Copy link
Copy Markdown
Contributor Author

I just realized: This change actually does have a visible effect in stable rust.

Consider the following code:

use std::cell::Cell;

struct Thing;
trait Trait {}
impl Trait for Thing {}

fn works<'a: 'b, 'b>(x: Cell<&'a Thing>) -> Cell<&'b dyn Trait> {
    x
}

fn fails<'a: 'b, 'b>(x: Cell<&'a mut Thing>) -> Cell<&'b mut dyn Trait> {
    x
}

Currently, the works function compiles, and the fails function does not compile.

error: lifetime may not live long enough
  --> src/lib.rs:12:5
   |
11 | fn fails<'a: 'b, 'b>(x: Cell<&'a mut Thing>) -> Cell<&'b mut dyn Trait> {
   |          --      -- lifetime `'b` defined here
   |          |
   |          lifetime `'a` defined here
12 |     x
   |     ^ function was supposed to return data with lifetime `'a` but it is returning data with lifetime `'b`
   |
   = help: consider adding the following bound: `'b: 'a`
   = note: requirement occurs because of the type `Cell<&mut dyn Trait>`, which makes the generic argument `&mut dyn Trait` invariant
   = note: the struct `Cell<T>` is invariant over the parameter `T`
   = help: see <https://doc.rust-lang.org/nomicon/subtyping.html> for more information about variance

This PR makes it so that both functions compile instead.

I am now unsure on what the best way forward is.

@theemathas theemathas added the needs-fcp This change is insta-stable, or significant enough to need a team FCP to proceed. label Feb 8, 2026
@theemathas

Copy link
Copy Markdown
Contributor Author

The inconsistent lifetimes in the impls were there since ancient times 843db01#diff-3da87c1e923c79167e692c9c84af74599a13c74a83e797b131aff23470a47411R1223-R1233

@theemathas

Copy link
Copy Markdown
Contributor Author

Even "no-op" unsize-coercions have strange behavior:

use std::cell::Cell;
struct Thing;
trait Trait {}

// currently errors
fn with_thing<'a: 'b, 'b>(x: Cell<&'a Thing>) -> Cell<&'b Thing> {
    x
}

// currently compiles
fn with_trait<'a: 'b, 'b>(x: Cell<&'a dyn Trait>) -> Cell<&'b dyn Trait> {
    x
}

// currently errors, will compile with this PR
fn with_trait_mut<'a: 'b, 'b>(x: Cell<&'a mut dyn Trait>) -> Cell<&'b mut dyn Trait> {
    x
}

@jackh726

Copy link
Copy Markdown
Member

I just realized: This change actually does have a visible effect in stable rust.

Please add this as a test here.

@jackh726

jackh726 commented Apr 10, 2026

Copy link
Copy Markdown
Member

Okay, discussion on Zulip hasn't raised any issues here. Going to fcp merge for types here. I'm going to ping @rust-lang/libs-api, but I think this is probably "just okay" to have as a types team PR. This isn't really a new "API surface" as much as a change to what coercions we accept.


This impl change allows (as an example) the following to compile:

// currently compiles
fn with_trait<'a: 'b, 'b>(x: Cell<&'a dyn Trait>) -> Cell<&'b dyn Trait> {
    x
}

// currently errors, will compile with this PR
fn with_trait_mut<'a: 'b, 'b>(x: Cell<&'a mut dyn Trait>) -> Cell<&'b mut dyn Trait> {
    x
}

@rfcbot merge types

@rust-rfcbot

rust-rfcbot commented Apr 10, 2026

Copy link
Copy Markdown
Collaborator

Team member @jackh726 has proposed to merge this. The next step is review by the rest of the tagged team members:

No concerns currently listed.

Once a majority of reviewers approve (and at most 2 approvals are outstanding), this will enter its final comment period. If you spot a major issue that hasn't been raised at any point in this process, please speak up!

See this document for info about what commands tagged team members can give me.

@rust-rfcbot rust-rfcbot added the proposed-final-comment-period Proposed to merge/close by relevant subteam, see T-<team> label. Will enter FCP once signed off. label Apr 10, 2026
@rustbot rustbot added S-waiting-on-author Status: This is awaiting some action (such as code changes or more information) from the author. and removed S-waiting-on-review Status: Awaiting review from the assignee but also interested parties. labels May 23, 2026
@rustbot

rustbot commented May 23, 2026

Copy link
Copy Markdown
Collaborator

Reminder, once the PR becomes ready for a review, use @rustbot ready.

This modifies the &mut -> &mut CoerceUnsized impl so that, it can
shorten the lifetime.

Note that there are already two impls that allow shortening the lifetime
like this (the &mut T -> &U and the &T -> &U impls). So this change
makes the impls consistent with each other.

I initially tried to also do the same to the CoerceUnsized impl for
core::cell::{Ref, RefMut}. However, this can't be done because
Ref and RefMut "store" the lifetime and the data in different fields,
and CoerceUnsized can only coerce one field.

This change has an effect on stable code, since it allows shortening
lifetimes inside invariant types via a &mut -> &mut unsize coercion.
@theemathas theemathas force-pushed the coerce-unsized-shorten branch from f4e5661 to 94c4323 Compare May 23, 2026 14:07
@theemathas

Copy link
Copy Markdown
Contributor Author

@rustbot ready

@rustbot rustbot added S-waiting-on-review Status: Awaiting review from the assignee but also interested parties. and removed S-waiting-on-author Status: This is awaiting some action (such as code changes or more information) from the author. labels May 23, 2026
@theemathas

Copy link
Copy Markdown
Contributor Author

@jackh726 This PR is still waiting for approval.

@theemathas

Copy link
Copy Markdown
Contributor Author

@rustbot reroll

@rustbot rustbot assigned camelid and unassigned jackh726 Jun 16, 2026
@theemathas theemathas dismissed jackh726’s stale review June 16, 2026 16:09

I've rewritten the test.

@camelid

camelid commented Jun 16, 2026

Copy link
Copy Markdown
Member

r? @jackh726

Re-assigning Jack since he has context for this and I do not.

@rustbot rustbot assigned jackh726 and unassigned camelid Jun 16, 2026
@jackh726

Copy link
Copy Markdown
Member

@bors r+ rollup

@rust-bors

rust-bors Bot commented Jun 17, 2026

Copy link
Copy Markdown
Contributor

📋 This PR cannot be approved because it currently has the following label: needs-fcp.

@theemathas theemathas removed the needs-fcp This change is insta-stable, or significant enough to need a team FCP to proceed. label Jun 17, 2026
@theemathas

Copy link
Copy Markdown
Contributor Author

@bors r=jackh726 rollup

@rust-bors

rust-bors Bot commented Jun 17, 2026

Copy link
Copy Markdown
Contributor

📌 Commit 94c4323 has been approved by jackh726

It is now in the queue for this repository.

@rust-bors rust-bors Bot added S-waiting-on-bors Status: Waiting on bors to run and complete tests. Bors will change the label on completion. and removed S-waiting-on-review Status: Awaiting review from the assignee but also interested parties. labels Jun 17, 2026
rust-bors Bot pushed a commit that referenced this pull request Jun 17, 2026
Rollup of 7 pull requests

Successful merges:

 - #157681 (Add instrument_fn attribute)
 - #149219 (Allow shortening lifetime in CoerceUnsized for &mut)
 - #157539 (Rename `RandomSource` -> `Rng`, `DefaultRandomSource` -> `SystemRng`)
 - #157980 (Some minor cleanups around hir ty/pat/expr)
 - #157988 (Fix incremental-finalize-fail proc macro test on AIX)
 - #157989 (run-make: handle AIX symbol cdylib export test)
 - #157998 (Add big disclaimer to the description of lint `explicit_outlives_requirements`)
@rust-bors rust-bors Bot merged commit 59f27a4 into rust-lang:main Jun 17, 2026
11 checks passed
@rustbot rustbot added this to the 1.98.0 milestone Jun 17, 2026
rust-timer added a commit that referenced this pull request Jun 17, 2026
Rollup merge of #149219 - theemathas:coerce-unsized-shorten, r=jackh726

Allow shortening lifetime in CoerceUnsized for &mut

This modifies the &mut -> &mut CoerceUnsized impl so that, it can shorten the lifetime.

Note that there are already two impls that allow shortening the lifetime like this (the &mut T -> &U and the &T -> &U impls). So this change makes the impls consistent with each other.

I initially tried to also do the same to the CoerceUnsized impl for core::cell::{Ref, RefMut}. However, this can't be done because Ref and RefMut "store" the lifetime and the data in different fields, and CoerceUnsized can only coerce one field.

This change has an effect on stable code, since it allows shortening lifetimes inside invariant types via a &mut -> &mut unsize coercion.
@theemathas theemathas deleted the coerce-unsized-shorten branch June 17, 2026 07:14
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment

Labels

A-coercions Area: implicit and explicit `expr as Type` coercions disposition-merge This issue / PR is in PFCP or FCP with a disposition to merge it. F-coerce_unsized The `CoerceUnsized` trait finished-final-comment-period The final comment period is finished for this PR / Issue. S-waiting-on-bors Status: Waiting on bors to run and complete tests. Bors will change the label on completion. T-libs-api Relevant to the library API team, which will review and decide on the PR/issue. T-types Relevant to the types team, which will review and decide on the PR/issue. to-announce Announce this issue on triage meeting

Projects

None yet

Development

Successfully merging this pull request may close these issues.

10 participants