如何在Windows上为我的操作系统编译和链接C和ASM

时间:2015-08-28 20:52:25

标签: c assembly x86 nasm tdm-gcc

我的32位保护模式操作系统项目Sinatra出现问题。我可以将源代码编译到目标文件,但我不知道如何将它们链接在一起。我在Windows上使用 NASM TDM-GCC 。我修复了我的代码问题所以它编译。为简洁起见,我删除了评论。

我的档案boot.asm

[BITS 32]
[global start]
[extern _JlMain]
start: 
    cli
    call _JlMain
    hlt

我的档案JSinatra.h

#ifndef __SINATRA_H__
#define __SINATRA_H__

#define JWhiteText 0x07
void JlMain();
void JlClearScreen();
unsigned int JlPrintF(char * message, unsigned int line);

#endif

我的档案JSinatra.c

#include "JSinatra.h"

void JlClearScreen() // clear entire screen
{
    char * vidmem = (char * ) 0xb8000;
    unsigned int i = 0;
    while (i < (80 * 25 * 2)) {
        vidmem[i] = ' ';
        i += 1;
        vidmem[i] = JWhiteText;
        i += 1;
    }
}
unsigned int JlPrintF(char * message, unsigned int line) {
    char * vidmem = (char * ) 0xb8000;
    unsigned int i = 0;
    i = line * 80 * 2;
    while ( * message != 0) {
        if ( * message == '\n') {
            line += 1;
            i = (line * 80 * 2); * message += 1;
        } else {
            vidmem[i] = * message; * message += 1;
            i += 1;
            vidmem[i] = JWhiteText;
            i += 1;
        }
    }
    return (1);
}
void JlMain() {
    JlClearScreen();
    JlPrintF("Sinatra v0 Virgin/Kernel Mode\n", 0);
}

我需要从绝对地址0x100000开始加载我的操作系统。如何正确编译和链接我的代码以创建二进制图像?

2 个答案:

答案 0 :(得分:2)

首先,如果您要编译为ELF,那么在组装函数之前,您不能添加初始下划线。

现在,为了将不同的源文件链接在一起,你显然必须让它们达到共同点,在这种情况下是目标代码。

所以,你要做的是:

  1. 将程序集源文件汇编到目标代码。
  2. 编译但不将C源文件链接到目标代码。在gcc中:gcc -c file.c -o file.o
  3. 将这些链接在一起。在gcc中:gcc cfile.o asfile.o -o app

答案 1 :(得分:0)

在Windows上使用GCC-TDM和NASM

因为您的目标是在没有C运行时的情况下以绝对地址加载操作系统,所以您需要确保编译为独立代码;您的 asm C 文件的目标是相同类型的对象(win32 / PECOFF);最后一步是将PECOFF文件转换为二进制图像。

要编译 C 文件,您可以使用以下内容:

gcc -m32 -ffreestanding -c JSinatra.c -o JSinatra.o

要汇编 asm 文件,您可以使用以下内容:

nasm -f win32 boot.asm -o boot.o

要将它们链接在一起,您必须分两步完成:

ld -m i386pe -T NUL -o sinatra.tmp -Ttext 0x100000 boot.o JSinatra.o

上面的ld命令将创建一个临时文件sinatra.tmp,它是一个32位的PECOFF可执行文件。然后,您需要使用以下命令将sinatra.tmp转换为二进制图像:

objcopy -O binary sinatra.tmp sinatra.img

然后,您应该在文件sinatra.img

中包含二进制图像