- 論壇徽章:
- 0
|
首先描述一下環(huán)境,X86_64,SMP system.
smp_processor_id() 可以最終化減成如下形式:
asm("movl %%gs:%P1, %0"
:"=q"(ret__)
:"m"(per_cpu__cpu_number));
DEFINE_PER_CPU(int, cpu_number),在 setup_per_cpu_areas() 函數(shù)中拷貝
vmlinux 中的每處理器變量,每個(gè)處理器都有一份拷貝,并將地址保存到
__per_cpu_offset數(shù)組中。
1. 那么我的問題來了,此時(shí)每個(gè)CPU 的每 cpu 變量豈不都相同?比如
cpu_number 都相同,還是每個(gè) cpu 都會(huì)去初始化自己的本地每CPU變量,比如
cpu 0 將 per_cpu__cpu_number 設(shè)置為 0
cpu 1 將 per_cpu__cpu_number 設(shè)置為 1.
在 do_boot_cpu() 中 cpu0(BP) 會(huì)首先為每個(gè) AP 建立 0 號(hào)進(jìn)程,然后發(fā)送 IPI給
AP,AP 在載入 gdt,idt 之后回根據(jù) thread.eip 跳到 start_secondary 中,可是
在 do_boot_cpu() 中有這么一句:
initial_gs = per_cpu_offset(cpu);
然后在 arch/x86/kernel/head_64.S 中(有些不理解它的作用):
movl $MSR_GS_BASE,%ecx
movq initial_gs(%rip),%rax
movq %rax,%rdx
shrq $32,%rdx
wrmsr
2. gs:per_cpu__cpu_number ,gs 是一個(gè)段選擇子,他的 GDT 段描述符中的地址字段 和 per_cpu_offset 是什么關(guān)系?它是在什么時(shí)候被設(shè)置的?換句話說 gs:per_cpu__cpu_number 是怎么定位到每 CPU 變量的本地拷貝的 ?
smp_processor_id() 是怎么最終獲得 cpu 編號(hào)的? 比如 CPU 2 .
非常感謝。。。 |
|