Skip to content

Commit 85e6421

Browse files
committed
Implement Rc/Arc conversions for string-like types
Provides the following conversion implementations: * `From<`{`CString`,`&CStr`}`>` for {`Arc`,`Rc`}`<CStr>` * `From<`{`OsString`,`&OsStr`}`>` for {`Arc`,`Rc`}`<OsStr>` * `From<`{`PathBuf`,`&Path`}`>` for {`Arc`,`Rc`}`<Path>` Closes #45008
1 parent 9cd994c commit 85e6421

File tree

7 files changed

+259
-0
lines changed

7 files changed

+259
-0
lines changed

src/libstd/ffi/c_str.rs

+57
Original file line numberDiff line numberDiff line change
@@ -19,8 +19,10 @@ use memchr;
1919
use ops;
2020
use os::raw::c_char;
2121
use ptr;
22+
use rc::Rc;
2223
use slice;
2324
use str::{self, Utf8Error};
25+
use sync::Arc;
2426
use sys;
2527

2628
/// A type representing an owned, C-compatible, nul-terminated string with no nul bytes in the
@@ -704,6 +706,42 @@ impl From<CString> for Box<CStr> {
704706
}
705707
}
706708

709+
#[stable(feature = "shared_from_slice2", since = "1.23.0")]
710+
impl From<CString> for Arc<CStr> {
711+
#[inline]
712+
fn from(s: CString) -> Arc<CStr> {
713+
let arc: Arc<[u8]> = Arc::from(s.into_inner());
714+
unsafe { Arc::from_raw(Arc::into_raw(arc) as *const CStr) }
715+
}
716+
}
717+
718+
#[stable(feature = "shared_from_slice2", since = "1.23.0")]
719+
impl<'a> From<&'a CStr> for Arc<CStr> {
720+
#[inline]
721+
fn from(s: &CStr) -> Arc<CStr> {
722+
let arc: Arc<[u8]> = Arc::from(s.to_bytes_with_nul());
723+
unsafe { Arc::from_raw(Arc::into_raw(arc) as *const CStr) }
724+
}
725+
}
726+
727+
#[stable(feature = "shared_from_slice2", since = "1.23.0")]
728+
impl From<CString> for Rc<CStr> {
729+
#[inline]
730+
fn from(s: CString) -> Rc<CStr> {
731+
let rc: Rc<[u8]> = Rc::from(s.into_inner());
732+
unsafe { Rc::from_raw(Rc::into_raw(rc) as *const CStr) }
733+
}
734+
}
735+
736+
#[stable(feature = "shared_from_slice2", since = "1.23.0")]
737+
impl<'a> From<&'a CStr> for Rc<CStr> {
738+
#[inline]
739+
fn from(s: &CStr) -> Rc<CStr> {
740+
let rc: Rc<[u8]> = Rc::from(s.to_bytes_with_nul());
741+
unsafe { Rc::from_raw(Rc::into_raw(rc) as *const CStr) }
742+
}
743+
}
744+
707745
#[stable(feature = "default_box_extra", since = "1.17.0")]
708746
impl Default for Box<CStr> {
709747
fn default() -> Box<CStr> {
@@ -1201,6 +1239,8 @@ mod tests {
12011239
use borrow::Cow::{Borrowed, Owned};
12021240
use hash::{Hash, Hasher};
12031241
use collections::hash_map::DefaultHasher;
1242+
use rc::Rc;
1243+
use sync::Arc;
12041244

12051245
#[test]
12061246
fn c_to_rust() {
@@ -1337,4 +1377,21 @@ mod tests {
13371377
let boxed = <Box<CStr>>::default();
13381378
assert_eq!(boxed.to_bytes_with_nul(), &[0]);
13391379
}
1380+
1381+
#[test]
1382+
fn into_rc() {
1383+
let orig: &[u8] = b"Hello, world!\0";
1384+
let cstr = CStr::from_bytes_with_nul(orig).unwrap();
1385+
let rc: Rc<CStr> = Rc::from(cstr);
1386+
let arc: Arc<CStr> = Arc::from(cstr);
1387+
1388+
assert_eq!(&*rc, cstr);
1389+
assert_eq!(&*arc, cstr);
1390+
1391+
let rc2: Rc<CStr> = Rc::from(cstr.to_owned());
1392+
let arc2: Arc<CStr> = Arc::from(cstr.to_owned());
1393+
1394+
assert_eq!(&*rc2, cstr);
1395+
assert_eq!(&*arc2, cstr);
1396+
}
13401397
}

src/libstd/ffi/os_str.rs

+58
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,8 @@ use fmt;
1313
use ops;
1414
use cmp;
1515
use hash::{Hash, Hasher};
16+
use rc::Rc;
17+
use sync::Arc;
1618

1719
use sys::os_str::{Buf, Slice};
1820
use sys_common::{AsInner, IntoInner, FromInner};
@@ -592,6 +594,42 @@ impl From<OsString> for Box<OsStr> {
592594
}
593595
}
594596

597+
#[stable(feature = "shared_from_slice2", since = "1.23.0")]
598+
impl From<OsString> for Arc<OsStr> {
599+
#[inline]
600+
fn from(s: OsString) -> Arc<OsStr> {
601+
let arc = s.inner.into_arc();
602+
unsafe { Arc::from_raw(Arc::into_raw(arc) as *const OsStr) }
603+
}
604+
}
605+
606+
#[stable(feature = "shared_from_slice2", since = "1.23.0")]
607+
impl<'a> From<&'a OsStr> for Arc<OsStr> {
608+
#[inline]
609+
fn from(s: &OsStr) -> Arc<OsStr> {
610+
let arc = s.inner.into_arc();
611+
unsafe { Arc::from_raw(Arc::into_raw(arc) as *const OsStr) }
612+
}
613+
}
614+
615+
#[stable(feature = "shared_from_slice2", since = "1.23.0")]
616+
impl From<OsString> for Rc<OsStr> {
617+
#[inline]
618+
fn from(s: OsString) -> Rc<OsStr> {
619+
let rc = s.inner.into_rc();
620+
unsafe { Rc::from_raw(Rc::into_raw(rc) as *const OsStr) }
621+
}
622+
}
623+
624+
#[stable(feature = "shared_from_slice2", since = "1.23.0")]
625+
impl<'a> From<&'a OsStr> for Rc<OsStr> {
626+
#[inline]
627+
fn from(s: &OsStr) -> Rc<OsStr> {
628+
let rc = s.inner.into_rc();
629+
unsafe { Rc::from_raw(Rc::into_raw(rc) as *const OsStr) }
630+
}
631+
}
632+
595633
#[stable(feature = "box_default_extra", since = "1.17.0")]
596634
impl Default for Box<OsStr> {
597635
fn default() -> Box<OsStr> {
@@ -793,6 +831,9 @@ mod tests {
793831
use super::*;
794832
use sys_common::{AsInner, IntoInner};
795833

834+
use rc::Rc;
835+
use sync::Arc;
836+
796837
#[test]
797838
fn test_os_string_with_capacity() {
798839
let os_string = OsString::with_capacity(0);
@@ -935,4 +976,21 @@ mod tests {
935976
assert_eq!(os_str, os_string);
936977
assert!(os_string.capacity() >= 123);
937978
}
979+
980+
#[test]
981+
fn into_rc() {
982+
let orig = "Hello, world!";
983+
let os_str = OsStr::new(orig);
984+
let rc: Rc<OsStr> = Rc::from(os_str);
985+
let arc: Arc<OsStr> = Arc::from(os_str);
986+
987+
assert_eq!(&*rc, os_str);
988+
assert_eq!(&*arc, os_str);
989+
990+
let rc2: Rc<OsStr> = Rc::from(os_str.to_owned());
991+
let arc2: Arc<OsStr> = Arc::from(os_str.to_owned());
992+
993+
assert_eq!(&*rc2, os_str);
994+
assert_eq!(&*arc2, os_str);
995+
}
938996
}

src/libstd/path.rs

+58
Original file line numberDiff line numberDiff line change
@@ -86,6 +86,8 @@ use hash::{Hash, Hasher};
8686
use io;
8787
use iter::{self, FusedIterator};
8888
use ops::{self, Deref};
89+
use rc::Rc;
90+
use sync::Arc;
8991

9092
use ffi::{OsStr, OsString};
9193

@@ -1452,6 +1454,42 @@ impl<'a> From<PathBuf> for Cow<'a, Path> {
14521454
}
14531455
}
14541456

1457+
#[stable(feature = "shared_from_slice2", since = "1.23.0")]
1458+
impl From<PathBuf> for Arc<Path> {
1459+
#[inline]
1460+
fn from(s: PathBuf) -> Arc<Path> {
1461+
let arc: Arc<OsStr> = Arc::from(s.into_os_string());
1462+
unsafe { Arc::from_raw(Arc::into_raw(arc) as *const Path) }
1463+
}
1464+
}
1465+
1466+
#[stable(feature = "shared_from_slice2", since = "1.23.0")]
1467+
impl<'a> From<&'a Path> for Arc<Path> {
1468+
#[inline]
1469+
fn from(s: &Path) -> Arc<Path> {
1470+
let arc: Arc<OsStr> = Arc::from(s.as_os_str());
1471+
unsafe { Arc::from_raw(Arc::into_raw(arc) as *const Path) }
1472+
}
1473+
}
1474+
1475+
#[stable(feature = "shared_from_slice2", since = "1.23.0")]
1476+
impl From<PathBuf> for Rc<Path> {
1477+
#[inline]
1478+
fn from(s: PathBuf) -> Rc<Path> {
1479+
let rc: Rc<OsStr> = Rc::from(s.into_os_string());
1480+
unsafe { Rc::from_raw(Rc::into_raw(rc) as *const Path) }
1481+
}
1482+
}
1483+
1484+
#[stable(feature = "shared_from_slice2", since = "1.23.0")]
1485+
impl<'a> From<&'a Path> for Rc<Path> {
1486+
#[inline]
1487+
fn from(s: &Path) -> Rc<Path> {
1488+
let rc: Rc<OsStr> = Rc::from(s.as_os_str());
1489+
unsafe { Rc::from_raw(Rc::into_raw(rc) as *const Path) }
1490+
}
1491+
}
1492+
14551493
#[stable(feature = "rust1", since = "1.0.0")]
14561494
impl ToOwned for Path {
14571495
type Owned = PathBuf;
@@ -2568,6 +2606,9 @@ impl Error for StripPrefixError {
25682606
mod tests {
25692607
use super::*;
25702608

2609+
use rc::Rc;
2610+
use sync::Arc;
2611+
25712612
macro_rules! t(
25722613
($path:expr, iter: $iter:expr) => (
25732614
{
@@ -3970,4 +4011,21 @@ mod tests {
39704011
assert_eq!(format!("a{:#<5}b", Path::new("").display()), "a#####b");
39714012
assert_eq!(format!("a{:#<5}b", Path::new("a").display()), "aa####b");
39724013
}
4014+
4015+
#[test]
4016+
fn into_rc() {
4017+
let orig = "hello/world";
4018+
let path = Path::new(orig);
4019+
let rc: Rc<Path> = Rc::from(path);
4020+
let arc: Arc<Path> = Arc::from(path);
4021+
4022+
assert_eq!(&*rc, path);
4023+
assert_eq!(&*arc, path);
4024+
4025+
let rc2: Rc<Path> = Rc::from(path.to_owned());
4026+
let arc2: Arc<Path> = Arc::from(path.to_owned());
4027+
4028+
assert_eq!(&*rc2, path);
4029+
assert_eq!(&*arc2, path);
4030+
}
39734031
}

src/libstd/sys/redox/os_str.rs

+24
Original file line numberDiff line numberDiff line change
@@ -15,6 +15,8 @@ use borrow::Cow;
1515
use fmt;
1616
use str;
1717
use mem;
18+
use rc::Rc;
19+
use sync::Arc;
1820
use sys_common::{AsInner, IntoInner};
1921
use std_unicode::lossy::Utf8Lossy;
2022

@@ -123,6 +125,16 @@ impl Buf {
123125
let inner: Box<[u8]> = unsafe { mem::transmute(boxed) };
124126
Buf { inner: inner.into_vec() }
125127
}
128+
129+
#[inline]
130+
pub fn into_arc(&self) -> Arc<Slice> {
131+
self.as_slice().into_arc()
132+
}
133+
134+
#[inline]
135+
pub fn into_rc(&self) -> Rc<Slice> {
136+
self.as_slice().into_rc()
137+
}
126138
}
127139

128140
impl Slice {
@@ -156,4 +168,16 @@ impl Slice {
156168
let boxed: Box<[u8]> = Default::default();
157169
unsafe { mem::transmute(boxed) }
158170
}
171+
172+
#[inline]
173+
pub fn into_arc(&self) -> Arc<Slice> {
174+
let arc: Arc<[u8]> = Arc::from(&self.inner);
175+
unsafe { Arc::from_raw(Arc::into_raw(arc) as *const Slice) }
176+
}
177+
178+
#[inline]
179+
pub fn into_rc(&self) -> Rc<Slice> {
180+
let rc: Rc<[u8]> = Rc::from(&self.inner);
181+
unsafe { Rc::from_raw(Rc::into_raw(rc) as *const Slice) }
182+
}
159183
}

src/libstd/sys/unix/os_str.rs

+24
Original file line numberDiff line numberDiff line change
@@ -15,6 +15,8 @@ use borrow::Cow;
1515
use fmt;
1616
use str;
1717
use mem;
18+
use rc::Rc;
19+
use sync::Arc;
1820
use sys_common::{AsInner, IntoInner};
1921
use std_unicode::lossy::Utf8Lossy;
2022

@@ -123,6 +125,16 @@ impl Buf {
123125
let inner: Box<[u8]> = unsafe { mem::transmute(boxed) };
124126
Buf { inner: inner.into_vec() }
125127
}
128+
129+
#[inline]
130+
pub fn into_arc(&self) -> Arc<Slice> {
131+
self.as_slice().into_arc()
132+
}
133+
134+
#[inline]
135+
pub fn into_rc(&self) -> Rc<Slice> {
136+
self.as_slice().into_rc()
137+
}
126138
}
127139

128140
impl Slice {
@@ -156,4 +168,16 @@ impl Slice {
156168
let boxed: Box<[u8]> = Default::default();
157169
unsafe { mem::transmute(boxed) }
158170
}
171+
172+
#[inline]
173+
pub fn into_arc(&self) -> Arc<Slice> {
174+
let arc: Arc<[u8]> = Arc::from(&self.inner);
175+
unsafe { Arc::from_raw(Arc::into_raw(arc) as *const Slice) }
176+
}
177+
178+
#[inline]
179+
pub fn into_rc(&self) -> Rc<Slice> {
180+
let rc: Rc<[u8]> = Rc::from(&self.inner);
181+
unsafe { Rc::from_raw(Rc::into_raw(rc) as *const Slice) }
182+
}
159183
}

src/libstd/sys/windows/os_str.rs

+24
Original file line numberDiff line numberDiff line change
@@ -15,6 +15,8 @@ use borrow::Cow;
1515
use fmt;
1616
use sys_common::wtf8::{Wtf8, Wtf8Buf};
1717
use mem;
18+
use rc::Rc;
19+
use sync::Arc;
1820
use sys_common::{AsInner, IntoInner};
1921

2022
#[derive(Clone, Hash)]
@@ -115,6 +117,16 @@ impl Buf {
115117
let inner: Box<Wtf8> = unsafe { mem::transmute(boxed) };
116118
Buf { inner: Wtf8Buf::from_box(inner) }
117119
}
120+
121+
#[inline]
122+
pub fn into_arc(&self) -> Arc<Slice> {
123+
self.as_slice().into_arc()
124+
}
125+
126+
#[inline]
127+
pub fn into_rc(&self) -> Rc<Slice> {
128+
self.as_slice().into_rc()
129+
}
118130
}
119131

120132
impl Slice {
@@ -144,4 +156,16 @@ impl Slice {
144156
pub fn empty_box() -> Box<Slice> {
145157
unsafe { mem::transmute(Wtf8::empty_box()) }
146158
}
159+
160+
#[inline]
161+
pub fn into_arc(&self) -> Arc<Slice> {
162+
let arc = self.inner.into_arc();
163+
unsafe { Arc::from_raw(Arc::into_raw(arc) as *const Slice) }
164+
}
165+
166+
#[inline]
167+
pub fn into_rc(&self) -> Rc<Slice> {
168+
let rc = self.inner.into_rc();
169+
unsafe { Rc::from_raw(Rc::into_raw(rc) as *const Slice) }
170+
}
147171
}

0 commit comments

Comments
 (0)