Linux内核——Linux内核体系模式(二)

1 Linux系统的中断机制

Linux内核将中断分为两类:硬件中断和软件中断(异常)。每个中断是由0-255之间的一个数字进行标识。

  • 中断int0-int31(0x00-0x1f)作为异常
  • int32-int255由用户自己设定
  • int32-int47对应与8259A中断控制芯片发出的硬件中断请求信号IRQ0-IRQ15,并把程序编程发出的系统调用中断设置为intl28(0x80)。系统调用中断是用户程序使用操作系统资源的唯一界面接口。

    1. 系统如何进行中断初始化
  • 内核在head.s程序中使用一个哑中断向量(中断描述符)对中断描述符表IDT中所有256个描述符进行了默认设置(boot/head.s,78)。哑中断向量指向一个默认的"无中断"处理过程(boot/head.s,150)
  • 当发生了一个中断而又没有重新设置过该中断向量就会显示信息"未知中断(Unknown Interrupt)"
  • 可以对所有256项都进行设置可以有效防止出现一般保护性错误(异常13)。否则当设置的IDT少于256项,那么在一个要求的中断所指定的描述符项大于设置的最大描述符项时,CPU就产生一个一般保护出错(异常13)
  • 如果硬件出问题没有吧设备的向量放到数据总线上,此时CPU会从数据总线上读入全1作为向量,将会造成一般保护出错。
  • 异常中断处理过程(int0-int31)都在traps.c的初始化函数中进行了重新设置(kernel/traps.c,181),系统调用中断int128在调度程序初始化函数中进行了重新设置(kernel/sched.c,385)。
  1. 中断和陷阱门的使用:

    • Linux内核使用中断门和陷阱门两种描述符来处理中断,区别在于对EFLAGS寄存器中的IF(中断允许)标志的影响。
    • 中断门会复位IF标志,避免其他中断干扰;陷阱门则不影响IF标志。
  2. 标志寄存器的中断标志管理:

    • Linux内核使用cli和sti指令来控制CPU的中断响应。
    • cli指令清除中断标志,禁止中断;sti指令设置中断标志,允许中断。
    • 这种方法用于保护临界代码区免受中断干扰,避免数据竞争和系统崩溃。

2 Linux的系统调用

系统调用

系统调用是用户程序与操作系统内核之间的接口,用于请求操作系统提供的服务。通常使用函数形式进行调用,可以带有参数,并通过返回值表示执行结果。

1 )系统调用号和错误处理:

每个系统调用都有一个唯一的功能号,这些功能号定义在include/unistd.h文件中。错误的系统调用会返回负值,并把错误类型码存储在全局变量errno中。

2)系统调用的实现机制:

系统调用通过中断0x80进入内核,eax寄存器存放系统调用号,ebx、ecx和edx寄存器用于传递参数。Linux内核中的系统调用处理函数以'sys_'为前缀,例如write系统调用的处理程序是sys_write。

系统调用处理过程

  • 当发出中断调用int 0x80后,system_call程序开始执行,它检查系统调用号的有效性,并从sys_call_table[]数组中调用相应的处理程序。
  • sys_call_table[]数组中的索引对应系统调用的功能号,每个索引处的函数指针指向相应的内核服务程序。
  • 库函数与直接系统调用:
    • 用户程序通常通过库函数间接调用系统调用,但也可以跳过库函数直接执行系统调用。
    • 直接执行系统调用可以使用宏syscalln(),其中n代表参数个数,例如syscall3表示三个参数。
    • 系统调用接口是内核与应用程序交互的唯一途径,它允许应用程序利用内核提供的服务来访问系统硬件资源。

(八股回答)

在Linux系统中,系统调用的处理过程是一个复杂的机制,涉及用户空间和内核空间之间的交互。

1. 触发系统调用 当用户程序需要操作系统提供的服务时,它会通过库函数(如C标准库中的函数)或直接通过内嵌汇编语句来触发一个系统调用。这通常涉及执行一个特殊的汇编指令(如int$0x80),该指令会产生一个异常,使得控制权转移到内核空间。

2. 传递系统调用号和参数 在执行系统调用指令之前,程序会将系统调用号存储在eax寄存器中,将任何需要的参数存储在ebxecxedx寄存器中。这些参数是系统调用可能需要的输入数据。

3. 进入内核空间 当CPU执行int $0x80指令时,它会切换到内核模式并跳转到异常处理程序的地址。在Linux中,这个处理程序是system_call函数,它位于kernel/system_calls文件中。

4. 系统调用分发 system_call函数首先检查eax寄存器中的系统调用号是否有效。然后,它使用系统调用号作为索引,从sys_call_table数组中查找相应的内核函数指针,并间接调用该函数。sys_call_table是一个包含所有系统调用处理函数指针的数组。

5. 执行系统调用处理程序 一旦找到并调用了相应的内核函数,它将执行所需的操作,可能会访问硬件设备、管理文件系统或执行其他内核级别的任务。

6. 返回用户空间 系统调用处理程序执行完毕后,它将返回值存储在eax寄存器中,并执行iret指令返回到用户空间。如果在系统调用过程中发生错误,一个负值会被存储在eax中,而错误号会被存储在errno全局变量中供应用程序检查。

7. 获取系统调用结果 一旦控制权返回给用户程序,它可以检查eax寄存器中的返回值以确定系统调用的结果。如果返回值是正数,表示系统调用成功;如果是负数,则表示系统调用失败,程序可以查看errno以获取错误信息。

3 Linux进程控制

  1. 进程调度

    Linux操作系统采用基于优先级的进程调度算法,常见的如完全公平队列(Completely Fair Scheduler, CFS)。CFS的目标是确保每个进程获得公平的CPU时间,同时考虑到进程的优先级。

    • 进程优先级:每个进程都有一个优先级,这个优先级决定了它在CPU上执行的时间。高优先级的进程会获得更多的CPU时间。
    • 时间片:CFS为每个进程分配一个运行的时间片,这个时间片基于进程的优先级。当一个进程的时间片用完后,它会被放到运行队列的末尾。
  2. 分时技术

    分时技术是一种操作系统管理多个进程的方法,使得每个进程都能在有限的时间内获得CPU的使用权。

    • 时间片轮转:操作系统为每个进程分配一个小的时间片(通常是几十毫秒),在这个时间片内,进程可以执行指令。时间片结束后,即使进程没有完成执行,也会被挂起,操作系统接着运行下一个进程的时间片。
    • 抢占式多任务:如果一个更高优先级的进程需要CPU,当前运行的低优先级进程会被抢占,即被中断并移动到运行队列的末尾,而高优先级进程将获得CPU
  3. 多级反馈队列(MLFQ)

    Linux还采用了多级反馈队列(Multi-Level Feedback Queue, MLFQ)来进一步优化进程调度。

    • 多级队列:进程根据其行为和优先级被分类到不同的队列中。例如,交互式进程可能被放在一个更高的优先级队列,而批处理进程则可能被放在一个较低的优先级队列。
    • 反馈机制:进程的行为(如CPU使用时间、阻塞频率等)会影响其在队列中的优先级。例如,经常使用CPU的进程可能会被降低优先级,而不经常使用的进程则可能被提升优先级。
  4. 用户控制与系统配置

    系统管理员可以调整系统的各种参数来控制进程调度的行为,以适应特定的工作负载和性能要求。

    • 调整优先级:可以通过nice值来调整进程的优先级,影响其在CPU上的运行时间。
    • 配置调度器:Linux允许管理员选择不同的进程调度器,或者调整现有调度器的参数,以优化特定类型的应用程序或工作负载。
  5. 任务数据结构

    进程表项是一个task_struct任务结构指针。

  6. 进程运行状态

  7. 进程初始化与创建进程
    【Linux】Linux进程的创建与管理
    Linux系统初始化进程及文件(带命令)

相关推荐
宁zz9 小时前
乌班图安装jenkins
运维·jenkins
大丈夫立于天地间10 小时前
ISIS协议中的数据库同步
运维·网络·信息与通信
cg501710 小时前
Spring Boot 的配置文件
java·linux·spring boot
暮云星影10 小时前
三、FFmpeg学习笔记
linux·ffmpeg
rainFFrain10 小时前
单例模式与线程安全
linux·运维·服务器·vscode·单例模式
GalaxyPokemon10 小时前
Muduo网络库实现 [九] - EventLoopThread模块
linux·服务器·c++
mingqian_chu11 小时前
ubuntu中使用安卓模拟器
android·linux·ubuntu
weixin_5088216511 小时前
1ms软延时,不用定时器,stm32
stm32·单片机·嵌入式硬件
@郭小茶12 小时前
docker-compose方式部署docker项目
运维·docker·容器