Skip to content

Commit 0701b37

Browse files
committed
Auto merge of #44678 - alexcrichton:rollup, r=alexcrichton
Rollup of 11 pull requests - Successful merges: #44364, #44466, #44537, #44548, #44640, #44651, #44657, #44661, #44668, #44671, #44675 - Failed merges:
2 parents 3a7b960 + 929215d commit 0701b37

Some content is hidden

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

74 files changed

+1985
-798
lines changed
Original file line numberDiff line numberDiff line change
@@ -1,14 +1,13 @@
11
# `splice`
22

3-
The tracking issue for this feature is: [#32310]
3+
The tracking issue for this feature is: [#44643]
44

5-
[#32310]: https://github.com/rust-lang/rust/issues/32310
5+
[#44643]: https://github.com/rust-lang/rust/issues/44643
66

77
------------------------
88

9-
The `splice()` method on `Vec` and `String` allows you to replace a range
10-
of values in a vector or string with another range of values, and returns
11-
the replaced values.
9+
The `splice()` method on `String` allows you to replace a range
10+
of values in a string with another range of values.
1211

1312
A simple example:
1413

@@ -20,4 +19,4 @@ let beta_offset = s.find('β').unwrap_or(s.len());
2019
// Replace the range up until the β from the string
2120
s.splice(..beta_offset, "Α is capital alpha; ");
2221
assert_eq!(s, "Α is capital alpha; β is beta");
23-
```
22+
```

src/liballoc/string.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -1451,7 +1451,7 @@ impl String {
14511451
/// s.splice(..beta_offset, "Α is capital alpha; ");
14521452
/// assert_eq!(s, "Α is capital alpha; β is beta");
14531453
/// ```
1454-
#[unstable(feature = "splice", reason = "recently added", issue = "32310")]
1454+
#[unstable(feature = "splice", reason = "recently added", issue = "44643")]
14551455
pub fn splice<R>(&mut self, range: R, replace_with: &str)
14561456
where R: RangeArgument<usize>
14571457
{

src/liballoc/vec.rs

+6-7
Original file line numberDiff line numberDiff line change
@@ -1943,15 +1943,14 @@ impl<T> Vec<T> {
19431943
/// # Examples
19441944
///
19451945
/// ```
1946-
/// #![feature(splice)]
19471946
/// let mut v = vec![1, 2, 3];
19481947
/// let new = [7, 8];
19491948
/// let u: Vec<_> = v.splice(..2, new.iter().cloned()).collect();
19501949
/// assert_eq!(v, &[7, 8, 3]);
19511950
/// assert_eq!(u, &[1, 2]);
19521951
/// ```
19531952
#[inline]
1954-
#[unstable(feature = "splice", reason = "recently added", issue = "32310")]
1953+
#[stable(feature = "vec_splice", since = "1.22.0")]
19551954
pub fn splice<R, I>(&mut self, range: R, replace_with: I) -> Splice<I::IntoIter>
19561955
where R: RangeArgument<usize>, I: IntoIterator<Item=T>
19571956
{
@@ -2554,13 +2553,13 @@ impl<'a, T> InPlace<T> for PlaceBack<'a, T> {
25542553
/// [`splice()`]: struct.Vec.html#method.splice
25552554
/// [`Vec`]: struct.Vec.html
25562555
#[derive(Debug)]
2557-
#[unstable(feature = "splice", reason = "recently added", issue = "32310")]
2556+
#[stable(feature = "vec_splice", since = "1.22.0")]
25582557
pub struct Splice<'a, I: Iterator + 'a> {
25592558
drain: Drain<'a, I::Item>,
25602559
replace_with: I,
25612560
}
25622561

2563-
#[unstable(feature = "splice", reason = "recently added", issue = "32310")]
2562+
#[stable(feature = "vec_splice", since = "1.22.0")]
25642563
impl<'a, I: Iterator> Iterator for Splice<'a, I> {
25652564
type Item = I::Item;
25662565

@@ -2573,18 +2572,18 @@ impl<'a, I: Iterator> Iterator for Splice<'a, I> {
25732572
}
25742573
}
25752574

2576-
#[unstable(feature = "splice", reason = "recently added", issue = "32310")]
2575+
#[stable(feature = "vec_splice", since = "1.22.0")]
25772576
impl<'a, I: Iterator> DoubleEndedIterator for Splice<'a, I> {
25782577
fn next_back(&mut self) -> Option<Self::Item> {
25792578
self.drain.next_back()
25802579
}
25812580
}
25822581

2583-
#[unstable(feature = "splice", reason = "recently added", issue = "32310")]
2582+
#[stable(feature = "vec_splice", since = "1.22.0")]
25842583
impl<'a, I: Iterator> ExactSizeIterator for Splice<'a, I> {}
25852584

25862585

2587-
#[unstable(feature = "splice", reason = "recently added", issue = "32310")]
2586+
#[stable(feature = "vec_splice", since = "1.22.0")]
25882587
impl<'a, I: Iterator> Drop for Splice<'a, I> {
25892588
fn drop(&mut self) {
25902589
// exhaust drain first

src/libcore/intrinsics.rs

+1-32
Original file line numberDiff line numberDiff line change
@@ -1381,38 +1381,7 @@ extern "rust-intrinsic" {
13811381
}
13821382

13831383
#[cfg(stage0)]
1384-
/// Computes the byte offset that needs to be applied to `ptr` in order to
1385-
/// make it aligned to `align`.
1386-
/// If it is not possible to align `ptr`, the implementation returns
1387-
/// `usize::max_value()`.
1388-
///
1389-
/// There are no guarantees whatsover that offsetting the pointer will not
1390-
/// overflow or go beyond the allocation that `ptr` points into.
1391-
/// It is up to the caller to ensure that the returned offset is correct
1392-
/// in all terms other than alignment.
1393-
///
1394-
/// # Examples
1395-
///
1396-
/// Accessing adjacent `u8` as `u16`
1397-
///
1398-
/// ```
1399-
/// # #![feature(core_intrinsics)]
1400-
/// # fn foo(n: usize) {
1401-
/// # use std::intrinsics::align_offset;
1402-
/// # use std::mem::align_of;
1403-
/// # unsafe {
1404-
/// let x = [5u8, 6u8, 7u8, 8u8, 9u8];
1405-
/// let ptr = &x[n] as *const u8;
1406-
/// let offset = align_offset(ptr as *const (), align_of::<u16>());
1407-
/// if offset < x.len() - n - 1 {
1408-
/// let u16_ptr = ptr.offset(offset as isize) as *const u16;
1409-
/// assert_ne!(*u16_ptr, 500);
1410-
/// } else {
1411-
/// // while the pointer can be aligned via `offset`, it would point
1412-
/// // outside the allocation
1413-
/// }
1414-
/// # } }
1415-
/// ```
1384+
/// remove me after the next release
14161385
pub unsafe fn align_offset(ptr: *const (), align: usize) -> usize {
14171386
let offset = ptr as usize % align;
14181387
if offset == 0 {

src/libcore/iter/traits.rs

+17
Original file line numberDiff line numberDiff line change
@@ -196,6 +196,23 @@ pub trait FromIterator<A>: Sized {
196196
/// assert_eq!(i as i32, n);
197197
/// }
198198
/// ```
199+
///
200+
/// It is common to use `IntoIterator` as a trait bound. This allows
201+
/// the input collection type to change, so long as it is still an
202+
/// iterator. Additional bounds can be specified by restricting on
203+
/// `Item`:
204+
///
205+
/// ```rust
206+
/// fn collect_as_strings<T>(collection: T) -> Vec<String>
207+
/// where T: IntoIterator,
208+
/// T::Item : std::fmt::Debug,
209+
/// {
210+
/// collection
211+
/// .into_iter()
212+
/// .map(|item| format!("{:?}", item))
213+
/// .collect()
214+
/// }
215+
/// ```
199216
#[stable(feature = "rust1", since = "1.0.0")]
200217
pub trait IntoIterator {
201218
/// The type of the elements being iterated over.

src/libcore/ptr.rs

+74-1
Original file line numberDiff line numberDiff line change
@@ -1064,7 +1064,43 @@ impl<T: ?Sized> *const T {
10641064
copy_nonoverlapping(self, dest, count)
10651065
}
10661066

1067-
1067+
/// Computes the byte offset that needs to be applied in order to
1068+
/// make the pointer aligned to `align`.
1069+
/// If it is not possible to align the pointer, the implementation returns
1070+
/// `usize::max_value()`.
1071+
///
1072+
/// There are no guarantees whatsover that offsetting the pointer will not
1073+
/// overflow or go beyond the allocation that the pointer points into.
1074+
/// It is up to the caller to ensure that the returned offset is correct
1075+
/// in all terms other than alignment.
1076+
///
1077+
/// # Examples
1078+
///
1079+
/// Accessing adjacent `u8` as `u16`
1080+
///
1081+
/// ```
1082+
/// # #![feature(align_offset)]
1083+
/// # fn foo(n: usize) {
1084+
/// # use std::mem::align_of;
1085+
/// # unsafe {
1086+
/// let x = [5u8, 6u8, 7u8, 8u8, 9u8];
1087+
/// let ptr = &x[n] as *const u8;
1088+
/// let offset = ptr.align_offset(align_of::<u16>());
1089+
/// if offset < x.len() - n - 1 {
1090+
/// let u16_ptr = ptr.offset(offset as isize) as *const u16;
1091+
/// assert_ne!(*u16_ptr, 500);
1092+
/// } else {
1093+
/// // while the pointer can be aligned via `offset`, it would point
1094+
/// // outside the allocation
1095+
/// }
1096+
/// # } }
1097+
/// ```
1098+
#[unstable(feature = "align_offset", issue = "44488")]
1099+
pub fn align_offset(self, align: usize) -> usize {
1100+
unsafe {
1101+
intrinsics::align_offset(self as *const _, align)
1102+
}
1103+
}
10681104
}
10691105

10701106
#[lang = "mut_ptr"]
@@ -1284,6 +1320,43 @@ impl<T: ?Sized> *mut T {
12841320
}
12851321
}
12861322

1323+
/// Computes the byte offset that needs to be applied in order to
1324+
/// make the pointer aligned to `align`.
1325+
/// If it is not possible to align the pointer, the implementation returns
1326+
/// `usize::max_value()`.
1327+
///
1328+
/// There are no guarantees whatsover that offsetting the pointer will not
1329+
/// overflow or go beyond the allocation that the pointer points into.
1330+
/// It is up to the caller to ensure that the returned offset is correct
1331+
/// in all terms other than alignment.
1332+
///
1333+
/// # Examples
1334+
///
1335+
/// Accessing adjacent `u8` as `u16`
1336+
///
1337+
/// ```
1338+
/// # #![feature(align_offset)]
1339+
/// # fn foo(n: usize) {
1340+
/// # use std::mem::align_of;
1341+
/// # unsafe {
1342+
/// let x = [5u8, 6u8, 7u8, 8u8, 9u8];
1343+
/// let ptr = &x[n] as *const u8;
1344+
/// let offset = ptr.align_offset(align_of::<u16>());
1345+
/// if offset < x.len() - n - 1 {
1346+
/// let u16_ptr = ptr.offset(offset as isize) as *const u16;
1347+
/// assert_ne!(*u16_ptr, 500);
1348+
/// } else {
1349+
/// // while the pointer can be aligned via `offset`, it would point
1350+
/// // outside the allocation
1351+
/// }
1352+
/// # } }
1353+
/// ```
1354+
#[unstable(feature = "align_offset", issue = "44488")]
1355+
pub fn align_offset(self, align: usize) -> usize {
1356+
unsafe {
1357+
intrinsics::align_offset(self as *const _, align)
1358+
}
1359+
}
12871360

12881361
/// Calculates the offset from a pointer (convenience for `.offset(count as isize)`).
12891362
///

src/libcore/str/mod.rs

+5-6
Original file line numberDiff line numberDiff line change
@@ -23,7 +23,6 @@ use fmt;
2323
use iter::{Map, Cloned, FusedIterator};
2424
use slice::{self, SliceIndex};
2525
use mem;
26-
use intrinsics::align_offset;
2726

2827
pub mod pattern;
2928

@@ -404,7 +403,7 @@ unsafe fn from_raw_parts_mut<'a>(p: *mut u8, len: usize) -> &'a mut str {
404403
#[inline]
405404
#[stable(feature = "rust1", since = "1.0.0")]
406405
pub unsafe fn from_utf8_unchecked(v: &[u8]) -> &str {
407-
mem::transmute(v)
406+
&*(v as *const [u8] as *const str)
408407
}
409408

410409
/// Converts a slice of bytes to a string slice without checking
@@ -429,7 +428,7 @@ pub unsafe fn from_utf8_unchecked(v: &[u8]) -> &str {
429428
#[inline]
430429
#[stable(feature = "str_mut_extras", since = "1.20.0")]
431430
pub unsafe fn from_utf8_unchecked_mut(v: &mut [u8]) -> &mut str {
432-
mem::transmute(v)
431+
&mut *(v as *mut [u8] as *mut str)
433432
}
434433

435434
#[stable(feature = "rust1", since = "1.0.0")]
@@ -1515,7 +1514,7 @@ fn run_utf8_validation(v: &[u8]) -> Result<(), Utf8Error> {
15151514
let ptr = v.as_ptr();
15161515
let align = unsafe {
15171516
// the offset is safe, because `index` is guaranteed inbounds
1518-
align_offset(ptr.offset(index as isize) as *const (), usize_bytes)
1517+
ptr.offset(index as isize).align_offset(usize_bytes)
15191518
};
15201519
if align == 0 {
15211520
while index < blocks_end {
@@ -2447,12 +2446,12 @@ impl StrExt for str {
24472446

24482447
#[inline]
24492448
fn as_bytes(&self) -> &[u8] {
2450-
unsafe { mem::transmute(self) }
2449+
unsafe { &*(self as *const str as *const [u8]) }
24512450
}
24522451

24532452
#[inline]
24542453
unsafe fn as_bytes_mut(&mut self) -> &mut [u8] {
2455-
mem::transmute(self)
2454+
&mut *(self as *mut str as *mut [u8])
24562455
}
24572456

24582457
fn find<'a, P: Pattern<'a>>(&'a self, pat: P) -> Option<usize> {

src/librustc/dep_graph/dep_node.rs

+14-2
Original file line numberDiff line numberDiff line change
@@ -603,12 +603,12 @@ trait DepNodeParams<'a, 'gcx: 'tcx + 'a, 'tcx: 'a> : fmt::Debug {
603603
}
604604

605605
impl<'a, 'gcx: 'tcx + 'a, 'tcx: 'a, T> DepNodeParams<'a, 'gcx, 'tcx> for T
606-
where T: HashStable<StableHashingContext<'a, 'gcx, 'tcx>> + fmt::Debug
606+
where T: HashStable<StableHashingContext<'gcx>> + fmt::Debug
607607
{
608608
default const CAN_RECONSTRUCT_QUERY_KEY: bool = false;
609609

610610
default fn to_fingerprint(&self, tcx: TyCtxt<'a, 'gcx, 'tcx>) -> Fingerprint {
611-
let mut hcx = StableHashingContext::new(tcx);
611+
let mut hcx = tcx.create_stable_hashing_context();
612612
let mut hasher = StableHasher::new();
613613

614614
self.hash_stable(&mut hcx, &mut hasher);
@@ -633,6 +633,18 @@ impl<'a, 'gcx: 'tcx + 'a, 'tcx: 'a> DepNodeParams<'a, 'gcx, 'tcx> for (DefId,) {
633633
}
634634
}
635635

636+
impl<'a, 'gcx: 'tcx + 'a, 'tcx: 'a> DepNodeParams<'a, 'gcx, 'tcx> for (DefIndex,) {
637+
const CAN_RECONSTRUCT_QUERY_KEY: bool = true;
638+
639+
fn to_fingerprint(&self, tcx: TyCtxt) -> Fingerprint {
640+
tcx.hir.definitions().def_path_hash(self.0).0
641+
}
642+
643+
fn to_debug_str(&self, tcx: TyCtxt<'a, 'gcx, 'tcx>) -> String {
644+
tcx.item_path_str(DefId::local(self.0))
645+
}
646+
}
647+
636648
impl<'a, 'gcx: 'tcx + 'a, 'tcx: 'a> DepNodeParams<'a, 'gcx, 'tcx> for (DefId, DefId) {
637649
const CAN_RECONSTRUCT_QUERY_KEY: bool = false;
638650

src/librustc/dep_graph/graph.rs

+25-11
Original file line numberDiff line numberDiff line change
@@ -9,11 +9,15 @@
99
// except according to those terms.
1010

1111
use rustc_data_structures::fx::FxHashMap;
12+
use rustc_data_structures::stable_hasher::{HashStable, StableHasher,
13+
StableHashingContextProvider};
1214
use session::config::OutputType;
1315
use std::cell::{Ref, RefCell};
1416
use std::rc::Rc;
1517
use util::common::{ProfileQueriesMsg, profq_msg};
1618

19+
use ich::Fingerprint;
20+
1721
use super::dep_node::{DepNode, DepKind, WorkProductId};
1822
use super::query::DepGraphQuery;
1923
use super::raii;
@@ -71,10 +75,6 @@ impl DepGraph {
7175
self.data.as_ref().map(|data| raii::IgnoreTask::new(&data.edges))
7276
}
7377

74-
pub fn in_task<'graph>(&'graph self, key: DepNode) -> Option<raii::DepTask<'graph>> {
75-
self.data.as_ref().map(|data| raii::DepTask::new(&data.edges, key))
76-
}
77-
7878
pub fn with_ignore<OP,R>(&self, op: OP) -> R
7979
where OP: FnOnce() -> R
8080
{
@@ -109,24 +109,38 @@ impl DepGraph {
109109
/// `arg` parameter.
110110
///
111111
/// [README]: README.md
112-
pub fn with_task<C, A, R>(&self,
113-
key: DepNode,
114-
cx: C,
115-
arg: A,
116-
task: fn(C, A) -> R)
117-
-> (R, DepNodeIndex)
118-
where C: DepGraphSafe
112+
pub fn with_task<C, A, R, HCX>(&self,
113+
key: DepNode,
114+
cx: C,
115+
arg: A,
116+
task: fn(C, A) -> R)
117+
-> (R, DepNodeIndex)
118+
where C: DepGraphSafe + StableHashingContextProvider<ContextType=HCX>,
119+
R: HashStable<HCX>,
119120
{
120121
if let Some(ref data) = self.data {
121122
data.edges.borrow_mut().push_task(key);
122123
if cfg!(debug_assertions) {
123124
profq_msg(ProfileQueriesMsg::TaskBegin(key.clone()))
124125
};
126+
127+
// In incremental mode, hash the result of the task. We don't
128+
// do anything with the hash yet, but we are computing it
129+
// anyway so that
130+
// - we make sure that the infrastructure works and
131+
// - we can get an idea of the runtime cost.
132+
let mut hcx = cx.create_stable_hashing_context();
133+
125134
let result = task(cx, arg);
126135
if cfg!(debug_assertions) {
127136
profq_msg(ProfileQueriesMsg::TaskEnd)
128137
};
129138
let dep_node_index = data.edges.borrow_mut().pop_task(key);
139+
140+
let mut stable_hasher = StableHasher::new();
141+
result.hash_stable(&mut hcx, &mut stable_hasher);
142+
let _: Fingerprint = stable_hasher.finish();
143+
130144
(result, dep_node_index)
131145
} else {
132146
(task(cx, arg), DepNodeIndex::INVALID)

0 commit comments

Comments
 (0)