linux内核设计的艺术-开始执行main函数

为了执行linux内的C语言main函数,上一篇讲到了,为了从汇编语言环境跳转到C语言环境下执行,将CPU工作模式从16位转变到32位模式(C语言是32位的),并且重新建立了GDT与IDT,但是此时GDT和IDT中并没有内容,所以不能进行内存寻址与中断,接下来就是初始化GDT和IDT了。

进入32位模式后,寄存器也将变为32位寄存器,下面的汇编语法和之前的intel汇编有些不同,为AT&T汇编,至于差别不在赘述。

Head.S
startup_32:
	//重设段寄存器内容
	movl $0x10,%eax
	mov %ax,%ds
	mov %ax,%es
	mov %ax,%fs
	mov %ax,%gs
    //初始化栈指针,即将栈顶指针指向栈底
	lss _stack_start,%esp
    //建立IDT
	call setup_idt
//建立GDT
	call setup_gdt
    //建立好GDT和IDT后重新设置段寄存器
	movl $0x10,%eax
	mov %ax,%ds
	mov %ax,%es
	mov %ax,%fs
	mov %ax,%gs
	lss _stack_start,%esp
	xorl %eax,%eax
    //检查A20是否成功打开
1:	incl %eax
	movl %eax,0x000000
	cmpl %eax,0x100000
	je 1b

    //检查8087数学协处理器后,开始进行内存内页
	call check_x87
	jmp after_page_tables

after_page_tables:
    //以下为将main函数的参数入栈
	pushl $0
	pushl $0
	pushl $0
	pushl $L6
	//将main函数的地址入栈
	pushl $_main
    //开始进行内存分页
	jmp setup_paging

上面的push _main操作很重要,因为在之后的ret指令后将会将栈中的_main函数地址装入cs,和ip中,参数也会装载到相应的寄存器中,这些操作都将在setup_paging后完成

setup_paging:
	//初始化前5k的内存
	movl $1024*5,%ecx
	xorl %eax,%eax
	xorl %edi,%edi
	cld;
rep;
stosl
//将0x0000处赋值为0x1000 + 7,7的意思是111代表访问权限,0x1000作为页表基地址,接下来就分别映射0x2000,0x3000,0x4000
	movl $pg0+7,_pg_dir
	movl $pg1+7,_pg_dir+4
	movl $pg2+7,_pg_dir+8
	movl $pg3+7,_pg_dir+12
	movl $pg3+4092,%edi
	movl $0xfff007,%eax
    //上述代码其实是建立4096字节的内核页表,每一个字节可以寻址到一个4k的页面,所以可以寻址的空间为16M
	std
1:	stosl
	subl $0x1000,%eax
	jge 1b
	xorl %eax,%eax
	movl %eax,%cr3
	movl %cr0,%eax
	orl $0x80000000,%eax
	movl %eax,%cr0
    //调用ret开始执行main函数
	ret

下面是一张内存分页的图,寻址是需要先根据pg_dir的地址找到页表目录,在页目录中找到pg页表,然后在页表中找到相应的页面

时间: 2024-10-12 22:52:54

linux内核设计的艺术-开始执行main函数的相关文章

《LINUX内核设计的艺术》第一章从开机家电到执行main函数之前的过程 学习笔记之一

从开机加电到实行main函数之前的过程 分为三步,目的是实现从启动盘加载操作系统程序,完成实现main函数的准备工作 启动BLOS,准备是模式下的中断向量表和中断服务程序 从启动盘加载操作系统程序到内存.加载操作系统程序就是靠第一步实现的 为实现32位的main函数做过度工作 1.1启动blos,准备实模式下的中断向量表和中断服务程序 由blos来加载软件操作系统的任务 1.1.1         BLOS的启动原理 0XFFFF0 由硬件来启动,CPU硬件设计逻辑设计为加电瞬间就强行将CS的值

linux内核设计的艺术--加载内核代码

在BIOS触发0x19中断将磁盘的第一个扇区(512B)加载到内存中后,计算机才真正开始执行磁盘上的程序.而这512B的程序bootsect.s中的第一批代码,此时处理器还处于实模式内存寻址的最大范围是1M(0x0000-0xFFFF),接下来我们看看在bootsect.s的第一批代码中做了些什么? SYSSIZE = 0x3000 //内核程序的大小 SETUPLEN = 4 //要加载的setup程序长度单位为扇区数 BOOTSEG = 0x07c0 //启动扇区被BIOS加载的位置,也就是

linux内核设计的艺术--系统启动第一步

计算机究竟是如何执行起来的呢,在我学习计算机的时候一直不是非常明确,可是近期借了本<linux内核设计的艺术>算是知道了计算机从按开机到启动操作系统之间究竟做了些什么. 这本书刚開始介绍的并非linux的启动,而是linux启动之前的一步,也就是在你按下了开机button之后进入系统引导之前计算机做的事情,这个时候做的事情是每台执行操作系统的计算机都须要做的.大致的步骤也差点儿相同,OK,进入正题. 事实上在当我们按了开机键的时候,硬件电路将CPU的CS设置成0XF000,IP设置成0XFFF

《Linux内核设计的艺术》学习笔记(一)从开机加电到执行main函数之前的过程

分享一个最近丢了手机心塞到爆炸的我,现在穷的只剩下满脑子的智慧了,好了,我要开始学习了. 首先,搭建一个linux0.11的系统环境,贴出结果图. 从开机加电到执行main函数之前的过程. 1. 启动BIOS,准备实模式下的中断向量表和中断服务程序; 2. 从启动盘加载操作系统程序到内存,加载操作系统程序的工作就是利用第一步中断服务程序实现的; 3. 为执行32位的main函数做过渡工作. 启动BIOS,准备实模式下的中断向量表和中断服务程序 cpu的硬件设计为加电即进入16位实模式下状态运行,

《Linux内核设计的艺术》学习笔记(一)从开机加电到执行main函数

  实验内核版本:0.11 ◆ 从开机到main函数的三步: ① 启动BIOS,准备实模式下的中断向量表和中断服务程序: ② 从启动盘加载OS程序到内存中,加载OS程序的工作就是利用第一步中的中断服务程序实现的: ③ 为执行保护模式下32位的main函数做过渡工作. ? Intel将所有80x86系列的CPU硬件都设计为加电即进入16位实模式状态运行: ? 将CPU硬件逻辑设计为在加电瞬间强行将CS置为0xFFFF,IP置为0x0000,即是CS:IP指向了0xFFFF0这个地址: 整个过程是一

linux内核设计的艺术--载入内核代码

在BIOS触发0x19中断将磁盘的第一个扇区(512B)载入到内存中后.计算机才真正開始运行磁盘上的程序.而这512B的程序就是bootsect.s,此时处理器还处于实模式内存寻址的最大范围是1M(0x0000-0xFFFF),接下来我们看看在bootsect.s的第一批代码中做了些什么? </pre><pre name="code" class="cpp">SYSSIZE = 0x3000 //内核程序的大小 SETUPLEN = 4 //

linux内核设计的艺术--从16位转向32位

上一篇讲到了将kernel模块加载到了内存的0x10000-0x10000+120KB处,接下来,将会执行setup.s中的代码了. 首先,setup要获取一系列系统硬件信息 mov ax,#INITSEG mov ds,ax //设置段地址 mov ah,#0x03 xor bh,bh int 0x10 //调用0x10中断获取屏幕光标位置 mov [0],dx //将屏幕光标保存至0x90000处 //调用0x15中断,获取内存信息保存至0x90002 mov ah,#0x88 int 0x

《Linux内核设计的艺术》学习笔记(五)INT 0x10中断

参考书籍:<IBM-PC汇编语言程序设计>   ◆ 设置显示方式: AH = 0 AL = 00 40 × 25 黑白文本,16级灰度 AL = 01 40 × 25 16色文本 AL = 02 80 × 25 黑白文本,16级灰度 AL = 03 80 × 25 16色文本 AL = 04 320 × 200 4色图形 AL = 05 320 × 200 黑白图形,4色灰度 AL = 06 640 × 200 黑白图形 AL = 07 80 × 25 黑白文本 AL = 08 160 × 2

《Linux内核设计的艺术》学习笔记(四)默认段和偏移寄存器

参考书籍:<Intel微处理器> 表1 默认的16位段 + 偏移寻址组合 段 偏移量 特殊用途 CS IP 指令地址 SS SP或BP 堆栈地址 DS BX.DI.SI.8位或16位数 数据地址 ES 串指令的DI 串目标地址 表2 默认的32位段 + 偏移寻址组合 段 偏移量 特殊用途 CS EIP 指令地址 SS ESP或EBP 堆栈地址 DS EAX.EBX.ECX.EDX. ESI.EDI,一个8位或32位数 数据地址 ES 串指令的EDI 串目标地址 FS 无默认值 一般地址 GS