linux-0.11分析:init文件 main.c的第一个初始化函数mem_int 第四篇随笔
init文件夹 mian.c
参考 [github这个博主的 厉害][ https://github.com/sunym1993/flash-linux0.11-talk ]
首先先看看这个
mian.c
中的主要几行代码吧void main(void)
{
ROOT_DEV = ORIG_ROOT_DEV;
drive_info = DRIVE_INFO; memory_end = (1<<20) + (EXT_MEM_K<<10);
memory_end &= 0xfffff000;
if (memory_end > 16*1024*1024)
memory_end = 16*1024*1024;
if (memory_end > 12*1024*1024)
buffer_memory_end = 4*1024*1024;
else if (memory_end > 6*1024*1024)
buffer_memory_end = 2*1024*1024;
else
buffer_memory_end = 1*1024*1024;
main_memory_start = buffer_memory_end;
#ifdef RAMDISK
main_memory_start += rd_init(main_memory_start, RAMDISK*1024);
#endif mem_init(main_memory_start,memory_end);
trap_init();
blk_dev_init();
chr_dev_init();
tty_init();
time_init();
sched_init();
buffer_init(buffer_memory_end);
hd_init();
floppy_init();
sti();
move_to_user_mode();
if (!fork()) { /* we count on this going ok */
init();
}
/*
注意!!对于任何其他任务,“pause()”意味着我们必须获得一个
*发出唤醒信号,但task0是唯一的例外(请参阅“schedule()”)
*因为任务0在每个空闲时刻被激活(当没有其他任务时
*可以运行)。对于task0,“pause()”只意味着我们去检查其他
*任务可以运行,否则我们返回这里
*/
for(;;) pause();
}
首先看看前两行代码:
ROOT_DEV = ORIG_ROOT_DEV; 这是在设置系统根文件的设备号
drive_info = DRIVE_INFO; 这里是在设置之前存在0x90000位置处的设备信息
再看看这些设备信息吧
地址 | 字节 | 存储的东西 |
---|---|---|
0x90000 | 1 | 光标列号 |
0x90001 | 1 | 光标行号 |
0x90002 | 2 | 内存大小 |
0x90004 | 2 | 显示页面 |
0x90006 | 1 | 模式模式 |
0x90007 | 1 | 窗口宽度 |
0x90008 | 2 | 配置参数 |
0x9000A | 2 | 配置参数 |
0x9000C | 2 | 配置参数 |
0x90080 | 16 | hd0硬盘信息 |
0x90010 | 16 | hd1硬盘信息 |
接下来看看中间那段代码
memory_end = (1<<20) + (EXT_MEM_K<<10);
memory_end &= 0xfffff000;
if (memory_end > 16*1024*1024) //内存是否大于16MB
memory_end = 16*1024*1024; //如果大于,就memory_end把置为16MB
if (memory_end > 12*1024*1024) //内存是否大于12MB
buffer_memory_end = 4*1024*1024;//如果大于,就把buffer_memory_end置为4MB
else if (memory_end > 6*1024*1024) //内存是否大于6MB
buffer_memory_end = 2*1024*1024;//如果大于,就buffer_memory_end把置为2MB
else
buffer_memory_end = 1*1024*1024;//如果都不满足,就buffer_memory_end把置为2MB main_memory_start = buffer_memory_end;//把buffer_memory_end值给main_memory_start
#ifdef RAMDISK
main_memory_start += rd_init(main_memory_start, RAMDISK*1024);
//main_memory_start加上一段距离
#endif
仔细看会发现,这就是在定义,且改变这
memory_end、main_memory_start、buffer_memory_end
这三个变量的值;首先前面有句:#define EXT_MEM_K (*(unsigned short *)0x90002)
在获取
memory_end
的内存大小后面就是一些判断,判断内存的大小并初始化数据;代码有解释
实际上就是这三个变量
memory_end、main_memory_start、buffer_memory_end
指向了不同的位置我们用内存为16MB来说明,有一张图来说明一哈
看看后面那一段init(初始化)代码,先看第一个
mem_init
1、第一个初始化函数,mem_init
()
mem_init(main_memory_start,memory_end);
可以看出传入之前定义的那两个参数:main_memory_start、memory_end
mm 文件夹 memory.c
那先看看这个函数把:这个函数的位置在:linux文件下mm文件下的memory.c中
#define invalidate() \
__asm__("movl %%eax,%%cr3"::"a" (0))
/* these are not to be changed without changing head.s etc */
#define LOW_MEM 0x100000
#define PAGING_MEMORY (15*1024*1024) //15728640
#define PAGING_PAGES (PAGING_MEMORY>>12) //3840
#define MAP_NR(addr) (((addr)-LOW_MEM)>>12)
#define USED 100
#define CODE_SPACE(addr) ((((addr)+4095)&~4095) < \
current->start_code + current->end_code)
static long HIGH_MEMORY = 0;
#define copy_page(from,to) \
__asm__("cld ; rep ; movsl"::"S" (from),"D" (to),"c" (1024):"cx","di","si")
static unsigned char mem_map [ PAGING_PAGES ] = {0,};
//start_mem = 4MB end_mem = 16MB
void mem_init(long start_mem, long end_mem)
{
int i;
HIGH_MEMORY = end_mem; //前面定义:HIGH_MEMORY = 0 赋值为传入的参数end_mem
for (i=0 ; i<PAGING_PAGES ; i++) //PAGING_PAGES前面也有
mem_map[i] = USED; //前面的定义的一个char数组
i = MAP_NR(start_mem); //MAP_NR前面定义了 start_mem这个位置减去LOW_MEM,再右移12位
end_mem -= start_mem; //更改传入的参数end_mem = end_mem - start_mem
end_mem >>= 12; // end_mem右移动12位
while (end_mem-->0)
mem_map[i++]=0; //
}
如果传入的start_mem = 4MB end_mem = 16MB
HIGH_MEMORY = 4MB PAGING_PAGES = 3840B
正好一个char型占1个字节也就是8位mem_map 数组的3840B全为100
i = 768B
end_mem = 3072B
mem_map数组 从 768B 开始全部添加3840B个0
使所以数组的长度为 3840 每个里面存4k数据的位置,正好为15MB
真相这些只是把数据存放在数组中,并不代表真实的内存
实际上就是
mem_map
数组存储哪些内存有值,哪些没有为0可以使用最终就是 1-3MB为100值;3~16MB为0
正好
mem_map
数组的前768存100,768~3840存0
看看下面这个图吧
linux-0.11分析:init文件 main.c的第一个初始化函数mem_int 第四篇随笔的更多相关文章
- linux-0.11分析:init文件 main.c的第二个初始化函数trap_init() 第五篇随笔
2.第二个初始化函数,trap_init() 参考 [github这个博主的 厉害][ https://github.com/sunym1993/flash-linux0.11-talk ] kern ...
- Linux 0.11下信号量的实现和应用
Linux 011下信号量的实现和应用 生产者-消费者问题 实现信号量 信号量的代码实现 关于sem_wait和sem_post sem_wait和sem_post函数的代码实现 信号量的完整代码 实 ...
- Linux 0.11源码阅读笔记-总览
Linux 0.11源码阅读笔记-总览 阅读源码的目的 加深对Linux操作系统的了解,了解Linux操作系统基本架构,熟悉进程管理.内存管理等主要模块知识. 通过阅读教复杂的代码,锻炼自己复杂项目代 ...
- Linux 0.11中write实现
看了一下Linux 0.11版本号write的实现,首先它在标准头文件unistd.h中有定义 int write(int fildes, const char * buf, off_t count) ...
- 【从头到脚品读 Linux 0.11 源码】第一回 最开始的两行代码
从这一篇开始,您就将跟着我一起进入这操作系统的梦幻之旅! 别担心,每一章的内容会非常的少,而且你也不要抱着很大的负担去学习,只需要像读小说一样,跟着我一章一章读下去就好. 话不多说,直奔主题.当你按下 ...
- Linux 0.11源码阅读笔记-总结
总结 Linux 0.11主要包含文件管理和进程管理两个部分.进程管理包括内存管理.进程管理.进程间通信模块.文件管理包含磁盘文件系统,打开文件内存数据.磁盘文件系统包括空闲磁盘块管理,文件数据块的管 ...
- Linux 0.11源码阅读笔记-文件管理
Linux 0.11源码阅读笔记-文件管理 文件系统 生磁盘 未安装文件系统的磁盘称之为生磁盘,生磁盘也可以作为文件读写,linux中一切皆文件. 磁盘分区 生磁盘可以被分区,分区中可以安装文件系统, ...
- Linux 0.11源码阅读笔记-中断过程
Linux 0.11源码阅读笔记-中断过程 是什么中断 中断发生时,计算机会停止当前运行的程序,转而执行中断处理程序,然后再返回原被中断的程序继续运行.中断包括硬件中断和软件中断,硬中断是由外设自动产 ...
- 分析入口文件main.php
在分析之前,需要了解php cli模式下的编程 1.了解getopt函数,php手册地址:http://php.net/manual/zh/function.getopt.php static pri ...
随机推荐
- Java获取特定区间随机数及产生不重复随机数
问题 有这样一种需求,在这样一个数组中String[] arr = new String[]{"电商", "互联网", "小程序", &qu ...
- 自学c语言
C 语言是一种通用的.面向过程式的计算机程序设计语言. 当前最新的 C 语言标准为 C18 前期准备 C 编译器 写在源文件中的源代码是人类可读的源.它需要"编译",转为机器语言 ...
- 基于surging网络组件多协议适配的平台化发展
前言 Surging 发展已经有快6年的时间,经过这些年的发展,功能框架也趋于成熟,但是针对于商业化需求还需要不断的打磨,前段时间客户找到我想升级成平台化,针对他的需求我 ...
- PyTorch DataSet Normalization torchvision.transforms.Normalize()
特征缩放, 在这种情况下,我们不仅仅考虑是一个值的数据集,我们考虑的是具有多个特征和相关的值的样本或元素的数据集. 假如正在处理一个人的数据集, 归一化数据集有许多不同的 ...
- 【python基础】第07回 运算符和流程控制 2
本章内容概要 1.逻辑运算符补充 2.循环结构 本章内容详解 1.逻辑运算符补充 两边都不为0的情况 or 直接取前面的值 and 直接取后面的值如果存在0的情况 and 直接取0 or 直接取非0 ...
- Vue动态组件的实践与原理探究
我司有一个工作台搭建产品,允许通过拖拽小部件的方式来搭建一个工作台页面,平台内置了一些常用小部件,另外也允许自行开发小部件上传使用,本文会从实践的角度来介绍其实现原理. ps.本文项目使用Vue CL ...
- Markdown第一次学习
# # Markdown学习 一级标题: #空格+标题名称+回车得到一级标题 ## 二级标题 一级标题方法中变成两个##号 ### 三级标题 变成三个###号,以此类推,最多到六级标题 ## 字体 h ...
- .NET服务治理之限流中间件-FireflySoft.RateLimit
概述 FireflySoft.RateLimit自2021年1月发布第一个版本以来,经历了多次升级迭代,目前已经十分稳定,被很多开发者应用到了生产系统中,最新发布的版本是3.0.0. Github:h ...
- Tapdata 在线研讨会:如何快速上手 Tapdata Cloud?
偶然接触到 Tapdata Cloud,据说不仅可以实现异构数据实时同步,还永久 100% 免费,但就是不知道怎么获取.怎么用? 打开相关文档逐渐陷入迷茫,术语."黑话"随处可见, ...
- Linux操作系统(7):rpm包管理和yum软件包在线管理
一.rpm 包的管理 介绍:一种用于互联网下载包的打包及安装工具,它包含在某些 Linux 分发版中.它生成具有.RPM 扩展名的文件.RPM 是 RedHat Package Manager(Red ...