破碎细胞 发表于 2009-02-17 13:23

[貌似已解决]gcc编译汇编和c的函数调用的问题

感谢大家热情的回复,我后来自己好好想了想,然后翻了翻资料,和同学讨论了下,貌似找到了问题的根源。
main.o的.text被ld贴到了512字节后面,但bios只会将第一个扇区读入内存,所以后面的压根就没有读入,自然一call就挂了。
不知道,我这样理解有没有错。
本来是想在写bootsector的时候就用C,看来还是不能偷懒阿,只有在loader再开始了。

我写了一个引导扇区的程序,想在其中调用用c写的函数,但是发现,一旦在汇编中调用了C的函数之后,系统就卡在那里不往下执行了,请问这怎么解决阿。
我是新手,还望大家多多帮忙阿。
boot.S:
#define MSG(x) leaw x,%si;call display

.code16
.section .text
.globl _start
_start:
    movw $0x2000,%sp
    movw %sp,%bp


    #set video mode
    movw $0x0003,%ax
    int $0x10
    
    MSG(sector_msg1)
    call main
    MSG(sector_msg2)
    
    
    _endless_loop:jmp _endless_loop

sector_msg1:
    .asciz "bootsector: looking for a loader... "
sector_msg2:
    .asciz "found\n\r"

.globl test
test:
    MSG(sector_msg2)
    ret

display:
    cld
    movb $0xe,%ah    /* use BIOS interrupt 10,e */
    movb $0xf,%bl    /* font color */
    display_start:
    lodsb
    orb %al,%al
    jz display_end
    int $0x10
    jmp display_start
    display_end:
    ret


.org 510
.word 0xaa55

main.c
extern void test();
void main()
{
    test();
    return;
}

链接脚本boot.ld:
OUTPUT_FORMAT("binary")
ENTRY(_start)
SECTIONS
{
    .text 0x7c00:
    {
      *(.text)
    }
}

Makefile:
CC=gcc
LD=ld
AS=as
LDSCRIPT=carrot.ld
DEBUG_DIR=./
INCLUDE_DIR=../include/

all:boot.img

boot.img:boot.bin
    dd if=/dev/zero of=$(DEBUG_DIR)emptydisk.img bs=512 count=2880
    dd if=boot.bin of=$(DEBUG_DIR)boot.img bs=512 count=1
    dd if=$(DEBUG_DIR)emptydisk.img of=$(DEBUG_DIR)boot.img skip=1 seek=1 bs=512 count=2879
boot.bin:boot.o
    $(LD) -Tboot.ld boot.o main.o -o boot.bin
boot.o:boot.S
    $(CC) -c -I$(INCLUDE_DIR) -o boot.o boot.S
    $(CC) -c -I$(INCLUDE_DIR) -o main.o main.c

clean:
    rm -rf *.bin *.o $(INCLUDE_DIR)/boot.img

[ 本帖最后由 破碎细胞 于 2009-2-19 17:18 编辑 ]

lixinwei1985@si 发表于 2009-02-17 14:01

好想没看到指定SP 是不这问题

破碎细胞 发表于 2009-02-17 17:50

原帖由 lixinwei1985@si 于 2009-2-17 14:01 发表 http://linux.chinaunix.net/bbs/images/common/back.gif
好想没看到指定SP 是不这问题
加上了这个
movw $0x2000,%sp
movw %sp,%bp
还是这个样子

cwqing1973 发表于 2009-02-17 18:31

这可能是常识的问题,是你没有注意,一个程序应该只有一个main函数!你这里C函数是用main!

beepbug 发表于 2009-02-17 18:43

原帖由 破碎细胞 于 2009-2-17 13:23 发表 http://linux.chinaunix.net/bbs/images/common/back.gif
我写了一个引导扇区的程序,想在其中调用用c写的函数,但是发现,一旦在汇编中调用了C的函数之后,系统就卡在那里不往下执行了,请问这怎么解决阿。
我是新手,还望大家多多帮忙阿。
boot.S:
#define MSG(x) ...
没看你的代码,因为你的概念就是错的。
当你执行“引导扇区的程序”时,内存里就只有ROM-BIOS和这个引导记录代码,C库函数代码还没有进来,你没法调用。

破碎细胞 发表于 2009-02-17 18:45

回复 #4 cwqing1973 的帖子

貌似不是这个问题,改成cmain结果是一样的。

cjaizss 发表于 2009-02-19 10:02

你用你的MSG宏插到main函数里去跟踪看看

破碎细胞 发表于 2009-02-19 16:51

原帖由 beepbug 于 2009-2-17 18:43 发表 http://linux.chinaunix.net/bbs/images/common/back.gif

没看你的代码,因为你的概念就是错的。
当你执行“引导扇区的程序”时,内存里就只有ROM-BIOS和这个引导记录代码,C库函数代码还没有进来,你没法调用。
并没有调用C的库函数。
我的目的仅仅是想用C来写一部分代码,这样轻松一点。
还是看了代码再说比较好一点。

破碎细胞 发表于 2009-02-19 17:11

原帖由 cjaizss 于 2009-2-19 10:02 发表 http://linux.chinaunix.net/bbs/images/common/back.gif
你用你的MSG宏插到main函数里去跟踪看看
我貌似大概知道什么问题了。
C的.text被贴到了512字节后面,而bios开始只读取引导扇区,也就是512个字节,后面的压根就读如内存,所以一call就挂了。不知道这样解释有没有道理。
其实,我的初衷是想在bootsector开始就用c来写部分代码,现在看来只有在loader开始了。

swordfish.cn 发表于 2009-02-19 17:16

Mark 一下,哈哈。不能偷懒哦。
页: [1] 2
查看完整版本: [貌似已解决]gcc编译汇编和c的函数调用的问题