Skip to content

Commit 9ed53cf

Browse files
author
Clar Charr
committed
Conversions between CStr/OsStr/Path and boxes.
1 parent 2425b22 commit 9ed53cf

File tree

8 files changed

+192
-1
lines changed

8 files changed

+192
-1
lines changed

src/liballoc/boxed.rs

+8
Original file line numberDiff line numberDiff line change
@@ -315,6 +315,14 @@ impl<T> Default for Box<[T]> {
315315
}
316316
}
317317

318+
#[stable(feature = "default_box_extra", since = "1.17.0")]
319+
impl Default for Box<str> {
320+
fn default() -> Box<str> {
321+
let default: Box<[u8]> = Default::default();
322+
unsafe { mem::transmute(default) }
323+
}
324+
}
325+
318326
#[stable(feature = "rust1", since = "1.0.0")]
319327
impl<T: Clone> Clone for Box<T> {
320328
/// Returns a new box with a `clone()` of this box's contents.

src/libstd/ffi/c_str.rs

+41-1
Original file line numberDiff line numberDiff line change
@@ -303,6 +303,12 @@ impl CString {
303303
&self.inner
304304
}
305305

306+
/// Converts this `CString` into a boxed `CStr`.
307+
#[unstable(feature = "into_boxed_c_str", issue = "0")]
308+
pub fn into_boxed_c_str(self) -> Box<CStr> {
309+
unsafe { mem::transmute(self.into_inner()) }
310+
}
311+
306312
// Bypass "move out of struct which implements `Drop` trait" restriction.
307313
fn into_inner(self) -> Box<[u8]> {
308314
unsafe {
@@ -380,6 +386,22 @@ impl Borrow<CStr> for CString {
380386
fn borrow(&self) -> &CStr { self }
381387
}
382388

389+
#[stable(feature = "box_from_c_str", since = "1.17.0")]
390+
impl<'a> From<&'a CStr> for Box<CStr> {
391+
fn from(s: &'a CStr) -> Box<CStr> {
392+
let boxed: Box<[u8]> = Box::from(s.to_bytes_with_nul());
393+
unsafe { mem::transmute(boxed) }
394+
}
395+
}
396+
397+
#[stable(feature = "default_box_extra", since = "1.17.0")]
398+
impl Default for Box<CStr> {
399+
fn default() -> Box<CStr> {
400+
let boxed: Box<[u8]> = Box::from([0]);
401+
unsafe { mem::transmute(boxed) }
402+
}
403+
}
404+
383405
impl NulError {
384406
/// Returns the position of the nul byte in the slice that was provided to
385407
/// `CString::new`.
@@ -686,7 +708,7 @@ impl ToOwned for CStr {
686708
type Owned = CString;
687709

688710
fn to_owned(&self) -> CString {
689-
CString { inner: self.to_bytes_with_nul().to_vec().into_boxed_slice() }
711+
CString { inner: self.to_bytes_with_nul().into() }
690712
}
691713
}
692714

@@ -847,4 +869,22 @@ mod tests {
847869
let cstr = CStr::from_bytes_with_nul(data);
848870
assert!(cstr.is_err());
849871
}
872+
873+
#[test]
874+
fn into_boxed() {
875+
let orig: &[u8] = b"Hello, world!\0";
876+
let cstr = CStr::from_bytes_with_nul(orig).unwrap();
877+
let cstring = cstr.to_owned();
878+
let box1: Box<CStr> = Box::from(cstr);
879+
let box2 = cstring.into_boxed_c_str();
880+
assert_eq!(cstr, &*box1);
881+
assert_eq!(box1, box2);
882+
assert_eq!(&*box2, cstr);
883+
}
884+
885+
#[test]
886+
fn boxed_default() {
887+
let boxed = <Box<CStr>>::default();
888+
assert_eq!(boxed.to_bytes_with_nul(), &[0]);
889+
}
850890
}

src/libstd/ffi/os_str.rs

+38
Original file line numberDiff line numberDiff line change
@@ -204,6 +204,12 @@ impl OsString {
204204
pub fn reserve_exact(&mut self, additional: usize) {
205205
self.inner.reserve_exact(additional)
206206
}
207+
208+
/// Converts this `OsString` into a boxed `OsStr`.
209+
#[unstable(feature = "into_boxed_os_str", issue = "0")]
210+
pub fn into_boxed_os_str(self) -> Box<OsStr> {
211+
unsafe { mem::transmute(self.inner.into_box()) }
212+
}
207213
}
208214

209215
#[stable(feature = "rust1", since = "1.0.0")]
@@ -445,6 +451,20 @@ impl OsStr {
445451
}
446452
}
447453

454+
#[stable(feature = "box_from_os_str", since = "1.17.0")]
455+
impl<'a> From<&'a OsStr> for Box<OsStr> {
456+
fn from(s: &'a OsStr) -> Box<OsStr> {
457+
unsafe { mem::transmute(s.inner.into_box()) }
458+
}
459+
}
460+
461+
#[stable(feature = "box_default_extra", since = "1.17.0")]
462+
impl Default for Box<OsStr> {
463+
fn default() -> Box<OsStr> {
464+
unsafe { mem::transmute(Slice::empty_box()) }
465+
}
466+
}
467+
448468
#[stable(feature = "osstring_default", since = "1.9.0")]
449469
impl<'a> Default for &'a OsStr {
450470
/// Creates an empty `OsStr`.
@@ -741,4 +761,22 @@ mod tests {
741761
let os_str: &OsStr = Default::default();
742762
assert_eq!("", os_str);
743763
}
764+
765+
#[test]
766+
fn into_boxed() {
767+
let orig = "Hello, world!";
768+
let os_str = OsStr::new(orig);
769+
let os_string = os_str.to_owned();
770+
let box1: Box<OsStr> = Box::from(os_str);
771+
let box2 = os_string.into_boxed_os_str();
772+
assert_eq!(os_str, &*box1);
773+
assert_eq!(box1, box2);
774+
assert_eq!(&*box2, os_str);
775+
}
776+
777+
#[test]
778+
fn boxed_default() {
779+
let boxed = <Box<OsStr>>::default();
780+
assert!(boxed.is_empty());
781+
}
744782
}

src/libstd/path.rs

+40
Original file line numberDiff line numberDiff line change
@@ -1194,6 +1194,28 @@ impl PathBuf {
11941194
pub fn into_os_string(self) -> OsString {
11951195
self.inner
11961196
}
1197+
1198+
/// Converts this `PathBuf` into a boxed `Path`.
1199+
#[unstable(feature = "into_boxed_path", issue = "0")]
1200+
pub fn into_boxed_path(self) -> Box<Path> {
1201+
unsafe { mem::transmute(self.inner.into_boxed_os_str()) }
1202+
}
1203+
}
1204+
1205+
#[stable(feature = "box_from_path", since = "1.17.0")]
1206+
impl<'a> From<&'a Path> for Box<Path> {
1207+
fn from(path: &'a Path) -> Box<Path> {
1208+
let boxed: Box<OsStr> = path.inner.into();
1209+
unsafe { mem::transmute(boxed) }
1210+
}
1211+
}
1212+
1213+
#[stable(feature = "box_default_extra", since = "1.17.0")]
1214+
impl Default for Box<Path> {
1215+
fn default() -> Box<Path> {
1216+
let boxed: Box<OsStr> = Default::default();
1217+
unsafe { mem::transmute(boxed) }
1218+
}
11971219
}
11981220

11991221
#[stable(feature = "rust1", since = "1.0.0")]
@@ -3676,4 +3698,22 @@ mod tests {
36763698
let actual = format!("{:?}", iter);
36773699
assert_eq!(expected, actual);
36783700
}
3701+
3702+
#[test]
3703+
fn into_boxed() {
3704+
let orig: &str = "some/sort/of/path";
3705+
let path = Path::new(orig);
3706+
let path_buf = path.to_owned();
3707+
let box1: Box<Path> = Box::from(path);
3708+
let box2 = path_buf.into_boxed_path();
3709+
assert_eq!(path, &*box1);
3710+
assert_eq!(box1, box2);
3711+
assert_eq!(&*box2, path);
3712+
}
3713+
3714+
#[test]
3715+
fn boxed_default() {
3716+
let boxed = <Box<Path>>::default();
3717+
assert!(boxed.as_os_str().is_empty());
3718+
}
36793719
}

src/libstd/sys/redox/os_str.rs

+16
Original file line numberDiff line numberDiff line change
@@ -94,6 +94,11 @@ impl Buf {
9494
pub fn push_slice(&mut self, s: &Slice) {
9595
self.inner.extend_from_slice(&s.inner)
9696
}
97+
98+
#[inline]
99+
pub fn into_box(self) -> Box<Slice> {
100+
unsafe { mem::transmute(self.inner.into_boxed_slice()) }
101+
}
97102
}
98103

99104
impl Slice {
@@ -116,4 +121,15 @@ impl Slice {
116121
pub fn to_owned(&self) -> Buf {
117122
Buf { inner: self.inner.to_vec() }
118123
}
124+
125+
#[inline]
126+
pub fn into_box(&self) -> Box<Slice> {
127+
let boxed: Box<[u8]> = self.inner.into();
128+
unsafe { mem::transmute(boxed) }
129+
}
130+
131+
pub fn empty_box() -> Box<Slice> {
132+
let boxed: Box<[u8]> = Default::default();
133+
unsafe { mem::transmute(boxed) }
134+
}
119135
}

src/libstd/sys/unix/os_str.rs

+16
Original file line numberDiff line numberDiff line change
@@ -94,6 +94,11 @@ impl Buf {
9494
pub fn push_slice(&mut self, s: &Slice) {
9595
self.inner.extend_from_slice(&s.inner)
9696
}
97+
98+
#[inline]
99+
pub fn into_box(self) -> Box<Slice> {
100+
unsafe { mem::transmute(self.inner.into_boxed_slice()) }
101+
}
97102
}
98103

99104
impl Slice {
@@ -116,4 +121,15 @@ impl Slice {
116121
pub fn to_owned(&self) -> Buf {
117122
Buf { inner: self.inner.to_vec() }
118123
}
124+
125+
#[inline]
126+
pub fn into_box(&self) -> Box<Slice> {
127+
let boxed: Box<[u8]> = self.inner.into();
128+
unsafe { mem::transmute(boxed) }
129+
}
130+
131+
pub fn empty_box() -> Box<Slice> {
132+
let boxed: Box<[u8]> = Default::default();
133+
unsafe { mem::transmute(boxed) }
134+
}
119135
}

src/libstd/sys/windows/os_str.rs

+14
Original file line numberDiff line numberDiff line change
@@ -88,6 +88,11 @@ impl Buf {
8888
pub fn reserve_exact(&mut self, additional: usize) {
8989
self.inner.reserve_exact(additional)
9090
}
91+
92+
#[inline]
93+
pub fn into_box(self) -> Box<Slice> {
94+
unsafe { mem::transmute(self.inner.into_box()) }
95+
}
9196
}
9297

9398
impl Slice {
@@ -108,4 +113,13 @@ impl Slice {
108113
buf.push_wtf8(&self.inner);
109114
Buf { inner: buf }
110115
}
116+
117+
#[inline]
118+
pub fn into_box(&self) -> Box<Slice> {
119+
unsafe { mem::transmute(self.inner.into_box()) }
120+
}
121+
122+
pub fn empty_box() -> Box<Slice> {
123+
unsafe { mem::transmute(self.inner.empty_box()) }
124+
}
111125
}

src/libstd/sys_common/wtf8.rs

+19
Original file line numberDiff line numberDiff line change
@@ -340,6 +340,12 @@ impl Wtf8Buf {
340340
}
341341
}
342342
}
343+
344+
/// Converts this `Wtf8Buf` into a boxed `Wtf8`.
345+
#[inline]
346+
pub fn into_box(self) -> Box<Wtf8> {
347+
unsafe { mem::transmute(self.bytes.into_boxed_slice()) }
348+
}
343349
}
344350

345351
/// Create a new WTF-8 string from an iterator of code points.
@@ -583,6 +589,19 @@ impl Wtf8 {
583589
_ => None
584590
}
585591
}
592+
593+
/// Boxes this `Wtf8`.
594+
#[inline]
595+
pub fn into_box(&self) -> Box<Wtf8> {
596+
let boxed: Box<[u8]> = self.bytes.into();
597+
unsafe { mem::transmute(boxed) }
598+
}
599+
600+
/// Creates a boxed, empty `Wtf8`.
601+
pub fn empty_box() -> Box<Wtf8> {
602+
let boxed: Box<[u8]> = Default::default();
603+
unsafe { mem::transmute(boxed) }
604+
}
586605
}
587606

588607

0 commit comments

Comments
 (0)