编写和运行简单的"Hello World"操作系统内核
通常编写一个操作系统内核是一项浩大的工程。但我今天的目标是制作一个简单的内核,用比较方便的方法在虚拟机上验证它能够被grub装载和运行,并且可通过gdb进行调试,为接下去的工作创造一个基础环境。
首先,为了方便运行和调试我们需要一个虚拟机。虚拟机有很多选择,这里用最简单的qemu。
先用dd创建一个文件作为虚拟盘,100MB就可以了:
$ dd if=/dev/zero of=disk.img count= bs=
然后对这个虚拟磁盘进行分区:
$ fdisk -c=dos disk.img
用命令n创建一个分区就可以了。通常情况下分区的起始扇区是2048(不带选项-c=dos),如果用老式的msdos格式分区表(命令c),就可以选择从63扇区开始。以前文章里提到63个扇区足够塞下grub的核心映像,所以为了测试下grub就选择了msdos模式。用命令w把变动写入虚拟盘。
把这个分区虚拟成设备文件:
$ sudo losetup -o /dev/loop1 disk.img
这里指定了起始扇区的偏移量。63个扇区,每个扇区512个字节,总共是32256字节。如果你的分区起始扇区是2048,那么偏移量应该是2048 * 512字节。
格式化:
$ sudo mkfs.ext4 /dev/loop1
挂载起来,这样就可以方便地往里面放kernel和grub需要的配置文件和模块什么的:
$ sudo mount /dev/loop1 /mnt
安装grub:
$ sudo grub-install --boot-directory=/mnt --modules="part_msdos" disk.img
使用qemu来启动虚拟机(我用的是64位系统):
$ qemu-system-x86_64 -hda disk.img -m -s &
这时候应该能够看到grub的提示符了。当然现在还没有grub菜单也没有kernel,我们暂时先关掉虚拟机。
接下来可以为grub建立个multiboot启动菜单:
$ sudo vi /mnt/grub/grub.cfg
制作菜单命令:
menuentry "Hello" {
multiboot (hd0,msdos1)/kernel
boot
}
确保数据写回了虚拟盘:
$ sync
这时候如果你再打开虚拟机,应该就可以看到启动菜单了,当然因为还没有kernel,选择菜单项后无法继续,会提示kernel找不到。
下一步,我们用c语言从头编写个最简单kernel程序。这个kernel没有实现操作系统的基本功能。但是可以被grub装载和运行。
kernel.c:
/* 在文件里嵌入一个签名。Grub在multiboot时会寻找这个签名 */
#define GRUB_MAGIC 0x1BADB002
#define GRUB_FLAGS 0x0
#define GRUB_CHECKSUM (-1 * (GRUB_MAGIC + GRUB_FLAGS)) struct grub_signature {
unsigned int magic;
unsigned int flags;
unsigned int checksum;
}; struct grub_signature gs __attribute__ ((section (".grub_sig"))) = {
GRUB_MAGIC,
GRUB_FLAGS,
GRUB_CHECKSUM
}; /* 显示字符的函数。因为我们什么库都不能用,只能直接写屏了。0xB8000是VGA彩色字符模式的数据缓存。每个字符用两个字节表示。前一个是Ascii码,后一个代表颜色。 */
void puts( const char *s, int color ){
volatile char *buffer = (volatile char*)0xB8000; while( *s != ) {
*buffer++ = *s++;
*buffer++ = color;
}
} /* 主函数,程序入口。最后用个死循环,把代码指针困在那里。*/
void main (void) {
puts("Hello World!", 0x7);
while () {}
}
有必要再写个链接模板,确保编译好的kernel装载在内存地址0x100000,这里是grub代码最后跳转到的区域,从这里我们的kernel接过了接力棒。另外,虽然我用的虚拟机是64位的,但是我需要生成一个32位的kernel,因为做64位的kernel还需要做额外的设置工作,比如从32位保护模式打开long mode,比较麻烦。所以先暂时用32位的kernel。
kernel.ld:
OUTPUT_FORMAT("elf32-i386")
ENTRY(main)
SECTIONS
{
.grub_sig 0x100000 : AT(0x100000)
{
*(.grub_sig)
}
.text :
{
*(.text)
}
.data :
{
*(.data)
}
.bss :
{
*(.bss)
}
/DISCARD/ :
{
*(.comment)
*(.eh_frame)
}
}
还有一个Makefile,主要是设置一些编译选项。
Makefile:
CC = gcc
LD = ld
CFLAGS = -std=c99 -pedantic -Wall -nostdlib -ffreestanding -m32
LDFLAGS = -T kernel.ld -nostdlib -n -melf_i386
OBJS = kernel.o
.PHONY: all
all: kernel
%.o: %.c
${CC} -c ${CFLAGS} $<
kernel: $(OBJS) kernel.ld
${LD} ${LDFLAGS} -o kernel ${OBJS}
clean:
rm -f ${OBJS} kernel
编译生成kernel并放入我们的虚拟盘里:
$ sudo cp kernel /mnt/
$ sync
再次启动虚拟机,在启动菜单里选择multiboot我们的kernel,应该就能看到Hello World!的字符显示在虚拟机屏幕上了。
如果想要调试,可以运行gdb。因为我们在启动qemu的时候使用了-s选项,所以qemu默认会打开tcp端口1234作为gdb调试端口。在gdb中可以使用target remote tcp::1234命令来连接。试试看连接,会发现cpu一直在执行0x100066处的指令。用objdump -D kernel看下kernel的汇编代码:
0010004c <main>:
10004c:55 push %ebp
10004d:89 e5 mov %esp,%ebp
10004f:83 ec 08 sub $0x8,%esp
:c7 44 24 04 07 00 00 movl $0x7,0x4(%esp)
:0005a:c7 04 24 68 00 10 00 movl $0x100068,(%esp)
:e8 a6 ff ff ff call 10000c <puts>
:eb fe jmp <main+0x1a>
0x100066处的指令正好是死循环的那条jmp指令。
编写和运行简单的"Hello World"操作系统内核的更多相关文章
- 使用KEIL C51实现的简单合作式多任务操作系统内核(单片机实现版本)
基于网上网友的代码,自己在单片机上实现, 特此记录分享之. 基于https://blog.csdn.net/yyx112358/article/details/78877523 //使用KEIL C5 ...
- 红帽企业版RHEL7.1在研域工控板上,开机没有登陆窗口 -- 编写xorg.conf 简单三行解决Ubuntu分辩率不可调的问题
红帽企业版RHEL7.1在研域工控板上,开机没有登陆窗口 没有登陆窗口 的原因分析: 没有登陆窗口的原因是因为有多个屏幕在工作,其中一个就是build-in 屏幕(内置的虚拟屏幕)和外接的显示器,并且 ...
- Java语言编写计算器(简单的计算器)
Java编写的一个简单计算器,本人还比较菜,只能这样了,有点代码冗余,不能连续计算. import javax.swing.*; import java.awt.*; import java.awt. ...
- 为Lua5.3编写C模块简单示例
为Lua5.3编写C模块简单示例 一.编译安装Lua5.3 MSVC 命令行安装脚本: @echo off md bin md lib md include cd src cl /c /nologo ...
- 【JavaScript】使用setInterval()函数作简单的轮询操作
轮询(Polling)是一种CPU决策怎样提供周边设备服务的方式,又称"程控输出入"(Programmed I/O). 轮询法的概念是.由CPU定时发出询问.依序询问每个周边设备是 ...
- Java入门——编写并运行第一个程序
Java入门——编写并运行第一个程序 摘要:本文主要介绍如何使用Java语言编写并通过DOS运行简单的程序. 编写简单的程序 在D盘新建一个文本文档,输入如下代码: class Hello { pub ...
- 基于mykernel 2.0编写一个操作系统内核
一.配置mykernel 2.0,熟悉Linux内核的编译 1.实验环境:VMware 15 Pro,Ubuntu 18.04.4 2.配置环境 1)在电脑上先下载好以下两个文件,之后通过共享文件夹, ...
- 基于mykernel2.0编写一个操作系统内核
基于mykernel2.0编写一个操作系统内核 一. 实验准备 详细要求 基于mykernel 2.0编写一个操作系统内核 按照https://github.com/mengning/mykernel ...
- 国产化之路-统信UOS + Nginx + Asp.Net MVC + EF Core 3.1 + 达梦DM8实现简单增删改查操作
专题目录 国产化之路-统信UOS操作系统安装 国产化之路-国产操作系统安装.net core 3.1 sdk 国产化之路-安装WEB服务器 国产化之路-安装达梦DM8数据库 国产化之路-统信UOS + ...
随机推荐
- java基础---->Java中异常的使用(二)
这一篇博客用例子讲述一下异常的处理过程.那些 我们一直惴惴不安 又充满好奇的未来 会在心里隐隐约约地觉得它们是明亮的. 异常的执行过程 一.实例一:return语句 public class Exce ...
- 【BZOJ4832】[Lydsy2017年4月月赛]抵制克苏恩 概率与期望
[BZOJ4832][Lydsy2017年4月月赛]抵制克苏恩 Description 小Q同学现在沉迷炉石传说不能自拔.他发现一张名为克苏恩的牌很不公平.如果你不玩炉石传说,不必担心,小Q同学会告诉 ...
- 【BZOJ3436】小K的农场 差分约束
[BZOJ3436]小K的农场 Description 背景 小K是个特么喜欢玩MC的孩纸... 描述 小K在MC里面建立很多很多的农场,总共n个,以至于他自己都忘记了每个农场中种植作物的具体数量了, ...
- 解决SecureCRT连接linux终端中文显示乱码
现象如下: 原因: SecureCRT的字符集编码不是Linux的默认编码:UTF-8 解决办法: 1.在“选项”找到“会话选项” 2.选择“外观”,设置字符编码为“UTF-8” 3.确定后,继续在终 ...
- Spark源码分析 – SchedulerBackend
SchedulerBackend, 两个任务, 申请资源和task执行和管理 对于SparkDeploySchedulerBackend, 基于actor模式, 主要就是启动和管理两个actor De ...
- 关于HashSet在 java7 与 java8的不同
作者:RednaxelaFX链接:https://www.zhihu.com/question/28414001/answer/40733996来源:知乎著作权归作者所有.商业转载请联系作者获得授权, ...
- HDFS基本命令行操作及上传文件的简单API
一.HDFS基本命令行操作: 1.HDFS集群修改SecondaryNameNode位置到hd09-2 (1)修改hdfs-site.xml <configuration> //配置元数据 ...
- Flask之flask-session
简介 flask-session是flask框架的session组件,由于原来flask内置session使用签名cookie保存,该组件则将支持session保存到多个地方,如: redis:保存数 ...
- SVM支持向量机总结
一.拉格朗日乘子法 一般,在有等式约束时使用拉格朗日乘子法,在有不等约束时使用KKT条件.这里我们先介绍拉格朗日乘子法,后面再介绍KKT条件. 比如考虑下面的组合优化的问题, 这是一个带等式约束的优化 ...
- wpa安装方法
1.openssl 2.lib 1.1.2 3.wpa lua 编译错误 http://www.blogjava.net/xiaomage234/archive/2013/09/13/404037.h ...