系统调用(上)

谈愈敏 原创作品转载请注明出处 《Linux内核分析》MOOC课程 http://mooc.study.163.com/course/USTC-1000029000

用户态、内核态和中断

系统调用是用户通过库函数方式:库函数帮我们把系统调用封装起来。

用户态&内核态(CPU执行级别)
  • 内核态:高级别执行,可以使用特权指令,访问任意的物理地址。对应x86 0级
  • 用户态:低级别执行,代码范围受到限制。对应x86 3级(x86CPU有0-3四个级别)

这种权限级别划分让系统更稳定。

如何区分用户态&内核态?

地址空间是显著标识,cs:eip[代码段选择寄存器:偏移量寄存器](指令地址)

  • 0xc0000000以上的地址只能在内核态下访问
  • 0x00000000-0xbfffffff两种状态都行

注意:这里指逻辑地址(进程地址空间)而不是物理地址

另:cs寄存器的最低两位,表示当前代码的特权级

中断处理是从用户态进入内核态主要的方式
  • 从用户态进入内核态:必须保存用户态的寄存器上下文
  • 中断/int指令在堆栈上保存寄存器的值:用户态/内核态栈顶地址(ss:esp)、状态字(eflags)、cs:eip值(内核态时指向中断服务程序入口)

系统调用是一种特殊的中断

中断发生后第一件事就是保存现场:

进入中断程序,保存寄存器数据
SAVE_ALL

中断结束前最后一件事就是恢复现场:

退出中断程序,恢复寄存器数据
RESTORE_ALL
iret(pop cs:eip/ss:esp/eflags from kernel stack)
#对应着中断信号或int指令,与发生时CPU动作相反

系统调用概述

系统调用的意义

操作系统为用户态进程与硬件设备进行交互提供了一组接口-系统调用

  • 用户不管硬件编程
  • 提高系统安全性
  • 用户程序可移植
API和系统调用
  • API:应用编程接口,是一个函数定义
  • 系统调用:通过软中断向内核发出明确请求

Libc库定义的一些API引用了封装例程(唯一目的就是发布系统调用)

  • 一般每个系统调用对应一个封装例程
  • 库用封装例程定义出给用户的API

不是每个API都对应一个特定的系统调用

  • API可直接提供用户态服务,如数学函数
  • 一个API可调用几个系统调用
  • 不同API可调用同一系统调用

返回值

  • 封装例程返回一个整数,含义依赖与相应系统调用
  • -1表示内核不能满足进程的要求
  • Libc定义的errno变量包含特定出错码
应用程序、封装例程、系统调用处理程序、系统调用服务例程之间的关系

xyz是一个函数,即API
应用程序编程接口里封装了一个系统调用,触发一个中断:中断向量0x80对应内核代码系统调用起点system_call
中断服务程序(sys_xyz)

系统调用三层皮:

  • API(xyz)
  • 中断向量(system_call)
  • 中断服务程序(sys_xyz)

用户态进程调用系统调用时,CPU切换到内核态执行内核函数(Linux中通过执行int $128来执行系统调用,产生向量为128的编程异常)

传参:进程指明需要哪个系统调用,传递系统调用号,使用eax传递

  • 系统调用号将xyz与sys_xyz关联起来
  • system_call是linux中所有系统调用的入口点,每个系统调用至少有一个参数,即系统调用号。

寄存器传递参数的限制:

  • 每个参数长度不能超过寄存器长度即32位
  • 在系统调用号eax之外,个数不能超过6个

超过6的话:某个寄存器中存储指针,指针会指向一个地址空间,存储参数。

使用库函数API和C代码中嵌入汇编代码触发同一个系统调用

使用库函数API获取系统当前时间

使用time():

#include<stdio.h>
#include<time.h> int main()
{
time_t tt;
struct tm *t;//构造一个结构体,将tt转换为这个方便读取时间的
tt = time(NULL);//time系统调用
t = localtime(&tt); //类型装换为我们熟知的时间格式
printf("time:%d:%d:%d:%d:%d:%d\n", t->tm_year+1900, t->tm_mon, t->tm_mday, t->tm_hour, t->tm_min, t->tm_sec);
return 0;
}

编译:gcc time.c -o time -m32

使用C代码中嵌入汇编代码触发系统调用获取系统当前时间
#include<stdio.h>
#include<time.h> int main()
{
time_t tt;
struct tm *t;
asm volatile(
"mov $0,%%ebx\n\t" //把ebx清零,系统调用传递第一个参数用ebx,这里是NULL
"mov $0xd,%%eax\n\t"//把0xd放入eax中,即eax传递系统调用号13——time
"int $0x80\n\t" //内核处理系统调用
"mov %%eax,%0\n\t" //系统调用返回值是在eax中,%0指tt,把返回值放到tt中去。
: "=m" (tt)
);
t = localtime(&tt);
printf("time:%d:%d:%d:%d:%d:%d\n", t->tm_year+1900, t->tm_mon, t->tm_mday, t->tm_hour, t->tm_min, t->tm_sec);
return 0;
}

这段代码让我们更清楚的知道用户态进程对内核态做了什么

  • 传递了一个系统调用号 - eax
  • 传递了参数 - ebx

实验

选择20号系统调用getpid来获取进程ID

使用库函数API获取系统当前进程ID

代码:

运行:

使用C代码中嵌入汇编代码触发系统调用获取系统当前进程ID

思想:和视频示例中类似,只是传递给eax的系统调用号为0x14,即20号系统调用getpid。

代码:

运行:

20135220谈愈敏Blog4_系统调用(上)的更多相关文章

  1. 20135220谈愈敏Blog5_系统调用(下)

    系统调用(下) 谈愈敏 原创作品转载请注明出处 <Linux内核分析>MOOC课程 http://mooc.study.163.com/course/USTC-1000029000 给Me ...

  2. 20135220谈愈敏Blog8_进程的切换和系统的一般执行过程

    进程的切换和系统的一般执行过程 谈愈敏 原创作品转载请注明出处 <Linux内核分析>MOOC课程 http://mooc.study.163.com/course/USTC-100002 ...

  3. 20135220谈愈敏Blog7_可执行程序的装载

    可执行程序的装载 谈愈敏 原创作品转载请注明出处 <Linux内核分析>MOOC课程 http://mooc.study.163.com/course/USTC-1000029000 一. ...

  4. 20135220谈愈敏Blog3_构造一个简单的Linux系统MenuOS

    构造一个简单的Linux系统MenuOS 谈愈敏 原创作品转载请注明出处 <Linux内核分析>MOOC课程 http://mooc.study.163.com/course/USTC-1 ...

  5. 20135220谈愈敏Blog6_进程的描述和创建

    进程的描述和创建 谈愈敏 原创作品转载请注明出处 <Linux内核分析>MOOC课程 http://mooc.study.163.com/course/USTC-1000029000 进程 ...

  6. 20135220谈愈敏Blog2_操作系统是如何工作的

    操作系统是如何工作的 谈愈敏 原创作品转载请注明出处 <Linux内核分析>MOOC课程http://mooc.study.163.com/course/USTC-1000029000 计 ...

  7. 20135220谈愈敏Linux_总结

    Linux_总结 具体博客链接 计算机是如何工作的 操作系统是如何工作的 构造一个简单的Linux系统MenuOS 系统调用(上) 系统调用(下) 进程的描述和创建 可执行程序的装载 进程的切换和系统 ...

  8. 20135220谈愈敏Linux Book_4

    进程调度 进程:程序的运行态表现形式 进程调度程序:确保进程能有效工作的一个内核子系统,决定将哪个进程投入运行.何时运行以及运行多长时间,在可运行态进程之间分配有限的处理器时间资源. 最大限度的利用处 ...

  9. 20135220谈愈敏Linux Book_3

    第3章 进程管理 进程是Unix操作系统抽象概念中最基本的一种,进程管理是操作系统的心脏所在. 3.1 进程 进程:处于执行期的程序以及相关的资源的总称. 线程:在进程中活动的对象,拥有独立的程序计数 ...

随机推荐

  1. SQLSERVER无排序生成序号

    实现方式:ROW_NUMBER() SELECT RowID=(ROW_NUMBER() OVER(ORDER BY(SELECT ))) FROM dbo.tbl_name 实现方式:IDENTIT ...

  2. dns bind配置教程

    实验环境 三台centos7虚拟机,一台ip为192.168.52.130,一台为192.168.52.131,最后一台为192.168.52.132 安装bind 使用yum -y insall b ...

  3. Alpha冲刺报告(9/12)(麻瓜制造者)

    今日已完成 邓弘立: 正在进行主页逻辑的编写 符天愉: 部署商品发布和物品需求的接口 江郑: 尝试完善接口文档,进行进一步测试 刘双玉: 编写接口说明 肖小强: 进行逻辑模块的编写 李佳铭: 修改了U ...

  4. 团队作业——Alpha冲刺 3/12

    团队作业--Alpha冲刺 冲刺任务安排 杨光海天 今日任务:完成Android开发环境的搭建,学习基础开发知识 明日任务:继续学习Android开发知识,与其他成员协商,了解自己需要完成的开发任务, ...

  5. vue2.0路由切换后页面滚动位置不变BUG

    最近项目中遇到这样一个问题,vue切换路由,页面到顶端的滚动距离仍会保持不变.  方法一: 监听路由 // app.vue export default { watch:{ '$route':func ...

  6. C结构体数组赋值

    #include <stdio.h> #include <stdlib.h> struct MyStruct { int a; char b; }; struct MyStru ...

  7. python第三十六课——1.可迭代对象

    1.可迭代对象: 满足前提: 只要能被循环操作的对象,就可以可迭代对象 举例: str.list.tuple.set.dict.range.generator... 高效的检测一个对象是否是可迭代对象 ...

  8. Docker技术入门与实战 第二版-学习笔记-8-网络功能network-3-容器访问控制和自定义网桥

    1)容器访问控制 容器的访问控制,主要通过 Linux 上的 iptables防火墙来进行管理和实现. iptables是 Linux 上默认的防火墙软件,在大部分发行版中都自带. 容器访问外部网络 ...

  9. [转]在C++中容易出现的#error No Target Architecture

    项目环境:win 7 64位,编译环境:VS2013 最近在写C++的项目,发现了自己很多不会的地方,这也使得我在C++中的成长变得比较快,下面我就说说我自己在写项目是遇到的一些问题,希望可以帮到一些 ...

  10. 关于javascript中对浮点加,减,乘,除的精度分析

    大学专业是计算机童鞋或多或小的知道 计算机是由二进制存储和处理数字的,不能精确到处理浮点数,且javascript也没有这样的方法 所以在浏览器计算的时候也会有误差,比如说 我想用 3.3 / 1.1 ...