15271139_陈思言 L7 2019-10-27 20:15:46 系统调用
451 0

“操作系统与Linux内核导读(北京航空航天大学)”实验报告

系统调用

关键代码

who.c

#include <errno.h>
#include <string.h>
#include <asm/segment.h>

static char buf[24];

int sys_iam(const char *name)
{
    int i = 0;
    char tmp[24];
    while ((tmp[i] = get_fs_byte(name + i)) != '\0' && i < 24)
        i++;
    if (i > 23)
        return -EINVAL;
    strcpy(buf, tmp);
    //printk("%s", buf);
    return i;
}

int sys_whoami(char *name, unsigned int size)
{
    unsigned int n = strlen(buf), i;
    if(n < size)
    {
        for(i = 0; i < n; i++)
            put_fs_byte(buf[i], &name[i]);
        return n;
    }
    else
        return -EINVAL;
}
copy

iam.c

#define __LIBRARY__
#include <unistd.h>

_syscall1(int, iam, const char*, name)

#include<stdio.h>

int main(int argc, char *argv[])
{
    if(argc != 2)
    {
        puts("Argument Error!");
        return -1;
    }
    if(iam(argv[1]) != -1)
        return 0;
    return -1;
}
copy

whoami.c

#define __LIBRARY__
#include <unistd.h>

_syscall2(int, whoami, char*, name, unsigned int, size)

#include<stdio.h>

int main()
{
    char name[24];
    if(whoami(name, 23) != -1)
        puts(name);
    return -1;
}
copy

完整代码在https://github.com/chensiyan96/oslab/tree/3

运行结果

环境截图

思考题

  1. 从Linux 0.11现在的机制看,它的系统调用最多能传递几个参数?你能想出办法来扩大这个限制吗? 答:从现有的unistd.h的定义来看,其中定义了0~3个参数的_syscall0、_syscall1、_syscall2、_syscall3,因此它的系统调用最多能传递3个参数。由于IA32一共有8个通用寄存器,esp是栈指针不能乱用,eax作为函数号,当前使用了ebx、ecx、edx来传参,因此简单的扩大这个限制的方法是将ebp、esi、edi也用来传递参数,这样可以比较方便地扩展到传递6个参数。也可以使用栈来传参,但是由于内核态和用户态并不使用同一个栈,需要再通过类似get_fs_byte拷贝的方法获取参数,十分麻烦低效。
  2. 用文字简要描述向Linux 0.11添加一个系统调用foo()的步骤。 答:先添加一个系统调用库函数foo(),同时在unistd.h中定义其系统调用号,然后在系统调用中断处理函数中扩大系统调用函数的数量,并将内核函数加入到内核函数表中,随后在内核中添加内核函数sys_foo()的具体实现,最后修改Makefile。
最新评论
暂无评论~