最近在看Linux源代码,Linux是将所有处理器的任务状态段存放在init_tss这一结构中,不料Linux内核将init_tss定义percpu data,所以深入对percpu data进行分析。以下是我的分析过程,其中我觉得最重要的就是那幅图和最后的计算分析。
介绍:在多处理器中,为了保证内存访问的正确性,保护某个cpu私有的数据不被其他cpu所访问,同时避免CPU同步的发生;将这些数据存放到指定的字节(section中,详见AT&T汇编,我在这里就将这一区域称为镜像),然后为每个CPU保留这个镜像的副本,每个cpu只能访问它自己的副本。下面以init_tss为例子详细讲解percpu数据。
init_tss数组中存放的是tss。在Linux内核中,每个cpu均有一个独立的全局段描述符表(GDT),每个GDT里面有唯一一个任务状态段(TSS),而所有CPU的TSS均存储在init_tss中。在Linux中,TSS段只用于在进程由用户态向内核态切换时获得内核栈的地址和I/O访问时的权限确定过程中(详情参见ULK中第三章进程中的任务状态段);这并不像原始的Intel设计那样,将任务切换的现场保存在TSS中,而是需要额外的字段(thread_struct)来保存进程切换的现场(详情参见ULK中第三章进程中的任务状态段)。
源代码分析:
首先在vimlinuz.lds.S中,为所有的“percpu”变量定义一个独立的字节