clone_cell provides a Cell
implementation that works with types whose Clone
implementations are guaranteed not to mutate the Cell
content through the self
reference the clone
method gets. This is done with a PureClone
trait which is a subtrait of Clone
(and a logical supertrait of Copy
). It is only implemented for types with compliant clone
methods.
This crate was largely inspired by the Swift programming language's class properties (fields in Rust speak), which have value semantics. In Swift, class types themselves have reference semantics and are shared. But methods on class types are mutating. The Swift compiler is able to guarantee memory safety in a single-threaded context because copy constructors are not defined by the user. Intead, the compiler automatically generates ones that simply perform a field-wise clone.
In Rust, to enable interior-mutating methods on a struct
stored in an Rc
without the overhead of a RefCell
, we can wrap each of the fields in a Cell
. But Cell::get
is only implemented for types that are Copy
. This is because if the clone
method obtains a reference to the Cell
's interior, it may be able to mutate its state, causing undefined behavior.
The Cell
implementation provided by this crate is intended to be a drop-in replacement of std::cell::Cell
. It can work with types that behave like "values", such as Rc<T>
, Weak<T>
(shared pointers themselves are like values; it is the pointees that behave like references), Option<T: PureClone>
, and more. Some motivating use cases include combining Cell
with clone-on-write or immutable collections to enable efficient mutable sharing of collections.
In this example below, we store an Rc<T>
in a Cell
.
```rust
let x = Cell::new(Rc::new(0));
x.set(Rc::new(42));
assert_eq!(*x.get(), 42); ```
We can also use the provided proc macro to automatically derive PureClone
for types when they only contain fields that are PureClone
.
```rust
struct Foo { x: i32, }
let f = Cell::new(Foo { x: 0 }); f.set(Foo { x: 42 });
assert_eq!(f.get().x, 42); ```
I believe this is sound, because PureClone
is unsafe to implement. This trait is implemented for:
1. Copy
types;
2. Types that perform a shallow clone such as Rc
and Weak
; and
3. Types whose clone
methods are known to be safe, such as those that only contain fields that are PureClone
.
Please let me know if you find any soundness issues!
Pull requests are welcome and any feedback is appreciated!