Skip to content
Snippets Groups Projects
syscall.c 4.46 KiB
Newer Older
rtm's avatar
rtm committed
#include "types.h"
#include "param.h"
#include "mmu.h"
#include "proc.h"
#include "defs.h"
#include "x86.h"
#include "traps.h"
#include "syscall.h"
rtm's avatar
rtm committed
#include "spinlock.h"
rtm's avatar
rtm committed

/*
 * User code makes a system call with INT T_SYSCALL.
 * System call number in %eax.
rtm's avatar
rtm committed
 * Arguments on the stack, from the user call to the C
 * library system call function. The saved user %esp points
 * to a saved frame pointer, a program counter, and then
 * the first argument.
rtm's avatar
rtm committed
 *
 * Return value? Error indication? Errno?
 */

rtm's avatar
rtm committed
/*
 * fetch 32 bits from a user-supplied pointer.
rsc's avatar
 
rsc committed
 * returns 0 if addr was OK, -1 if illegal.
rtm's avatar
rtm committed
 */
int
fetchint(struct proc *p, unsigned addr, int *ip)
{
  *ip = 0;

  if(addr > p->sz - 4)
rsc's avatar
 
rsc committed
    return -1;
rtm's avatar
rtm committed
  memcpy(ip, p->mem + addr, 4);
rsc's avatar
 
rsc committed
  return 0;
rtm's avatar
rtm committed
}

int
fetcharg(int argno, int *ip)
{
  unsigned esp;

  esp = (unsigned) curproc[cpu()]->tf->tf_esp;
  return fetchint(curproc[cpu()], esp + 4 + 4*argno, ip);
rtm's avatar
rtm committed
}

rtm's avatar
rtm committed
int
putint(struct proc *p, unsigned addr, int ip)
{
  if(addr > p->sz - 4)
rsc's avatar
 
rsc committed
    return -1;
rtm's avatar
rtm committed
  memcpy(p->mem + addr, &ip, 4);
rsc's avatar
 
rsc committed
  return 0;
rtm's avatar
rtm committed
}

int
rsc's avatar
rsc committed
sys_pipe(void)
rtm's avatar
rtm committed
{
  struct fd *rfd = 0, *wfd = 0;
  int f1 = -1, f2 = -1;
  struct proc *p = curproc[cpu()];
  unsigned fdp;

  if(pipe_alloc(&rfd, &wfd) < 0)
    goto oops;
  if((f1 = fd_ualloc()) < 0)
    goto oops;
  p->fds[f1] = rfd;
  if((f2 = fd_ualloc()) < 0)
    goto oops;
  p->fds[f2] = wfd;
  if(fetcharg(0, &fdp) < 0)
    goto oops;
  if(putint(p, fdp, f1) < 0)
    goto oops;
  if(putint(p, fdp+4, f2) < 0)
    goto oops;
  return 0;

 oops:
  cprintf("sys_pipe failed\n");
  if(rfd)
    fd_close(rfd);
  if(wfd)
    fd_close(wfd);
  if(f1 >= 0)
    p->fds[f1] = 0;
  if(f2 >= 0)
    p->fds[f2] = 0;
  return -1;
}

int
rsc's avatar
rsc committed
sys_write(void)
rtm's avatar
rtm committed
{
rtm's avatar
rtm committed
  int fd, n, ret;
rtm's avatar
rtm committed
  unsigned addr;
  struct proc *p = curproc[cpu()];

  if(fetcharg(0, &fd) < 0 || fetcharg(1, &addr) < 0 || fetcharg(2, &n) < 0)
    return -1;
  if(fd < 0 || fd >= NOFILE)
    return -1;
  if(p->fds[fd] == 0)
    return -1;
  if(addr + n > p->sz)
    return -1;
rtm's avatar
rtm committed
  ret = fd_write(p->fds[fd], p->mem + addr, n);
  return ret;
rtm's avatar
rtm committed
}

int
rsc's avatar
rsc committed
sys_read(void)
rtm's avatar
rtm committed
{
rtm's avatar
rtm committed
  int fd, n, ret;
rtm's avatar
rtm committed
  unsigned addr;
  struct proc *p = curproc[cpu()];

  if(fetcharg(0, &fd) < 0 || fetcharg(1, &addr) < 0 || fetcharg(2, &n) < 0)
    return -1;
  if(fd < 0 || fd >= NOFILE)
    return -1;
  if(p->fds[fd] == 0)
    return -1;
  if(addr + n > p->sz)
    return -1;
rtm's avatar
rtm committed
  ret = fd_read(p->fds[fd], p->mem + addr, n);
  return ret;
}

int
rsc's avatar
rsc committed
sys_close(void)
rtm's avatar
rtm committed
{
  int fd;
  struct proc *p = curproc[cpu()];

  if(fetcharg(0, &fd) < 0)
    return -1;
  if(fd < 0 || fd >= NOFILE)
    return -1;
  if(p->fds[fd] == 0)
    return -1;
  fd_close(p->fds[fd]);
  p->fds[fd] = 0;
  return 0;
rtm's avatar
rtm committed
}

rtm's avatar
rtm committed
int
rsc's avatar
rsc committed
sys_fork(void)
rtm's avatar
rtm committed
  struct proc *np;

  np = newproc();
rtm's avatar
rtm committed
  if(np){
    np->state = RUNNABLE;
    return np->pid;
  } else {
    return -1;
  }
rtm's avatar
rtm committed
int
rsc's avatar
rsc committed
sys_exit(void)
  return 0;  // not reached
rtm's avatar
rtm committed
int
rsc's avatar
rsc committed
sys_wait(void)
rtm's avatar
rtm committed
{
rtm's avatar
rtm committed

rtm's avatar
rtm committed

  fetcharg(0, &pid);
  return proc_kill(pid);
rtm's avatar
rtm committed
}

rtm's avatar
rtm committed
int
rsc's avatar
rsc committed
sys_cons_putc(void)
rtm's avatar
rtm committed
{
  int c;

  fetcharg(0, &c);
  cons_putc(c & 0xff);
rtm's avatar
rtm committed
  return 0;
rtm's avatar
rtm committed
}

int
sys_block(void)
{
kaashoek's avatar
kaashoek committed
  char buf[512];
  int i, j;
kaashoek's avatar
kaashoek committed
  void *c;

  cprintf("%d: call sys_block\n", cpu());
kaashoek's avatar
kaashoek committed
  for (i = 0; i < 100; i++) {
kaashoek's avatar
kaashoek committed
    if ((c = ide_start_read(i, buf, 1)) == 0) {
kaashoek's avatar
kaashoek committed
      panic("couldn't start read\n");
    }
    cprintf("call sleep\n");
rtm's avatar
rtm committed
    sleep (c, 0);
kaashoek's avatar
kaashoek committed
    if (ide_finish_read(c)) {
kaashoek's avatar
kaashoek committed
      panic("couldn't do read\n");
    }
    cprintf("sector %d: ", i);
    for (j = 0; j < 2; j++)
      cprintf("%x ", buf[j] & 0xff);
    cprintf("\n");
  }
rtm's avatar
rtm committed
int
rsc's avatar
rsc committed
sys_panic(void)
rtm's avatar
rtm committed
{
  struct proc *p = curproc[cpu()];
  unsigned int addr;

  fetcharg(0, &addr);
  panic(p->mem + addr);
  return 0;
}

rtm's avatar
rtm committed
void
rsc's avatar
rsc committed
syscall(void)
rtm's avatar
rtm committed
  struct proc *cp = curproc[cpu()];
  int num = cp->tf->tf_regs.reg_eax;
rtm's avatar
rtm committed
  int ret = -1;
rtm's avatar
rtm committed
  //cprintf("%x sys %d\n", cp, num);
rtm's avatar
rtm committed
  switch(num){
  case SYS_fork:
rtm's avatar
rtm committed
    ret = sys_fork();
rtm's avatar
rtm committed
    break;
  case SYS_exit:
rtm's avatar
rtm committed
    ret = sys_exit();
rtm's avatar
rtm committed
    break;
rtm's avatar
rtm committed
  case SYS_wait:
rtm's avatar
rtm committed
    ret = sys_wait();
rtm's avatar
rtm committed
    break;
rtm's avatar
rtm committed
  case SYS_cons_putc:
rtm's avatar
rtm committed
    ret = sys_cons_putc();
rtm's avatar
rtm committed
    break;
rtm's avatar
rtm committed
  case SYS_pipe:
    ret = sys_pipe();
    break;
  case SYS_write:
    ret = sys_write();
    break;
  case SYS_read:
    ret = sys_read();
    break;
rtm's avatar
rtm committed
  case SYS_close:
    ret = sys_close();
    break;
  case SYS_block:
    ret = sys_block();
    break;
  case SYS_kill:
    ret = sys_kill();
    break;
rtm's avatar
rtm committed
  case SYS_panic:
    ret = sys_panic();
    break;
rtm's avatar
rtm committed
  default:
    cprintf("unknown sys call %d\n", num);
    // XXX fault
    break;
  }
rtm's avatar
rtm committed
  cp->tf->tf_regs.reg_eax = ret;