实验2:系统调用

实验2:系统调用

实验2: 系统调用

实验内容

在 Linux 0.11 上添加两个系统调用,并编写两个简单的应用程序测试它们。

1. iam()

第一个系统调用是 iam(),其原型为:

int iam(const char * name);

完成的功能是将字符串参数 name 的内容拷贝到内核中保存下来。要求 name 的长度不能超过 23 个字符。返回值是拷贝的字符数。如果 name 的字符个数超过了 23,则返回 “-1”,并置 errno 为 EINVAL。

kernal/who.c 中实现此系统调用。

2. whoami()

第二个系统调用是 whoami(),其原型为:

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

它将内核中由 iam() 保存的名字拷贝到 name 指向的用户地址空间中,同时确保不会对 name 越界访存(name 的大小由 size 说明)。返回值是拷贝的字符数。如果 size 小于需要的空间,则返回“-1”,并置 errno 为 EINVAL。

也是在 kernal/who.c 中实现。

3. 测试程序

运行添加过新系统调用的 Linux 0.11,在其环境下编写两个测试程序 iam.c 和 whoami.c。最终的运行结果是:

$ ./iam lizhijun
$ ./whoami
lizhijun

系统调用简介

系统调用和一般的函数调用在使用方式上没有区别,区别在于调用之后函数内部如何处理。

系统调用中使用int 0x80来切换到内核态,使用内核中的系统函数来完成系统调用,如下面的步骤:

  1. 应用程序调用库函数(API);
  2. API 将系统调用号存入 EAX,然后通过中断调用使系统进入内核态;
  3. 内核中的中断处理函数根据系统调用号,调用对应的内核函数(系统调用);
  4. 系统调用完成相应功能,将返回值存入 EAX,返回到中断处理函数;
  5. 中断处理函数返回到 API 中;
  6. API 将 EAX 返回给应用程序。

下图是一个系统调用在内核里面的调用流程:

需要修改的文件(以linux-0.11为主文件夹):

linux-0.11/include/linux/sys.h
linux-0.11/kernel/system_call.s
linux-0.11/kernel/Makefile
在linux-0.11/kernel中添加文件
who.c

下面的需要在虚拟机中的linux系统中修改:
在linux-0.11的文件目录下修改
/usr/root/include/unistd.h
~/iam.c
~/whoami.c

修改sys.h

sys.h中维护了一个sys_call_table是一个函数指针数组,通过系统调用号,然后在数组中找到相应的中断处理函数。我们要将自己编写的系统调用添加到里面,就如以下内容:

extern int sys_iam();
extern int sys_whoami(
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值