xv6 系统调用

1. 系统调用的实现

  开发程序需所有的接口在user.h中,包含两部分system call和ulib

  user.h中的系统接口函数在usys.S中通过汇编实现

#define SYSCALL(name) \
  .globl name; \
  name: \
    movl $SYS_ ## name, %eax; \
    int $T_SYSCALL; \
    ret

SYSCALL(fork)
.......

  把系统调用ID放入eax中,使用int T_SYSCALL中断

2. int 指令

  int n 指令是调用n号中断的中断过程。最终效果和函数调用类似,中断过程执行结束过后返回。

3. xv6 中断处理

  在tvinit中生成了中断描述表(IDT),然后在idtinit中加载该中断描述表,注册0-255中断的处理过程。在tvinit函数中我们可以看到T_SYSCALL中断要求的调用权限是 DPL_USER,所有我们可以在用户空间调用T_SYSCALL中断。

  每个中断的处理过程在vectors.S(vector.pl生成)中实现,压入不同参数过后执行alltraps。在alltraps中准备参数trapframe,然后调用c实现的函数trap。

  trap函数实现如下:

  

void trap(struct trapframe *tf)
{
  if(tf->trapno == T_SYSCALL){
    if(myproc()->killed)
      exit();
    myproc()->tf = tf;
    syscall();
    if(myproc()->killed)
      exit();
    return;
  }
    .....
}

  当中断是T_SYSCALL时执行函数syscall, syscall根据eax中的系统调用id,确定具体函数,执行相应函数,把执行结果放入eax中。

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

  num = curproc->tf->eax;
  if(num > 0 && num < NELEM(syscalls) && syscalls[num]) {
    curproc->tf->eax = syscalls[num]();
  } else {
    cprintf("%d %s: unknown sys call %d\n",
            curproc->pid, curproc->name, num);
    curproc->tf->eax = -1;
  }
}

//TODO trapframe 是如何构建的

 

参考:https://th0ar.gitbooks.io/xv6-chinese/content/content/chapter0.html

转载于:https://www.cnblogs.com/hygblog/p/9337643.html

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
xv6 中添加系统调用需要进行以下步骤: 1. 在 `syscall.h` 文件中添加系统调用号。具体来说,你需要在 `SYS_hoge` 列表中添加一个新的系统调用号,其中 `hoge` 是你的系统调用的名称。例如,如果你的系统调用名称是 `mycall`,则可以在 `syscall.h` 文件中添加以下代码: ``` #define SYS_mycall 22 ``` 2. 在 `syscall.c` 文件中添加系统调用的具体实现。具体来说,你需要在 `syscalls[]` 数组中添加一个新的结构体,其中包含你的系统调用的名称和具体的实现函数。例如,如果你的系统调用名称是 `mycall`,则可以在 `syscall.c` 文件中添加以下代码: ``` static int sys_mycall(void) { // 在这里编写你的系统调用的具体实现 return 0; } static struct { int nargs; int (*func)(void); } syscalls[] = { { 0, sys_fork }, { 1, sys_exit }, { 0, sys_wait }, { 1, sys_pipe }, { 4, sys_read }, { 4, sys_write }, { 2, sys_close }, { 1, sys_mycall }, // 添加新的系统调用 }; ``` 3. 在 `user.h` 文件中添加用户空间的系统调用函数声明。具体来说,你需要添加一个新的函数声明,其中包含你的系统调用的名称和参数。例如,如果你的系统调用名称是 `mycall`,则可以在 `user.h` 文件中添加以下代码: ``` int mycall(void); ``` 4. 在用户程序中调用你的系统调用。具体来说,你可以在用户程序中调用你的系统调用函数,例如: ``` #include "user.h" int main(int argc, char *argv[]) { mycall(); // 调用你的系统调用 exit(); } ``` 5. 在内核态的系统调用函数中添加系统调用的具体实现。具体来说,你需要在 `syscall()` 函数中添加一个新的 `case` 分支,其中包含你的系统调用的具体实现。例如,如果你的系统调用名称是 `mycall`,则可以在 `syscall()` 函数中添加以下代码: ``` case SYS_mycall: argint(0, &arg1); // 在这里调用你的系统调用的具体实现 return 0; ``` 6. 最后,重新编译 xv6 并运行你的用户程序。在 xv6 的命令行下输入你的程序名称即可运行。例如,如果你的用户程序名称是 `myprog`,则可以通过输入 `myprog` 命令来运行它,其中包含了你的系统调用调用
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值