page_pde_offset = (__PAGE_OFFSET >> 20);                /* __PAGE_OFFSET是0xc0000000,page_pde_offset = 3072 = 0xc00,是页目录中的偏移 */
36
37 movl $(pg0 - __PAGE_OFFSET), %edi /* 将pg0对应的物理地址送到edi */
38 movl $(swapper_pg_dir - __PAGE_OFFSET), %edx /* 将swapper_pg_dir(存放临时页全局目录的地址)送到edx */
39 movl $0x007, %eax /* 0x007 = PRESENT+RW+USER */
40 10:
41 leal 0x007(%edi),%ecx /* Create PDE entry —— 构造一个页目录项(地址+标志位),把edi指向的物理地址加上0x007放入ecx中 */
42 movl %ecx,(%edx) /* 第一次循环时把ecx中的内容放入swapper_pg_dir的第0项里,第二次循环时把ecx中的内容放入swapper_pg_dir的第1项里 */
43 movl %ecx,page_pde_offset(%edx) /* Store kernel PDE entry */
44 addl $4,%edx /* edx加上一个页目录项长度(4字节),指向页全局目录的下一个页目录项地址 */
45 movl $1024, %ecx /* 初始化1024个页目录项,设置计数器 */
46 11:
47 stosl /* eax -> [edi]; edi = edi + 4 */
48 addl $0x1000,%eax /* 更改eax的值,为下次stosl作准备 */
49 loop 11b /* 循环操作,其实就是初始化1024个页表项 */
50 /* End condition: we must map up to and including INIT_MAP_BEYOND_END */
51 /* bytes beyond the end of our own page tables; the +0x007 is the attribute bits */
52 leal (INIT_MAP_BEYOND_END+0x007)(%edi),%ebp /* ebp = edi指向的物理地址 + INIT_MAP_BEYOND_END(128K)+ 0x007 */
53 cmpl %ebp,%eax /* 比较ebp与eax */
54 jb 10b /* 如果eax < ebp,则跳到上面10的地方 */
55 movl %edi,(init_pg_tables_end - __PAGE_OFFSET) /* 此时的edi中存放pg0+0x2000,将此值存入init_pg_tables_end中,表示页表初始化结束 */
56

下面的是较详细的解释(结合我的作业题目):

初始化临时内核页表是在startup_32汇编语言函数中完成的。在ULK所述中,假设内核能容纳于RAM的前8MB空间,然后对RAM的前8MB进行恒等映射
(例如用户地址0x00003000映射物理地址0x00003000,0xc0003000映射到物理地址0x00003000),来初始化临时页全局目录swapper_pg_dir和相应的页表。
映射8MB只需要填充swapper_pg_dir中第0项,1项,768项和769项。前两项是给用户线性地址映射,后两项给内核线性地址映射。用页全局目录里的两项就能对8MB映射的
理由是2×1024(页表有1024项)×4K(一页的大小)=8M。实际上初始化内核页表来对RAM的前8MB映射不是个硬性的规定。
这取决于你的内核的配置(我认为大多数情况下是对8MB映射)。在startup_32中可以看到,对多少内存进行映射是通过pg0动态判断的。   linux/arch/i386/kernel/head.S   page_pde_offset = (__PAGE_OFFSET >> 20);   /*__PAGE_OFFSET是0xc0000000,内核线性空间的起始地址。   page_pde_offset=0xc00(十进制为3072)*/   movl $(pg0 - __PAGE_OFFSET), %edi   /*假设pg0的物理地址(00567000),放入edi中。*/   movl $(swapper_pg_dir - __PAGE_OFFSET), %edx   /*swapper_pg_dir的物理地址,放入edx中。*/   movl $PTE-ATTR, %eax   /* PTE-ATTR=0x007 页目录项和页表项的低12位是标志位,把标志位0x007放入eax中。*/   10:   leal PDE-ATTR(%edi),%ecx /* Create PDE entry */   /*第一循环时把edi指向的pg0的物理地址加上PDE-ATTR放入ecx中。   第二次循环时把edi指向的物理地址0x567000加上PDE-ATTR放入ecx中。
  linux2.6/arch/x86/kernel/head_32.S 中的PDE-ATTR=0x067
*/   movl %ecx,(%edx) /* Store identity PDE entry */   /*第一次循环时把ecx中的内容放入swapper_pg_dir的第0项里。   第二次循环时把ecx中的内容放入swapper_pg_dir的第1项里。*/   movl %ecx,page_pde_offset(%edx) /* Store kernel PDE entry */   /*第一次循环时把ecx中的内容放入swapper_pg_dir的第768项里。因为前面算出page_pde_offset的值为3072,而swapper_pg_dir中每项是4个字节,所以3072/4=768。   第二次循环时把ecx中的内容放入swapper_pg_dir的第769项里。*/   addl $4,%edx   /*第一次循环时,此时edx指向swapper_pg_dir的第1项。   第二次循环时,此时edx指向swapper_pg_dir的第2项。*/   movl $1024, %ecx   /*为初始化1024个页表项设置计数*/   11:   stosl   /*把eax中的内容放入edi指向的物理地址中,然后edi+4。*/   addl $0x1000,%eax   loop 11b   /*跳到上面的11处循环。   第一次执行1024次后,从pg0物理地址(0x567000)开始存放的是0x007,0x1007,0x2007,...,0x3ff007,也就是当前能够映射到物理地址从0x000到0x3fffff处。此时edi中的值为0x567000。   第二次执行1024次后,从物理地址(0x568000)开始存放的是0x400007,0x401007,0x402007,...,7ff007,也就是当前能够映射到物理地址0x000到7fffff处,正好8MB。此时edi中的值为0x568000。*/   leal (INIT_MAP_BEYOND_END+PTE-ATTR)(%edi),%ebp   /*INIT_MAP_BEYOND_END的值为128k,在此文件中的一个宏定义。把edi指向的物理地址加上128k加上0x007放入edp中。*/   cmpl %ebp,%eax   /*在第一次循环中ebp中的值为0x567007,eax中的值为0x400007小于0x515007。当前所映射到的最大物理地址为0x3fffff没有包含0x567007,所以没有映射完。   在第二次循环中ebp中的值为0x568007,eax中的值为0x800007大于0x568007。当前所映射到的最大物理地址为0x7fffff包含了0x568007,所以物理地址映射完毕。*/   jb 10b   /*第一次循环做完时跳到上面的10处继续循环   第二次循环做完时跳出循环。*/

题目:

 在32位pc中,结合Linux2.6.26/arch/x86/kernel/head_32.S中228-251行相关代码,关于临时2级页表的初始化过程,假设pg0所在的物理地址是0x567000,回答下列问题(以下涉及到数值的地方,请用16进制表示):
1.填写在swapper_pg_dir中第0x0项的内容是什么,有什么含义?
2.若填写了swapper_pg_dir中第0x1项,则此内容是什么? 
3.填写在pg0的第0x0项、第0x1项和第0x3FF项的内容是多少,有什么含义?
4.根据swapper_pg_dir的第0x0项和pg0的内容,这个临时页表所代表的地址空间中,0~4MB-1的空间被映射的物理地址空间范围是什么?
5.若内核地址空间从3G开始,那么填写在swapper_pd_dir中第0x300项和0x301项的内容是什么,与上述第0项和第1项有什么关系,有什么含义?

综上分析:

1):填写在swapper_pg_dir中的第0项的内容是:0x567067
2):填写在swapper_pg_dir中的第1项的内容是:0x568067
3):填写在pg0的第0项是0x007
   填写在pg0的第1项是0x1007
   填写在pg0的第0x3FF项是0x3FF007

4):0-4M-1的空间对应的物理地址空间为:0x000-0x3fffff

5):swapper_gd_dir的第768项和769项的内容分别为:0x567067和0x568067

和第0项和第1项是相同的。因为第0,1项是给用户线性地址映射的。而前者是为内核线性地址映射的。

中间有什么错误,望同学指出!

临时2级页表的初始化过程 head_32.S 相关代码解释的更多相关文章

  1. linux内存管理解析1----linux物理,线性内存布局及页表的初始化

    主要议题: 1分页,分段模式及实模式 2Linux分页 3linux内存线性地址空间布局及物理内存空间布局 4linux页表初始化及代码解析 1.1.1内存寻址和保护模式 在X86平台上,内存控制单元 ...

  2. java代码的初始化过程研究

        刚刚在ITeye上看到一篇关于java代码初始化的文章,看到代码我试着推理了下结果,虽然是大学时代学的知识了,没想到还能做对.(看来自己大学时掌握的基础还算不错,(*^__^*) 嘻嘻……)但 ...

  3. Java对象相关元素的初始化过程

    1.类的成员变量.构造函数.成员方法的初始化过程 当一个类使用new关键字来创建新的对象的时候,比如Person per = new Person();JVM根据Person()寻找匹配的类,然后找到 ...

  4. linux内存源码分析 - 页表的初始化

    本文为原创,转载请注明:http://www.cnblogs.com/tolimit/ 本文章中系统我们假设为x86下的32位系统,暂且不分析64位系统的页表结构. linux分页 linux下采用四 ...

  5. KVM地址翻译流程及EPT页表的建立过程

    本博文为原创,遵循CC3.0协议,转载请注明出处:http://blog.csdn.net/lux_veritas/article/details/9284635 ------------------ ...

  6. Linux内存管理 (2)页表的映射过程

    专题:Linux内存管理专题 关键词:swapper_pd_dir.ARM PGD/PTE.Linux PGD/PTE.pgd_offset_k. Linux下的页表映射分为两种,一是Linux自身的 ...

  7. 启动期间的内存管理之初始化过程概述----Linux内存管理(九)

    在内存管理的上下文中, 初始化(initialization)可以有多种含义. 在许多CPU上, 必须显式设置适用于Linux内核的内存模型. 例如在x86_32上需要切换到保护模式, 然后内核才能检 ...

  8. ABP中模块初始化过程(二)

    在上一篇介绍在StartUp类中的ConfigureService()中的AddAbp方法后我们再来重点说一说在Configure()方法中的UserAbp()方法,还是和前面的一样我们来通过代码来进 ...

  9. linux文件系统初始化过程(3)---加载initrd(上)

    一.目的 本文主要讲述linux3.10文件系统初始化过程的第二阶段:加载initrd. initrd是一个临时文件系统,由bootload负责加载到内存中,里面包含了基本的可执行程序和驱动程序.在l ...

随机推荐

  1. 优雅地对泛型List 进行深拷贝

    public class People { public string Name; public int Age; public People(string name, int age) { this ...

  2. grep in linux

    1.作用linux系统中grep命令是一种强大的文本搜索工具,它能使用正则表达式搜索文本,并把匹配的行打印出来.grep全称是Global Regular Expression Print,表示全局正 ...

  3. homework-03

    1.分工准备 这次的工作是结对编程,在第二次作业中我是使用python完成的作业,而小明是使用C完成的作业.因为打算使用动态链接库的方式将第二次的代码嵌入到本次的作业中,而python生成动态链接库不 ...

  4. JodaTime用法简介

    JodaTime用法简介 Java的Date和Calendar用起来简直就是灾难,跟C#的DateTime差距太明显了,幸好有JodaTime 本文简单罗列JodaTime的用法 package co ...

  5. 删除MySQL重复数据

    删除MySQL重复数据 项目背景 在最近做的一个linux性能采集项目中,发现线程的程序入库很慢,再仔细定位,发现数据库里面很多冗余数据.因为在采集中,对于同一台设备,同一个时间点应该只有一个数据,然 ...

  6. CSS 文本格式

    整理自:(http://www.w3school.com.cn/css/css_text.asp) Text Color 颜色属性被用来设置文字的颜色. 颜色是通过CSS最经常的指定: 十六进制值 - ...

  7. DNS添加/修改/查询/删除A记录

    #查询DNS可用类 Get-WmiObject -Namespace root\MicrosoftDNS -List #查询所有资源记录 $mydns = [WMIClass]"ROOT\M ...

  8. Java模拟网站登录

    web登陆无非就是网页获取,cookie 的管理,post和get方式的模拟. 1.网页内容获取 java.io.InputStream in; java.net.URL url = new java ...

  9. delphi 01设置 字体属性

    设置/获取 字体属性 名称 大小 粗体 斜体 下划线 删除线 颜色1 颜色2   uses MSHTML;   //设置 //------------------------------------- ...

  10. C# Func&lt;&gt;托付

    曾经我们为了可以调用一个方法.必须比照这种方法定义一个对应的delegate. 原先我们定义delegate // 托付声明 -- 定义一个签名: delegate double MathAction ...