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

arbitrary_self_type: insert implied Receiver bound on Deref #138952

Open
wants to merge 1 commit into
base: master
Choose a base branch
from

Conversation

dingxiangfei2009
Copy link
Contributor

r? @nikomatsakis

This is an experimental setup which allows us to assess the impact from inserting an implied supertrait bound Deref: Receiver.

The observations are as follows.

  • Thanks to the trait solver, not too many hacks are needed to implement the idea behind this comment, where inductive cycles are admissible.
  • We want to allow users to continue use the dyn Deref type as Deref has been dyn-compatible. Following the current dyn-compatibility rule, it would have been impossible to use dyn Deref<Target = ..> because one would need to write dyn Deref<..> + Receiver<..> while neither Deref nor Receiver is an auto trait. This is the main change that this PR proposes: we fill in the missing projection bound that Receiver demands from a dyn Deref and this will remain the only exception.
  • Finally, the type (pretty-)printing is adjusted so as to not present the frivolous Receiver::Target = .. bound. It will always be the same term as Deref::Target.

I am proposing a crater experiment to see the extent of impact on the existing ecosystem and explore corner cases that I have missed.

@rustbot rustbot added S-waiting-on-review Status: Awaiting review from the assignee but also interested parties. T-compiler Relevant to the compiler team, which will review and decide on the PR/issue. T-libs Relevant to the library team, which will review and decide on the PR/issue. labels Mar 25, 2025
@rustbot
Copy link
Collaborator

rustbot commented Mar 25, 2025

HIR ty lowering was modified

cc @fmease

@dingxiangfei2009
Copy link
Contributor Author

@rustbot labels +F-arbitrary_self_types

@rustbot rustbot added the F-arbitrary_self_types `#![feature(arbitrary_self_types)]` label Mar 25, 2025
@dingxiangfei2009 dingxiangfei2009 changed the title arbitrary_self_type: insert implied bound on Deref arbitrary_self_type: insert implied Receiver bound on Deref Mar 25, 2025
@rust-log-analyzer
Copy link
Collaborator

The job x86_64-gnu-tools failed! Check out the build log: (web) (plain)

Click to see the possible cause of the failure (guessed by this bot)
tests/ui/ref_option/ref_option.all.fixed ... ok
tests/ui/non_std_lazy_static/non_std_lazy_static_fixable.fixed ... ok

FAILED TEST: tests/ui/unnecessary_to_owned.rs
command: CLIPPY_CONF_DIR="tests" RUSTC_ICE="0" "/checkout/obj/build/x86_64-unknown-linux-gnu/stage1-tools/x86_64-unknown-linux-gnu/release/clippy-driver" "--error-format=json" "--emit=metadata" "-Aunused" "-Ainternal_features" "-Zui-testing" "-Zdeduplicate-diagnostics=no" "-Dwarnings" "-Ldependency=/checkout/obj/build/x86_64-unknown-linux-gnu/stage1-tools/x86_64-unknown-linux-gnu/release/deps" "--extern=clippy_config=/checkout/obj/build/x86_64-unknown-linux-gnu/stage1-tools/x86_64-unknown-linux-gnu/release/deps/libclippy_config-37899240293f779f.rlib" "--extern=clippy_lints=/checkout/obj/build/x86_64-unknown-linux-gnu/stage1-tools/x86_64-unknown-linux-gnu/release/deps/libclippy_lints-3be15972e23ff692.rlib" "--extern=clippy_utils=/checkout/obj/build/x86_64-unknown-linux-gnu/stage1-tools/x86_64-unknown-linux-gnu/release/deps/libclippy_utils-dc5e0558d6401eca.rlib" "--extern=futures=/checkout/obj/build/x86_64-unknown-linux-gnu/stage1-tools/x86_64-unknown-linux-gnu/release/deps/libfutures-40b624a0d8607be6.rlib" "--extern=if_chain=/checkout/obj/build/x86_64-unknown-linux-gnu/stage1-tools/x86_64-unknown-linux-gnu/release/deps/libif_chain-9ed8a3357ffcec70.rlib" "--extern=itertools=/checkout/obj/build/x86_64-unknown-linux-gnu/stage1-tools/x86_64-unknown-linux-gnu/release/deps/libitertools-7f8b2af5daa98f7b.rlib" "--extern=parking_lot=/checkout/obj/build/x86_64-unknown-linux-gnu/stage1-tools/x86_64-unknown-linux-gnu/release/deps/libparking_lot-a444b065634e4b61.rlib" "--extern=quote=/checkout/obj/build/x86_64-unknown-linux-gnu/stage1-tools/x86_64-unknown-linux-gnu/release/deps/libquote-9dc8d553ca6fe129.rlib" "--extern=regex=/checkout/obj/build/x86_64-unknown-linux-gnu/stage1-tools/x86_64-unknown-linux-gnu/release/deps/libregex-7c4e26a9b860012c.rlib" "--extern=serde=/checkout/obj/build/x86_64-unknown-linux-gnu/stage1-tools/x86_64-unknown-linux-gnu/release/deps/libserde-520c5bcf4877870c.rlib" "--extern=serde_derive=/checkout/obj/build/x86_64-unknown-linux-gnu/stage1/lib/rustlib/x86_64-unknown-linux-gnu/lib/libserde_derive-7bfb9b74de07a025.so" "--extern=syn=/checkout/obj/build/x86_64-unknown-linux-gnu/stage1-tools/x86_64-unknown-linux-gnu/release/deps/libsyn-a2dd6e67e3656997.rlib" "--extern=tokio=/checkout/obj/build/x86_64-unknown-linux-gnu/stage1-tools/x86_64-unknown-linux-gnu/release/deps/libtokio-c39f2d6297d35d2d.rlib" "-Ldependency=/checkout/obj/build/x86_64-unknown-linux-gnu/stage1-tools/release/deps" "--out-dir" "/checkout/obj/build/x86_64-unknown-linux-gnu/stage1-tools/ui_test/0/tests/ui" "tests/ui/unnecessary_to_owned.rs" "--edition" "2021"

error: actual output differed from expected
Execute `cargo uibless` to update `tests/ui/unnecessary_to_owned.stderr` to the actual output
--- tests/ui/unnecessary_to_owned.stderr
+++ <stderr output>
 error: redundant clone
   --> tests/ui/unnecessary_to_owned.rs:217:64
... 170 lines skipped ...
 
 error: unnecessary use of `to_owned`
-  --> tests/ui/unnecessary_to_owned.rs:109:25
-   |
-LL |     require_deref_c_str(c_str.to_owned());
-   |                         ^^^^^^^^^^^^^^^^ help: use: `c_str`
-
-error: unnecessary use of `to_owned`
-  --> tests/ui/unnecessary_to_owned.rs:111:26
-   |
-LL |     require_deref_os_str(os_str.to_owned());
---
-
-error: unnecessary use of `to_owned`
-  --> tests/ui/unnecessary_to_owned.rs:120:30
-   |
-LL |     require_impl_deref_c_str(c_str.to_owned());
-   |                              ^^^^^^^^^^^^^^^^ help: use: `c_str`
-
-error: unnecessary use of `to_owned`
-  --> tests/ui/unnecessary_to_owned.rs:122:31
-   |
-LL |     require_impl_deref_os_str(os_str.to_owned());
---
-error: unnecessary use of `to_owned`
   --> tests/ui/unnecessary_to_owned.rs:138:26
    |
... 290 lines skipped ...
    |              ^^^^^^^^^^^^^^^^^^ help: replace it with: `(&["b"]).as_slice()`
 
-error: aborting due to 82 previous errors
+error: aborting due to 68 previous errors
 

Full unnormalized output:
error: redundant clone
##[error]  --> tests/ui/unnecessary_to_owned.rs:217:64
   |
LL |     require_c_str(&CString::from_vec_with_nul(vec![0]).unwrap().to_owned());
   |                                                                ^^^^^^^^^^^ help: remove this
   |
note: this value is dropped without further use
  --> tests/ui/unnecessary_to_owned.rs:217:20
   |
LL |     require_c_str(&CString::from_vec_with_nul(vec![0]).unwrap().to_owned());
   |                    ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
   = note: `-D clippy::redundant-clone` implied by `-D warnings`
   = help: to override `-D warnings` add `#[allow(clippy::redundant_clone)]`

error: redundant clone
##[error]  --> tests/ui/unnecessary_to_owned.rs:219:40
   |
LL |     require_os_str(&OsString::from("x").to_os_string());
   |                                        ^^^^^^^^^^^^^^^ help: remove this
   |
note: this value is dropped without further use
  --> tests/ui/unnecessary_to_owned.rs:219:21
   |
LL |     require_os_str(&OsString::from("x").to_os_string());
   |                     ^^^^^^^^^^^^^^^^^^^

error: redundant clone
##[error]  --> tests/ui/unnecessary_to_owned.rs:221:48
   |
LL |     require_path(&std::path::PathBuf::from("x").to_path_buf());
   |                                                ^^^^^^^^^^^^^^ help: remove this
   |
note: this value is dropped without further use
  --> tests/ui/unnecessary_to_owned.rs:221:19
   |
LL |     require_path(&std::path::PathBuf::from("x").to_path_buf());
   |                   ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^

error: redundant clone
##[error]  --> tests/ui/unnecessary_to_owned.rs:223:35
   |
LL |     require_str(&String::from("x").to_string());
   |                                   ^^^^^^^^^^^^ help: remove this
   |
note: this value is dropped without further use
  --> tests/ui/unnecessary_to_owned.rs:223:18
   |
LL |     require_str(&String::from("x").to_string());
   |                  ^^^^^^^^^^^^^^^^^

error: redundant clone
##[error]  --> tests/ui/unnecessary_to_owned.rs:225:39
   |
LL |     require_slice(&[String::from("x")].to_owned());
   |                                       ^^^^^^^^^^^ help: remove this
   |
note: this value is dropped without further use
  --> tests/ui/unnecessary_to_owned.rs:225:20
   |
LL |     require_slice(&[String::from("x")].to_owned());
   |                    ^^^^^^^^^^^^^^^^^^^

error: unnecessary use of `into_owned`
##[error]  --> tests/ui/unnecessary_to_owned.rs:65:36
   |
LL |     require_c_str(&Cow::from(c_str).into_owned());
   |                                    ^^^^^^^^^^^^^ help: remove this
   |
   = note: `-D clippy::unnecessary-to-owned` implied by `-D warnings`
   = help: to override `-D warnings` add `#[allow(clippy::unnecessary_to_owned)]`

error: unnecessary use of `to_owned`
##[error]  --> tests/ui/unnecessary_to_owned.rs:67:19
   |
LL |     require_c_str(&c_str.to_owned());
   |                   ^^^^^^^^^^^^^^^^^ help: use: `c_str`

error: unnecessary use of `to_os_string`
##[error]  --> tests/ui/unnecessary_to_owned.rs:70:20
   |
LL |     require_os_str(&os_str.to_os_string());
   |                    ^^^^^^^^^^^^^^^^^^^^^^ help: use: `os_str`

error: unnecessary use of `into_owned`
##[error]  --> tests/ui/unnecessary_to_owned.rs:72:38
   |
LL |     require_os_str(&Cow::from(os_str).into_owned());
   |                                      ^^^^^^^^^^^^^ help: remove this

error: unnecessary use of `to_owned`
##[error]  --> tests/ui/unnecessary_to_owned.rs:74:20
   |
---

error: unnecessary use of `into_owned`
##[error]  --> tests/ui/unnecessary_to_owned.rs:79:34
   |
LL |     require_path(&Cow::from(path).into_owned());
   |                                  ^^^^^^^^^^^^^ help: remove this

error: unnecessary use of `to_owned`
##[error]  --> tests/ui/unnecessary_to_owned.rs:81:18
   |
---

error: unnecessary use of `into_owned`
##[error]  --> tests/ui/unnecessary_to_owned.rs:86:30
   |
LL |     require_str(&Cow::from(s).into_owned());
   |                              ^^^^^^^^^^^^^ help: remove this

error: unnecessary use of `to_owned`
##[error]  --> tests/ui/unnecessary_to_owned.rs:88:17
   |
LL |     require_str(&s.to_owned());
   |                 ^^^^^^^^^^^^^ help: use: `s`

error: unnecessary use of `to_string`
##[error]  --> tests/ui/unnecessary_to_owned.rs:90:17
   |
LL |     require_str(&x_ref.to_string());
   |                 ^^^^^^^^^^^^^^^^^^ help: use: `x_ref.as_ref()`

error: unnecessary use of `to_vec`
##[error]  --> tests/ui/unnecessary_to_owned.rs:93:19
   |
LL |     require_slice(&slice.to_vec());
   |                   ^^^^^^^^^^^^^^^ help: use: `slice`

error: unnecessary use of `into_owned`
##[error]  --> tests/ui/unnecessary_to_owned.rs:95:36
   |
LL |     require_slice(&Cow::from(slice).into_owned());
   |                                    ^^^^^^^^^^^^^ help: remove this

error: unnecessary use of `to_owned`
##[error]  --> tests/ui/unnecessary_to_owned.rs:97:19
   |
LL |     require_slice(&array.to_owned());
   |                   ^^^^^^^^^^^^^^^^^ help: use: `array.as_ref()`

error: unnecessary use of `to_owned`
##[error]  --> tests/ui/unnecessary_to_owned.rs:99:19
   |
LL |     require_slice(&array_ref.to_owned());
   |                   ^^^^^^^^^^^^^^^^^^^^^ help: use: `array_ref.as_ref()`

error: unnecessary use of `to_owned`
##[error]  --> tests/ui/unnecessary_to_owned.rs:101:19
   |
LL |     require_slice(&slice.to_owned());
   |                   ^^^^^^^^^^^^^^^^^ help: use: `slice`

error: unnecessary use of `into_owned`
##[error]  --> tests/ui/unnecessary_to_owned.rs:105:42
   |
LL |     require_x(&Cow::<X>::Owned(x.clone()).into_owned());
   |                                          ^^^^^^^^^^^^^ help: remove this

error: unnecessary use of `to_owned`
##[error]  --> tests/ui/unnecessary_to_owned.rs:138:26
   |
LL |     require_as_ref_c_str(c_str.to_owned());
   |                          ^^^^^^^^^^^^^^^^ help: use: `c_str`

error: unnecessary use of `to_owned`
##[error]  --> tests/ui/unnecessary_to_owned.rs:140:27
   |
LL |     require_as_ref_os_str(os_str.to_owned());
---

error: unnecessary use of `to_owned`
##[error]  --> tests/ui/unnecessary_to_owned.rs:155:31
   |
LL |     require_impl_as_ref_c_str(c_str.to_owned());
   |                               ^^^^^^^^^^^^^^^^ help: use: `c_str`

error: unnecessary use of `to_owned`
##[error]  --> tests/ui/unnecessary_to_owned.rs:157:32
   |
LL |     require_impl_as_ref_os_str(os_str.to_owned());
---

error: unnecessary use of `to_owned`
##[error]  --> tests/ui/unnecessary_to_owned.rs:172:30
   |
LL |     require_as_ref_str_slice(s.to_owned(), array.to_owned());
   |                              ^^^^^^^^^^^^ help: use: `s`

error: unnecessary use of `to_owned`
##[error]  --> tests/ui/unnecessary_to_owned.rs:172:44
   |
LL |     require_as_ref_str_slice(s.to_owned(), array.to_owned());
   |                                            ^^^^^^^^^^^^^^^^ help: use: `array`

error: unnecessary use of `to_owned`
##[error]  --> tests/ui/unnecessary_to_owned.rs:175:30
   |
LL |     require_as_ref_str_slice(s.to_owned(), array_ref.to_owned());
   |                              ^^^^^^^^^^^^ help: use: `s`

error: unnecessary use of `to_owned`
##[error]  --> tests/ui/unnecessary_to_owned.rs:175:44
   |
LL |     require_as_ref_str_slice(s.to_owned(), array_ref.to_owned());
   |                                            ^^^^^^^^^^^^^^^^^^^^ help: use: `array_ref`

error: unnecessary use of `to_owned`
##[error]  --> tests/ui/unnecessary_to_owned.rs:178:30
   |
---

error: unnecessary use of `to_string`
##[error]  --> tests/ui/unnecessary_to_owned.rs:191:20
   |
LL |     let _ = x.join(&x_ref.to_string());
   |                    ^^^^^^^^^^^^^^^^^^ help: use: `x_ref`

error: unnecessary use of `to_vec`
##[error]  --> tests/ui/unnecessary_to_owned.rs:194:13
   |
LL |     let _ = slice.to_vec().into_iter();
---
   |
LL |     let _ = IntoIterator::into_iter(slice.to_owned());
   |             ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ help: use: `slice.iter().copied()`

error: allocating a new `String` only to create a temporary `&str` from it
##[error]  --> tests/ui/unnecessary_to_owned.rs:229:26
   |
LL |     let _ref_str: &str = &String::from_utf8(slice.to_vec()).expect("not UTF-8");
   |                          ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
   |
help: convert from `&[u8]` to `&str` directly
   |
LL -     let _ref_str: &str = &String::from_utf8(slice.to_vec()).expect("not UTF-8");
LL +     let _ref_str: &str = core::str::from_utf8(&slice).expect("not UTF-8");
   |

error: allocating a new `String` only to create a temporary `&str` from it
##[error]  --> tests/ui/unnecessary_to_owned.rs:231:26
   |
LL |     let _ref_str: &str = &String::from_utf8(b"foo".to_vec()).unwrap();
   |                          ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
   |
help: convert from `&[u8]` to `&str` directly
   |
LL -     let _ref_str: &str = &String::from_utf8(b"foo".to_vec()).unwrap();
LL +     let _ref_str: &str = core::str::from_utf8(b"foo").unwrap();
   |

error: allocating a new `String` only to create a temporary `&str` from it
##[error]  --> tests/ui/unnecessary_to_owned.rs:233:26
   |
LL |     let _ref_str: &str = &String::from_utf8(b"foo".as_slice().to_owned()).unwrap();
   |                          ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
   |
help: convert from `&[u8]` to `&str` directly
   |
LL -     let _ref_str: &str = &String::from_utf8(b"foo".as_slice().to_owned()).unwrap();
LL +     let _ref_str: &str = core::str::from_utf8(b"foo".as_slice()).unwrap();
   |

error: unnecessary use of `to_vec`
##[error]  --> tests/ui/unnecessary_to_owned.rs:291:14
   |
LL |     for t in file_types.to_vec() {
   |              ^^^^^^^^^^^^^^^^^^^
   |
help: remove any references to the binding
   |
LL ~     for t in file_types {
LL |
LL ~         let path = match get_file_path(t) {
   |

error: unnecessary use of `to_string`
##[error]  --> tests/ui/unnecessary_to_owned.rs:357:24
   |
LL |         Box::new(build(y.to_string()))
   |                        ^^^^^^^^^^^^^ help: use: `y`

error: unnecessary use of `to_string`
##[error]  --> tests/ui/unnecessary_to_owned.rs:467:12
   |
LL |         id("abc".to_string())
   |            ^^^^^^^^^^^^^^^^^ help: use: `"abc"`

error: unnecessary use of `to_vec`
##[error]  --> tests/ui/unnecessary_to_owned.rs:611:37
   |
LL |         IntoFuture::into_future(foo([].to_vec(), &0));
   |                                     ^^^^^^^^^^^ help: use: `[]`

error: unnecessary use of `to_vec`
##[error]  --> tests/ui/unnecessary_to_owned.rs:622:18
   |
LL |         s.remove(&a.to_vec());
   |                  ^^^^^^^^^^^ help: replace it with: `a`

error: unnecessary use of `to_owned`
##[error]  --> tests/ui/unnecessary_to_owned.rs:627:14
   |
LL |     s.remove(&"b".to_owned());
   |              ^^^^^^^^^^^^^^^ help: replace it with: `"b"`

error: unnecessary use of `to_string`
##[error]  --> tests/ui/unnecessary_to_owned.rs:629:14
   |
LL |     s.remove(&"b".to_string());
   |              ^^^^^^^^^^^^^^^^ help: replace it with: `"b"`

error: unnecessary use of `to_vec`
##[error]  --> tests/ui/unnecessary_to_owned.rs:635:14
   |
LL |     s.remove(&["b"].to_vec());
   |              ^^^^^^^^^^^^^^^ help: replace it with: `["b"].as_slice()`

error: unnecessary use of `to_vec`
##[error]  --> tests/ui/unnecessary_to_owned.rs:637:14
   |
LL |     s.remove(&(&["b"]).to_vec());
   |              ^^^^^^^^^^^^^^^^^^ help: replace it with: `(&["b"]).as_slice()`

error: aborting due to 68 previous errors



---
full stderr:
error: redundant clone
##[error]  --> tests/ui/unnecessary_to_owned.rs:217:64
   |
LL |     require_c_str(&CString::from_vec_with_nul(vec![0]).unwrap().to_owned());
   |                                                                ^^^^^^^^^^^ help: remove this
   |
note: this value is dropped without further use
  --> tests/ui/unnecessary_to_owned.rs:217:20
   |
LL |     require_c_str(&CString::from_vec_with_nul(vec![0]).unwrap().to_owned());
   |                    ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
   = note: `-D clippy::redundant-clone` implied by `-D warnings`
   = help: to override `-D warnings` add `#[allow(clippy::redundant_clone)]`

error: redundant clone
##[error]  --> tests/ui/unnecessary_to_owned.rs:219:40
   |
LL |     require_os_str(&OsString::from("x").to_os_string());
   |                                        ^^^^^^^^^^^^^^^ help: remove this
   |
note: this value is dropped without further use
  --> tests/ui/unnecessary_to_owned.rs:219:21
   |
LL |     require_os_str(&OsString::from("x").to_os_string());
   |                     ^^^^^^^^^^^^^^^^^^^

error: redundant clone
##[error]  --> tests/ui/unnecessary_to_owned.rs:221:48
   |
LL |     require_path(&std::path::PathBuf::from("x").to_path_buf());
   |                                                ^^^^^^^^^^^^^^ help: remove this
   |
note: this value is dropped without further use
  --> tests/ui/unnecessary_to_owned.rs:221:19
   |
LL |     require_path(&std::path::PathBuf::from("x").to_path_buf());
   |                   ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^

error: redundant clone
##[error]  --> tests/ui/unnecessary_to_owned.rs:223:35
   |
LL |     require_str(&String::from("x").to_string());
   |                                   ^^^^^^^^^^^^ help: remove this
   |
note: this value is dropped without further use
  --> tests/ui/unnecessary_to_owned.rs:223:18
   |
LL |     require_str(&String::from("x").to_string());
   |                  ^^^^^^^^^^^^^^^^^

error: redundant clone
##[error]  --> tests/ui/unnecessary_to_owned.rs:225:39
   |
LL |     require_slice(&[String::from("x")].to_owned());
   |                                       ^^^^^^^^^^^ help: remove this
   |
note: this value is dropped without further use
  --> tests/ui/unnecessary_to_owned.rs:225:20
   |
LL |     require_slice(&[String::from("x")].to_owned());
   |                    ^^^^^^^^^^^^^^^^^^^

error: unnecessary use of `into_owned`
##[error]  --> tests/ui/unnecessary_to_owned.rs:65:36
   |
LL |     require_c_str(&Cow::from(c_str).into_owned());
   |                                    ^^^^^^^^^^^^^ help: remove this
   |
   = note: `-D clippy::unnecessary-to-owned` implied by `-D warnings`
   = help: to override `-D warnings` add `#[allow(clippy::unnecessary_to_owned)]`

error: unnecessary use of `to_owned`
##[error]  --> tests/ui/unnecessary_to_owned.rs:67:19
   |
LL |     require_c_str(&c_str.to_owned());
   |                   ^^^^^^^^^^^^^^^^^ help: use: `c_str`

error: unnecessary use of `to_os_string`
##[error]  --> tests/ui/unnecessary_to_owned.rs:70:20
   |
LL |     require_os_str(&os_str.to_os_string());
   |                    ^^^^^^^^^^^^^^^^^^^^^^ help: use: `os_str`

error: unnecessary use of `into_owned`
##[error]  --> tests/ui/unnecessary_to_owned.rs:72:38
   |
LL |     require_os_str(&Cow::from(os_str).into_owned());
   |                                      ^^^^^^^^^^^^^ help: remove this

error: unnecessary use of `to_owned`
##[error]  --> tests/ui/unnecessary_to_owned.rs:74:20
   |
---

error: unnecessary use of `into_owned`
##[error]  --> tests/ui/unnecessary_to_owned.rs:79:34
   |
LL |     require_path(&Cow::from(path).into_owned());
   |                                  ^^^^^^^^^^^^^ help: remove this

error: unnecessary use of `to_owned`
##[error]  --> tests/ui/unnecessary_to_owned.rs:81:18
   |
---

error: unnecessary use of `into_owned`
##[error]  --> tests/ui/unnecessary_to_owned.rs:86:30
   |
LL |     require_str(&Cow::from(s).into_owned());
   |                              ^^^^^^^^^^^^^ help: remove this

error: unnecessary use of `to_owned`
##[error]  --> tests/ui/unnecessary_to_owned.rs:88:17
   |
LL |     require_str(&s.to_owned());
   |                 ^^^^^^^^^^^^^ help: use: `s`

error: unnecessary use of `to_string`
##[error]  --> tests/ui/unnecessary_to_owned.rs:90:17
   |
LL |     require_str(&x_ref.to_string());
   |                 ^^^^^^^^^^^^^^^^^^ help: use: `x_ref.as_ref()`

error: unnecessary use of `to_vec`
##[error]  --> tests/ui/unnecessary_to_owned.rs:93:19
   |
LL |     require_slice(&slice.to_vec());
   |                   ^^^^^^^^^^^^^^^ help: use: `slice`

error: unnecessary use of `into_owned`
##[error]  --> tests/ui/unnecessary_to_owned.rs:95:36
   |
LL |     require_slice(&Cow::from(slice).into_owned());
   |                                    ^^^^^^^^^^^^^ help: remove this

error: unnecessary use of `to_owned`
##[error]  --> tests/ui/unnecessary_to_owned.rs:97:19
   |
LL |     require_slice(&array.to_owned());
   |                   ^^^^^^^^^^^^^^^^^ help: use: `array.as_ref()`

error: unnecessary use of `to_owned`
##[error]  --> tests/ui/unnecessary_to_owned.rs:99:19
   |
LL |     require_slice(&array_ref.to_owned());
   |                   ^^^^^^^^^^^^^^^^^^^^^ help: use: `array_ref.as_ref()`

error: unnecessary use of `to_owned`
##[error]  --> tests/ui/unnecessary_to_owned.rs:101:19
   |
LL |     require_slice(&slice.to_owned());
   |                   ^^^^^^^^^^^^^^^^^ help: use: `slice`

error: unnecessary use of `into_owned`
##[error]  --> tests/ui/unnecessary_to_owned.rs:105:42
   |
LL |     require_x(&Cow::<X>::Owned(x.clone()).into_owned());
   |                                          ^^^^^^^^^^^^^ help: remove this

error: unnecessary use of `to_owned`
##[error]  --> tests/ui/unnecessary_to_owned.rs:138:26
   |
LL |     require_as_ref_c_str(c_str.to_owned());
   |                          ^^^^^^^^^^^^^^^^ help: use: `c_str`

error: unnecessary use of `to_owned`
##[error]  --> tests/ui/unnecessary_to_owned.rs:140:27
   |
LL |     require_as_ref_os_str(os_str.to_owned());
---

error: unnecessary use of `to_owned`
##[error]  --> tests/ui/unnecessary_to_owned.rs:155:31
   |
LL |     require_impl_as_ref_c_str(c_str.to_owned());
   |                               ^^^^^^^^^^^^^^^^ help: use: `c_str`

error: unnecessary use of `to_owned`
##[error]  --> tests/ui/unnecessary_to_owned.rs:157:32
   |
LL |     require_impl_as_ref_os_str(os_str.to_owned());
---

error: unnecessary use of `to_owned`
##[error]  --> tests/ui/unnecessary_to_owned.rs:172:30
   |
LL |     require_as_ref_str_slice(s.to_owned(), array.to_owned());
   |                              ^^^^^^^^^^^^ help: use: `s`

error: unnecessary use of `to_owned`
##[error]  --> tests/ui/unnecessary_to_owned.rs:172:44
   |
LL |     require_as_ref_str_slice(s.to_owned(), array.to_owned());
   |                                            ^^^^^^^^^^^^^^^^ help: use: `array`

error: unnecessary use of `to_owned`
##[error]  --> tests/ui/unnecessary_to_owned.rs:175:30
   |
LL |     require_as_ref_str_slice(s.to_owned(), array_ref.to_owned());
   |                              ^^^^^^^^^^^^ help: use: `s`

error: unnecessary use of `to_owned`
##[error]  --> tests/ui/unnecessary_to_owned.rs:175:44
   |
LL |     require_as_ref_str_slice(s.to_owned(), array_ref.to_owned());
   |                                            ^^^^^^^^^^^^^^^^^^^^ help: use: `array_ref`

error: unnecessary use of `to_owned`
##[error]  --> tests/ui/unnecessary_to_owned.rs:178:30
   |
---

error: unnecessary use of `to_string`
##[error]  --> tests/ui/unnecessary_to_owned.rs:191:20
   |
LL |     let _ = x.join(&x_ref.to_string());
   |                    ^^^^^^^^^^^^^^^^^^ help: use: `x_ref`

error: unnecessary use of `to_vec`
##[error]  --> tests/ui/unnecessary_to_owned.rs:194:13
   |
LL |     let _ = slice.to_vec().into_iter();
---
   |
LL |     let _ = IntoIterator::into_iter(slice.to_owned());
   |             ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ help: use: `slice.iter().copied()`

error: allocating a new `String` only to create a temporary `&str` from it
##[error]  --> tests/ui/unnecessary_to_owned.rs:229:26
   |
LL |     let _ref_str: &str = &String::from_utf8(slice.to_vec()).expect("not UTF-8");
   |                          ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
   |
help: convert from `&[u8]` to `&str` directly
   |
LL -     let _ref_str: &str = &String::from_utf8(slice.to_vec()).expect("not UTF-8");
LL +     let _ref_str: &str = core::str::from_utf8(&slice).expect("not UTF-8");
   |

error: allocating a new `String` only to create a temporary `&str` from it
##[error]  --> tests/ui/unnecessary_to_owned.rs:231:26
   |
LL |     let _ref_str: &str = &String::from_utf8(b"foo".to_vec()).unwrap();
   |                          ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
   |
help: convert from `&[u8]` to `&str` directly
   |
LL -     let _ref_str: &str = &String::from_utf8(b"foo".to_vec()).unwrap();
LL +     let _ref_str: &str = core::str::from_utf8(b"foo").unwrap();
   |

error: allocating a new `String` only to create a temporary `&str` from it
##[error]  --> tests/ui/unnecessary_to_owned.rs:233:26
   |
LL |     let _ref_str: &str = &String::from_utf8(b"foo".as_slice().to_owned()).unwrap();
   |                          ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
   |
help: convert from `&[u8]` to `&str` directly
   |
LL -     let _ref_str: &str = &String::from_utf8(b"foo".as_slice().to_owned()).unwrap();
LL +     let _ref_str: &str = core::str::from_utf8(b"foo".as_slice()).unwrap();
   |

error: unnecessary use of `to_vec`
##[error]  --> tests/ui/unnecessary_to_owned.rs:291:14
   |
LL |     for t in file_types.to_vec() {
   |              ^^^^^^^^^^^^^^^^^^^
   |
help: remove any references to the binding
   |
LL ~     for t in file_types {
LL |
LL ~         let path = match get_file_path(t) {
   |

error: unnecessary use of `to_string`
##[error]  --> tests/ui/unnecessary_to_owned.rs:357:24
   |
LL |         Box::new(build(y.to_string()))
   |                        ^^^^^^^^^^^^^ help: use: `y`

error: unnecessary use of `to_string`
##[error]  --> tests/ui/unnecessary_to_owned.rs:467:12
   |
LL |         id("abc".to_string())
   |            ^^^^^^^^^^^^^^^^^ help: use: `"abc"`

error: unnecessary use of `to_vec`
##[error]  --> tests/ui/unnecessary_to_owned.rs:611:37
   |
LL |         IntoFuture::into_future(foo([].to_vec(), &0));
   |                                     ^^^^^^^^^^^ help: use: `[]`

error: unnecessary use of `to_vec`
##[error]  --> tests/ui/unnecessary_to_owned.rs:622:18
   |
LL |         s.remove(&a.to_vec());
   |                  ^^^^^^^^^^^ help: replace it with: `a`

error: unnecessary use of `to_owned`
##[error]  --> tests/ui/unnecessary_to_owned.rs:627:14
   |
LL |     s.remove(&"b".to_owned());
   |              ^^^^^^^^^^^^^^^ help: replace it with: `"b"`

error: unnecessary use of `to_string`
##[error]  --> tests/ui/unnecessary_to_owned.rs:629:14
   |
LL |     s.remove(&"b".to_string());
   |              ^^^^^^^^^^^^^^^^ help: replace it with: `"b"`

error: unnecessary use of `to_vec`
##[error]  --> tests/ui/unnecessary_to_owned.rs:635:14
   |
LL |     s.remove(&["b"].to_vec());
   |              ^^^^^^^^^^^^^^^ help: replace it with: `["b"].as_slice()`

error: unnecessary use of `to_vec`
##[error]  --> tests/ui/unnecessary_to_owned.rs:637:14
   |
LL |     s.remove(&(&["b"]).to_vec());
   |              ^^^^^^^^^^^^^^^^^^ help: replace it with: `(&["b"]).as_slice()`

error: aborting due to 68 previous errors


full stdout:

@compiler-errors
Copy link
Member

@bors try

bors added a commit to rust-lang-ci/rust that referenced this pull request Mar 26, 2025
…mpl, r=<try>

arbitrary_self_type: insert implied Receiver bound on Deref

r? `@nikomatsakis`

This is an experimental setup which allows us to assess the impact from inserting an implied supertrait bound `Deref: Receiver`.

The observations are as follows.
- Thanks to the trait solver, not too many hacks are needed to implement the idea behind [this comment](rust-lang#135881 (comment)), where inductive cycles are admissible.
- We want to allow users to continue use the `dyn Deref` type as `Deref` has been `dyn`-compatible. Following the current `dyn`-compatibility rule, it would have been impossible to use `dyn Deref<Target = ..>` because one would need to write `dyn Deref<..> + Receiver<..>` while neither `Deref` nor `Receiver` is an `auto` trait. This is the main change that this PR proposes: we fill in the missing projection bound that `Receiver` demands from a `dyn Deref` and this will remain the only exception.
- Finally, the type (pretty-)printing is adjusted so as to not present the frivolous `Receiver::Target = ..` bound. It will always be the same term as `Deref::Target`.

I am proposing a crater experiment to see the extent of impact on the existing ecosystem and explore corner cases that I have missed.
@bors
Copy link
Collaborator

bors commented Mar 26, 2025

⌛ Trying commit 81e08c1 with merge af54b60...

#[must_use]
#[stable(feature = "rust1", since = "1.0.0")]
#[rustc_diagnostic_item = "deref_method"]
fn deref(&self) -> &<Self as Deref>::Target;
Copy link
Member

Choose a reason for hiding this comment

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

Why did you need to change this? Isn't the point of the hack in hir lowering to avoid breakage around T::Target shorthand projections?

@@ -6,8 +6,8 @@ use rustc_hir::def::{DefKind, Res};
use rustc_lint_defs::builtin::UNUSED_ASSOCIATED_TYPE_BOUNDS;
use rustc_middle::ty::elaborate::ClauseWithSupertraitSpan;
use rustc_middle::ty::{
self, BottomUpFolder, DynKind, ExistentialPredicateStableCmpExt as _, Ty, TyCtxt, TypeFoldable,
TypeVisitableExt, Upcast,
self, BottomUpFolder, DynKind, ExistentialPredicateStableCmpExt as _, Interner as _, Ty,
Copy link
Member

Choose a reason for hiding this comment

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

Why are you importing Interner? There should be no reason to use that trait.

@@ -112,9 +112,29 @@ impl<'tcx> dyn HirTyLowerer<'tcx> + '_ {
// let _: &dyn Alias<Assoc = u32> = /* ... */;
// ```
let mut projection_bounds = FxIndexMap::default();
let mut auto_fill_bounds: SmallVec<[_; 4]> = smallvec![];
let mut push_auto_fill_receiver_bound = |pred: ty::Binder<'tcx, _>| {
Copy link
Member

Choose a reason for hiding this comment

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

This is a bit overkill. Just take the projection and replace its def id with the one for Receiver::Target.

At that point, this doesn't need to be a callback, just an additional call to insert into the projection bounds map that you can duplicate twice.

pred.skip_binder().def_id(),
rustc_hir::LangItem::DerefTarget,
) {
// We need to fill in a `Receiver<Target = ?>` bound because
Copy link
Member

@compiler-errors compiler-errors Mar 26, 2025

Choose a reason for hiding this comment

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

This comment is missing the more important part that a supertrait like Deref: Receiver<Target = Self::Target> does not automatically elaborate into a projection bound for Reciever's Target type.

.filter(|item| !tcx.generics_require_sized_self(item.def_id))
.count()
})
let expected_count = obj.principal_def_id().map_or(0, |principal_def_id| {
Copy link
Member

Choose a reason for hiding this comment

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

why is this changed?

@bors
Copy link
Collaborator

bors commented Mar 26, 2025

☀️ Try build successful - checks-actions
Build commit: af54b60 (af54b6062f33b99e418444700f3a823ef0214052)

@compiler-errors
Copy link
Member

@craterbot check

@craterbot
Copy link
Collaborator

👌 Experiment pr-138952 created and queued.
🤖 Automatically detected try build af54b60
🔍 You can check out the queue and this experiment's details.

ℹ️ Crater is a tool to run experiments across parts of the Rust ecosystem. Learn more

@craterbot
Copy link
Collaborator

🚧 Experiment pr-138952 is now running

ℹ️ Crater is a tool to run experiments across parts of the Rust ecosystem. Learn more

@craterbot craterbot added S-waiting-on-crater Status: Waiting on a crater run to be completed. and removed S-waiting-on-review Status: Awaiting review from the assignee but also interested parties. labels Mar 26, 2025
@craterbot
Copy link
Collaborator

🎉 Experiment pr-138952 is completed!
📊 35 regressed and 4 fixed (604269 total)
📰 Open the full report.

⚠️ If you notice any spurious failure please add them to the denylist!
ℹ️ Crater is a tool to run experiments across parts of the Rust ecosystem. Learn more

@craterbot craterbot added S-waiting-on-review Status: Awaiting review from the assignee but also interested parties. and removed S-waiting-on-crater Status: Waiting on a crater run to be completed. labels Mar 27, 2025
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
F-arbitrary_self_types `#![feature(arbitrary_self_types)]` S-waiting-on-review Status: Awaiting review from the assignee but also interested parties. T-compiler Relevant to the compiler team, which will review and decide on the PR/issue. T-libs Relevant to the library team, which will review and decide on the PR/issue.
Projects
None yet
Development

Successfully merging this pull request may close these issues.

7 participants