Skip to content

Commit c9fd3d4

Browse files
committed
Make LinkedList and its read-only iterators covariant
CC #30642
1 parent 8aee582 commit c9fd3d4

File tree

2 files changed

+15
-7
lines changed

2 files changed

+15
-7
lines changed

src/libcollections/lib.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -46,7 +46,7 @@
4646
#![feature(num_bits_bytes)]
4747
#![feature(oom)]
4848
#![feature(pattern)]
49-
#![feature(ptr_as_ref)]
49+
#![feature(shared)]
5050
#![feature(slice_bytes)]
5151
#![feature(slice_patterns)]
5252
#![feature(staged_api)]

src/libcollections/linked_list.rs

+14-6
Original file line numberDiff line numberDiff line change
@@ -27,7 +27,7 @@ use core::fmt;
2727
use core::hash::{Hasher, Hash};
2828
use core::iter::FromIterator;
2929
use core::mem;
30-
use core::ptr;
30+
use core::ptr::Shared;
3131

3232
/// A doubly-linked list.
3333
#[stable(feature = "rust1", since = "1.0.0")]
@@ -40,7 +40,7 @@ pub struct LinkedList<T> {
4040
type Link<T> = Option<Box<Node<T>>>;
4141

4242
struct Rawlink<T> {
43-
p: *mut T,
43+
p: Option<Shared<T>>,
4444
}
4545

4646
impl<T> Copy for Rawlink<T> {}
@@ -93,12 +93,12 @@ pub struct IntoIter<T> {
9393
impl<T> Rawlink<T> {
9494
/// Like Option::None for Rawlink
9595
fn none() -> Rawlink<T> {
96-
Rawlink { p: ptr::null_mut() }
96+
Rawlink { p: None }
9797
}
9898

9999
/// Like Option::Some for Rawlink
100100
fn some(n: &mut T) -> Rawlink<T> {
101-
Rawlink { p: n }
101+
unsafe { Rawlink { p: Some(Shared::new(n)) } }
102102
}
103103

104104
/// Convert the `Rawlink` into an Option value
@@ -108,7 +108,7 @@ impl<T> Rawlink<T> {
108108
/// - Dereference of raw pointer.
109109
/// - Returns reference of arbitrary lifetime.
110110
unsafe fn resolve<'a>(&self) -> Option<&'a T> {
111-
self.p.as_ref()
111+
self.p.map(|p| &**p)
112112
}
113113

114114
/// Convert the `Rawlink` into an Option value
@@ -118,7 +118,7 @@ impl<T> Rawlink<T> {
118118
/// - Dereference of raw pointer.
119119
/// - Returns reference of arbitrary lifetime.
120120
unsafe fn resolve_mut<'a>(&mut self) -> Option<&'a mut T> {
121-
self.p.as_mut()
121+
self.p.map(|p| &mut **p)
122122
}
123123

124124
/// Return the `Rawlink` and replace with `Rawlink::none()`
@@ -984,6 +984,14 @@ impl<A: Hash> Hash for LinkedList<A> {
984984
}
985985
}
986986

987+
// Ensure that `LinkedList` and its read-only iterators are covariant in their type parameters.
988+
#[allow(dead_code)]
989+
fn assert_covariance() {
990+
fn a<'a>(x: LinkedList<&'static str>) -> LinkedList<&'a str> { x }
991+
fn b<'i, 'a>(x: Iter<'i, &'static str>) -> Iter<'i, &'a str> { x }
992+
fn c<'a>(x: IntoIter<&'static str>) -> IntoIter<&'a str> { x }
993+
}
994+
987995
#[cfg(test)]
988996
mod tests {
989997
use std::clone::Clone;

0 commit comments

Comments
 (0)