|
18 | 18 |
|
19 | 19 | use clone::Clone;
|
20 | 20 | use intrinsics;
|
21 |
| -use ops::Deref; |
| 21 | +use ops::{CoerceUnsized, Deref}; |
22 | 22 | use fmt;
|
23 | 23 | use hash;
|
24 | 24 | use option::Option::{self, Some, None};
|
25 |
| -use marker::{PhantomData, Send, Sized, Sync}; |
| 25 | +use marker::{Copy, PhantomData, Send, Sized, Sync, Unsize}; |
26 | 26 | use mem;
|
27 | 27 | use nonzero::NonZero;
|
28 | 28 |
|
@@ -532,3 +532,68 @@ impl<T> fmt::Pointer for Unique<T> {
|
532 | 532 | fmt::Pointer::fmt(&*self.pointer, f)
|
533 | 533 | }
|
534 | 534 | }
|
| 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 | +} |
0 commit comments