通常编写一个操作系统内核是一项浩大的工程。但我今天的目标是制作一个简单的内核,用比较方便的方法在虚拟机上验证它能够被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"操作系统内核的更多相关文章

  1. 使用KEIL C51实现的简单合作式多任务操作系统内核(单片机实现版本)

    基于网上网友的代码,自己在单片机上实现, 特此记录分享之. 基于https://blog.csdn.net/yyx112358/article/details/78877523 //使用KEIL C5 ...

  2. 红帽企业版RHEL7.1在研域工控板上,开机没有登陆窗口 -- 编写xorg.conf 简单三行解决Ubuntu分辩率不可调的问题

    红帽企业版RHEL7.1在研域工控板上,开机没有登陆窗口 没有登陆窗口 的原因分析: 没有登陆窗口的原因是因为有多个屏幕在工作,其中一个就是build-in 屏幕(内置的虚拟屏幕)和外接的显示器,并且 ...

  3. Java语言编写计算器(简单的计算器)

    Java编写的一个简单计算器,本人还比较菜,只能这样了,有点代码冗余,不能连续计算. import javax.swing.*; import java.awt.*; import java.awt. ...

  4. 为Lua5.3编写C模块简单示例

    为Lua5.3编写C模块简单示例 一.编译安装Lua5.3 MSVC 命令行安装脚本: @echo off md bin md lib md include cd src cl /c /nologo ...

  5. 【JavaScript】使用setInterval()函数作简单的轮询操作

    轮询(Polling)是一种CPU决策怎样提供周边设备服务的方式,又称"程控输出入"(Programmed I/O). 轮询法的概念是.由CPU定时发出询问.依序询问每个周边设备是 ...

  6. Java入门——编写并运行第一个程序

    Java入门——编写并运行第一个程序 摘要:本文主要介绍如何使用Java语言编写并通过DOS运行简单的程序. 编写简单的程序 在D盘新建一个文本文档,输入如下代码: class Hello { pub ...

  7. 基于mykernel 2.0编写一个操作系统内核

    一.配置mykernel 2.0,熟悉Linux内核的编译 1.实验环境:VMware 15 Pro,Ubuntu 18.04.4 2.配置环境 1)在电脑上先下载好以下两个文件,之后通过共享文件夹, ...

  8. 基于mykernel2.0编写一个操作系统内核

    基于mykernel2.0编写一个操作系统内核 一. 实验准备 详细要求 基于mykernel 2.0编写一个操作系统内核 按照https://github.com/mengning/mykernel ...

  9. 国产化之路-统信UOS + Nginx + Asp.Net MVC + EF Core 3.1 + 达梦DM8实现简单增删改查操作

    专题目录 国产化之路-统信UOS操作系统安装 国产化之路-国产操作系统安装.net core 3.1 sdk 国产化之路-安装WEB服务器 国产化之路-安装达梦DM8数据库 国产化之路-统信UOS + ...

随机推荐

  1. Excel 信息对比_数组版

    Sub LOOKUP_UChur() Dim i As Long '=== sourceWorksheet = 数据源表名称 Dim sourceWorksheet As Worksheet Dim ...

  2. redhad linux 7 安装ftp服务

    1. 查看有没有安装 rpm -qa|grep vsftpd 2.安装vsftp yum install vsftpd -y 3. 启动vsftp /sbin/service vsftpd start ...

  3. [2011WorldFinal]Chips Challenge[流量平衡]

    Chips Challenge Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others) ...

  4. 【BZOJ3831】[Poi2014]Little Bird 单调队列

    [BZOJ3831][Poi2014]Little Bird Description In the Byteotian Line Forest there are   trees in a row. ...

  5. 160419、CSS3实现32种基本图形

    CSS3可以实现很多漂亮的图形,我收集了32种图形,在下面列出.直接用CSS3画出这些图形,要比贴图性能更好,体验更加,是一种非常好的网页美观方式. 这32种图形分别为圆形,椭圆形,三角形,倒三角形, ...

  6. Spoken English Practice(1、This is between you and me, Don't let it out. 2、Don't let your dreams be dreams, no matter how hard it gets, say to yourself, I'm going to make it.)

    绿色:连读:                  红色:略读:               蓝色:浊化:               橙色:弱读     下划线_为浊化 口语蜕变(2017/7/12) ...

  7. linux文件与目录管理命令(ubuntu)

    ls:列出目录 选项与参数: -a:全部文件,隐藏档(开头为.的文件)也会列出: -d:仅列出目录本身(也就是 . ),而不是目录下的所有文件及目录: -l:长字符串列出,包括文件的属性.权限等数据.

  8. 如何使用 libtorch 实现 AlexNet 网络?

    如何使用 libtorch 实现 AlexNet 网络? 按照图片上流程写即可.输入的图片大小必须 227x227 3 通道彩色图片 // Define a new Module. struct Ne ...

  9. Node.js REST 工具 Restify

    Restify 是一个 Node.JS 模块,可以让你创建正确的 REST web services.它借鉴了很多 express 的设计,因为它是 node.js web 应用事实上的标准 API. ...

  10. 转 Merkle Tree(默克尔树)算法解析

    Merkle Tree概念  Merkle Tree,通常也被称作Hash Tree,顾名思义,就是存储hash值的一棵树.Merkle树的叶子是数据块(例如,文件或者文件的集合)的hash值.非叶节 ...