免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
最近访问板块 发新帖
查看: 32427 | 回复: 4
打印 上一主题 下一主题

[CPU及多核] 『求助』 关于 smp_processor_id() [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2012-08-06 10:58 |只看该作者 |倒序浏览
首先描述一下环境,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() 函数中拷贝                                                                                                                                                                           
vmlinux 中的每处理器变量,每个处理器都有一份拷贝,并将地址保存到                                                                                                                                                                              
__per_cpu_offset数组中。                                                                                                                                                                                                                       
                                                                                                                                                                                                                                               
1. 那么我的问题来了,此时每个CPU 的每 cpu 变量岂不都相同?比如                                                                                                                                                                                 
cpu_number 都相同,还是每个 cpu 都会去初始化自己的本地每CPU变量,比如                                                                                                                                                                          
cpu 0 将 per_cpu__cpu_number 设置为 0                                                                                                                                                                                                         
cpu 1 将 per_cpu__cpu_number 设置为 1.                                                                                                                                                                                                         
                                                                                                                                                                                                                                               
                                                                                                                                                                                                                                               
在 do_boot_cpu() 中 cpu0(BP) 会首先为每个 AP 建立 0 号进程,然后发送 IPI给                                                                                                                                                                     
AP,AP 在载入 gdt,idt 之后回根据 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 是一个段选择子,他的 GDT 段描述符中的地址字段 和 per_cpu_offset 是什么关系?它是在什么时候被设置的?换句话说 gs:per_cpu__cpu_number 是怎么定位到每 CPU 变量的本地拷贝的 ?                                       
   smp_processor_id() 是怎么最终获得 cpu 编号的? 比如 CPU 2 .               


非常感谢!!!!                                                                                    

论坛徽章:
6
金牛座
日期:2013-10-08 10:19:10技术图书徽章
日期:2013-10-14 16:24:09CU十二周年纪念徽章
日期:2013-10-24 15:41:34狮子座
日期:2013-11-24 19:26:19未羊
日期:2014-01-23 15:50:002015年亚洲杯之阿联酋
日期:2015-05-09 14:36:15
2 [报告]
发表于 2012-08-06 13:04 |只看该作者
回复 1# zd零
. 那么我的问题来了,此时每个CPU 的每 cpu 变量岂不都相同?比如                                                                                                                                                                                 
cpu_number 都相同,还是每个 cpu 都会去初始化自己的本地每CPU变量,比如                                                                                                                                                                          
cpu 0 将 per_cpu__cpu_number 设置为 0                                                                                                                                                                                                         
cpu 1 将 per_cpu__cpu_number 设置为 1.                                                                                                                                                                                                         


既然是每cpu变量,当然是每个cpu都有单独的本地CPU变量。

论坛徽章:
0
3 [报告]
发表于 2012-08-06 14:17 |只看该作者
回复 2# 瀚海书香


是啊,肯定每个 cpu 都有自己的单独的 CPU 变量啊,这一点我已经说了,我的问题是:
1. 可以肯定的是每个 CPU 的 per cpu 变量肯定都是不同的,比如 per_cpu__cpu_number,cpu 0 它的值为 0 ,cpu 1 它的值为 1,那我想问的是这个值是在何时被修改的呢?
2. movl %%gs:%P1, %0 是怎么完成对 current cpu 的 per_cpu__cpu_number 读取 ?

论坛徽章:
8
羊年新春福章
日期:2015-03-19 02:03:312015亚冠之北京国安
日期:2015-06-16 22:04:45程序设计版块每日发帖之星
日期:2015-06-23 22:20:00每日论坛发贴之星
日期:2015-06-23 22:20:002015亚冠之首尔
日期:2015-06-24 19:18:072015亚冠之广州恒大
日期:2015-08-06 10:29:442015亚冠之柏太阳神
日期:2015-11-02 11:21:0515-16赛季CBA联赛之辽宁
日期:2015-12-09 15:05:02
4 [报告]
发表于 2019-01-30 00:53 |只看该作者
本帖最后由 firocu 于 2019-01-30 01:29 编辑

下面初始化 GS BASE 的代码是 所有secondary cpu串行 执行的, 所以不会有问题。        movl    $MSR_GS_BASE,%ecx
        movl    initial_gs(%rip),%eax
        movl    initial_gs+4(%rip),%edx
        wrmsr

具体串行方法, check cpu_callin_mask in do_boot_cpu() and smp_callin() in start_secondary()





论坛徽章:
0
5 [报告]
发表于 2021-05-03 17:53 |只看该作者
2.6.25用的是fs,原理应该差不多。都是记录偏移,然后定位,类似重定位。
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP