Implement skeleton for swapping frames into disk
This commit is contained in:
@@ -63,8 +63,8 @@ userprog_SRC += userprog/tss.c # TSS management.
|
|||||||
|
|
||||||
# Virtual memory code.
|
# Virtual memory code.
|
||||||
vm_SRC += vm/frame.c # Frame table manager.
|
vm_SRC += vm/frame.c # Frame table manager.
|
||||||
|
vm_SRC += vm/page.c # Page table manager.
|
||||||
vm_SRC += devices/swap.c # Swap block manager.
|
vm_SRC += devices/swap.c # Swap block manager.
|
||||||
#vm_SRC = vm/file.c # Some other file.
|
|
||||||
|
|
||||||
# Filesystem code.
|
# Filesystem code.
|
||||||
filesys_SRC = filesys/filesys.c # Filesystem core.
|
filesys_SRC = filesys/filesys.c # Filesystem core.
|
||||||
|
|||||||
@@ -116,7 +116,7 @@ process_execute (const char *cmd)
|
|||||||
return tid;
|
return tid;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void *get_usr_kpage (enum palloc_flags flags);
|
static void *get_usr_kpage (enum palloc_flags flags, void *upage);
|
||||||
static void free_usr_kpage (void *kpage);
|
static void free_usr_kpage (void *kpage);
|
||||||
static bool install_page (void *upage, void *kpage, bool writable);
|
static bool install_page (void *upage, void *kpage, bool writable);
|
||||||
|
|
||||||
@@ -257,12 +257,13 @@ process_init_stack (char *cmd_saveptr, void **esp, char *file_name)
|
|||||||
int pages_needed = DIV_CEIL (overflow_bytes, PGSIZE);
|
int pages_needed = DIV_CEIL (overflow_bytes, PGSIZE);
|
||||||
|
|
||||||
/* Allocate the pages and map them to the user process. */
|
/* Allocate the pages and map them to the user process. */
|
||||||
|
void *upage;
|
||||||
|
uint8_t *kpage;
|
||||||
for (int i = 1; i < pages_needed + 1; i++)
|
for (int i = 1; i < pages_needed + 1; i++)
|
||||||
{
|
{
|
||||||
uint8_t *kpage = get_usr_kpage (PAL_ZERO);
|
upage = ((uint8_t *) PHYS_BASE) - PGSIZE * (i + 1);
|
||||||
if (!install_page (((uint8_t *) PHYS_BASE) - PGSIZE * (i + 1),
|
kpage = get_usr_kpage (PAL_ZERO, upage);
|
||||||
kpage, true))
|
if (!install_page (upage, kpage, true)) return false;
|
||||||
return false;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -710,7 +711,7 @@ load_segment (struct file *file, off_t ofs, uint8_t *upage,
|
|||||||
if (kpage == NULL){
|
if (kpage == NULL){
|
||||||
|
|
||||||
/* Get a new page of memory. */
|
/* Get a new page of memory. */
|
||||||
kpage = get_usr_kpage (0);
|
kpage = get_usr_kpage (0, upage);
|
||||||
if (kpage == NULL){
|
if (kpage == NULL){
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@@ -753,10 +754,12 @@ setup_stack (void **esp)
|
|||||||
uint8_t *kpage;
|
uint8_t *kpage;
|
||||||
bool success = false;
|
bool success = false;
|
||||||
|
|
||||||
kpage = get_usr_kpage (PAL_ZERO);
|
void *upage = ((uint8_t *) PHYS_BASE) - PGSIZE;
|
||||||
|
|
||||||
|
kpage = get_usr_kpage (PAL_ZERO, upage);
|
||||||
if (kpage != NULL)
|
if (kpage != NULL)
|
||||||
{
|
{
|
||||||
success = install_page (((uint8_t *) PHYS_BASE) - PGSIZE, kpage, true);
|
success = install_page (upage, kpage, true);
|
||||||
if (success)
|
if (success)
|
||||||
*esp = PHYS_BASE;
|
*esp = PHYS_BASE;
|
||||||
else
|
else
|
||||||
@@ -765,14 +768,20 @@ setup_stack (void **esp)
|
|||||||
return success;
|
return success;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Claims a page from the user pool and returns its kernel address,
|
/* Claims a page from the user pool for ownership by the current thread
|
||||||
updating the frame table if VM is enabled. */
|
and returns its kernel address, updating the frame table if VM
|
||||||
|
is enabled. Requires the intended virtual address for where the page
|
||||||
|
will be installed. */
|
||||||
static void *
|
static void *
|
||||||
get_usr_kpage (enum palloc_flags flags)
|
get_usr_kpage (enum palloc_flags flags, void *upage)
|
||||||
{
|
{
|
||||||
void *page;
|
void *page;
|
||||||
#ifdef VM
|
#ifdef VM
|
||||||
page = frame_alloc (flags);
|
struct thread *t = thread_current ();
|
||||||
|
if (pagedir_get_page (t->pagedir, upage) != NULL)
|
||||||
|
return NULL;
|
||||||
|
else
|
||||||
|
page = frame_alloc (flags, upage, t);
|
||||||
#else
|
#else
|
||||||
page = palloc_get_page (flags | PAL_USER);
|
page = palloc_get_page (flags | PAL_USER);
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
@@ -1,10 +1,14 @@
|
|||||||
#include <debug.h>
|
#include <debug.h>
|
||||||
#include <hash.h>
|
#include <hash.h>
|
||||||
#include <list.h>
|
#include <list.h>
|
||||||
|
#include <string.h>
|
||||||
|
|
||||||
#include "frame.h"
|
#include "frame.h"
|
||||||
|
#include "page.h"
|
||||||
#include "threads/malloc.h"
|
#include "threads/malloc.h"
|
||||||
|
#include "threads/vaddr.h"
|
||||||
#include "threads/synch.h"
|
#include "threads/synch.h"
|
||||||
|
#include "devices/swap.h"
|
||||||
|
|
||||||
/* Hash table that maps every active frame's kernel virtual address
|
/* Hash table that maps every active frame's kernel virtual address
|
||||||
to its corresponding 'frame_metadata'.*/
|
to its corresponding 'frame_metadata'.*/
|
||||||
@@ -22,13 +26,14 @@ struct list active_list;
|
|||||||
struct list inactive_list;
|
struct list inactive_list;
|
||||||
|
|
||||||
/* Synchronisation variables. */
|
/* Synchronisation variables. */
|
||||||
/* Ensures mutual exclusion to accessing the 'head' and first element of
|
/* Protects access to the 'inactive' list. */
|
||||||
'inactive_list', which is accessed every time a frame is allocated. */
|
struct lock inactive_lock;
|
||||||
struct lock inactive_head_lock;
|
|
||||||
|
|
||||||
struct frame_metadata
|
struct frame_metadata
|
||||||
{
|
{
|
||||||
void *frame; /* The kernel virtual address holding the frame. */
|
void *frame; /* The kernel virtual address holding the frame. */
|
||||||
|
void *upage; /* The user virtual address pointing to the frame. */
|
||||||
|
struct thread *owner; /* Pointer to the thread that owns the frame. */
|
||||||
struct hash_elem hash_elem; /* Tracks the position of the frame metadata
|
struct hash_elem hash_elem; /* Tracks the position of the frame metadata
|
||||||
within 'frame_table', whose key is the
|
within 'frame_table', whose key is the
|
||||||
kernel virtual address of the frame. */
|
kernel virtual address of the frame. */
|
||||||
@@ -40,6 +45,8 @@ struct frame_metadata
|
|||||||
hash_hash_func frame_metadata_hash;
|
hash_hash_func frame_metadata_hash;
|
||||||
hash_less_func frame_metadata_less;
|
hash_less_func frame_metadata_less;
|
||||||
|
|
||||||
|
static struct frame_metadata *get_victim (void);
|
||||||
|
|
||||||
/* Initialize the frame system by initializing the frame (hash) table with
|
/* Initialize the frame system by initializing the frame (hash) table with
|
||||||
the frame_metadata hashing and comparison functions, as well as initializing
|
the frame_metadata hashing and comparison functions, as well as initializing
|
||||||
the active & inactive lists. Also initializes the system's synchronisation
|
the active & inactive lists. Also initializes the system's synchronisation
|
||||||
@@ -51,7 +58,7 @@ frame_init (void)
|
|||||||
list_init (&active_list);
|
list_init (&active_list);
|
||||||
list_init (&inactive_list);
|
list_init (&inactive_list);
|
||||||
|
|
||||||
lock_init (&inactive_head_lock);
|
lock_init (&inactive_lock);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Attempt to allocate a frame for a user process, either by direct
|
/* Attempt to allocate a frame for a user process, either by direct
|
||||||
@@ -60,25 +67,42 @@ frame_init (void)
|
|||||||
processes is fulled and storing it in swap. If swap is full in
|
processes is fulled and storing it in swap. If swap is full in
|
||||||
the former case, panic the kernel. */
|
the former case, panic the kernel. */
|
||||||
void *
|
void *
|
||||||
frame_alloc (enum palloc_flags flags)
|
frame_alloc (enum palloc_flags flags, void *upage, struct thread *owner)
|
||||||
{
|
{
|
||||||
flags |= PAL_USER;
|
flags |= PAL_USER;
|
||||||
|
|
||||||
void *frame = palloc_get_page (flags);
|
void *frame = palloc_get_page (flags);
|
||||||
|
|
||||||
|
/* If a frame couldn't be allocated we must be out of main memory. Thus,
|
||||||
|
obtain a victim page to replace with our page, and swap the victim
|
||||||
|
into disk. */
|
||||||
if (frame == NULL)
|
if (frame == NULL)
|
||||||
{
|
{
|
||||||
/* TODO: Find victim page to replace, and swap it with this new page. */
|
/* TODO: Deal with race condition wherein a page may be evicted in one
|
||||||
return NULL;
|
thread while it's in the middle of being evicted in another. */
|
||||||
|
struct frame_metadata *victim = get_victim ();
|
||||||
|
if (victim == NULL)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
size_t swap_slot = swap_out (victim->frame);
|
||||||
|
page_set_swap (victim->owner, victim->upage, swap_slot);
|
||||||
|
|
||||||
|
/* If zero flag is set, zero out the victim page. */
|
||||||
|
if (flags & PAL_ZERO)
|
||||||
|
memset (victim->frame, 0, PGSIZE);
|
||||||
|
|
||||||
|
frame = victim->frame;
|
||||||
}
|
}
|
||||||
|
|
||||||
struct frame_metadata *frame_metadata =
|
struct frame_metadata *frame_metadata =
|
||||||
malloc (sizeof (struct frame_metadata));
|
malloc (sizeof (struct frame_metadata));
|
||||||
frame_metadata->frame = frame;
|
frame_metadata->frame = frame;
|
||||||
|
frame_metadata->upage = upage;
|
||||||
|
frame_metadata->owner = owner;
|
||||||
|
|
||||||
/* Newly faulted pages begin at the head of the inactive list. */
|
/* Newly faulted pages begin at the head of the inactive list. */
|
||||||
lock_acquire (&inactive_head_lock);
|
lock_acquire (&inactive_lock);
|
||||||
list_push_front (&inactive_list, &frame_metadata->list_elem);
|
list_push_front (&inactive_list, &frame_metadata->list_elem);
|
||||||
lock_release (&inactive_head_lock);
|
lock_release (&inactive_lock);
|
||||||
|
|
||||||
/* Finally, insert frame metadata within the frame table, with the key as its
|
/* Finally, insert frame metadata within the frame table, with the key as its
|
||||||
allocated kernel address. */
|
allocated kernel address. */
|
||||||
@@ -109,6 +133,24 @@ frame_free (void *frame)
|
|||||||
palloc_free_page (frame);
|
palloc_free_page (frame);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Obtain a pointer to the metadata of the frame we should evict next. */
|
||||||
|
static struct frame_metadata *
|
||||||
|
get_victim (void)
|
||||||
|
{
|
||||||
|
lock_acquire (&inactive_lock);
|
||||||
|
if (list_empty (&inactive_list))
|
||||||
|
{
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
struct list_elem *victim_elem = list_pop_back (&inactive_list);
|
||||||
|
lock_release (&inactive_lock);
|
||||||
|
|
||||||
|
return list_entry (victim_elem, struct frame_metadata, list_elem);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/* Hash function for frame metadata, used for storing entries in the
|
/* Hash function for frame metadata, used for storing entries in the
|
||||||
frame table. */
|
frame table. */
|
||||||
unsigned
|
unsigned
|
||||||
|
|||||||
@@ -1,10 +1,11 @@
|
|||||||
#ifndef VM_FRAME_H
|
#ifndef VM_FRAME_H
|
||||||
#define VM_FRAME_H
|
#define VM_FRAME_H
|
||||||
|
|
||||||
|
#include "threads/thread.h"
|
||||||
#include "threads/palloc.h"
|
#include "threads/palloc.h"
|
||||||
|
|
||||||
void frame_init (void);
|
void frame_init (void);
|
||||||
void *frame_alloc (enum palloc_flags);
|
void *frame_alloc (enum palloc_flags, void *, struct thread *);
|
||||||
void frame_free (void *frame);
|
void frame_free (void *frame);
|
||||||
|
|
||||||
#endif /* vm/frame.h */
|
#endif /* vm/frame.h */
|
||||||
|
|||||||
20
src/vm/page.c
Normal file
20
src/vm/page.c
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
#include "page.h"
|
||||||
|
|
||||||
|
/* Updates the 'owner' thread's page table entry for virtual address 'upage'
|
||||||
|
to have a present bit of 0 and stores the specified swap slot value in the
|
||||||
|
entry for later retrieval from disk. */
|
||||||
|
void
|
||||||
|
page_set_swap (struct thread *owner, void *upage, size_t swap_slot)
|
||||||
|
{
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Given that the page with user address 'upage' owned by 'owner' is flagged
|
||||||
|
to be in the swap disk via the owner's page table, returns its stored
|
||||||
|
swap slot. Otherwise panics the kernel. */
|
||||||
|
size_t
|
||||||
|
page_get_swap (struct thread *owner, void *upage)
|
||||||
|
{
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
9
src/vm/page.h
Normal file
9
src/vm/page.h
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
#ifndef VM_PAGE_H
|
||||||
|
#define VM_PAGE_H
|
||||||
|
|
||||||
|
#include "threads/thread.h"
|
||||||
|
|
||||||
|
void page_set_swap (struct thread *, void *, size_t);
|
||||||
|
size_t page_get_swap (struct thread *, void *);
|
||||||
|
|
||||||
|
#endif /* vm/frame.h */
|
||||||
Reference in New Issue
Block a user