我有一个关于Linux深入工作的问题。
假设在CPU中正在执行多线程进程。在这种情况下,我们将有一个在CPU上执行的线程。在更广泛的图片中,我们将属于Process的相应页面加载到RAM中以供执行。
让我们说线程进行系统调用。我对此之后的运作有点不清楚。中断将生成一个呼叫。我的一个问题是谁将接听这个电话?
让我们说系统有m:n用户级线程到内核级线程映射,我假设相应的内核级线程将回答此调用。
因此,内核将查找中断向量表并获取需要执行的例程。我的下一个问题是在执行中断时将使用哪个堆栈?它是内核线程的堆栈还是用户级别的线程堆栈? (我假设它将是内核线程的堆栈。)
回到程序流程,可以说操作是使用fopen
打开文件。我接下来的问题是如何从ISR跳转到系统调用?或者我们的ISR是否映射到系统调用?
当内核线程正在执行时,我还假设RAM上的“OS区域”将用于容纳正在执行系统调用的页面。
再次从不同的角度看待它(希望你还在我身边)最后我假设相应的内核线程正在由CPU调度程序处理,其中在上下文切换中会发生从用户级线程到相应的正在回答fopen
系统调用时的内核级线程。
我已经做了很多假设,如果有人能够清除疑虑或者至少引导我朝着正确的方向前进,那绝对是太棒了。
答案 0 :(得分:7)
注意:我主要使用ARM计算机,因此其中一些可能是ARM特定的。另外,我将尽可能地尝试简化它。随意纠正任何可能错误或过度简化的内容。
让我们说线程进行系统调用。我对此之后的运作有点不清楚。中断将生成一个呼叫。我的一个问题是谁将接听这个电话?
通常,处理器将在内核模式的某个预定位置开始执行。内核将保存当前进程状态并查看用户空间寄存器以确定请求了哪个系统调用并将其分派给正确的系统调用处理程序。
因此,内核将查找中断向量表并获取需要执行的例程。我的下一个问题是在执行中断时将使用哪个堆栈?它是内核线程的堆栈还是用户级别的线程堆栈? (我假设它将是内核线程的堆栈。)
我很确定它会切换到内核堆栈。如果使用用户空间堆栈,那么信息泄漏会有一些非常严重的安全问题。
回到程序流程,可以说操作是使用fopen打开文件。我接下来的问题是如何从ISR跳转到系统调用?或者我们的ISR是否映射到系统调用?
fopen()
实际上是一个libc函数,而不是系统调用本身。它可能(并且在大多数情况下会)在其实现中调用open()
系统调用。
所以,过程(粗略地)是:
fopen()
fopen
对open()
当内核线程正在执行时,我还假设RAM上的“OS区域”将用于容纳正在执行系统调用的页面。
页面不执行任何操作:)通常,在Linux中,映射到0xC0000000以上的任何地址都属于内核。
再次从不同的角度看待它(希望你还在我身边)最后我假设相应的内核线程正在由CPU调度程序处理,其中在上下文切换中会发生从用户级线程到相应的内核级别线程在应答fopen系统调用时。
使用抢占式内核,有效的线程不会受到歧视。根据我的理解,没有为服务系统调用创建新线程 - 它只是在请求系统调用的同一线程中运行,除了内核模式。
这意味着处于为系统调用提供服务的内核模式的线程可以与任何其他线程一样进行调度。因此,这是您在为内核开发时听到“用户空间上下文”的地方。这意味着它在用户模式线程上以内核模式执行。
这有点难以解释,所以我希望我做对了。