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. 设计模式系列 1——StaticFactory(静态工厂),AbstractFactory(抽象工厂)

    本文出自 代码大湿 代码大湿 本系列持续更新,敬请关注. 1 静态工厂 静态工厂介绍: 静态工厂模式可以实现接口封装隔离的原则.在客户端只知接口而不知实现的时候可以使用静态工厂模式. 源码请点击我 角 ...

  2. flash图标插件

    http://pullmonkey.com/projects/open_flash_chart2/

  3. 第二百九十三天 how can I 坚持

    总感觉怪怪的,换了个领导,好烦,虽然对我没用影响. 其实,还是智商低,不懂人情世故,就像...算了,不说了,只能当自己傻. 最近好冷啊,十年不遇的寒冬. 心情有些压抑. 不玩游戏了,看了集康熙来了.小 ...

  4. Java邮件服务学习之三:邮箱服务客户端-Java Mail

    一.java mail的两个JAR包 1.mail.jar:不在JDK中,核心功能依赖JDK4及以上,该jar包已经加入到java EE5: 下载地址:http://www.oracle.com/te ...

  5. 转载JSON格式化工具

    格式化工具地址 http://tool.liuxianan.com/

  6. [Windows驱动开发](一)序言

    笔者学习驱动编程是从两本书入门的.它们分别是<寒江独钓——内核安全编程>和<Windows驱动开发技术详解>.两本书分别从不同的角度介绍了驱动程序的制作方法. 在我理解,驱动程 ...

  7. PHP+MySQL多语句执行<转自wooyun>

    发起这个帖子,估计就很多人看到题目就表示不屑了.一直以来PHP+MySQL环境下,无论是写程序或者是注入攻击,是无法多语句执行的,这么广为人知的常识,没理由会有人不知道.可权威就是用来被挑战的,常识也 ...

  8. 从零开始学android开发-setBackgroundDrawable与setBackgroundResource的区别

    setBackgroundDrawable和setBackgroundResource的区别很多网友不知道View类提供的setBackgroundDrawable和setBackgroundReso ...

  9. Codeforces Round #282 (Div. 1) A. Treasure 水题

    A. Treasure Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://codeforces.com/contest/494/problem/A ...

  10. HDU 5512 Meeting 博弈论

    Meeting Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://acm.hdu.edu.cn/showproblem.php?pid=5512 ...