- Gavin Song [email protected]
- Jun Tan [email protected]
- Pengcheng Liu [email protected]
/* Buffer Cache */
void *cache_base; /* Points to the base of the buffer cache. */
size_t clock_hand; /* Used for clock replacement. */
struct bitmap *refbits; /* Reference bits for clock replacement. */
struct keyval *hash_table[64]; /* Maps sector indices to cache entries. */
struct lock cache_lock; /* Acquire before accessing cache metadata. */
struct cond cache_queue; /* Block when all cache entries are in use. */
uint32_t use_cnt[64]; /* Number of processes that are "using" each entry. */
struct keyval {
uint32_t key,
uint32_t val,
struct keyval *next,
}
void *get_cache_block (block_sector_t sector_idx);
void release_cache_block (block_sector_t sector_idx);
We will allocate 8 contiguous pages of memory to the buffer cache, using palloc_get_multiple ()
. The pointer cache_base
will be set to the base of this region, such that the start of the *(i+1)*th cache entry is located at cache_base + BLOCK_SECTOR_SIZE * i
.
We will use a hashmap to map sectors to entries in the buffer cache. We will use the six least significant bits in the sector index for the hash.
get_cache_block ()
starts by acquiring the cache_lock
and then looking up the sector in the hash table. If the sector cannot be found in the hash table, we will use the clock algorithm to select a block for eviction. Our clock algorithm will ignore blocks that are currently "in use" (which are tracked by use_cnt
). If there are currently no blocks that are not "in use", we will wait in the cache_queue
instead of running the clock algorithm. When the thread unblocks, it repeats from the step following the acquisition of cache_lock
.
We put a lock on the buffer cache, so that the full set of actions required to "get" or "release" a cache block has to finish before moving on to the next set of actions. This deals with numerous synchronization issues. For example, when a block is found in the hash table, we expect it to not be evicted before we update the use bit and the ref bit. Conversely, when a block is not found in the hash table, we expect it not to be added to the cache before we add it to the cache ourselves, and we expect the metadata to reflect that. Also, the clock algorithm should execute once at a time.
In get_cache_block ()
, the "use count" corresponding to the returned block is incremented. We have a separate function release_cache_block ()
for decrementing this value, as well as signaling cache_queue
if this causes the count to hit zero. The purpose of the use count is to ensure that the block is not evicted from the cache before the kernel is done with it, while still allowing other threads to simultaneously access the cache block.
We use a simple hashmap to improve lookup speeds. We use the six least significant bytes of the sector index to avoid collisions due to spatial locality. It's also slightly faster using a masking operation. We have 64 entries in the hash table because it is a power of two and gives us a load factor of 1... which is probably good enough.
Every entry in the hash table is a linked list of key-value pairs. However, we don't use the linked list implementation provided by Pintos - for three reasons. One, we don't expect these lists to be very long. In fact, we expect them to contain one entry on average. Since every Pintos linked list has a head and a tail node, this would make each list two or three times larger than necessary. Two, a list_elem
needs to be embedded inside of another struct in order to be associated with actual data. Since we would have to create a separate struct anyways, it makes little sense to include a list_elem
when we could accomplish the same thing with a pointer to the next key-value pair. And finally, three, we only need to be able to traverse the list forwards, one element at a time, so we have no need for the additional functionality provided by list.h
.
Part of the reason we have a "use count" per cache entry is to allow several operations to access the entry simultaneously. Are simultaneous operations on the same sector allowed? If not, does this mean we have to set up a lock for every cache entry?
struct inode_disk {
...
block_sector_t direct[122]; /* Direct pointers. */
block_sector_t indirect; /* Indirect pointer. */
block_sector_t doubly_indirect; /* Doubly indirect pointer. */
unsigned magic; /* Note: magic has a different offset now. */
uint8_t unused[3];
};
struct inode {
/* REMOVE struct inode_disk data. */
}
static struct lock *free_map_lock; /* Lock for free map operations. */
/* Allocates CNT sectors from the free map and stores each
of them in sectors array. Returns true if successful,
false if not enough consecutive sectors were available
or if the free_map file could not be written. */
bool free_map_allocate_nc (size_t cnt, block_sector_t *sectors);
The data
member of the inode
struct will be removed. Instead, the data in inode_disk
will always be accessed through the buffer cache. The code for that will look something like this:
struct inode_disk *data = get_cache_block (inode->sector);
/* Read or write to data here. */
release_cache_block (inode->sector);
Currently, free_map_allocate ()
to allocate blocks for inodes, failing if it cannot find a single continuous region large enough to hold all of the blocks. This makes us susceptible to external fragmentation, so we would like to be able to allocate nonconsecutive blocks. We will add a function free_map_allocate_nc ()
to accomplish this. First, it will count the number of free blocks available, failing early if there are not enough. If successful, it will allocate cnt
blocks and store their sector indices as an array.
When writing past the EOF of a file in inode_write_at ()
, we need to allocate new blocks on the fly. We also need to modify inode_create ()
to accomodate to the new inode structure. These will both use free_map_allocate_nc ()
.
We're putting a lock on the freemap. Using free_map_allocate_nc ()
instead of repeatedly allocating single blocks of memory will ensure that either all blocks are allocated or none at all.
We packed inode_disk
with direct pointers to maximize the benefit of faster accesses for short files. This was an arbitrary decision otherwise.
struct inode_disk {
block_sector_t parent_dir; /* inode_disk sector of the parent directory. */
bool is_dir; /* True if this file is a directory. */
uint32_t num_files; /* The number of subdirectories or files. */
...
};
struct thread {
...
#ifdef USERPROG
struct inode *cwd; /* The current working directory. */
...
};
/* Added a third parameter. */
bool inode_create (block_sector_t, off_t, bool is_dir);
/* Added a fourth parameter. */
bool filesys_create (const char *name, off_t initial_size, bool is_dir);
/* Stores the inode corresponding to the path within INODE.
Returns false if the path is invalid. */
bool follow_path (const char *path, struct inode **inode);
Note that filesys_create()
and inode_create()
each take an additional parameter is_dir
, which is True if the file being created is a directory. This is to correctly set the is_dir
member of the inode_disk
structure on creation. We are going to modify dir_open ()
to fail when it tries to open a non-directory file.
The crux of our code will lie in our follow_path ()
function. We will use the get_next_part()
function (provided in the spec) to iterate through the names in the path. If the path starts with a /
, we will assume it is an absolute path and begin traversing from the root directory. Otherwise, we will begin traversing at the current process' cwd
instead.
When we encounter a ..
in the path, then we traverse to the parent directory, whose location is stored in the parent_dir
member of inode_disk
. Otherwise, we will use dir_lookup ()
on the name. If this fails, we fail. If it succeeds, we run dir_open ()
on the returned inode. This should fail if the file is not a directory.
Finally, we're going to modify most of the functions in filesys.c
to use follow_path ()
instead of dir_open_root ()
.
We decided to disallow removing files or directories for which there is an inode
with an open_cnt
larger than 0. Also, we decided to treat each thread as an opener of its current working directory. This prevents current working directories from being deleted. Furthermore, we decided to disallow removing directories that are not empty.
We added a num_files
member to the inode_disk
structure in order to provide an easy way to determine whether a directory was empty and safe to remove.
One way to implement write behind is to have the timer interrupt write dirty blocks to disk every 30 seconds. Each block in the cache would have an associated lock, which the timer interrupt would acquire before writing the block to disk, to prevent user programs from overwriting the data part way through.
One way to implement read ahead is to use ide.c found in the devices folder. Create a list called read_ahead in the kernel that contains pointers to the blocks to be read in the future. Inode_read will populate this list whenever the length of text left to read is greater than the size of the current block it is reading. Then, in the interrupt_handler in ide.c, check the read_ahead list, and load the blocks into the cache buffer. This way, future blocks can be read into the cache buffer asynchronously.