Skip to content

Commit 914adf0

Browse files
committed
Auto merge of #71447 - cuviper:unsized_cow, r=dtolnay
impl From<Cow> for Box, Rc, and Arc These forward `Borrowed`/`Owned` values to existing `From` impls. - `Box<T>` is a fundamental type, so it would be a breaking change to add a blanket impl. Therefore, `From<Cow>` is only implemented for `[T]`, `str`, `CStr`, `OsStr`, and `Path`. - For `Rc<T>` and `Arc<T>`, `From<Cow>` is implemented for everything that implements `From` the borrowed and owned types separately.
2 parents 5943351 + 22efd95 commit 914adf0

File tree

9 files changed

+141
-5
lines changed

9 files changed

+141
-5
lines changed

src/liballoc/boxed.rs

+23
Original file line numberDiff line numberDiff line change
@@ -146,6 +146,7 @@ use core::ptr::{self, NonNull, Unique};
146146
use core::task::{Context, Poll};
147147

148148
use crate::alloc::{self, AllocInit, AllocRef, Global};
149+
use crate::borrow::Cow;
149150
use crate::raw_vec::RawVec;
150151
use crate::str::from_boxed_utf8_unchecked;
151152
use crate::vec::Vec;
@@ -800,6 +801,17 @@ impl<T: Copy> From<&[T]> for Box<[T]> {
800801
}
801802
}
802803

804+
#[stable(feature = "box_from_cow", since = "1.45.0")]
805+
impl<T: Copy> From<Cow<'_, [T]>> for Box<[T]> {
806+
#[inline]
807+
fn from(cow: Cow<'_, [T]>) -> Box<[T]> {
808+
match cow {
809+
Cow::Borrowed(slice) => Box::from(slice),
810+
Cow::Owned(slice) => Box::from(slice),
811+
}
812+
}
813+
}
814+
803815
#[stable(feature = "box_from_slice", since = "1.17.0")]
804816
impl From<&str> for Box<str> {
805817
/// Converts a `&str` into a `Box<str>`
@@ -818,6 +830,17 @@ impl From<&str> for Box<str> {
818830
}
819831
}
820832

833+
#[stable(feature = "box_from_cow", since = "1.45.0")]
834+
impl From<Cow<'_, str>> for Box<str> {
835+
#[inline]
836+
fn from(cow: Cow<'_, str>) -> Box<str> {
837+
match cow {
838+
Cow::Borrowed(s) => Box::from(s),
839+
Cow::Owned(s) => Box::from(s),
840+
}
841+
}
842+
}
843+
821844
#[stable(feature = "boxed_str_conv", since = "1.19.0")]
822845
impl From<Box<str>> for Box<[u8]> {
823846
/// Converts a `Box<str>>` into a `Box<[u8]>`

src/liballoc/rc.rs

+16
Original file line numberDiff line numberDiff line change
@@ -252,6 +252,7 @@ use core::ptr::{self, NonNull};
252252
use core::slice::from_raw_parts_mut;
253253

254254
use crate::alloc::{box_free, handle_alloc_error, AllocInit, AllocRef, Global, Layout};
255+
use crate::borrow::{Cow, ToOwned};
255256
use crate::string::String;
256257
use crate::vec::Vec;
257258

@@ -1497,6 +1498,21 @@ impl<T> From<Vec<T>> for Rc<[T]> {
14971498
}
14981499
}
14991500

1501+
#[stable(feature = "shared_from_cow", since = "1.45.0")]
1502+
impl<'a, B> From<Cow<'a, B>> for Rc<B>
1503+
where
1504+
B: ToOwned + ?Sized,
1505+
Rc<B>: From<&'a B> + From<B::Owned>,
1506+
{
1507+
#[inline]
1508+
fn from(cow: Cow<'a, B>) -> Rc<B> {
1509+
match cow {
1510+
Cow::Borrowed(s) => Rc::from(s),
1511+
Cow::Owned(s) => Rc::from(s),
1512+
}
1513+
}
1514+
}
1515+
15001516
#[stable(feature = "boxed_slice_try_from", since = "1.43.0")]
15011517
impl<T, const N: usize> TryFrom<Rc<[T]>> for Rc<[T; N]>
15021518
where

src/liballoc/sync.rs

+16
Original file line numberDiff line numberDiff line change
@@ -25,6 +25,7 @@ use core::sync::atomic;
2525
use core::sync::atomic::Ordering::{Acquire, Relaxed, Release, SeqCst};
2626

2727
use crate::alloc::{box_free, handle_alloc_error, AllocInit, AllocRef, Global, Layout};
28+
use crate::borrow::{Cow, ToOwned};
2829
use crate::boxed::Box;
2930
use crate::rc::is_dangling;
3031
use crate::string::String;
@@ -2128,6 +2129,21 @@ impl<T> From<Vec<T>> for Arc<[T]> {
21282129
}
21292130
}
21302131

2132+
#[stable(feature = "shared_from_cow", since = "1.45.0")]
2133+
impl<'a, B> From<Cow<'a, B>> for Arc<B>
2134+
where
2135+
B: ToOwned + ?Sized,
2136+
Arc<B>: From<&'a B> + From<B::Owned>,
2137+
{
2138+
#[inline]
2139+
fn from(cow: Cow<'a, B>) -> Arc<B> {
2140+
match cow {
2141+
Cow::Borrowed(s) => Arc::from(s),
2142+
Cow::Owned(s) => Arc::from(s),
2143+
}
2144+
}
2145+
}
2146+
21312147
#[stable(feature = "boxed_slice_try_from", since = "1.43.0")]
21322148
impl<T, const N: usize> TryFrom<Arc<[T]>> for Arc<[T; N]>
21332149
where

src/liballoc/tests/borrow.rs

+47
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,47 @@
1+
use std::borrow::{Cow, ToOwned};
2+
use std::ffi::{CStr, OsStr};
3+
use std::path::Path;
4+
use std::rc::Rc;
5+
use std::sync::Arc;
6+
7+
macro_rules! test_from_cow {
8+
($value:ident => $($ty:ty),+) => {$(
9+
let borrowed = <$ty>::from(Cow::Borrowed($value));
10+
let owned = <$ty>::from(Cow::Owned($value.to_owned()));
11+
assert_eq!($value, &*borrowed);
12+
assert_eq!($value, &*owned);
13+
)+};
14+
($value:ident : & $ty:ty) => {
15+
test_from_cow!($value => Box<$ty>, Rc<$ty>, Arc<$ty>);
16+
}
17+
}
18+
19+
#[test]
20+
fn test_from_cow_slice() {
21+
let slice: &[i32] = &[1, 2, 3];
22+
test_from_cow!(slice: &[i32]);
23+
}
24+
25+
#[test]
26+
fn test_from_cow_str() {
27+
let string = "hello";
28+
test_from_cow!(string: &str);
29+
}
30+
31+
#[test]
32+
fn test_from_cow_c_str() {
33+
let string = CStr::from_bytes_with_nul(b"hello\0").unwrap();
34+
test_from_cow!(string: &CStr);
35+
}
36+
37+
#[test]
38+
fn test_from_cow_os_str() {
39+
let string = OsStr::new("hello");
40+
test_from_cow!(string: &OsStr);
41+
}
42+
43+
#[test]
44+
fn test_from_cow_path() {
45+
let path = Path::new("hello");
46+
test_from_cow!(path: &Path);
47+
}

src/liballoc/tests/lib.rs

+1
Original file line numberDiff line numberDiff line change
@@ -21,6 +21,7 @@ use std::hash::{Hash, Hasher};
2121

2222
mod arc;
2323
mod binary_heap;
24+
mod borrow;
2425
mod boxed;
2526
mod btree;
2627
mod cow_str;

src/libstd/ffi/c_str.rs

+11
Original file line numberDiff line numberDiff line change
@@ -730,6 +730,17 @@ impl From<&CStr> for Box<CStr> {
730730
}
731731
}
732732

733+
#[stable(feature = "box_from_cow", since = "1.45.0")]
734+
impl From<Cow<'_, CStr>> for Box<CStr> {
735+
#[inline]
736+
fn from(cow: Cow<'_, CStr>) -> Box<CStr> {
737+
match cow {
738+
Cow::Borrowed(s) => Box::from(s),
739+
Cow::Owned(s) => Box::from(s),
740+
}
741+
}
742+
}
743+
733744
#[stable(feature = "c_string_from_box", since = "1.18.0")]
734745
impl From<Box<CStr>> for CString {
735746
/// Converts a [`Box`]`<CStr>` into a [`CString`] without copying or allocating.

src/libstd/ffi/os_str.rs

+11
Original file line numberDiff line numberDiff line change
@@ -850,6 +850,17 @@ impl From<&OsStr> for Box<OsStr> {
850850
}
851851
}
852852

853+
#[stable(feature = "box_from_cow", since = "1.45.0")]
854+
impl From<Cow<'_, OsStr>> for Box<OsStr> {
855+
#[inline]
856+
fn from(cow: Cow<'_, OsStr>) -> Box<OsStr> {
857+
match cow {
858+
Cow::Borrowed(s) => Box::from(s),
859+
Cow::Owned(s) => Box::from(s),
860+
}
861+
}
862+
}
863+
853864
#[stable(feature = "os_string_from_box", since = "1.18.0")]
854865
impl From<Box<OsStr>> for OsString {
855866
/// Converts a [`Box`]`<`[`OsStr`]`>` into a `OsString` without copying or

src/libstd/path.rs

+11
Original file line numberDiff line numberDiff line change
@@ -1433,6 +1433,17 @@ impl From<&Path> for Box<Path> {
14331433
}
14341434
}
14351435

1436+
#[stable(feature = "box_from_cow", since = "1.45.0")]
1437+
impl From<Cow<'_, Path>> for Box<Path> {
1438+
#[inline]
1439+
fn from(cow: Cow<'_, Path>) -> Box<Path> {
1440+
match cow {
1441+
Cow::Borrowed(path) => Box::from(path),
1442+
Cow::Owned(path) => Box::from(path),
1443+
}
1444+
}
1445+
}
1446+
14361447
#[stable(feature = "path_buf_from_box", since = "1.18.0")]
14371448
impl From<Box<Path>> for PathBuf {
14381449
/// Converts a `Box<Path>` into a `PathBuf`

src/test/ui/const-generics/array-impls/alloc-types-no-impls-length-33.stderr

+5-5
Original file line numberDiff line numberDiff line change
@@ -18,7 +18,7 @@ LL | let boxed_array = <Box<[i32; 33]>>::try_from(boxed_slice);
1818
<std::boxed::Box<(dyn std::error::Error + 'static)> as std::convert::From<&str>>
1919
<std::boxed::Box<(dyn std::error::Error + 'static)> as std::convert::From<std::borrow::Cow<'a, str>>>
2020
<std::boxed::Box<(dyn std::error::Error + 'static)> as std::convert::From<std::string::String>>
21-
and 16 others
21+
and 21 others
2222
= note: required because of the requirements on the impl of `std::convert::Into<std::boxed::Box<[i32; 33]>>` for `std::boxed::Box<[i32]>`
2323
= note: required because of the requirements on the impl of `std::convert::TryFrom<std::boxed::Box<[i32]>>` for `std::boxed::Box<[i32; 33]>`
2424

@@ -38,11 +38,11 @@ LL | let boxed_array = <Rc<[i32; 33]>>::try_from(boxed_slice);
3838
| ^^^^^^^^^^^^^^^^^^^^^^^^^ the trait `std::convert::From<std::rc::Rc<[i32]>>` is not implemented for `std::rc::Rc<[i32; 33]>`
3939
|
4040
= help: the following implementations were found:
41+
<std::rc::Rc<B> as std::convert::From<std::borrow::Cow<'a, B>>>
4142
<std::rc::Rc<T> as std::convert::From<T>>
4243
<std::rc::Rc<T> as std::convert::From<std::boxed::Box<T>>>
4344
<std::rc::Rc<[T]> as std::convert::From<&[T]>>
44-
<std::rc::Rc<[T]> as std::convert::From<std::vec::Vec<T>>>
45-
and 8 others
45+
and 9 others
4646
= note: required because of the requirements on the impl of `std::convert::Into<std::rc::Rc<[i32; 33]>>` for `std::rc::Rc<[i32]>`
4747
= note: required because of the requirements on the impl of `std::convert::TryFrom<std::rc::Rc<[i32]>>` for `std::rc::Rc<[i32; 33]>`
4848

@@ -62,11 +62,11 @@ LL | let boxed_array = <Arc<[i32; 33]>>::try_from(boxed_slice);
6262
| ^^^^^^^^^^^^^^^^^^^^^^^^^^ the trait `std::convert::From<std::sync::Arc<[i32]>>` is not implemented for `std::sync::Arc<[i32; 33]>`
6363
|
6464
= help: the following implementations were found:
65+
<std::sync::Arc<B> as std::convert::From<std::borrow::Cow<'a, B>>>
6566
<std::sync::Arc<T> as std::convert::From<T>>
6667
<std::sync::Arc<T> as std::convert::From<std::boxed::Box<T>>>
6768
<std::sync::Arc<[T]> as std::convert::From<&[T]>>
68-
<std::sync::Arc<[T]> as std::convert::From<std::vec::Vec<T>>>
69-
and 8 others
69+
and 9 others
7070
= note: required because of the requirements on the impl of `std::convert::Into<std::sync::Arc<[i32; 33]>>` for `std::sync::Arc<[i32]>`
7171
= note: required because of the requirements on the impl of `std::convert::TryFrom<std::sync::Arc<[i32]>>` for `std::sync::Arc<[i32; 33]>`
7272

0 commit comments

Comments
 (0)