• 嵌入式操作系统课程作业之Write OS


    嵌入式操作系统课程作业之Write OS
    [作者]:菩提树下的杨过 [来源]:互联网 [收录时间]:2007-8-1 20:18:38

    嵌入式操作系统课程作业之Write OS

    姓名:唐良学号: 024304xxxx

    .编译器的下载

    1. Djgpp

    Windows开发环境下,没有直接提供gcc编译器,所以需要从自己去下载WINDOWS版本的gcc编译器.Windows下有cygwin, devcpp, djgpp等工具都带有gcc编译器.但是djgpp最小,而且可以产生最单纯的binary代码,所以很多关于操作系统编写的网站上都推荐使用djgpp.

    2. Nasm

    Nasmmasm,以及as86等都很类似,都是支持16位和32位的汇编编译器.但是nasm使用起来比较轻活,能够产生多种中间代码格式,比如*.obj,*.o,所以很多操作系统编写的网站上也都推荐使用nasm作为汇编编译器.

    . boot启动代码的编写

    Boot启动代码主要完成转载kernel,进入32为模式等工作.除开Linux标准代码外,网上关于boot启动的模板代码多不胜数,不过我使用的还是哈工大的pyos的启动代码.

    Boot.s的代码如下:

    [BITS 16]

    [ORG 0x7C00]

    jmpmain

    ; ----------------------------------------------------------------------------------------------

    ; 数据定义

    bootdrivedb0

    ; ----------------------------------------------------------------------------------------------

    ; GDT 定义

    gdt:

    gdt_null:

    dd0

    dd0; 空描述符全是0

    gdt_code_addr equ $ - gdt; 数据段在GDT表中的位置

    gdt_code:

    dw0xffff; 段大小为4GB

    dw0; 基址的低16

    db0; 基址的高八位

    db10011010b

    db11001111b

    db0

    gdt_data_addr equ $ - gdt; 数据段在GDT表中的位置

    gdt_data:

    dw0xffff

    dw0x0000

    db0

    db10010010b

    db11001111b

    db0

    gdt_end:

    gdt_addr:

    dwgdt_end - gdt - 1; GDT 表的大小

    ddgdt; GDT 表的位置

    ; --------------------------------------------------------------------------------------

    main:

    mov [bootdrive] , dl; 他得到启动的驱动器号

    xor ax , ax; 设置 DS

    mov ds , ax

    ; 清屏

    ;mov ax , 3; 设置清屏功能号

    ;int 0x10; 调用 BIOS 10 号中断清屏

    .ResetFloppy; 重置磁盘

    mov ax , 0; 设置重置磁盘的功能号

    mov dl , [bootdrive]; 选择启动磁盘

    int 0x13

    jc .ResetFloppy; 如果出错则重试

    .ReadFloppy; 读内核到内存中 0000:9000 (es:bx)

    xor ax , ax; 设置 es 寄存器

    mov es , ax

    mov bx , 0x9000

    mov ah , 2; 设置读磁盘功能号

    mov dl , [bootdrive]; 设置欲读驱动器号

    mov ch , 0; 磁头号

    mov cl , 2; 起始扇区号

    mov al , 17; 读入扇区数量

    int 13h

    jc .ReadFloppy

    mov dl , [bootdrive]; 停止驱动器

    mov edx , 0x3f2

    mov al , 0x0c

    out dx , al

    cli; 关中断

    lgdt [gdt_addr]; 载入 GDT 的描述符

    mov eax , cr0; 下面三句设置 cr0 的第 0 位(PE位)为1,表示进入保护模式

    or eax , 1

    mov cr0 , eax

    jmp gdt_code_addr:code_32; 跳入32位的代码段中

    [BITS 32]

    code_32:

    mov ax , gdt_data_addr; 以下三句设置 DSESSSFSGS的置为数据段描述表的位置

    mov ds , ax

    mov es , ax

    mov ss , ax

    mov fs , ax

    mov gs , ax

    mov esp , 0xffff; 设置堆栈的头指针

    jmp gdt_code_addr:0x9000; 跳入内核

    ;---------------------------------------------------------------------------

    times 510-($-$$) db 0

    db 0x55

    db 0xAA

    . 编译boot启动代码

    nasm十分简单. 可以很容易生成bin文件.它默认生成的也是bin原始代码.直接输入nasmw boot.s就能够得到boot原始代码.

    然后将通过WinHexboot的全部代码复制到一张1.44MB的软盘镜像文件的头512的字节中去.可以通过Bochs,Virtual PC,VMWare来建立软件镜像文件.这些软件的软盘镜像文件都是原始数据文件,没有任何关于软盘的配置数据,所以直接通过WinHex中的Ctrl+CCtrl+B(千万不能是Ctrl+V)就可以完成复制了.

    . 编写显示内核进入装载的C代码

    如果按照pyos的第二个实验来做,我得到的C语言生成的代码是.data数据段放在了.text代码段前面,就不能直接一下jmp 0x9000进入kernel的初始程序的代码段.后来我参考了网上一些做法,有种办法是编写link.script的连接脚本,有些是增加一个类似crt0.sC语言启动汇编.我选择的是后者.

    1. 编写C代码

    char* msg = "Welcome to SCU Operation System!Version 0.0001 by tangl_99" ;

    void k_main()

    {

    unsigned char* videomem = ( unsigned char* )0xb8000 ;

    while( *msg != '"0' ){

    *videomem++ = *msg++ ;

    *videomem++ = 0x1b ;

    }

    for(;;);

    }

    这里使用k_main作为入口函数,有别于通常的main入口函数.

    2. 编写crt0.s启动代码

    [BITS 32]

    [global start]

    [extern _k_main] ; this is in the c file

    start:

    call _k_main

    cli; stop interrupts

    hlt ; halt the CPU

    3. 编译连接kernel初始程序

    编译连接部分我就没有参照pyos的做法了.根据网络的通常的做法,我的编译连接命令如下:

    setdjgpp d:"djgpp d:/djgpp# djgpp需要预先设置一下

    gcc –c kernel.c

    nasmw –f aout crt0.s

    ld -nostartfiles --oformat binary -Ttext 0x9000 -o kernel crt0.o kernel.o

    最后会生成1024字节大小的 kernel原始代码文件.还是通过WinHex将其复制到软盘镜像文件,要从第512个字节位置开始复制.

    . Virtual PC下启动运行生成OS 软盘镜像文件

    Virtual PC的使用就不再多说了,Virtual PC应该是Windows下最容易使用的模拟器.

    下面是运行0.0001版本的截图:

    嵌入式操作系统课程作业之Write OS

    点击查看大图
  • 相关阅读:
    model number
    LeetCode: Largest Rectangle in Histogram 解题报告
    Leetcode:Edit Distance 解题报告
    Leetcode:Interleaving String 解题报告
    Leetcode:Scramble String 解题报告
    Leetcode:【DP】Longest Palindromic Substring 解题报告
    Leetcode:Longest Substring Without Repeating Characters 解题报告
    Leetcode: Remove Duplicates from Sorted List II 解题报告
    Leetcode:Flatten Binary Tree to Linked List 解题报告
    LeetCode: Convert Sorted Array to Binary Search Tree 解题报告
  • 原文地址:https://www.cnblogs.com/adylee/p/1066780.html
Copyright © 2020-2023  润新知