Skip to content

syscall.c

TIP

系统调用相关的实现。 xv6-riscv 内核通过 ecall 指令提供系统调用服务。 ecall 指令会进入内核态,并跳转到 kernelvec 处的代码。 kernelvec 会保存用户寄存器,并调用 trap 函数。 trap 函数会判断陷阱类型,如果是系统调用,则会调用 syscall 函数。 syscall 函数会根据系统调用号,在 syscalls 表中查找并执行相应的处理函数。 处理函数执行完毕后,返回值会存入当前进程陷阱帧的 a0 寄存器。 最后,通过 sret 指令返回用户态,用户程序从 a0 寄存器中获取返回值。

#include "types.h"
#include "param.h"
#include "memlayout.h"
#include "riscv.h"
#include "spinlock.h"
#include "proc.h"
#include "syscall.h"
#include "defs.h"

TIP

从用户空间获取一个64位整数。 用户虚拟地址 addr 处的数据将被拷贝到内核变量 *ip 中。 成功时返回 0,失败时返回 -1。

int
fetchaddr(uint64 addr, uint64 *ip)
{
  struct proc *p = myproc();

TIP

检查地址是否在当前进程的地址空间范围内。 必须同时检查 addraddr + sizeof(uint64),以防止地址溢出。

  if(addr >= p->sz || addr+sizeof(uint64) > p->sz)
    return -1;

TIP

使用 copyin 函数安全地从用户空间拷贝数据到内核空间。

  if(copyin(p->pagetable, (char *)ip, addr, sizeof(*ip)) != 0)
    return -1;
  return 0;
}

TIP

从用户空间获取一个以 null 结尾的字符串。 用户虚拟地址 addr 处的字符串将被拷贝到内核缓冲区 buf 中,最多 max 字节。 成功时返回字符串的长度(不包括结尾的 null 字符),失败时返回 -1。

int
fetchstr(uint64 addr, char *buf, int max)
{
  struct proc *p = myproc();

TIP

使用 copyinstr 函数安全地从用户空间拷贝字符串到内核空间。

  if(copyinstr(p->pagetable, buf, addr, max) < 0)
    return -1;
  return strlen(buf);
}

TIP

从陷阱帧中获取第 n 个原始系统调用参数(存储在 a0-a5 寄存器中)。 RISC-V 架构规定,系统调用的前 6 个参数通过 a0-a5 寄存器传递。

static uint64
argraw(int n)
{
  struct proc *p = myproc();
  switch (n) {
  case 0:
    return p->trapframe->a0;
  case 1:
    return p->trapframe->a1;
  case 2:
    return p->trapframe->a2;
  case 3:
    return p->trapframe->a3;
  case 4:
    return p->trapframe->a4;
  case 5:
    return p->trapframe->a5;
  }

TIP

如果 n 不在 0-5 范围内,说明代码逻辑有误,触发 panic。

  panic("argraw");
  return -1;
}

TIP

获取第 n 个 32 位整型系统调用参数,并将其存入 *ip。

void
argint(int n, int *ip)
{
  *ip = argraw(n);
}

TIP

获取第 n 个指针类型的系统调用参数,并将其存入 *ip。 这个函数本身不检查地址的合法性。 地址的合法性检查会由后续的 copyincopyout 等函数完成。

void
argaddr(int n, uint64 *ip)
{
  *ip = argraw(n);
}

TIP

获取第 n 个字符串类型的系统调用参数。 先通过 argaddr 获取字符串在用户空间的地址, 然后通过 fetchstr 将字符串从用户空间拷贝到内核缓冲区 buf 中。 成功时返回字符串长度,失败时返回 -1。

int
argstr(int n, char *buf, int max)
{
  uint64 addr;
  argaddr(n, &addr);
  return fetchstr(addr, buf, max);
}

TIP

声明所有系统调用处理函数的原型。 这些函数的具体实现在 sysproc.csysfile.c 文件中。

extern uint64 sys_fork(void);
extern uint64 sys_exit(void);
extern uint64 sys_wait(void);
extern uint64 sys_pipe(void);
extern uint64 sys_read(void);
extern uint64 sys_kill(void);
extern uint64 sys_exec(void);
extern uint64 sys_fstat(void);
extern uint64 sys_chdir(void);
extern uint64 sys_dup(void);
extern uint64 sys_getpid(void);
extern uint64 sys_sbrk(void);
extern uint64 sys_sleep(void);
extern uint64 sys_uptime(void);
extern uint64 sys_open(void);
extern uint64 sys_write(void);
extern uint64 sys_mknod(void);
extern uint64 sys_unlink(void);
extern uint64 sys_link(void);
extern uint64 sys_mkdir(void);
extern uint64 sys_close(void);
extern uint64 sys_shutdown(void);
extern uint64 sys_reboot(void);

TIP

系统调用处理函数指针数组。 syscall.h 中定义的系统调用号 (例如 SYS_fork) 作为该数组的索引, 映射到对应的处理函数 (例如 sys_fork)。

static uint64 (*syscalls[])(void) = {
[SYS_fork]    sys_fork,
[SYS_exit]    sys_exit,
[SYS_wait]    sys_wait,
[SYS_pipe]    sys_pipe,
[SYS_read]    sys_read,
[SYS_kill]    sys_kill,
[SYS_exec]    sys_exec,
[SYS_fstat]   sys_fstat,
[SYS_chdir]   sys_chdir,
[SYS_dup]     sys_dup,
[SYS_getpid]  sys_getpid,
[SYS_sbrk]    sys_sbrk,
[SYS_sleep]   sys_sleep,
[SYS_uptime]  sys_uptime,
[SYS_open]    sys_open,
[SYS_write]   sys_write,
[SYS_mknod]   sys_mknod,
[SYS_unlink]  sys_unlink,
[SYS_link]    sys_link,
[SYS_mkdir]   sys_mkdir,
[SYS_close]   sys_close,
[SYS_shutdown] sys_shutdown,
[SYS_reboot]   sys_reboot,
};

TIP

系统调用分发函数。 当用户程序执行 ecall 指令引发陷阱时,此函数被调用。

void
syscall(void)
{
  int num;
  struct proc *p = myproc();

TIP

RISC-V 约定,系统调用号存储在 a7 寄存器中。 从当前进程的陷阱帧中读取 a7 寄存器的值。

  num = p->trapframe->a7;
  

TIP

检查系统调用号是否合法,并且对应的处理函数是否存在。

  if(num > 0 && num < NELEM(syscalls) && syscalls[num]) {

TIP

调用对应的系统调用处理函数。 将处理函数的返回值存入 a0 寄存器,以便返回给用户程序。

    p->trapframe->a0 = syscalls[num]();
  } else {

TIP

如果系统调用号无效,则打印错误信息,并返回 -1。

    printf("%d %s: unknown sys call %d\n",
            p->pid, p->name, num);
    p->trapframe->a0 = -1;
  }
}