ucore Lab2

lab 2 直接执行make qemu-nox会显示 assert 失败:

kernel panic at kern/mm/default_pmm.c:277:
assertion failed: (p0 = alloc_page()) == p2 - 1

1 连续物理内存管理

1.1 page 概览

对物理内存的管理,为了节省空间,也是为了配合接下来的虚拟内存管理,通常以某个比 byte 大一些的单位进行管理,我们称这一单位内存为一"页(page)",通常是 4KB.待 pages 初始化完毕后,物理内存示意图如下:

其中绿色代表可以分配的内存,红色代表不可被分配的内存.注意,ucore规定物理内存可用范围最大不超过KERNSIZE.函数page_init的主要作用就是初始化pages也就是所有page的所有信息.

注意,pages以全局指针的形式存在,因为最开始无法知道 page 的数量,所以无法写成数量确定的数组.此数量必须尽快确认,否则后期无法管理.

如何确定 page 的数量npage呢?

1.1 探测物理内存布局,获取 pages 大小

npages可由最大物理内存边界/PGSIZE 得出.

而最大物理内存边界可以借助 BIOS 可以探测并计算出来,参考探测系统物理内存布局实现物理内存探测.可以获取到最大可用物理内存边界maxpa, 但maxpa 最终必须<=KMEMSIZE.

于是确认了npagepages:

npage = maxpa / PGSIZE;
pages = (struct Page *)ROUNDUP((void *)end, PGSIZE);// 从end向上取整页

1.2 确定每个 page 的状态

每个page 的状态即其四个字段:

struct Page {
int ref; // 虚存引用计数, 实际上是被页表项引用的数量.每有一个页表项指向此 page,ref 就+1
uint32_t flags; // 当前页状态位.解释见下
unsigned int property; // (对于 first fit)用于可用区域内存的第一个 page,记录其之后有多少个 page 是 free 的
list_entry_t page_link; // 上/下一个空闲 page
};

对于状态的解释 : 在当前阶段,此字段理解为是否可用.通过SetPageReserved来标记其为保留(即不可用)的.

注意使用free_list作为双良链表的表头,应使用相关函数维护好关系.

参考以页为单位管理物理内存

练习 1 重写内存管理函数

参考 严蔚敏老师的 数据结构(C 语言版) 8.2 节 "首次拟合法".

first fit 算法要求空闲 block 按起始地址有序排列.

default 系列的内存管理函数,注释已经描述的非常清楚了,这里再描述下算法原理.

first-fit 的分配算法, 核心步骤如下:

  1. 找到 property > n 的节点 base_page
  2. 从 base_page 步进 n 个 page 找到 p
  3. 设置 p 的 property,值为 property - n
  4. 把 p 作为新节点直接加入链表
  5. 删除 base_page

释放的情况比较多,释放page后有四种情况:

  1. 空闲块与其之前的空闲块相邻,与之后的空闲块不相邻
  2. 空闲块与其之前的空闲块相邻,与之后的空闲块相邻
  3. 空闲块与其之前的空闲块不相邻,与之后的空闲块不相邻
  4. 空闲块与其之前的空闲块相邻,与之后的空闲块也相邻

练习 2 实现寻找虚拟地址对应的页表项

完成函数:

pte_t *
get_pte(pde_t *pgdir, uintptr_t la, bool create) {

此函数的作用是向上提供一个几乎透明的操作,返回指定 linear address 对应的 page table entry.注意写这个函数时有坑,此函数并非要初始化对应的页表项,而是单纯的获取其地址! 初始化页表项的在boot_map_segment中进行.

  1. 得到 page dir 的 index: PDX(la),于是对应的 pde 是 pgdir[PDX(la)].

  2. 考察 pde 具体的结构:

高 20 位是 page table 地址.

pgdir[PSX(la)]& ~0xFFF即可得到 page table 的物理地址.

在取值之前,首先要判断存在位PTE_P.

  1. 如果存在,说明之前已经初始化过了 pd,只需计算la对应的页表项即可.
  2. 如果不存在,说明当前 page directory entry 除了 PTE_P 外都是空的!更别说页表了,肯定也是不存在.此时需要新申请一块物理内存,作为新的页表.但如果 create=0 的话就直接返回 NULL 就行了.

当前的页目录是__boot_pgdir,而我们之前只初始化了虚拟地址的[0,4M)KERNBASE + [0 ~ 4M)的页表,而未初始化其他部分.其他部分是 0.新申请page作为页表之后,只是把页表的(物理)基址写到 pd 项中而已.此函数可能更新pd,但不会更新pt.

pte_t *
get_pte(pde_t *pgdir, uintptr_t la, bool create) {
// 1. 由线性地址取page directory 中对应的条目
pde_t *pdep = &pgdir[PDX(la)];
// 2.1 若存在位为 0,则需要判断 create 选项.
if (!(*pdep & PTE_P)) {
struct Page *page;
// 2.1.1 若 create=0 则返回 NULL
if (!create)
return NULL;
// 2.1.2 若 create=1 则分配一块物理内存,作为新的页表
if (page = alloc_page() == NULL) {
return NULL;
}
// 2.1.3 设置此 page 的引用计数
set_page_ref(page, 1);
// 2.1.4 修改 page directory 项的标志位,把新页表地址写入此项.
uintptr_t pa = page2pa(page);
memset(KADDR(pa), 0, PGSIZE);
*pdep = pa | PTE_U | PTE_W | PTE_P;
}
// 2.2 若存在位不为 0,则返回页表项地址.
// 1. 对 *pdep 取高 20 位得到页表(物理)基址
// 2. 用KADDR将页表物理基址换算为内核虚拟地址
// 2. 从页表虚拟基址取 PTX(la) 个偏移量得到页表项,返回它的地址.
return &((pte_t *)KADDR(PDE_ADDR(*pdep)))[PTX(la)];
}

练习3 释放某虚地址所在的页并取消对应二级页表项的映射

编写函数:

// 释放给定页表ptep关联的page
// 去使能地址 la 对应的 tlb.
static inline void
page_remove_pte(pde_t *pgdir, uintptr_t la, pte_t *ptep)

用到函数:page_ref_dec, 即减少引用.

    // 排除页表不存在的情况
if (*ptep & PTE_P) {
struct Page *page = pte2page(*ptep);
if (page_ref_dec(page) == 0) {
free_page(page);
}
*ptep = 0;
tlb_invalidate(pgdir, la);
}

ucore Lab2 实验笔记的更多相关文章

  1. ucore操作系统学习笔记(二) ucore lab2物理内存管理分析

    一.lab2物理内存管理介绍 操作系统的一个主要职责是管理硬件资源,并向应用程序提供具有良好抽象的接口来使用这些资源. 而内存作为重要的计算机硬件资源,也必然需要被操作系统统一的管理.最初没有操作系统 ...

  2. 《ucore lab2》实验报告

    资源 ucore在线实验指导书 我的ucore实验代码 练习1:实现 first-fit 连续物理内存分配算法 题目 在实现first fit 内存分配算法的回收函数时,要考虑地址连续的空闲块之间的合 ...

  3. ucore操作系统学习笔记(一) ucore lab1系统启动流程分析

    一.ucore操作系统介绍 操作系统作为一个基础系统软件,对下控制硬件(cpu.内存.磁盘网卡等外设),屏蔽了底层复杂多样的硬件差异:对上则提供封装良好的应用程序接口,简化应用程序开发者的使用难度.站 ...

  4. SQL*Loader实验笔记【二】

      所有SQL*Loader实验笔记 实验案例总结(1-7):     SQL*Loader实验笔记[一] 实验案例总结(8-13):   SQL*Loader实验笔记[二] 实验案例总结(14-19 ...

  5. centos救援模式实验笔记

    1.  首先在BIOS中把启动选项设置成DVD光驱启动或者USB启动也是可以的 2.  从光盘启动之后再出现的选项中选择“Rescue installed system”然后按回车确认,具体图下图: ...

  6. 【目标检测大集合】R-FCN、SSD、YOLO2、faster-rcnn和labelImg实验笔记

    R-FCN.SSD.YOLO2.faster-rcnn和labelImg实验笔记 转自:https://ask.julyedu.com/question/7490 R-FCNpaper:https:/ ...

  7. Openstack实验笔记

    Openstack实验笔记 制作人:全心全意 Openstack:提供可靠的云部署方案及良好的扩展性 Openstack简单的说就是云操作系统,或者说是云管理平台,自身并不提供云服务,只是提供部署和管 ...

  8. Huawei-R&S-网络工程师实验笔记20190615-IP基础(AR201上配置IP)

    >Huawei-R&S-网络工程师实验笔记20190615-IP基础(AR201上配置IP) >>实验开始,先上拓扑图参考: >>>一般正常配置IP操作如下 ...

  9. Huawei-R&S-网络工程师实验笔记20190609-VLAN划分综合(Hybrid端口)

    >Huawei-R&S-网络工程师实验笔记20190609-VLAN划分综合(Hybrid端口) >>实验开始,先上拓扑图参考: >>>实验目标:分别实现主 ...

随机推荐

  1. 将文件夹上传到FTP服务器,遍历上传,,,文件夹不能直接上传到FTP服务器上。。。

    <? $ftp_ip = "FTP"; $ftp_user = "user"; $ftp_pwd = "password"; $con ...

  2. 利用docker启动 wordpress

    网上有很多教程哈,我只是记录自己怎么玩的,没啥教学意义 查看镜像说明的mysql/data目录,方便挂载 [root@docker ~]# docker inspect -f {{.Config.Vo ...

  3. Appium测试框架

    介绍 读作['æpɪəm],是selenium的扩展,同样基于WebDriver协议,详见:http://appium.io/. 关于WebDriver终端操作,详见:https://www.w3.o ...

  4. 万能的TextView,实现常用布局样式

    package com.loaderman.textviewdemo; import android.content.Context; import android.content.res.Typed ...

  5. 代码实现:输入3个数a,b,c,按大小顺序输出。

    import java.util.Arrays; import java.util.Scanner; //输入3个数a,b,c,按大小顺序输出. public class Test34 { publi ...

  6. CentOS 系统安装 nodejs 及相关配置

    概述 今天我在我的 AWS EC2 服务器上安装了 nodejs.没想到竟然这么麻烦,比在 windows 和 mac 上麻烦多了.所以我把心得记录下来,供以后开发时参考,相信对其他人也有用. 参考资 ...

  7. hash模块MD5加密

    MD5加密:获取32位加密字符串: 示例(MD5加密'123456')import hashlibhashObject=hashlib.md5(b'123456') #实例化,加密字符串不能直接加密, ...

  8. Nova rebuild for boot from volume issue

    目录 文章目录 目录 Nova boot from volume Rebuild Instance Rebuild for boot from volume Nova boot from volume ...

  9. Python OOP知识积累

    目录 目录 前言 对象 类 面向对象 Python 面向对象编程三个基本特征 封装 继承 继承的作用 泛化与特化 实现继承的方式 多重继承 多态 方法多态 最后 前言 Python是一个功能非常强大的 ...

  10. ABAP 实现内表自定义的F4功能

    “实现多列内容的F4功能 REPORT Z_TAB_TEST.   TYPES: shlp_descr TYPE shlp_descr .   DATA: BEGIN OF itab OCCURS 0 ...