Skip to content

Commit d6bd8d8

Browse files
committed
Add Shared pointer and have {Arc, Rc} use it
This change has two consequences: 1. It makes `Arc<T>` and `Rc<T>` covariant in `T`. 2. It causes the compiler to reject code that was unsound with respect to dropck. See compile-fail/issue-29106.rs for an example of code that no longer compiles. Because of this, this is a [breaking-change]. Fixes #29037. Fixes #29106.
1 parent 05cfd72 commit d6bd8d8

File tree

8 files changed

+148
-12
lines changed

8 files changed

+148
-12
lines changed

src/liballoc/arc.rs

+6-5
Original file line numberDiff line numberDiff line change
@@ -79,9 +79,8 @@ use core::cmp::Ordering;
7979
use core::mem::{align_of_val, size_of_val};
8080
use core::intrinsics::{drop_in_place, abort};
8181
use core::mem;
82-
use core::nonzero::NonZero;
8382
use core::ops::{Deref, CoerceUnsized};
84-
use core::ptr;
83+
use core::ptr::{self, Shared};
8584
use core::marker::Unsize;
8685
use core::hash::{Hash, Hasher};
8786
use core::{usize, isize};
@@ -124,12 +123,13 @@ const MAX_REFCOUNT: usize = (isize::MAX) as usize;
124123
pub struct Arc<T: ?Sized> {
125124
// FIXME #12808: strange name to try to avoid interfering with
126125
// field accesses of the contained type via Deref
127-
_ptr: NonZero<*mut ArcInner<T>>,
126+
_ptr: Shared<ArcInner<T>>,
128127
}
129128

130129
unsafe impl<T: ?Sized + Sync + Send> Send for Arc<T> { }
131130
unsafe impl<T: ?Sized + Sync + Send> Sync for Arc<T> { }
132131

132+
#[cfg(not(stage0))] // remove cfg after new snapshot
133133
impl<T: ?Sized + Unsize<U>, U: ?Sized> CoerceUnsized<Arc<U>> for Arc<T> {}
134134

135135
/// A weak pointer to an `Arc`.
@@ -141,12 +141,13 @@ impl<T: ?Sized + Unsize<U>, U: ?Sized> CoerceUnsized<Arc<U>> for Arc<T> {}
141141
pub struct Weak<T: ?Sized> {
142142
// FIXME #12808: strange name to try to avoid interfering with
143143
// field accesses of the contained type via Deref
144-
_ptr: NonZero<*mut ArcInner<T>>,
144+
_ptr: Shared<ArcInner<T>>,
145145
}
146146

147147
unsafe impl<T: ?Sized + Sync + Send> Send for Weak<T> { }
148148
unsafe impl<T: ?Sized + Sync + Send> Sync for Weak<T> { }
149149

150+
#[cfg(not(stage0))] // remove cfg after new snapshot
150151
impl<T: ?Sized + Unsize<U>, U: ?Sized> CoerceUnsized<Weak<U>> for Weak<T> {}
151152

152153
#[stable(feature = "rust1", since = "1.0.0")]
@@ -190,7 +191,7 @@ impl<T> Arc<T> {
190191
weak: atomic::AtomicUsize::new(1),
191192
data: data,
192193
};
193-
Arc { _ptr: unsafe { NonZero::new(Box::into_raw(x)) } }
194+
Arc { _ptr: unsafe { Shared::new(Box::into_raw(x)) } }
194195
}
195196

196197
/// Unwraps the contained value if the `Arc<T>` has only one strong reference.

src/liballoc/lib.rs

+1
Original file line numberDiff line numberDiff line change
@@ -90,6 +90,7 @@
9090
#![feature(placement_in_syntax)]
9191
#![feature(placement_new_protocol)]
9292
#![feature(raw)]
93+
#![feature(shared)]
9394
#![feature(staged_api)]
9495
#![feature(unboxed_closures)]
9596
#![feature(unique)]

src/liballoc/rc.rs

+6-5
Original file line numberDiff line numberDiff line change
@@ -163,9 +163,8 @@ use core::hash::{Hasher, Hash};
163163
use core::intrinsics::{assume, drop_in_place, abort};
164164
use core::marker::{self, Unsize};
165165
use core::mem::{self, align_of_val, size_of_val, forget};
166-
use core::nonzero::NonZero;
167166
use core::ops::{CoerceUnsized, Deref};
168-
use core::ptr;
167+
use core::ptr::{self, Shared};
169168

170169
use heap::deallocate;
171170

@@ -184,12 +183,13 @@ struct RcBox<T: ?Sized> {
184183
pub struct Rc<T: ?Sized> {
185184
// FIXME #12808: strange names to try to avoid interfering with field
186185
// accesses of the contained type via Deref
187-
_ptr: NonZero<*mut RcBox<T>>,
186+
_ptr: Shared<RcBox<T>>,
188187
}
189188

190189
impl<T: ?Sized> !marker::Send for Rc<T> {}
191190
impl<T: ?Sized> !marker::Sync for Rc<T> {}
192191

192+
#[cfg(not(stage0))] // remove cfg after new snapshot
193193
impl<T: ?Sized+Unsize<U>, U: ?Sized> CoerceUnsized<Rc<U>> for Rc<T> {}
194194

195195
impl<T> Rc<T> {
@@ -210,7 +210,7 @@ impl<T> Rc<T> {
210210
// pointers, which ensures that the weak destructor never frees
211211
// the allocation while the strong destructor is running, even
212212
// if the weak pointer is stored inside the strong one.
213-
_ptr: NonZero::new(Box::into_raw(box RcBox {
213+
_ptr: Shared::new(Box::into_raw(box RcBox {
214214
strong: Cell::new(1),
215215
weak: Cell::new(1),
216216
value: value,
@@ -712,12 +712,13 @@ impl<T> fmt::Pointer for Rc<T> {
712712
pub struct Weak<T: ?Sized> {
713713
// FIXME #12808: strange names to try to avoid interfering with
714714
// field accesses of the contained type via Deref
715-
_ptr: NonZero<*mut RcBox<T>>,
715+
_ptr: Shared<RcBox<T>>,
716716
}
717717

718718
impl<T: ?Sized> !marker::Send for Weak<T> {}
719719
impl<T: ?Sized> !marker::Sync for Weak<T> {}
720720

721+
#[cfg(not(stage0))] // remove cfg after new snapshot
721722
impl<T: ?Sized+Unsize<U>, U: ?Sized> CoerceUnsized<Weak<U>> for Weak<T> {}
722723

723724
impl<T: ?Sized> Weak<T> {

src/libcore/ptr.rs

+67-2
Original file line numberDiff line numberDiff line change
@@ -18,11 +18,11 @@
1818

1919
use clone::Clone;
2020
use intrinsics;
21-
use ops::Deref;
21+
use ops::{CoerceUnsized, Deref};
2222
use fmt;
2323
use hash;
2424
use option::Option::{self, Some, None};
25-
use marker::{PhantomData, Send, Sized, Sync};
25+
use marker::{Copy, PhantomData, Send, Sized, Sync, Unsize};
2626
use mem;
2727
use nonzero::NonZero;
2828

@@ -532,3 +532,68 @@ impl<T> fmt::Pointer for Unique<T> {
532532
fmt::Pointer::fmt(&*self.pointer, f)
533533
}
534534
}
535+
536+
/// A wrapper around a raw `*mut T` that indicates that the possessor
537+
/// of this wrapper has shared ownership of the referent. Useful for
538+
/// building abstractions like `Rc<T>` or `Arc<T>`, which internally
539+
/// use raw pointers to manage the memory that they own.
540+
#[unstable(feature = "shared", reason = "needs an RFC to flesh out design",
541+
issue = "0")]
542+
pub struct Shared<T: ?Sized> {
543+
pointer: NonZero<*const T>,
544+
// NOTE: this marker has no consequences for variance, but is necessary
545+
// for dropck to understand that we logically own a `T`.
546+
//
547+
// For details, see:
548+
// https://github.com/rust-lang/rfcs/blob/master/text/0769-sound-generic-drop.md#phantom-data
549+
_marker: PhantomData<T>,
550+
}
551+
552+
/// `Shared` pointers are not `Send` because the data they reference may be aliased.
553+
// NB: This impl is unnecessary, but should provide better error messages.
554+
#[unstable(feature = "shared", issue = "0")]
555+
impl<T: ?Sized> !Send for Shared<T> { }
556+
557+
/// `Shared` pointers are not `Sync` because the data they reference may be aliased.
558+
// NB: This impl is unnecessary, but should provide better error messages.
559+
#[unstable(feature = "shared", issue = "0")]
560+
impl<T: ?Sized> !Sync for Shared<T> { }
561+
562+
#[unstable(feature = "shared", issue = "0")]
563+
impl<T: ?Sized> Shared<T> {
564+
/// Creates a new `Shared`.
565+
pub unsafe fn new(ptr: *mut T) -> Self {
566+
Shared { pointer: NonZero::new(ptr), _marker: PhantomData }
567+
}
568+
}
569+
570+
#[unstable(feature = "shared", issue = "0")]
571+
impl<T: ?Sized> Clone for Shared<T> {
572+
fn clone(&self) -> Self {
573+
*self
574+
}
575+
}
576+
577+
#[unstable(feature = "shared", issue = "0")]
578+
impl<T: ?Sized> Copy for Shared<T> { }
579+
580+
#[cfg(not(stage0))] // remove cfg after new snapshot
581+
#[unstable(feature = "shared", issue = "0")]
582+
impl<T: ?Sized, U: ?Sized> CoerceUnsized<Shared<U>> for Shared<T> where T: Unsize<U> { }
583+
584+
#[unstable(feature = "shared", issue = "0")]
585+
impl<T: ?Sized> Deref for Shared<T> {
586+
type Target = *mut T;
587+
588+
#[inline]
589+
fn deref(&self) -> &*mut T {
590+
unsafe { mem::transmute(&*self.pointer) }
591+
}
592+
}
593+
594+
#[unstable(feature = "shared", issue = "0")]
595+
impl<T> fmt::Pointer for Shared<T> {
596+
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
597+
fmt::Pointer::fmt(&*self.pointer, f)
598+
}
599+
}

src/libstd/sync/mutex.rs

+1
Original file line numberDiff line numberDiff line change
@@ -293,6 +293,7 @@ impl<T: ?Sized> Mutex<T> {
293293

294294
#[stable(feature = "rust1", since = "1.0.0")]
295295
impl<T: ?Sized> Drop for Mutex<T> {
296+
#[unsafe_destructor_blind_to_params]
296297
fn drop(&mut self) {
297298
// This is actually safe b/c we know that there is no further usage of
298299
// this mutex (it's up to the user to arrange for a mutex to get

src/libstd/sync/rwlock.rs

+1
Original file line numberDiff line numberDiff line change
@@ -314,6 +314,7 @@ impl<T: ?Sized> RwLock<T> {
314314

315315
#[stable(feature = "rust1", since = "1.0.0")]
316316
impl<T: ?Sized> Drop for RwLock<T> {
317+
#[unsafe_destructor_blind_to_params]
317318
fn drop(&mut self) {
318319
// IMPORTANT: This code needs to be kept in sync with `RwLock::into_inner`.
319320
unsafe { self.inner.lock.destroy() }

src/test/compile-fail/issue-29106.rs

+34
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,34 @@
1+
// Copyright 2015 The Rust Project Developers. See the COPYRIGHT
2+
// file at the top-level directory of this distribution and at
3+
// http://rust-lang.org/COPYRIGHT.
4+
//
5+
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
6+
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
7+
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
8+
// option. This file may not be copied, modified, or distributed
9+
// except according to those terms.
10+
11+
use std::rc::Rc;
12+
use std::sync::Arc;
13+
14+
struct Foo<'a>(&'a String);
15+
16+
impl<'a> Drop for Foo<'a> {
17+
fn drop(&mut self) {
18+
println!("{:?}", self.0);
19+
}
20+
}
21+
22+
fn main() {
23+
{
24+
let (y, x);
25+
x = "alive".to_string();
26+
y = Arc::new(Foo(&x)); //~ ERROR `x` does not live long enough
27+
}
28+
29+
{
30+
let (y, x);
31+
x = "alive".to_string();
32+
y = Rc::new(Foo(&x)); //~ ERROR `x` does not live long enough
33+
}
34+
}

src/test/run-pass/issue-29037.rs

+32
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,32 @@
1+
// Copyright 2015 The Rust Project Developers. See the COPYRIGHT
2+
// file at the top-level directory of this distribution and at
3+
// http://rust-lang.org/COPYRIGHT.
4+
//
5+
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
6+
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
7+
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
8+
// option. This file may not be copied, modified, or distributed
9+
// except according to those terms.
10+
11+
// This test ensures that each pointer type `P<X>` is covariant in `X`.
12+
13+
use std::rc::Rc;
14+
use std::sync::Arc;
15+
16+
fn a<'r>(x: Box<&'static str>) -> Box<&'r str> {
17+
x
18+
}
19+
20+
fn b<'r, 'w>(x: &'w &'static str) -> &'w &'r str {
21+
x
22+
}
23+
24+
fn c<'r>(x: Arc<&'static str>) -> Arc<&'r str> {
25+
x
26+
}
27+
28+
fn d<'r>(x: Rc<&'static str>) -> Rc<&'r str> {
29+
x
30+
}
31+
32+
fn main() {}

0 commit comments

Comments
 (0)