Expand description

This library provides wrapper types that permit sending non Send types to other threads and use runtime checks to ensure safety.

It provides three types: Fragile<T> and Sticky<T> which are similar in nature but have different behaviors with regards to how destructors are executed and the extra SemiSticky<T> type which uses Sticky<T> if the value has a destructor and Fragile<T> if it does not.

Both types wrap a value and provide a Send bound. Neither of the types permit access to the enclosed value unless the thread that wrapped the value is attempting to access it. The difference between the two types starts playing a role once destructors are involved.

A Fragile<T> will actually send the T from thread to thread but will only permit the original thread to invoke the destructor. If the value gets dropped in a different thread, the destructor will panic.

A Sticky<T> on the other hand does not actually send the T around but keeps it stored in the original thread’s thread local storage. If it gets dropped in the originating thread it gets cleaned up immediately, otherwise it leaks until the thread shuts down naturally.

Example usage

use std::thread;
use fragile::Fragile;

// creating and using a fragile object in the same thread works
let val = Fragile::new(true);
assert_eq!(*val.get(), true);
assert!(val.try_get().is_ok());

// once send to another thread it stops working
thread::spawn(move || {
    assert!(val.try_get().is_err());
}).join()
    .unwrap();

Why?

Most of the time trying to use this crate is going to indicate some code smell. But there are situations where this is useful. For instance you might have a bunch of non Send types but want to work with a Send error type. In that case the non sendable extra information can be contained within the error and in cases where the error did not cross a thread boundary yet extra information can be obtained.

Features

By default the crate has no dependencies. Optionally the slab feature can be enabled which optimizes the internal storage of the Sticky type to make it use a slab instead.

Structs

A Fragile<T> wraps a non sendable T to be safely send to other threads.

Returned when borrowing fails.

A SemiSticky<T> keeps a value T stored in a thread if it has a drop.

A Sticky<T> keeps a value T stored in a thread.