Skip to content

Commit 14b1552

Browse files
committed
Auto merge of rust-lang#71483 - Dylan-DPC:rollup-c2h9s8b, r=Dylan-DPC
Rollup of 7 pull requests Successful merges: - rust-lang#70633 (Confusing suggestion on incorrect closing `}`) - rust-lang#71404 (Don't fuse Chain in its second iterator) - rust-lang#71408 (Check code blocks tags) - rust-lang#71442 (Add a "by reference" adaptor for `AllocRef`) - rust-lang#71446 (Only use read_unaligned in transmute_copy if necessary) - rust-lang#71470 (Fix doc links) - rust-lang#71479 (add back Scalar::null_ptr) Failed merges: r? @ghost
2 parents 413a129 + b107eb5 commit 14b1552

30 files changed

+939
-73
lines changed

src/liballoc/collections/vec_deque.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -73,7 +73,7 @@ pub struct VecDeque<T> {
7373
/// It produces the following sequence of matching slices:
7474
///
7575
/// ([0 1], [a b])
76-
/// ([2], [c])
76+
/// (\[2\], \[c\])
7777
/// ([3 4], [d e])
7878
///
7979
/// and the uneven remainder of either A or B is skipped.

src/liballoc/rc.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -1024,7 +1024,7 @@ unsafe fn set_data_ptr<T: ?Sized, U>(mut ptr: *mut T, data: *mut U) -> *mut T {
10241024
}
10251025

10261026
impl<T> Rc<[T]> {
1027-
/// Copy elements from slice into newly allocated Rc<[T]>
1027+
/// Copy elements from slice into newly allocated Rc<\[T\]>
10281028
///
10291029
/// Unsafe because the caller must either take ownership or bind `T: Copy`
10301030
unsafe fn copy_from_slice(v: &[T]) -> Rc<[T]> {

src/liballoc/sync.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -902,7 +902,7 @@ unsafe fn set_data_ptr<T: ?Sized, U>(mut ptr: *mut T, data: *mut U) -> *mut T {
902902
}
903903

904904
impl<T> Arc<[T]> {
905-
/// Copy elements from slice into newly allocated Arc<[T]>
905+
/// Copy elements from slice into newly allocated Arc<\[T\]>
906906
///
907907
/// Unsafe because the caller must either take ownership or bind `T: Copy`.
908908
unsafe fn copy_from_slice(v: &[T]) -> Arc<[T]> {

src/libcore/alloc/mod.rs

+47
Original file line numberDiff line numberDiff line change
@@ -364,4 +364,51 @@ pub unsafe trait AllocRef {
364364
}
365365
}
366366
}
367+
368+
/// Creates a "by reference" adaptor for this instance of `AllocRef`.
369+
///
370+
/// The returned adaptor also implements `AllocRef` and will simply borrow this.
371+
#[inline(always)]
372+
fn by_ref(&mut self) -> &mut Self {
373+
self
374+
}
375+
}
376+
377+
#[unstable(feature = "allocator_api", issue = "32838")]
378+
unsafe impl<A> AllocRef for &mut A
379+
where
380+
A: AllocRef + ?Sized,
381+
{
382+
#[inline]
383+
fn alloc(&mut self, layout: Layout, init: AllocInit) -> Result<MemoryBlock, AllocErr> {
384+
(**self).alloc(layout, init)
385+
}
386+
387+
#[inline]
388+
unsafe fn dealloc(&mut self, ptr: NonNull<u8>, layout: Layout) {
389+
(**self).dealloc(ptr, layout)
390+
}
391+
392+
#[inline]
393+
unsafe fn grow(
394+
&mut self,
395+
ptr: NonNull<u8>,
396+
layout: Layout,
397+
new_size: usize,
398+
placement: ReallocPlacement,
399+
init: AllocInit,
400+
) -> Result<MemoryBlock, AllocErr> {
401+
(**self).grow(ptr, layout, new_size, placement, init)
402+
}
403+
404+
#[inline]
405+
unsafe fn shrink(
406+
&mut self,
407+
ptr: NonNull<u8>,
408+
layout: Layout,
409+
new_size: usize,
410+
placement: ReallocPlacement,
411+
) -> Result<MemoryBlock, AllocErr> {
412+
(**self).shrink(ptr, layout, new_size, placement)
413+
}
367414
}

src/libcore/ffi.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -282,7 +282,7 @@ impl<'a, 'f: 'a> DerefMut for VaList<'a, 'f> {
282282
mod sealed_trait {
283283
/// Trait which whitelists the allowed types to be used with [VaList::arg]
284284
///
285-
/// [VaList::va_arg]: struct.VaList.html#method.arg
285+
/// [VaList::arg]: ../struct.VaList.html#method.arg
286286
#[unstable(
287287
feature = "c_variadic",
288288
reason = "the `c_variadic` feature has not been properly tested on \

src/libcore/iter/adapters/chain.rs

+22-10
Original file line numberDiff line numberDiff line change
@@ -18,6 +18,9 @@ pub struct Chain<A, B> {
1818
// adapter because its specialization for `FusedIterator` unconditionally descends into the
1919
// iterator, and that could be expensive to keep revisiting stuff like nested chains. It also
2020
// hurts compiler performance to add more iterator layers to `Chain`.
21+
//
22+
// Only the "first" iterator is actually set `None` when exhausted, depending on whether you
23+
// iterate forward or backward. If you mix directions, then both sides may be `None`.
2124
a: Option<A>,
2225
b: Option<B>,
2326
}
@@ -43,6 +46,17 @@ macro_rules! fuse {
4346
};
4447
}
4548

49+
/// Try an iterator method without fusing,
50+
/// like an inline `.as_mut().and_then(...)`
51+
macro_rules! maybe {
52+
($self:ident . $iter:ident . $($call:tt)+) => {
53+
match $self.$iter {
54+
Some(ref mut iter) => iter.$($call)+,
55+
None => None,
56+
}
57+
};
58+
}
59+
4660
#[stable(feature = "rust1", since = "1.0.0")]
4761
impl<A, B> Iterator for Chain<A, B>
4862
where
@@ -54,7 +68,7 @@ where
5468
#[inline]
5569
fn next(&mut self) -> Option<A::Item> {
5670
match fuse!(self.a.next()) {
57-
None => fuse!(self.b.next()),
71+
None => maybe!(self.b.next()),
5872
item => item,
5973
}
6074
}
@@ -85,7 +99,7 @@ where
8599
}
86100
if let Some(ref mut b) = self.b {
87101
acc = b.try_fold(acc, f)?;
88-
self.b = None;
102+
// we don't fuse the second iterator
89103
}
90104
Try::from_ok(acc)
91105
}
@@ -114,7 +128,7 @@ where
114128
}
115129
self.a = None;
116130
}
117-
fuse!(self.b.nth(n))
131+
maybe!(self.b.nth(n))
118132
}
119133

120134
#[inline]
@@ -123,7 +137,7 @@ where
123137
P: FnMut(&Self::Item) -> bool,
124138
{
125139
match fuse!(self.a.find(&mut predicate)) {
126-
None => fuse!(self.b.find(predicate)),
140+
None => maybe!(self.b.find(predicate)),
127141
item => item,
128142
}
129143
}
@@ -174,7 +188,7 @@ where
174188
#[inline]
175189
fn next_back(&mut self) -> Option<A::Item> {
176190
match fuse!(self.b.next_back()) {
177-
None => fuse!(self.a.next_back()),
191+
None => maybe!(self.a.next_back()),
178192
item => item,
179193
}
180194
}
@@ -190,7 +204,7 @@ where
190204
}
191205
self.b = None;
192206
}
193-
fuse!(self.a.nth_back(n))
207+
maybe!(self.a.nth_back(n))
194208
}
195209

196210
#[inline]
@@ -199,7 +213,7 @@ where
199213
P: FnMut(&Self::Item) -> bool,
200214
{
201215
match fuse!(self.b.rfind(&mut predicate)) {
202-
None => fuse!(self.a.rfind(predicate)),
216+
None => maybe!(self.a.rfind(predicate)),
203217
item => item,
204218
}
205219
}
@@ -216,7 +230,7 @@ where
216230
}
217231
if let Some(ref mut a) = self.a {
218232
acc = a.try_rfold(acc, f)?;
219-
self.a = None;
233+
// we don't fuse the second iterator
220234
}
221235
Try::from_ok(acc)
222236
}
@@ -236,8 +250,6 @@ where
236250
}
237251

238252
// Note: *both* must be fused to handle double-ended iterators.
239-
// Now that we "fuse" both sides, we *could* implement this unconditionally,
240-
// but we should be cautious about committing to that in the public API.
241253
#[stable(feature = "fused", since = "1.26.0")]
242254
impl<A, B> FusedIterator for Chain<A, B>
243255
where

src/libcore/mem/mod.rs

+6-1
Original file line numberDiff line numberDiff line change
@@ -924,7 +924,12 @@ pub fn drop<T>(_x: T) {}
924924
#[inline]
925925
#[stable(feature = "rust1", since = "1.0.0")]
926926
pub unsafe fn transmute_copy<T, U>(src: &T) -> U {
927-
ptr::read_unaligned(src as *const T as *const U)
927+
// If U has a higher alignment requirement, src may not be suitably aligned.
928+
if align_of::<U>() > align_of::<T>() {
929+
ptr::read_unaligned(src as *const T as *const U)
930+
} else {
931+
ptr::read(src as *const T as *const U)
932+
}
928933
}
929934

930935
/// Opaque type representing the discriminant of an enum.

src/libcore/tests/iter.rs

+39-25
Original file line numberDiff line numberDiff line change
@@ -207,50 +207,64 @@ fn test_iterator_chain_find() {
207207
assert_eq!(iter.next(), None);
208208
}
209209

210-
#[test]
211-
fn test_iterator_chain_size_hint() {
212-
struct Iter {
213-
is_empty: bool,
214-
}
210+
struct Toggle {
211+
is_empty: bool,
212+
}
215213

216-
impl Iterator for Iter {
217-
type Item = ();
214+
impl Iterator for Toggle {
215+
type Item = ();
218216

219-
// alternates between `None` and `Some(())`
220-
fn next(&mut self) -> Option<Self::Item> {
221-
if self.is_empty {
222-
self.is_empty = false;
223-
None
224-
} else {
225-
self.is_empty = true;
226-
Some(())
227-
}
217+
// alternates between `None` and `Some(())`
218+
fn next(&mut self) -> Option<Self::Item> {
219+
if self.is_empty {
220+
self.is_empty = false;
221+
None
222+
} else {
223+
self.is_empty = true;
224+
Some(())
228225
}
226+
}
229227

230-
fn size_hint(&self) -> (usize, Option<usize>) {
231-
if self.is_empty { (0, Some(0)) } else { (1, Some(1)) }
232-
}
228+
fn size_hint(&self) -> (usize, Option<usize>) {
229+
if self.is_empty { (0, Some(0)) } else { (1, Some(1)) }
233230
}
231+
}
234232

235-
impl DoubleEndedIterator for Iter {
236-
fn next_back(&mut self) -> Option<Self::Item> {
237-
self.next()
238-
}
233+
impl DoubleEndedIterator for Toggle {
234+
fn next_back(&mut self) -> Option<Self::Item> {
235+
self.next()
239236
}
237+
}
240238

239+
#[test]
240+
fn test_iterator_chain_size_hint() {
241241
// this chains an iterator of length 0 with an iterator of length 1,
242242
// so after calling `.next()` once, the iterator is empty and the
243243
// state is `ChainState::Back`. `.size_hint()` should now disregard
244244
// the size hint of the left iterator
245-
let mut iter = Iter { is_empty: true }.chain(once(()));
245+
let mut iter = Toggle { is_empty: true }.chain(once(()));
246246
assert_eq!(iter.next(), Some(()));
247247
assert_eq!(iter.size_hint(), (0, Some(0)));
248248

249-
let mut iter = once(()).chain(Iter { is_empty: true });
249+
let mut iter = once(()).chain(Toggle { is_empty: true });
250250
assert_eq!(iter.next_back(), Some(()));
251251
assert_eq!(iter.size_hint(), (0, Some(0)));
252252
}
253253

254+
#[test]
255+
fn test_iterator_chain_unfused() {
256+
// Chain shouldn't be fused in its second iterator, depending on direction
257+
let mut iter = NonFused::new(empty()).chain(Toggle { is_empty: true });
258+
iter.next().unwrap_none();
259+
iter.next().unwrap();
260+
iter.next().unwrap_none();
261+
262+
let mut iter = Toggle { is_empty: true }.chain(NonFused::new(empty()));
263+
iter.next_back().unwrap_none();
264+
iter.next_back().unwrap();
265+
iter.next_back().unwrap_none();
266+
}
267+
254268
#[test]
255269
fn test_zip_nth() {
256270
let xs = [0, 1, 2, 4, 5];

src/libcore/tests/lib.rs

+1
Original file line numberDiff line numberDiff line change
@@ -42,6 +42,7 @@
4242
#![feature(unwrap_infallible)]
4343
#![feature(leading_trailing_ones)]
4444
#![feature(const_forget)]
45+
#![feature(option_unwrap_none)]
4546

4647
extern crate test;
4748

src/librustc_lint/lib.rs

+3-1
Original file line numberDiff line numberDiff line change
@@ -61,7 +61,8 @@ use rustc_middle::ty::query::Providers;
6161
use rustc_middle::ty::TyCtxt;
6262
use rustc_session::lint::builtin::{
6363
BARE_TRAIT_OBJECTS, ELIDED_LIFETIMES_IN_PATHS, EXPLICIT_OUTLIVES_REQUIREMENTS,
64-
INTRA_DOC_LINK_RESOLUTION_FAILURE, MISSING_DOC_CODE_EXAMPLES, PRIVATE_DOC_TESTS,
64+
INTRA_DOC_LINK_RESOLUTION_FAILURE, INVALID_CODEBLOCK_ATTRIBUTE, MISSING_DOC_CODE_EXAMPLES,
65+
PRIVATE_DOC_TESTS,
6566
};
6667
use rustc_span::Span;
6768

@@ -299,6 +300,7 @@ fn register_builtins(store: &mut LintStore, no_interleave_lints: bool) {
299300
add_lint_group!(
300301
"rustdoc",
301302
INTRA_DOC_LINK_RESOLUTION_FAILURE,
303+
INVALID_CODEBLOCK_ATTRIBUTE,
302304
MISSING_DOC_CODE_EXAMPLES,
303305
PRIVATE_DOC_TESTS
304306
);

src/librustc_middle/mir/interpret/value.rs

+5
Original file line numberDiff line numberDiff line change
@@ -188,6 +188,11 @@ impl<'tcx, Tag> Scalar<Tag> {
188188
}
189189
}
190190

191+
#[inline]
192+
pub fn null_ptr(cx: &impl HasDataLayout) -> Self {
193+
Scalar::Raw { data: 0, size: cx.data_layout().pointer_size.bytes() as u8 }
194+
}
195+
191196
#[inline]
192197
pub fn zst() -> Self {
193198
Scalar::Raw { data: 0, size: 0 }

0 commit comments

Comments
 (0)