通常编写一个操作系统内核是一项浩大的工程。但我今天的目标是制作一个简单的内核,用比较方便的方法在虚拟机上验证它能够被grub装载和运行,并且可通过gdb进行调试,为接下去的工作创造一个基础环境。

首先,为了方便运行和调试我们需要一个虚拟机。虚拟机有很多选择,这里用最简单的qemu。

先用dd创建一个文件作为虚拟盘,100MB就可以了:

  1. $ dd if=/dev/zero of=disk.img count= bs=

然后对这个虚拟磁盘进行分区:

  1. $ fdisk -c=dos disk.img

用命令n创建一个分区就可以了。通常情况下分区的起始扇区是2048(不带选项-c=dos),如果用老式的msdos格式分区表(命令c),就可以选择从63扇区开始。以前文章里提到63个扇区足够塞下grub的核心映像,所以为了测试下grub就选择了msdos模式。用命令w把变动写入虚拟盘。

把这个分区虚拟成设备文件:

  1. $ sudo losetup -o /dev/loop1 disk.img

这里指定了起始扇区的偏移量。63个扇区,每个扇区512个字节,总共是32256字节。如果你的分区起始扇区是2048,那么偏移量应该是2048 * 512字节。

格式化:

  1. $ sudo mkfs.ext4 /dev/loop1

挂载起来,这样就可以方便地往里面放kernel和grub需要的配置文件和模块什么的:

  1. $ sudo mount /dev/loop1 /mnt

安装grub:

  1. $ sudo grub-install --boot-directory=/mnt --modules="part_msdos" disk.img

使用qemu来启动虚拟机(我用的是64位系统):

  1. $ qemu-system-x86_64 -hda disk.img -m -s &

这时候应该能够看到grub的提示符了。当然现在还没有grub菜单也没有kernel,我们暂时先关掉虚拟机。

接下来可以为grub建立个multiboot启动菜单:

  1. $ sudo vi /mnt/grub/grub.cfg

制作菜单命令:

  1. menuentry "Hello" {
  2.   multiboot (hd0,msdos1)/kernel
  3.   boot
  4. }

确保数据写回了虚拟盘:

  1. $ sync

这时候如果你再打开虚拟机,应该就可以看到启动菜单了,当然因为还没有kernel,选择菜单项后无法继续,会提示kernel找不到。

下一步,我们用c语言从头编写个最简单kernel程序。这个kernel没有实现操作系统的基本功能。但是可以被grub装载和运行。

kernel.c:

  1. /* 在文件里嵌入一个签名。Grub在multiboot时会寻找这个签名 */
  2. #define GRUB_MAGIC 0x1BADB002
  3. #define GRUB_FLAGS 0x0
  4. #define GRUB_CHECKSUM (-1 * (GRUB_MAGIC + GRUB_FLAGS))
  5.  
  6. struct grub_signature {
  7. unsigned int magic;
  8. unsigned int flags;
  9. unsigned int checksum;
  10. };
  11.  
  12. struct grub_signature gs __attribute__ ((section (".grub_sig"))) = {
  13. GRUB_MAGIC,
  14. GRUB_FLAGS,
  15. GRUB_CHECKSUM
  16. };
  17.  
  18. /* 显示字符的函数。因为我们什么库都不能用,只能直接写屏了。0xB8000是VGA彩色字符模式的数据缓存。每个字符用两个字节表示。前一个是Ascii码,后一个代表颜色。 */
  19. void puts( const char *s, int color ){
  20. volatile char *buffer = (volatile char*)0xB8000;
  21.  
  22. while( *s != ) {
  23. *buffer++ = *s++;
  24. *buffer++ = color;
  25. }
  26. }
  27.  
  28. /* 主函数,程序入口。最后用个死循环,把代码指针困在那里。*/
  29. void main (void) {
  30. puts("Hello World!", 0x7);
  31. while () {}
  32. }

有必要再写个链接模板,确保编译好的kernel装载在内存地址0x100000,这里是grub代码最后跳转到的区域,从这里我们的kernel接过了接力棒。另外,虽然我用的虚拟机是64位的,但是我需要生成一个32位的kernel,因为做64位的kernel还需要做额外的设置工作,比如从32位保护模式打开long mode,比较麻烦。所以先暂时用32位的kernel。

kernel.ld:

  1. OUTPUT_FORMAT("elf32-i386")
  2. ENTRY(main)
  3. SECTIONS
  4. {
  5. .grub_sig 0x100000 : AT(0x100000)
  6. {
  7. *(.grub_sig)
  8. }
  9. .text :
  10. {
  11. *(.text)
  12. }
  13. .data :
  14. {
  15. *(.data)
  16. }
  17. .bss :
  18. {
  19. *(.bss)
  20. }
  21. /DISCARD/ :
  22. {
  23. *(.comment)
  24. *(.eh_frame)
  25. }
  26. }

还有一个Makefile,主要是设置一些编译选项。

Makefile:

  1. CC = gcc
  2. LD = ld
  3. CFLAGS = -std=c99 -pedantic -Wall -nostdlib -ffreestanding -m32
  4. LDFLAGS = -T kernel.ld -nostdlib -n -melf_i386
  5. OBJS = kernel.o
  6. .PHONY: all
  7. all: kernel
  8. %.o: %.c
  9. ${CC} -c ${CFLAGS} $<
  10. kernel: $(OBJS) kernel.ld
  11. ${LD} ${LDFLAGS} -o kernel ${OBJS}
  12. clean:
  13. rm -f ${OBJS} kernel

编译生成kernel并放入我们的虚拟盘里:

  1. $ sudo cp kernel /mnt/
  2. $ sync

再次启动虚拟机,在启动菜单里选择multiboot我们的kernel,应该就能看到Hello World!的字符显示在虚拟机屏幕上了。

如果想要调试,可以运行gdb。因为我们在启动qemu的时候使用了-s选项,所以qemu默认会打开tcp端口1234作为gdb调试端口。在gdb中可以使用target remote tcp::1234命令来连接。试试看连接,会发现cpu一直在执行0x100066处的指令。用objdump -D kernel看下kernel的汇编代码:

  1. 0010004c <main>:
  2. 10004c:55 push %ebp
  3. 10004d:89 e5 mov %esp,%ebp
  4. 10004f:83 ec 08 sub $0x8,%esp
  5. :c7 44 24 04 07 00 00 movl $0x7,0x4(%esp)
  6. :0005a:c7 04 24 68 00 10 00 movl $0x100068,(%esp)
  7. :e8 a6 ff ff ff call 10000c <puts>
  8. :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. 2D绘图引擎比较

    这个问题很普遍.最近在研究这个问题,在网上搜了一些资料,再结合自己的经验,谈谈自己的一些想法. 一.双缓存能提高绘图效率吗? 网上有篇文章:绘图效率完整解决方案——三种手段提高GDI/GDI+绘图效率 ...

  2. 去掉第一次ssh连接的yes问题

    第一次执行的时候,有的机器可能会提醒输入ssh初次登录询问yes/no,如果要去掉这个yes/no的询问环节, 只需要修改本机的/etc/ssh/ssh_config文件中的"# Stric ...

  3. 【BZOJ2616】SPOJ PERIODNI 笛卡尔树+树形DP

    [BZOJ2616]SPOJ PERIODNI Description Input 第1行包括两个正整数N,K,表示了棋盘的列数和放的车数. 第2行包含N个正整数,表示了棋盘每列的高度. Output ...

  4. 【BZOJ2553】[BeiJing2011]禁忌 AC自动机+期望DP+矩阵乘法

    [BZOJ2553][BeiJing2011]禁忌 Description Magic Land上的人们总是提起那个传说:他们的祖先John在那个东方岛屿帮助Koishi与其姐姐Satori最终战平. ...

  5. 理解Javascript__理解undefined和null

    来自普遍的回答: 其实在 ECMAScript 的原始类型中,是有Undefined 和 Null 类型的. 这两种类型都分别对应了属于自己的唯一专用值,即undefined 和 null. 值 un ...

  6. Oracle之rman常用命令及维护(51CTO风哥rman课程)

    list 查看数据库备份的信息 查询数据库对应物 list incarnation; list backup summary; 列出当前备份信息及汇总 B是备份 F是全备 A是归档 第三个A是是否有效 ...

  7. 搜索R包和查看包的技巧

    1.R怎么搜包 # 安装sos包 install.packages("sos") # 导入sos包 library(sos) # 使用函数findFn() 函数里面传入关键字来搜索 ...

  8. xcode6 下载

    百度云下载 http://pan.baidu.com/s/1qWpuIC0提取码:ip9o 苹果下载: http://adcdownload.apple.com//wwdc_2014/xcode_6_ ...

  9. jsp ----- form表单

    jsp页面form表单中的action的值,最前面不加“/”

  10. 剪花布条---hdu2087(kmp模板)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=2087 kmp模板题: #include <cstdio> #include <cst ...