C编译器不生成有效的程序集输出

时间:2012-03-04 15:39:06

标签: c assembly kernel

我在C和ASM中编写一个简单的内核(调用C编译生成的汇编指令)。我正在关注osdev.org上的这篇优秀文章。在阅读了用汇编语言编写并编译并执行的内核之后,它运行良好,我看到在qemu上运行的kernel.bin非常棒。但我真的想用C语言写。我在同一个网页上找到了an example

我编译了它并尝试在qemu上运行kernel.bin但是我从所有设备上“启动失败”:硬盘,floopd和CD-ROM。然后我反汇编了由ks.o生成的kernel.bin(使用nasm汇编的kernel.asm)和使用ld汇编到bin文件的kernel.o(使用gcc汇编的kernel.c)。 kernel.bin:

http://pastebin.com/Y0pLFvij我看不到我的字符串或任何组装指令在视频上写我的字符串。

生成它的C代码:

#define WHITE_TXT 0x07 // white on black text
unsigned int k_printf(char *message, unsigned int line);
void k_clear_screen();

k_main() 
{
    k_clear_screen();
    k_printf("Hi!\nHow's this for a starter OS?", 0);
};

unsigned int k_printf(char *message, unsigned int line) // the message and then the line #
{
    char *vidmem = (char *) 0xb8000;
    unsigned int i=0;

    i=(line*80*2);

    while(*message!=0)
    {
        if(*message=='\n') // check for a new line
        {
            line++;
            i=(line*80*2);
            *message++;
        } else {
            vidmem[i]=*message;
            *message++;
            i++;
            vidmem[i]=WHITE_TXT;
            i++;
        };
    };

    return(1);
};

void k_clear_screen() // clear the entire text screen
{
    char *vidmem = (char *) 0xb8000;
    unsigned int i=0;
    while(i < (80*25*2))
    {
        vidmem[i]=' ';
        i++;
        vidmem[i]=WHITE_TXT;
        i++;
    };

和调用它的assembly

[bits 32]

[global start]
[extern k_main] ; this is in the c file

start:
  call k_main

  cli  ; stop interrupts
  hlt ; halt the CPU

link.ld

OUTPUT_FORMAT("binary")

ENTRY(start)

SECTIONS

{

  .text  0x100000 : {

    code = .; _code = .; __code = .;

    *(.text)

    . = ALIGN(4096);

  }

  .data  : {

    data = .; _data = .; __data = .;

    *(.data)

    . = ALIGN(4096);

  }

  .bss  :

  {

    bss = .; _bss = .; __bss = .;

    *(.bss)

    . = ALIGN(4096);

  }

  end = .; _end = .; __end = .;

}

和makefile:

bin:
    nasm -f aout -o ks.o kernelbase.asm 
    gcc -Wall -c -o kernel.o kernel.c    
    ld -T link.ld -o kernel.bin ks.o kernel.o 
run1:
    qemu kernel.bin

clear:
    rm -f *.o

cbin:
    rm -f *.bin

和纯汇编内核,反汇编:http://pastebin.com/Bkvkq3YQ 这对我来说很好,我可以看到在视频上写的指示。

我的操作系统:Ubuntu 10.4
编译器:gcc
汇编程序:NASM

我哪里错了?

1 个答案:

答案 0 :(得分:1)

我尝试做那个教程,代码也失败了。最后我这样做了:

将这两个文件编译为32位:

nasm -f elf32 ks.asm -o kernelbase.o
gcc -m32 -c kernel.c -o kernel.o

并链接:

gcc -m32 kernelbase.o kernel.o -o Kernel.bin

过了一段时间后,我发现使用gcc进行链接时没有出现任何错误。

希望我帮助过: - )