Skip to content

Latest commit

 

History

History
45 lines (32 loc) · 2.06 KB

README.md

File metadata and controls

45 lines (32 loc) · 2.06 KB

Example programs for Perennial

CI

dir - single directory

Manages a fixed number of inodes, which support block appends and reads.

Illustrates the shadow update pattern. To append to an inode, we allocate a block, update it with full ownership, and then atomically install. However, while updating we still have a crash obligation to the allocator.

Illustrates modularity. We're able to structure this into three components: an allocator, an inode, and a directory. The inode uses the allocator but does not manage it (it does not restore the allocator, and shares it).

  • The allocator is an in-memory structure that serializes access to the free space. It has to be restored during recovery by figuring out what has been allocated.
  • The inode is a durable, append-only structure. It calls into an allocator which is shared among all the inodes.
  • The directory composes multiple inodes and an allocator.
  • The single-inode package is a simple client of the inode; as an alternative to the directory, it only has a single inode.

replicated block

Self-contained example at replicated_block. Intended to resemble the Perennial replicated disk example.

Toy example that replicates a single logical disk block across two blocks. Reads can pick which replica to use, while writes update both. For consistency, we lock the disk addresses and recovery by syncing. This requires a crash lock, and its crash invariant allows the addresses to be out-of-sync.

This is a modular proof that can be instantiated by the caller for multiple addresses, building a replicated disk out of tiny pieces.

Note that this finally takes the 6.826 example and makes it horizontally modular, vertically composable, concurrent, and written in Go.

toy example

A fragment of code to illustrate transferring a disk block from one thread to another with a crash invariant.