xv6是一个支持多处理器的Unix-like操作系统,

近日阅读源码时发现xv6在记录当前CPU和进程状态时非常tricky

首先,上代码:

  1. extern struct cpu cpus[NCPU];
  2. extern int ncpu;
  3.  
  4. // Per-CPU variables, holding pointers to the
  5. // current cpu and to the current process.
  6. // The asm suffix tells gcc to use "%gs:0" to refer to cpu
  7. // and "%gs:4" to refer to proc. seginit sets up the
  8. // %gs segment register so that %gs refers to the memory
  9. // holding those two variables in the local cpu's struct cpu.
  10. // This is similar to how thread-local variables are implemented
  11. // in thread libraries such as Linux pthreads.
  12. extern struct cpu *cpu asm("%gs:0"); // &cpus[cpunum()]
  13. extern struct proc *proc asm("%gs:4"); // cpus[cpunum()].proc

其中struct cpu是一个用来保存每个cpu运行状态的结构体,

代码第一行定义了结构体数组cpus[NCPU],NCPU对应cpu的总数(最大为8),也就是说cpus用来存储所有cpu的运行状态

那么问题来了:上面的内核代码是运行于每个cpu之中的,那每个cpu如何知道自身的当前运行状态呢?

对于这个问题,我们可以通过lapic获取cpu自身编号,再利用编号对cpus寻址即可,

也就是说,对于任意一个cpu,自身状态的存储位置可以这样获得: struct cpu *c = &cpus[cpunum()];

然而,第二个问题来了:我们不可能每次引用cpu自身状态时都通过lapic获取编号啊,能不能弄一个全局变量把状态位置一次性存储下来呢?

像是这样, struct cpu *cpu; //全局变量,存储cpu自身状态 ,然后在初始化代码中 cpu = c ;

对于记录每个cpu正在运行的进程也有这样的问题,能不能写成: struct proc *proc; //全局变量,存储当前cpu正在运行的进程状态

那么,第三个问题来了:每个cpu是独立并行的,在每个cpu上运行的内核代码都是一样的,页表也一样,

这意味着全局变量cpu和proc的地址也是一样的,这样便不可以用来区分不同cpu的状态了。

因此,我们需要一种方法,可以让我们在每个cpu中都用同一个符号记录状态,但这些符号却是映射到不同的地址。

既然页表一样,我们自然不能用一个绝对的数值来寻址啦,仔细想想,页表之上有什么?页表之上,还有段表啊。

所以我们需要用segment register来寻址,只要我们在建立段表时把该段都映射到不同的内存区域不就可以了,所以我们有了以下声明:

  1. extern struct cpu *cpu asm("%gs:0"); // &cpus[cpunum()]
  2. extern struct proc *proc asm("%gs:4"); // cpus[cpunum()].proc

我们用gs作为段寄存器,cpu指向[%gs],proc指向[%gs+4],

其中为什么开头要用extern呢?我问过某大神,他说是因为gs段是在外部建立的,相当于外部定义的。。。

OK,最后一个问题来了,gs段应该指向哪,才能确保每个cpu的gs段都位于不同的区域?

最直观的想法当然是指向对应的cpus[num]内部啦,所以在struct cpu尾部增加两个域:

  1. struct cpu{
  2. ........ //cpu状态
  3.  
  4. // Cpu-local storage variables; see below
  5. struct cpu *cpu;
  6. struct proc *proc; // The currently-running process.
  7. }

然后在建立段表时,增加gs段,并映射至尾部这两个域:

  1. c = &cpus[cpunum()];
  2.  
  3. ......... //建立其他段
  4.  
  5. // 建立gs段,共两个域(存储cpu和proc地址),起始地址为&c->cpu
  6. c->gdt[SEG_KCPU] = SEG(STA_W, &c->cpu, , );
  7.  
  8. //加载gdt
  9. lgdt(c->gdt, sizeof(c->gdt));
  10. //加载gs
  11. loadgs(SEG_KCPU << );
  12.  
  13. // 把当前cpu和proc状态的地址赋给cpu和proc全局变量
  14. //而cpu变量实质为%gs:0, proc变量实质为%gs:4
  15. cpu = c;
  16. proc = ;

其实在这里cpu和proc变量跟线程局部存储的性质差不多,每个处理器都可以引用同一个变量,但这些变量都对应不同的存储区域。

有可能这种实现技巧跟TLS(线程局部存储)差不多,有空研究下TLS的实现看看是不是。

xv6中存储cpu和进程信息的技巧的更多相关文章

  1. 从零自学Java-3.在程序中存储和修改变量信息

    1.创建变量: 2.使用不同类型的变量: 3.在变量中存储值: 4.在数学表达式中使用变量: 5.把一个变量的值赋给另一个变量: 6.递增/递减变量的值. 程序Variable:使用不同类型的变量并赋 ...

  2. MySQL 调优基础(一) CPU与进程

    一般而言,MySQL 的调优可以分为两个层面,一个是在MySQL层面上进行的调优,比如SQL改写,索引的添加,MySQL各种参数的配置:另一个层面是从操作系统的层面和硬件的层面来进行调优.操作系统的层 ...

  3. linux中使用top获取进程的资源占用信息

    在linux中使用top获取进程的资源占用信息: Cpu(s):  1.0%us,  0.0%sy,  0.0%ni, 98.3%id,  0.7%wa,  0.0%hi,  0.0%si,  0.0 ...

  4. 【Linux】 linux中的进程信息相关的一些内容

    _ linux进程信息 ■ top top命令用于动态地查看系统的进程和其他一些资源的信息.开启top的时候可以加上-t <sec>来设置top更新的频率高低.进入top界面之后,可以输入 ...

  5. Android中获取系统内存信息以及进程信息-----ActivityManager的使用(一)

    本节内容主要是讲解ActivityManager的使用,通过ActivityManager我们可以获得系统里正在运行的activities,包括 进程(Process)等.应用程序/包.服务(Serv ...

  6. 在Linux中通过Top运行进程查找最高内存和CPU使用率

    按内存使用情况查找前15个进程,在批处理模式下为"top" 使用top命令查看有关当前状态,系统使用情况的更详细信息:正常运行时间,负载平均值和进程总数. 分类:Linux命令操作 ...

  7. C#获取CPU占用率、内存占用、磁盘占用、进程信息

    代码: using System; using System.Collections.Generic; using System.Diagnostics; using System.Threading ...

  8. JAVA 在程序中存储和修改信息

    1.语句和表达式 计算机程序是一组告诉计算机什么的指令,每一个指令称为语句. 2.指定变量类型 变量名.变量存储的信息类型 整型int(-2.14*109~2.14*109).浮点型float(38位 ...

  9. Linux中查看CPU信息【转】

    [转自]:http://blog.chinaunix.net/uid-23622436-id-3311579.html cat /proc/cpuinfo中的信息 processor       逻辑 ...

随机推荐

  1. HDU-2126 Buy the souvenirs

    数组01背包. http://acm.hdu.edu.cn/showproblem.php?pid=2126 http://blog.csdn.net/crazy_ac/article/details ...

  2. Linux学习笔记15——GDB 命令详细解释【转】

    GDB 命令详细解释 Linux中包含有一个很有用的调试工具--gdb(GNU Debuger),它可以用来调试C和C++程序,功能不亚于Windows下的许多图形界面的调试工具. 和所有常用的调试工 ...

  3. unity3d AI's sight

    just finished -----by wolf96

  4. HTML---网页编程(1)

    前 言 HTML需要和CSS还有JS一起用,才能提现强大. 所以,学了HTML.最好去学学CSS还有JS(JavaScript) ☆静态页面和动态页面 网站页面分为静态页面和动态页面两种 • 静态页面 ...

  5. 你不一定能做对的JavaScript闭包面试题

    由工作中演变而来的面试题 这是一个我工作当中的遇到的一个问题,似乎很有趣,就当做了一道题去面试,发现几乎没人能全部答对并说出原因,遂拿出来聊一聊吧. 先看题目代码: function fun(n,o) ...

  6. winform 跨窗体给控件传值 分类: WinForm 2014-08-02 16:33 195人阅读 评论(0) 收藏

    两个窗体 FormA,FormB; FormA窗体中有一文本框控件:TextBox; FormB窗体中有一变量:txtJSJ 目的:把变量赋值给文本框 实现: 设置TextBox属性: Modifie ...

  7. javascript 函数参数

    1.javascript函数参数的个数以及类型没有强制规定,调用时不必严格按照函数的参数或类型,函数的参数只是在调用函数的时候提供了便利,但不是必须的! 2.参数在javascript内部是用数组ar ...

  8. 构建ASP.NET MVC4+EF5+EasyUI+Unity2.x注入的后台管理系统(29)-T4模版

    原文:构建ASP.NET MVC4+EF5+EasyUI+Unity2.x注入的后台管理系统(29)-T4模版 这讲适合所有的MVC程序 很荣幸,我们的系统有了体验的地址了.演示地址 之前我们发布了一 ...

  9. BFS-hdu-4101-Ali and Baba

    题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=4101 题目大意: 给一个矩阵,0表示空的可走,-1宝藏的位置(只有一个),其余的正整数表示该位置石头 ...

  10. Android 颜色渲染(十) ComposeShader组合渲染

    版权声明:本文为博主原创文章,未经博主允许不得转载. 目录(?)[+] Android 颜色处理(十) ComposeShader组合渲染 public ComposeShader(Shader sh ...