《Linux内核设计与实现》课本第五章学习笔记——20135203齐岳

时间:2021-07-13 09:19:15

《Linux内核设计与实现》课本第五章学习笔记

By20135203齐岳

与内核通信

用户空间进程和硬件设备之间通过系统调用来交互,其主要作用有三个。

  • 为用户空间提供了硬件的抽象接口。
  • 保证了系统的稳定和安全。
  • 实现多任务和虚拟内存。保证良好的稳定性和安全性。

系统调用是用户空间访问内核的唯一手段;除异常和陷入外,是内核唯一合法的入口。

API、POSIX和C库

应用程序通过在用户空间实现的应用编程接口(API)而非直接通过系统调用来编程。

POSIX是应用编程接口的一个国际标准,C库提供了POSIX的绝大部分API。

《Linux内核设计与实现》课本第五章学习笔记——20135203齐岳

Unix接口设计的特点:提供机制(需要实现什么功能)而非策略(怎样实现这些功能)。Unix系统调用抽象出了用于完成某种确定的目的的函数,而至于函数是如何实现功能的则并不关心,从程序员的角度来看,只需通过接口便可实现功能。

系统调用

要访问系统调用,通常通过C库定义的函数调用来进行。例如,getpid()系统调用,在内核中的实现为:

SYSCALL_DEFINED0(getpid)
{
return task_tgid_vnr(current);//return current->tgid
}

SYSCALL _ DEFINED0只是一个宏,定义一个无参数的系统调用,展开后的代码如下

asmlinkage long sys_getpid(void)

asmlinkage为限定词,是一个编译指令,通知编译器仅从栈中提取该函数的参数。函数返回值在用户态时为int,在内核态为long。

系统调用号

在Linux系统中每个系统调用被赋予一个系统调用号,当用户空间的进程执行一个系统调用时,系统调用号用来指明执行哪个系统调用

系统调用号一旦分配就不会再更改,被删除的系统调用号也不许再回收。

sys _ ni _ syscall()专门针对无效的系统调用而设立的,只负责返回-ENOSYS。

系统调用号被定义在arch/i386/kernel/syscall_64.c文件中。

指定恰当的系统调用

X86中,系统调用号通过eax寄存器传递给内核,system _ call()函数通过将给定的系统调用与NR _syscalls作比较来检查其有效性,如果大于或等于NR _syscalls就返回-ENOSYS,否则就执行相应的系统调用:

call *sys_call_table(,%eax,8)//基址+偏移量*8

参数传递

上一篇博客中有详细的记录,这里不再赘述。见http://www.cnblogs.com/July0207/p/5277774.html

系统调用的实现

参数验证

由于系统调用在内核空间执行,所以必须验证其参数是否合法有效,而且必须是正确的。

检查指针是否有效

检查用户提供的指针是否有效,在接收这个指针之前,必须保证:

  • 指针所指向的内存区域属于用户空间。
  • 指针所指向的内存区域在进程的地址空间之内。
  • 如果是读,该内存应被标记为可读,如果是写,该内存应被标记为可写,如果是可执行,该内存应被标记为可执行。进程决不能绕过内存访问限制。
检查内核空间与用户空间数据的来回拷贝

内核提供了两个方法来完成必须的检查内核空间与用户空间数据的来回拷贝。

  • 为了向用户空间写数据,内核提供了copy _ to _user(),它需要三个参数,第一个是进程空间中的目的内存地址,第二个是内核空间内的源地址,第三个是需要拷贝的数据长度(字节数)。

  • 为了从用户空间读数据,内核提供了copy _ from _user(),该函数把第二个参数指定位置上的数据拷贝到第一个参数的指定位置,第三个是需要拷贝的数据长度(字节数)。

如果运行成功,则返回0,如果失败,则返回没能拷贝成功的字节数。

检查针对是否有合法权限

调用capable()函数检查用户是否有权对指定资源进行操作,返回非0值则有权限,返回0无权限。

<linux/capability.h>中包含一份所有权能和其对应的权限列表。

系统调用上下文

内核在执行系统调用时处于进程上下文。在进程上下文中,内核可以:

  • 休眠:说明系统调用可以使用内核提供的绝大部分功能。
  • 可以被抢占:要求保证该系统调用是可重入的。

绑定一个系统调用的最后步骤

  1. 在系统调用表的最后加入一个表项。
  2. 对于所支持的各种体系结构,系统调用号都必须定义于<asm/unistd.h>中。
  3. 系统调用必须被编译进内核映像,不能被编译成模块。只要将其放进kernel/下的一个相关文件中即可,例如sys.c。

从用户空间访问系统调用

Linux本身提供了一组宏,用于直接对系统调用进行访问。他会设置好寄存器并调用陷入指令。该宏必须了解到底有多少参数按照怎样的顺序压入寄存器。

	_syscalln()     //n的范围从0到6,代表需要传递给系统调用的参数个数。

例如,open()系统调用的形式是:

long open(const char *filename, int flags, int mode)

等价于

#define NR_open 5
_syscall3(long,open,const char*,filename, int,flags, int,mode)

对于每个宏来说,都有(2+2xn)个参数:

1.系统调用的返回值类型

2.系统调用的名称

3及以后是按照系统调用参数的顺序排列每个参数的类型和名称。

_ NR _open在<asm/unistd.h>中定义。这个宏会被扩展成为内嵌汇编的C函数。

建立一个新系统调用

好处:

  • 系统调用创建容易并且使用方便
  • linux系统调用的高性能

问题:

  • 占用系统调用号
  • 固化,不允许改动接口
  • 需要分别注册到每个需要支持的体系结构中
  • 脚本中不易调用,文件系统中也不能直接访问
  • 在主内核树外难以维护使用

替代方法:

  • 某些接口,例如信号量,用文件描述符表示
  • 把增加的信息作为一个文件放在sysfs的合适位置。

《Linux内核设计与实现》课本第五章学习笔记——20135203齐岳的更多相关文章

  1. 《Linux内核设计与实现》第五章学习笔记

    <Linux内核设计与实现>第五章学习笔记 姓名:王玮怡  学号:20135116 一.与内核通信     在Linux中,系统调用是用户空间访问内核的唯一手段:除异常和陷入外,它们是内核 ...

  2. 《Linux内核设计与实现》 第五章学习笔记

    第五章 系统调用 在现代操作系统中,内核提供了进程与内核进行交互的一组接口.有如下作用: 让应用程序受限的访问硬件设备 提供了创新进程并与已有进程进行通信的机制 提供了申请操作系统其它资源的能力 保证 ...

  3. 《Linux内核设计与实现》第四章学习笔记

    <Linux内核设计与实现>第四章学习笔记           ——进程调度 姓名:王玮怡  学号:20135116 一.多任务 1.多任务操作系统的含义 多任务操作系统就是能同时并发地交 ...

  4. 《Linux内核设计与实现》第四章学习笔记——进程调度

                                                                        <Linux内核设计与实现>第四章学习笔记——进程调 ...

  5. 《Linux内核设计与实现》 第一二章学习笔记

    <Linux内核设计与实现> 第一二章学习笔记 第一章 Linux内核简介 1.1 Unix的历史 Unix的特点 Unix很简洁,所提供的系统调用都有很明确的设计目的. Unix中一切皆 ...

  6. 《Linux内核设计与实现》课本第三章自学笔记——20135203齐岳

    <Linux内核设计与实现>课本第三章自学笔记 进程管理 By20135203齐岳 进程 进程:处于执行期的程序.包括代码段和打开的文件.挂起的信号.内核内部数据.处理器状态一个或多个具有 ...

  7. 《Linux内核设计与实现》课本第四章自学笔记——20135203齐岳

    <Linux内核设计与实现>课本第四章自学笔记 进程调度 By20135203齐岳 4.1 多任务 多任务操作系统就是能同时并发的交互执行多个进程的操作系统.多任务操作系统使多个进程处于堵 ...

  8. 《Linux内核设计与实现》第五章读书笔记

    第五章  系统调用 5.1与内核通信 1. 系统调用 让应用程序受限的访问硬件设备 提供创建新进程并与已有进程通信的机制 提供申请操作系统其他资源能力是用户空间进程和硬件设备之间的中间层 2. 系统调 ...

  9. 《Linux内核设计与实现》第三章学习笔记

    第三章  进程管理 姓名:王玮怡  学号:20135116 一.进程 1.进程的含义 进程是处于执行期的程序以及相关资源的总称,程序本身并不是进程,实际上就是正在执行的代码的实时结果.Linux内核通 ...

随机推荐

  1. js const

    js const const 声明创建一个只读的常量.这不意味着常量指向的值不可变,而是变量标识符的值只能赋值一次. const state = { notes: [], activeNote: {} ...

  2. mysql 查询一条记录的下一条和上一条记录

    如果ID是主键或者有索引,可以直接查找: 方法一: 查询上一条记录的SQL语句(如果有其他的查询条件记得加上other_conditions以免出现不必要的错误): select * from tab ...

  3. Windows 8 动手实验系列教程 实验7:磁贴和通知

    动手实验 实验7:磁贴和通知 2012年9月 简介 磁贴是Windows应用商店应用用户体验的重要元素.当应用程序被安装后,它的磁贴将在Windows 8开始屏幕被创建.该磁贴(称为主磁贴)作为启动应 ...

  4. 消息队列queue

    一.queue 在多线程编程中,程序的解耦往往是一个麻烦的问题,以及在socket网络编程中也会有这样的问题.recv 和send之间,如果服务端有消息,问题需要发送给客户端,而那边的recv 被主程 ...

  5. 启动软件丢失 MSVCR100&period;dll 系列,缺少库的问题

    做安装包时,Installshield2015工具,里可以添加需要的库. 不然需要自己一个个处理. 参考链接:https://zhidao.baidu.com/question/338311071.h ...

  6. Cracking The Coding Interview 1&period;5

    //原文: // // Write a method to replace all spaces in a string with '%20'. // #include <iostream&gt ...

  7. 看这一篇就够了&comma;css选择器知识汇总

    对大多技术人员来说都比较熟悉CSS选择器,举一例子来说,假设给一个p标签增加一个类(class),可是执行后该class中的有些属性并没有起作用.通过Firebug查看,发现没有起作用的属性被覆盖了, ...

  8. Js基础知识7-Es6新增对象Map和set数据结构

    前言 JavaScript中对象的本质是键值对的集合,ES5中的数据结构,主要是用Array和Object,但是键只能是字符串.为了弥补这种缺憾,ES6带来了一种新的数据结构Map. Map也是键值对 ...

  9. 简单的Cooki案例——记录用户上次访问该网页的时间

    功能: 帮助网站实现提示客户端计算机上次访问网站的时间 实现原理: 将每一个会话作为一次访问过程,将每次会话的开始时间作为每次访问网站的时间,然后将这个时间以Cookie的形式存储到客户端的计算机中, ...

  10. 自动化维护任务 – Automated Maintenance Task (转)

    1. Oracle有三个已定义好的automated maintenance tasks. Automatic Optimizer Statistics Collection—用于收集各种数据库对象的 ...