• 09 | 系统调用:公司成立好了就要开始接项目


    1、glibc 对系统调用的封装

    int open(const char *pathname, int flags, mode_t mode)glibc 里面的 open 函数

    make-syscall.sh  syscall-template.S

    2、系统调用图解

    3、总结

    - glibc 将系统调用封装成更友好的接口
    - 本节解析 glibc 函数如何调用到内核的 open
    ---
    - 用户进程调用 open 函数
        - glibc 的 syscal.list 列出 glibc 函数对应的系统调用
        - glibc 的脚本 make_syscall.sh 根据 syscal.list 生成对应的宏定义(函数映射到系统调用)
        - glibc 的 syscal-template.S 使用这些宏, 定义了系统调用的调用方式(也是通过宏)
        - 其中会调用 DO_CALL (也是一个宏), 32位与 64位实现不同
    ---
    - 32位 DO_CALL (位于 i386 目录下 sysdep.h)
        - 将调用参数放入寄存器中, 由系统调用名得到系统调用号, 放入 eax
        - 执行 ENTER_KERNEL(一个宏), 对应 int $0x80 触发软中断, 进入内核
        - 调用软中断处理函数 entry_INT80_32(内核启动时, 由 trap_init() 配置)
        - entry_INT80_32 将用户态寄存器存入 pt_regs 中(保存现场以及系统调用参数), 调用 do_syscall_32_iraq_on 
        - do_syscall_32_iraq_on 从 pt_regs 中取系统调用号(eax), 从系统调用表得到对应实现函数, 取 pt_regs 中存储的参数, 调用系统调用
        - entry_INT80_32 调用 INTERRUPT_RUTURN(一个宏)对应 iret 指令, 系统调用结果存在 pt_regs 的 eax 位置, 根据 pt_regs 恢复用户态进程
    ---
    - 64位 DO_CALL (位于 x86_64 目录下 sysdep.h)
        - 通过系统调用名得到系统调用号, 存入 rax; 不同中断, 执行 syscall 指令
        - MSR(特殊模块寄存器), 辅助完成某些功能(包括系统调用)
        - trap_init() 会调用 cpu_init->syscall_init 设置该寄存器
        - syscall 从 MSR 寄存器中, 拿出函数地址进行调用, 即调用 entry_SYSCALL_64
        - entry_SYSCALL_64 先保存用户态寄存器到 pt_regs 中
        - 调用 entry_SYSCALL64_slow_pat->do_syscall_64
        - do_syscall_64 从 rax 取系统调用号, 从系统调用表得到对应实现函数, 取 pt_regs 中存储的参数, 调用系统调用
        - 返回执行 USERGS_SYSRET64(一个宏), 对应执行 swapgs 和 sysretq 指令; 系统调用结果存在 pt_regs 的 ax 位置, 根据 pt_regs 恢复用户态进程
    ---
    - 系统调用表 sys_call_table
        - 32位 定义在 arch/x86/entry/syscalls/syscall_32.tbl 
        - 64位 定义在 arch/x86/entry/syscalls/syscall_64.tbl
        - syscall_*.tbl 内容包括: 系统调用号, 系统调用名, 内核实现函数名(以 sys 开头)
        - 内核实现函数的声明: include/linux/syscall.h
        - 内核实现函数的实现: 某个 .c 文件, 例如 sys_open 的实现在 fs/open.c
            - .c 文件中, 以宏的方式替代函数名, 用多层宏构建函数头
        - 编译过程中, 通过 syscall_*.tbl 生成 unistd_*.h 文件
            - unistd_*.h 包含系统调用与实现函数的对应关系
        - syscall_*.h include 了 unistd_*.h 头文件, 并定义了系统调用表(数组)

  • 相关阅读:
    洛谷P2221 [HAOI2012]高速公路(线段树+概率期望)
    洛谷P2254 [NOI2005]瑰丽华尔兹(单调队列)
    洛谷P2607 [ZJOI2008]骑士(基环树)
    洛谷P1505 [国家集训队]旅游(树剖+线段树)
    洛谷P3250 [HNOI2016]网络(整体二分+树状数组+树剖)
    洛谷P3833 [SHOI2012]魔法树(树链剖分)
    洛谷P4216 [SCOI2015]情报传递(树剖+主席树)
    Ansible 利用playbook批量部署Nginx
    Ansible实现zabbix服务器agent端批量部署
    时间序列模型:ARIMA
  • 原文地址:https://www.cnblogs.com/mzyc/p/10716208.html
Copyright © 2020-2023  润新知