AskOverflow.Dev

AskOverflow.Dev Logo AskOverflow.Dev Logo

AskOverflow.Dev Navigation

  • 主页
  • 系统&网络
  • Ubuntu
  • Unix
  • DBA
  • Computer
  • Coding
  • LangChain

Mobile menu

Close
  • 主页
  • 系统&网络
    • 最新
    • 热门
    • 标签
  • Ubuntu
    • 最新
    • 热门
    • 标签
  • Unix
    • 最新
    • 标签
  • DBA
    • 最新
    • 标签
  • Computer
    • 最新
    • 标签
  • Coding
    • 最新
    • 标签
主页 / coding / 问题 / 79017890
Accepted
Gyo
Gyo
Asked: 2024-09-24 17:25:23 +0800 CST2024-09-24 17:25:23 +0800 CST 2024-09-24 17:25:23 +0800 CST

系统调用号、陷阱处理以及 C 库在系统调用中的作用

  • 772

在阅读OSTEP这本书时,我有一个疑问。

摘自本书:

为了指定确切的系统调用,通常会为每个系统调用分配一个系统调用号。因此,用户代码负责将所需的系统调用号放在寄存器中或堆栈上的指定位置;操作系统在陷阱处理程序中处理系统调用时,会检查此号码,确保其有效,如果有效,则执行相应的代码。这种间接级别是一种保护形式;用户代码无法指定要跳转到的确切地址,而必须通过号码请求特定服务。

我试图更好地理解系统调用号在系统调用和陷阱处理上下文中的工作方式。以下是我目前所理解的内容:

  1. 系统调用号的作用类似于陷阱表中的键或索引,允许 CPU 跳转到正确的陷阱处理程序。
  2. 在陷阱处理程序内部,操作系统验证系统调用号。
  3. C 库的系统调用 API 在调用陷阱之前将系统调用号存储在寄存器或堆栈中。

我的理解准确吗?有人可以澄清或提供其他详细信息吗,尤其是关于 C 库的作用和验证步骤?

c
  • 2 2 个回答
  • 83 Views

2 个回答

  • Voted
  1. John Bollinger
    2024-09-24T22:36:11+08:002024-09-24T22:36:11+08:00

    有关大多数详细信息,请参阅@MikeNakis 的回答。我只想回应这一​​点:

    C 库的系统调用 API 在调用陷阱之前将系统调用号存储在寄存器或堆栈中。

    这有点令人担忧。也许这只是措辞不当,但如果这里确实存在误解,那么:

    您的书中描述的系统调用设计与任何用户空间库都没有直接关系。原则上,任何用户空间程序都可以通过引用中描述的机制的适当操作系统特定版本来请求系统调用,而无需借助库函数。

    但是,如果您的系统提供托管的 C 实现,那么它提供的 C 标准库中的许多函数确实会执行所描述的系统调用。此外,您可能特别想到某些系统的 C 实现作为扩展提供的系统调用包装函数。这些函数除了为特定系统调用提供方便的接口外,几乎没有其他作用。此外,某些系统的 C 标准库提供了一个通用的系统调用包装函数,要求调用者指定系统调用编号并以原始系统调用所需的形式提供适当的参数。除了最后一个之外,我认为所有这些都不能很好地描述为“系统调用 API”。

    • 2
  2. Best Answer
    Mike Nakis
    2024-09-24T20:38:53+08:002024-09-24T20:38:53+08:00
    1. 系统调用号的作用类似于陷阱表中的键或索引,允许 CPU 跳转到正确的陷阱处理程序。

    不。

    在支持指令的 CPU 架构中可以找到“陷阱表”,TRAP N其中N是陷阱编号。在此类架构中,陷阱表将陷阱编号映射到陷阱处理程序的地址。操作系统通常定义一个陷阱编号来提供系统调用。

    假设我们有一个支持多个陷阱处理程序的假想 CPU,以及一个用于系统调用的操作系统TRAP 42。

    系统调用号是调用操作系统陷阱之前必须设置的。例如:

    LOAD R1, system_call_number
    TRAP 42
    

    因此,CPU 使用它42在陷阱表中查找陷阱处理程序,并跳转到该处理程序。这将是操作系统陷阱处理程序。在该处理程序中,操作系统使用的值在另一个R1表(称为“系统调用”表)中查找系统函数的地址,并跳转到该地址。

    请注意,如果 CPU 支持使用无参数指令来调用操作系统,则该过程非常相似:

    LOAD R1, system_call_number
    SYSCALL
    

    在这种情况下,只有一个处理程序,没有陷阱表。但是,仍然可以进行许多不同的系统调用,因此可以通过系统调用表解析系统调用号的机制仍然存在。

    1. 在陷阱处理程序内部,操作系统验证系统调用号。

    嗯,当然。如果系统调用号与实际有效的系统函数不对应,操作系统可能会对您的进程造成不良影响。

    1. C 库的系统调用 API 在调用陷阱之前将系统调用号存储在寄存器或堆栈中。

    我不知道您说的“C 库的系统调用 API”是什么意思。无论如何,任何有兴趣调用系统调用的人都必须在调用操作系统之前将系统调用号存储在某个地方。操作系统定义了应该将数字存储在何处。它通常在寄存器中。

    • 1

相关问题

  • 比 * 更快的乘法

  • 在 C 中的 scanf() 格式说明符中使用宏获取字符串长度

  • 如何将#define的数据类型设置为long double?

  • 不兼容的常量指针

  • OpenGL 中的非渐变颜色变化

Sidebar

Stats

  • 问题 205573
  • 回答 270741
  • 最佳答案 135370
  • 用户 68524
  • 热门
  • 回答
  • Marko Smith

    Vue 3:创建时出错“预期标识符但发现‘导入’”[重复]

    • 1 个回答
  • Marko Smith

    为什么这个简单而小的 Java 代码在所有 Graal JVM 上的运行速度都快 30 倍,但在任何 Oracle JVM 上却不行?

    • 1 个回答
  • Marko Smith

    具有指定基础类型但没有枚举器的“枚举类”的用途是什么?

    • 1 个回答
  • Marko Smith

    如何修复未手动导入的模块的 MODULE_NOT_FOUND 错误?

    • 6 个回答
  • Marko Smith

    `(表达式,左值) = 右值` 在 C 或 C++ 中是有效的赋值吗?为什么有些编译器会接受/拒绝它?

    • 3 个回答
  • Marko Smith

    何时应使用 std::inplace_vector 而不是 std::vector?

    • 3 个回答
  • Marko Smith

    在 C++ 中,一个不执行任何操作的空程序需要 204KB 的堆,但在 C 中则不需要

    • 1 个回答
  • Marko Smith

    PowerBI 目前与 BigQuery 不兼容:Simba 驱动程序与 Windows 更新有关

    • 2 个回答
  • Marko Smith

    AdMob:MobileAds.initialize() - 对于某些设备,“java.lang.Integer 无法转换为 java.lang.String”

    • 1 个回答
  • Marko Smith

    我正在尝试仅使用海龟随机和数学模块来制作吃豆人游戏

    • 1 个回答
  • Martin Hope
    Aleksandr Dubinsky 为什么 InetAddress 上的 switch 模式匹配会失败,并出现“未涵盖所有可能的输入值”? 2024-12-23 06:56:21 +0800 CST
  • Martin Hope
    Phillip Borge 为什么这个简单而小的 Java 代码在所有 Graal JVM 上的运行速度都快 30 倍,但在任何 Oracle JVM 上却不行? 2024-12-12 20:46:46 +0800 CST
  • Martin Hope
    Oodini 具有指定基础类型但没有枚举器的“枚举类”的用途是什么? 2024-12-12 06:27:11 +0800 CST
  • Martin Hope
    sleeptightAnsiC `(表达式,左值) = 右值` 在 C 或 C++ 中是有效的赋值吗?为什么有些编译器会接受/拒绝它? 2024-11-09 07:18:53 +0800 CST
  • Martin Hope
    The Mad Gamer 何时应使用 std::inplace_vector 而不是 std::vector? 2024-10-29 23:01:00 +0800 CST
  • Martin Hope
    Chad Feller 在 5.2 版中,bash 条件语句中的 [[ .. ]] 中的分号现在是可选的吗? 2024-10-21 05:50:33 +0800 CST
  • Martin Hope
    Wrench 为什么双破折号 (--) 会导致此 MariaDB 子句评估为 true? 2024-05-05 13:37:20 +0800 CST
  • Martin Hope
    Waket Zheng 为什么 `dict(id=1, **{'id': 2})` 有时会引发 `KeyError: 'id'` 而不是 TypeError? 2024-05-04 14:19:19 +0800 CST
  • Martin Hope
    user924 AdMob:MobileAds.initialize() - 对于某些设备,“java.lang.Integer 无法转换为 java.lang.String” 2024-03-20 03:12:31 +0800 CST
  • Martin Hope
    MarkB 为什么 GCC 生成有条件执行 SIMD 实现的代码? 2024-02-17 06:17:14 +0800 CST

热门标签

python javascript c++ c# java typescript sql reactjs html

Explore

  • 主页
  • 问题
    • 最新
    • 热门
  • 标签
  • 帮助

Footer

AskOverflow.Dev

关于我们

  • 关于我们
  • 联系我们

Legal Stuff

  • Privacy Policy

Language

  • Pt
  • Server
  • Unix

© 2023 AskOverflow.DEV All Rights Reserve