Skip to content

Commit

Permalink
Updated release notes and README for the next release.
Browse files Browse the repository at this point in the history
  • Loading branch information
orium committed Sep 11, 2019
1 parent 9f932a9 commit eb7a94d
Show file tree
Hide file tree
Showing 3 changed files with 123 additions and 8 deletions.
59 changes: 55 additions & 4 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -21,7 +21,7 @@ To use rpds add the following to your `Cargo.toml`:
rpds = "<version>"
```

## Data Structures
## Data structures

This crate offers the following data structures:

Expand Down Expand Up @@ -238,9 +238,9 @@ assert!(!set_positive.contains(&"zero"));
assert_eq!(set_positive.first(), Some(&"one"));
```

## Other Features
## Other features

### Mutable Methods
### Mutable methods

When you change a data structure you often do not need its previous versions. For those cases
rpds offers you mutable methods which are generally faster:
Expand All @@ -257,7 +257,7 @@ let set_0_1 = set.clone();
let set_0_1_2 = set.insert("two");
```

### Initialization Macros
### Initialization macros

There are convenient initialization macros for all data structures:

Expand All @@ -270,6 +270,57 @@ let map = ht_map!["orange" => "orange", "banana" => "yellow"];

Check the documentation for initialization macros of other data structures.

### Thread safety

All data structures in this crate can be shared between threads, but that is an opt-in ability.
This is because there is a performance cost to make data structures thread safe. That cost
is worth avoiding when you are not actually sharing them between threads.

Of course if you try to share a rpds data structure across different threads you can count on
the rust compiler to ensure that it is safe to do so. If you are using the version of the data
structure that is not thread safe you will get a compile-time error.

To create a thread-safe version of any data structure you can create it with `new_sync()`:

```rust
let vec = Vector::new_sync()
.push_back(42);
```

Or use the `_sync` variant of the initialization macro:

```rust
let vec = vector_sync!(42);
```

#### Further details

Internally the data structures in this crate maintain a lot of reference-counting pointers.
These pointers are used both for links between the internal nodes of the data structure as well
as for the values it stores.

There are two implementations of reference-counting pointers in the standard library:
[`Rc`](https://doc.rust-lang.org/std/rc/struct.Rc.html) and
[`Arc`](https://doc.rust-lang.org/std/sync/struct.Arc.html). They behave the same way, but
`Arc` allows you to share the data it points to across multiple threads. The downside is that
it is significantly slower to clone and drop than `Rc`, and persistent data structures do a
lot of those operations. In some microbenchmarks with rpds data structure we can see that
using `Rc` instead of `Arc` can make some operations twice as fast! You can see this for
yourself by running `cargo bench`.

To implement this we parameterize the type of reference-counting pointer (`Rc` or `Arc`) as a
type argument of the data structure. We use the [archery](https://github.com/orium/archery/)
crate to do this in a convenient way.

The pointer type can be parameterized like this:

```rust
let vec: Vector<u32, archery::ArcK> = Vector::new_with_ptr_kind();
//
// This will use `Arc` pointers.
// Change it to `archery::RcK` to use a `Rc` pointer.
```

### Serialization

We support serialization through [serde](https://crates.io/crates/serde). To use it
Expand Down
7 changes: 7 additions & 0 deletions release-notes.md
Original file line number Diff line number Diff line change
@@ -1,5 +1,12 @@
# Release notes

## 0.7.0

* Now the shared pointer type of all data structures use can be parameterizable. See the
[Thread safety](./README.md#thread-safety) section in the README for details.
([#7](https://github.com/orium/rpds/issues/7))
* Fix bug where dropping long lists would cause a stack overflow. ([#46](https://github.com/orium/rpds/issues/46))

## 0.6.0

* Implemented `RedBlackTree{Map,Set}::range()` iterator.
Expand Down
65 changes: 61 additions & 4 deletions src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -27,7 +27,7 @@
//! rpds = "<version>"
//! ```
//!
//! ## Data Structures
//! ## Data structures
//!
//! This crate offers the following data structures:
//!
Expand Down Expand Up @@ -244,9 +244,9 @@
//! assert_eq!(set_positive.first(), Some(&"one"));
//! ```
//!
//! ## Other Features
//! ## Other features
//!
//! ### Mutable Methods
//! ### Mutable methods
//!
//! When you change a data structure you often do not need its previous versions. For those cases
//! rpds offers you mutable methods which are generally faster:
Expand All @@ -263,7 +263,7 @@
//! let set_0_1_2 = set.insert("two");
//! ```
//!
//! ### Initialization Macros
//! ### Initialization macros
//!
//! There are convenient initialization macros for all data structures:
//!
Expand All @@ -276,6 +276,63 @@
//!
//! Check the documentation for initialization macros of other data structures.
//!
//! ### Thread safety
//!
//! All data structures in this crate can be shared between threads, but that is an opt-in ability.
//! This is because there is a performance cost to make data structures thread safe. That cost
//! is worth avoiding when you are not actually sharing them between threads.
//!
//! Of course if you try to share a rpds data structure across different threads you can count on
//! the rust compiler to ensure that it is safe to do so. If you are using the version of the data
//! structure that is not thread safe you will get a compile-time error.
//!
//! To create a thread-safe version of any data structure you can create it with `new_sync()`:
//!
//! ```rust
//! # /*DROP_LINE_IN_README*/ use rpds::Vector;
//! # /*DROP_LINE_IN_README*/
//! let vec = Vector::new_sync()
//! .push_back(42);
//! ```
//!
//! Or use the `_sync` variant of the initialization macro:
//!
//! ```rust
//! # /*DROP_LINE_IN_README*/ use rpds::vector_sync;
//! # /*DROP_LINE_IN_README*/
//! let vec = vector_sync!(42);
//! ```
//!
//! #### Further details
//!
//! Internally the data structures in this crate maintain a lot of reference-counting pointers.
//! These pointers are used both for links between the internal nodes of the data structure as well
//! as for the values it stores.
//!
//! There are two implementations of reference-counting pointers in the standard library:
//! [`Rc`](https://doc.rust-lang.org/std/rc/struct.Rc.html) and
//! [`Arc`](https://doc.rust-lang.org/std/sync/struct.Arc.html). They behave the same way, but
//! `Arc` allows you to share the data it points to across multiple threads. The downside is that
//! it is significantly slower to clone and drop than `Rc`, and persistent data structures do a
//! lot of those operations. In some microbenchmarks with rpds data structure we can see that
//! using `Rc` instead of `Arc` can make some operations twice as fast! You can see this for
//! yourself by running `cargo bench`.
//!
//! To implement this we parameterize the type of reference-counting pointer (`Rc` or `Arc`) as a
//! type argument of the data structure. We use the [archery](https://github.com/orium/archery/)
//! crate to do this in a convenient way.
//!
//! The pointer type can be parameterized like this:
//!
//! ```rust
//! # /*DROP_LINE_IN_README*/ use rpds::Vector;
//! # /*DROP_LINE_IN_README*/
//! let vec: Vector<u32, archery::ArcK> = Vector::new_with_ptr_kind();
//! // ↖
//! // This will use `Arc` pointers.
//! // Change it to `archery::RcK` to use a `Rc` pointer.
//! ```
//!
//! ### Serialization
//!
//! We support serialization through [serde](https://crates.io/crates/serde). To use it
Expand Down

0 comments on commit eb7a94d

Please sign in to comment.