#include "userprog/syscall.h" #include #include #include "threads/interrupt.h" #include "threads/thread.h" #include "filesys/filesys.h" #include "filesys/off_t.h" #include "threads/init.h" #include "threads/malloc.h" static void syscall_handler (struct intr_frame *); void syscall_init (void) { intr_register_int (0x30, 3, INTR_ON, syscall_handler, "syscall"); } static void halt (void) { power_off(); } static bool create (const char *filename, off_t initial_size) { return filesys_create (filename, initial_size); } static int open (const char *filename) { struct thread *t = thread_current (); if (t->max_files == 0) { t->max_files = 128; printf("[%s] open: allocating %d file descriptors\n", t->name, t->max_files); t->files = (struct fd *) calloc (t->max_files, sizeof (struct fd)); } if (t->num_files == t->max_files) { // not enough space for more file descriptors printf("[%s] open: too many file descriptors\n", t->name); return -1; } struct file *file = filesys_open(filename); if (!file) { printf("[%s] open: couldn't find file %s\n", t->name, filename); return -1; } for (int i = 0; i < t->max_files; i++) { struct fd *fd = t->files + i; if (!fd->active) { fd->active = true; fd->file = file; return i + 2; // 0 and 1 are reserved for stdin and stdout } } free(file); printf("[%s] open: unable to find empty file descriptor\n", t->name); return -1; } static void write (const char *buf) { printf ("printf: %s", buf); } // cast to TYPE and deref argument N from f->esp #define INTR_ESP(N, TYPE) *(TYPE *)(f->esp+(4*(N))) static void syscall_handler (struct intr_frame *f UNUSED) { int syscall_number = INTR_ESP (0, int); switch (syscall_number) { case 0: // halt halt (); break; case 1: // exit break; case 4: // create f->eax = create (INTR_ESP (1, char *), INTR_ESP (2, off_t)); break; case 6: // open f->eax = open (INTR_ESP (1, char *)); break; case 8: // read break; case 9: // write write(INTR_ESP(2, char *)); break; case 12: // close break; default: printf ("kernel: unknown syscall '%d'\n", syscall_number); break; } } #undef INTR_ESP