Skip to content

Commit 8389253

Browse files
committed
Add generic conversion traits
This commit: * Introduces `std::convert`, providing an implementation of RFC 529. * Deprecates the `AsPath`, `AsOsStr`, and `IntoBytes` traits, all in favor of the corresponding generic conversion traits. Consequently, various IO APIs now take `AsRef<Path>` rather than `AsPath`, and so on. Since the types provided by `std` implement both traits, this should cause relatively little breakage. * Deprecates many `from_foo` constructors in favor of `from`. * Changes `PathBuf::new` to take no argument (creating an empty buffer, as per convention). The previous behavior is now available as `PathBuf::from`. * De-stabilizes `IntoCow`. It's not clear whether we need this separate trait. Closes #22751 Closes #14433 [breaking-change]
1 parent b0aad7d commit 8389253

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

69 files changed

+666
-196
lines changed

src/compiletest/compiletest.rs

+6-4
Original file line numberDiff line numberDiff line change
@@ -20,6 +20,8 @@
2020
#![feature(std_misc)]
2121
#![feature(test)]
2222
#![feature(path_ext)]
23+
#![feature(convert)]
24+
#![feature(str_char)]
2325

2426
#![deny(warnings)]
2527

@@ -115,7 +117,7 @@ pub fn parse_config(args: Vec<String> ) -> Config {
115117

116118
fn opt_path(m: &getopts::Matches, nm: &str) -> PathBuf {
117119
match m.opt_str(nm) {
118-
Some(s) => PathBuf::new(&s),
120+
Some(s) => PathBuf::from(&s),
119121
None => panic!("no option (=path) found for {}", nm),
120122
}
121123
}
@@ -130,18 +132,18 @@ pub fn parse_config(args: Vec<String> ) -> Config {
130132
compile_lib_path: matches.opt_str("compile-lib-path").unwrap(),
131133
run_lib_path: matches.opt_str("run-lib-path").unwrap(),
132134
rustc_path: opt_path(matches, "rustc-path"),
133-
clang_path: matches.opt_str("clang-path").map(|s| PathBuf::new(&s)),
135+
clang_path: matches.opt_str("clang-path").map(|s| PathBuf::from(&s)),
134136
valgrind_path: matches.opt_str("valgrind-path"),
135137
force_valgrind: matches.opt_present("force-valgrind"),
136-
llvm_bin_path: matches.opt_str("llvm-bin-path").map(|s| PathBuf::new(&s)),
138+
llvm_bin_path: matches.opt_str("llvm-bin-path").map(|s| PathBuf::from(&s)),
137139
src_base: opt_path(matches, "src-base"),
138140
build_base: opt_path(matches, "build-base"),
139141
aux_base: opt_path(matches, "aux-base"),
140142
stage_id: matches.opt_str("stage-id").unwrap(),
141143
mode: matches.opt_str("mode").unwrap().parse().ok().expect("invalid mode"),
142144
run_ignored: matches.opt_present("ignored"),
143145
filter: filter,
144-
logfile: matches.opt_str("logfile").map(|s| PathBuf::new(&s)),
146+
logfile: matches.opt_str("logfile").map(|s| PathBuf::from(&s)),
145147
runtool: matches.opt_str("runtool"),
146148
host_rustcflags: matches.opt_str("host-rustcflags"),
147149
target_rustcflags: matches.opt_str("target-rustcflags"),

src/compiletest/header.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -328,10 +328,10 @@ fn parse_exec_env(line: &str) -> Option<(String, String)> {
328328

329329
fn parse_pp_exact(line: &str, testfile: &Path) -> Option<PathBuf> {
330330
match parse_name_value_directive(line, "pp-exact") {
331-
Some(s) => Some(PathBuf::new(&s)),
331+
Some(s) => Some(PathBuf::from(&s)),
332332
None => {
333333
if parse_name_directive(line, "pp-exact") {
334-
testfile.file_name().map(|s| PathBuf::new(s))
334+
testfile.file_name().map(|s| PathBuf::from(s))
335335
} else {
336336
None
337337
}

src/compiletest/runtest.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -1440,7 +1440,7 @@ fn aux_output_dir_name(config: &Config, testfile: &Path) -> PathBuf {
14401440
}
14411441

14421442
fn output_testname(testfile: &Path) -> PathBuf {
1443-
PathBuf::new(testfile.file_stem().unwrap())
1443+
PathBuf::from(testfile.file_stem().unwrap())
14441444
}
14451445

14461446
fn output_base_name(config: &Config, testfile: &Path) -> PathBuf {

src/libcollections/borrow.rs

+9-2
Original file line numberDiff line numberDiff line change
@@ -14,6 +14,7 @@
1414

1515
use core::clone::Clone;
1616
use core::cmp::{Eq, Ord, Ordering, PartialEq, PartialOrd};
17+
use core::convert::AsRef;
1718
use core::hash::{Hash, Hasher};
1819
use core::marker::Sized;
1920
use core::ops::Deref;
@@ -291,10 +292,9 @@ impl<'a, B: ?Sized> Hash for Cow<'a, B> where B: Hash + ToOwned
291292
}
292293

293294
/// Trait for moving into a `Cow`
294-
#[stable(feature = "rust1", since = "1.0.0")]
295+
#[unstable(feature = "into_cow", reason = "may be replaced by `convert::Into`")]
295296
pub trait IntoCow<'a, B: ?Sized> where B: ToOwned {
296297
/// Moves `self` into `Cow`
297-
#[stable(feature = "rust1", since = "1.0.0")]
298298
fn into_cow(self) -> Cow<'a, B>;
299299
}
300300

@@ -304,3 +304,10 @@ impl<'a, B: ?Sized> IntoCow<'a, B> for Cow<'a, B> where B: ToOwned {
304304
self
305305
}
306306
}
307+
308+
#[stable(feature = "rust1", since = "1.0.0")]
309+
impl<'a, T: Clone> AsRef<T> for Cow<'a, T> {
310+
fn as_ref(&self) -> &T {
311+
self
312+
}
313+
}

src/libcollections/lib.rs

+1
Original file line numberDiff line numberDiff line change
@@ -36,6 +36,7 @@
3636
#![feature(unsafe_no_drop_flag)]
3737
#![feature(step_by)]
3838
#![feature(str_char)]
39+
#![feature(convert)]
3940
#![cfg_attr(test, feature(rand, rustc_private, test))]
4041
#![cfg_attr(test, allow(deprecated))] // rand
4142

src/libcollections/slice.rs

+6-5
Original file line numberDiff line numberDiff line change
@@ -88,6 +88,7 @@
8888
#![stable(feature = "rust1", since = "1.0.0")]
8989

9090
use alloc::boxed::Box;
91+
use core::convert::AsRef;
9192
use core::clone::Clone;
9293
use core::cmp::Ordering::{self, Greater, Less};
9394
use core::cmp::{self, Ord, PartialEq};
@@ -1088,23 +1089,23 @@ pub trait SliceConcatExt<T: ?Sized, U> {
10881089
fn connect(&self, sep: &T) -> U;
10891090
}
10901091

1091-
impl<T: Clone, V: AsSlice<T>> SliceConcatExt<T, Vec<T>> for [V] {
1092+
impl<T: Clone, V: AsRef<[T]>> SliceConcatExt<T, Vec<T>> for [V] {
10921093
fn concat(&self) -> Vec<T> {
1093-
let size = self.iter().fold(0, |acc, v| acc + v.as_slice().len());
1094+
let size = self.iter().fold(0, |acc, v| acc + v.as_ref().len());
10941095
let mut result = Vec::with_capacity(size);
10951096
for v in self {
1096-
result.push_all(v.as_slice())
1097+
result.push_all(v.as_ref())
10971098
}
10981099
result
10991100
}
11001101

11011102
fn connect(&self, sep: &T) -> Vec<T> {
1102-
let size = self.iter().fold(0, |acc, v| acc + v.as_slice().len());
1103+
let size = self.iter().fold(0, |acc, v| acc + v.as_ref().len());
11031104
let mut result = Vec::with_capacity(size + self.len());
11041105
let mut first = true;
11051106
for v in self {
11061107
if first { first = false } else { result.push(sep.clone()) }
1107-
result.push_all(v.as_slice())
1108+
result.push_all(v.as_ref())
11081109
}
11091110
result
11101111
}

src/libcollections/str.rs

+12-16
Original file line numberDiff line numberDiff line change
@@ -61,10 +61,10 @@ use core::iter::AdditiveIterator;
6161
use core::iter::{Iterator, IteratorExt, Extend};
6262
use core::option::Option::{self, Some, None};
6363
use core::result::Result;
64-
use core::slice::AsSlice;
6564
use core::str as core_str;
6665
use unicode::str::{UnicodeStr, Utf16Encoder};
6766

67+
use core::convert::AsRef;
6868
use vec_deque::VecDeque;
6969
use borrow::{Borrow, ToOwned};
7070
use string::String;
@@ -86,51 +86,47 @@ pub use core::str::{Searcher, ReverseSearcher, DoubleEndedSearcher, SearchStep};
8686
Section: Creating a string
8787
*/
8888

89-
impl<S: Str> SliceConcatExt<str, String> for [S] {
89+
impl<S: AsRef<str>> SliceConcatExt<str, String> for [S] {
9090
fn concat(&self) -> String {
91-
let s = self.as_slice();
92-
93-
if s.is_empty() {
91+
if self.is_empty() {
9492
return String::new();
9593
}
9694

9795
// `len` calculation may overflow but push_str will check boundaries
98-
let len = s.iter().map(|s| s.as_slice().len()).sum();
96+
let len = self.iter().map(|s| s.as_ref().len()).sum();
9997
let mut result = String::with_capacity(len);
10098

101-
for s in s {
102-
result.push_str(s.as_slice())
99+
for s in self {
100+
result.push_str(s.as_ref())
103101
}
104102

105103
result
106104
}
107105

108106
fn connect(&self, sep: &str) -> String {
109-
let s = self.as_slice();
110-
111-
if s.is_empty() {
107+
if self.is_empty() {
112108
return String::new();
113109
}
114110

115111
// concat is faster
116112
if sep.is_empty() {
117-
return s.concat();
113+
return self.concat();
118114
}
119115

120116
// this is wrong without the guarantee that `self` is non-empty
121117
// `len` calculation may overflow but push_str but will check boundaries
122-
let len = sep.len() * (s.len() - 1)
123-
+ s.iter().map(|s| s.as_slice().len()).sum();
118+
let len = sep.len() * (self.len() - 1)
119+
+ self.iter().map(|s| s.as_ref().len()).sum();
124120
let mut result = String::with_capacity(len);
125121
let mut first = true;
126122

127-
for s in s {
123+
for s in self {
128124
if first {
129125
first = false;
130126
} else {
131127
result.push_str(sep);
132128
}
133-
result.push_str(s.as_slice());
129+
result.push_str(s.as_ref());
134130
}
135131
result
136132
}

src/libcollections/string.rs

+23
Original file line numberDiff line numberDiff line change
@@ -814,6 +814,7 @@ impl<'a, 'b> PartialEq<Cow<'a, str>> for &'b str {
814814
}
815815

816816
#[unstable(feature = "collections", reason = "waiting on Str stabilization")]
817+
#[allow(deprecated)]
817818
impl Str for String {
818819
#[inline]
819820
#[stable(feature = "rust1", since = "1.0.0")]
@@ -973,6 +974,27 @@ impl<T: fmt::Display + ?Sized> ToString for T {
973974
}
974975
}
975976

977+
#[stable(feature = "rust1", since = "1.0.0")]
978+
impl AsRef<str> for String {
979+
fn as_ref(&self) -> &str {
980+
self
981+
}
982+
}
983+
984+
#[stable(feature = "rust1", since = "1.0.0")]
985+
impl<'a> From<&'a str> for String {
986+
fn from(s: &'a str) -> String {
987+
s.to_string()
988+
}
989+
}
990+
991+
#[stable(feature = "rust1", since = "1.0.0")]
992+
impl Into<Vec<u8>> for String {
993+
fn into(self) -> Vec<u8> {
994+
self.into_bytes()
995+
}
996+
}
997+
976998
#[stable(feature = "rust1", since = "1.0.0")]
977999
impl IntoCow<'static, str> for String {
9781000
#[inline]
@@ -989,6 +1011,7 @@ impl<'a> IntoCow<'a, str> for &'a str {
9891011
}
9901012
}
9911013

1014+
#[allow(deprecated)]
9921015
impl<'a> Str for Cow<'a, str> {
9931016
#[inline]
9941017
fn as_slice<'b>(&'b self) -> &'b str {

src/libcollections/vec.rs

+45-7
Original file line numberDiff line numberDiff line change
@@ -1369,7 +1369,7 @@ impl<T> ops::Index<ops::RangeFull> for Vec<T> {
13691369
type Output = [T];
13701370
#[inline]
13711371
fn index(&self, _index: &ops::RangeFull) -> &[T] {
1372-
self.as_slice()
1372+
self
13731373
}
13741374
}
13751375

@@ -1406,7 +1406,13 @@ impl<T> ops::IndexMut<ops::RangeFull> for Vec<T> {
14061406
impl<T> ops::Deref for Vec<T> {
14071407
type Target = [T];
14081408

1409-
fn deref(&self) -> &[T] { self.as_slice() }
1409+
fn deref(&self) -> &[T] {
1410+
unsafe {
1411+
let p = *self.ptr;
1412+
assume(p != 0 as *mut T);
1413+
slice::from_raw_parts(p, self.len)
1414+
}
1415+
}
14101416
}
14111417

14121418
#[stable(feature = "rust1", since = "1.0.0")]
@@ -1548,6 +1554,7 @@ impl<T: Ord> Ord for Vec<T> {
15481554
}
15491555
}
15501556

1557+
#[allow(deprecated)]
15511558
impl<T> AsSlice<T> for Vec<T> {
15521559
/// Returns a slice into `self`.
15531560
///
@@ -1562,11 +1569,7 @@ impl<T> AsSlice<T> for Vec<T> {
15621569
#[inline]
15631570
#[stable(feature = "rust1", since = "1.0.0")]
15641571
fn as_slice(&self) -> &[T] {
1565-
unsafe {
1566-
let p = *self.ptr;
1567-
assume(p != 0 as *mut T);
1568-
slice::from_raw_parts(p, self.len)
1569-
}
1572+
self
15701573
}
15711574
}
15721575

@@ -1614,6 +1617,41 @@ impl<T: fmt::Debug> fmt::Debug for Vec<T> {
16141617
}
16151618
}
16161619

1620+
#[stable(feature = "rust1", since = "1.0.0")]
1621+
impl<T> AsRef<Vec<T>> for Vec<T> {
1622+
fn as_ref(&self) -> &Vec<T> {
1623+
self
1624+
}
1625+
}
1626+
1627+
#[stable(feature = "rust1", since = "1.0.0")]
1628+
impl<T> Into<Vec<T>> for Vec<T> {
1629+
fn into(self) -> Vec<T> {
1630+
self
1631+
}
1632+
}
1633+
1634+
#[stable(feature = "rust1", since = "1.0.0")]
1635+
impl<T> AsRef<[T]> for Vec<T> {
1636+
fn as_ref(&self) -> &[T] {
1637+
self
1638+
}
1639+
}
1640+
1641+
#[stable(feature = "rust1", since = "1.0.0")]
1642+
impl<'a, T: Clone> From<&'a [T]> for Vec<T> {
1643+
fn from(s: &'a [T]) -> Vec<T> {
1644+
s.to_vec()
1645+
}
1646+
}
1647+
1648+
#[stable(feature = "rust1", since = "1.0.0")]
1649+
impl<'a> From<&'a str> for Vec<u8> {
1650+
fn from(s: &'a str) -> Vec<u8> {
1651+
s.as_bytes().to_vec()
1652+
}
1653+
}
1654+
16171655
////////////////////////////////////////////////////////////////////////////////
16181656
// Clone-on-write
16191657
////////////////////////////////////////////////////////////////////////////////

0 commit comments

Comments
 (0)