HariboteOS/day15/memory.c
2021-04-07 00:40:52 +08:00

192 lines
4.4 KiB
C
Raw Permalink Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

#include "memory.h"
#include "io.h"
unsigned int memtest(unsigned int start, unsigned int end) {
char flg486 = 0;
unsigned int eflg, cr0;
eflg = io_load_eflags();
eflg |= EFLAGS_AC_BIT; // AC-bit = 1
io_store_eflags(eflg);
eflg = io_load_eflags();
if ((eflg & EFLAGS_AC_BIT)) {
// 如果是i386即使设定AC=1也会自动变回到0
flg486 = 1;
}
eflg &= ~EFLAGS_AC_BIT; // AC-bit = 0
io_store_eflags(eflg);
if (flg486) {
cr0 = io_load_cr0();
cr0 |= CR0_CACHE_DISABLE; // 禁止缓存
io_store_cr0(cr0);
}
unsigned int ret = memtest_sub(start, end);
if (flg486) {
cr0 = io_load_cr0();
cr0 &= ~CR0_CACHE_DISABLE; // 允许缓存
io_store_cr0(cr0);
}
return ret;
}
/**
* 运行正常,便没有改写成汇编实现
*/
unsigned int memtest_sub(unsigned int start, unsigned int end) {
unsigned int i, *p, old, pat0 = 0xaa55aa55, pat1 = 0x55aa55aa;
for (i = start; i <= end; i += 0x1000) {
p = (unsigned int *)(i + 0xffc);
old = *p; // 先记住修改前的值
*p = pat0; // 试写
*p ^= 0xffffffff; // 反转
if (*p != pat1) {
// 检查反转结果
not_memory:
*p = old;
break;
}
*p ^= 0xffffffff; // 再次反转
if (*p != pat0) {
// 检查值是否恢复
goto not_memory;
}
*p = old;
}
return i;
}
void memman_init(struct MemMan *man) {
man->frees = 0; // 可用信息数目
man->maxfrees = 0; // 用于观察可用状况frees的最大值
man->lostsize = 0; // 释放失败的内存的大小总和
man->losts = 0; // 释放失败次数
}
/**
* 报告空余内存大小的合计
*/
unsigned int memman_total(struct MemMan *man) {
unsigned int i, t = 0;
for (i = 0; i < man->frees; i++) {
t += man->free[i].size;
}
return t;
}
/**
* 分配
*/
unsigned int memman_alloc(struct MemMan *man, unsigned int size) {
for (unsigned int i = 0; i < man->frees; i++) {
if (man->free[i].size >= size) {
// 找到了足够大的内存
unsigned int a = man->free[i].addr;
man->free[i].addr += size;
man->free[i].size -= size;
if (man->free[i].size == 0) {
// 如果free[i]变成了0就减掉一条可用信息
man->frees--;
for (; i < man->frees; i++) {
man->free[i] = man->free[i + 1]; // 代入结构体
}
}
return a;
}
}
return 0; // 没有可用空间
}
unsigned int memman_alloc_4k(struct MemMan *man, unsigned int size) {
size = (size + 0xfff) & 0xfffff000;
return memman_alloc(man, size);
}
/**
* 释放
*/
int memman_free(struct MemMan *man, unsigned int addr, unsigned int size) {
int i, j;
for (i = 0; i < man->frees; i++) {
if (man->free[i].addr > addr) {
break;
}
}
// free[i - 1].addr < addr < free[i].addr
if (i > 0) {
// 前面有可用内存
if (man->free[i - 1].addr + man->free[i - 1].size == addr) {
// 可以与前面的可用内存归纳到一起
man->free[i - 1].size += size;
if (i < man->frees) {
// 后面也是
if (addr + size == man->free[i].addr) {
// 也可以与后面的可用内存归纳到一起
man->free[i - 1].size += man->free[i].size;
// man->free[i]删除
// free[i]变成0后归纳到前面去
man->frees--;
for (; i < man->frees; i++) {
man->free[i] = man->free[i + 1]; // 结构体赋值
}
}
}
}
}
// 不能与前面的可用空间归纳到一起
if (i < man->frees) {
// 后面还有
if (addr + size == man->free[i].addr) {
man->free[i].addr = addr;
man->free[i].size += size;
return 0; // 成功完成
}
}
// 既不能与前面归纳到一起,也不能与后面归纳到一起
if (man->frees < MEMMAN_FREES) {
// free[i]之后的,向后移动,腾出一点可用空间
for (j = man->frees; j > i; j--) {
man->free[j] = man->free[j - 1];
}
man->frees++;
if (man->maxfrees < man->frees) {
man->maxfrees = man->frees; // 更新最大值
}
man->free[i].addr = addr;
man->free[i].size = size;
return 0; // 成功完成
}
// 不能往后移动
man->losts++;
man->lostsize += size;
return -1;
}
int memman_free_4k(struct MemMan *man, unsigned int addr, unsigned int size) {
size = (size + 0xfff) & 0xfffff000;
return memman_free(man, addr, size);
}