![ffd07516798748fcf561bfc990ead1b2.png](https://i-blog.csdnimg.cn/blog_migrate/811bc2005f0b64b39c4fd353f427e195.jpeg)
接上节,前文请见“一步步编写操作系统 69 汇编语言和c语言共同协作”,
本节是对前文的代码说明,代码再贴过来,C_with_S_c.c如下:
1extern void asm_print(char*,int); 2void c_print(char* str) { 3 int len=0; 4 while(str[len++]); 5 asm_print(str, len); 6}
代码C_with_S_c.c的第1行是声明外部函数asm_print,通知编译器这个函数并不在当前文件中定义。我们知道它定义在文件C_with_S_S.S中,但编译器是不知道的,所以只能在链接阶段将此函数重新定位,编排地址。
第2~6行是c_print的实现,在它的内部,它又调用了汇编代码C_with_S_S.S中的函数asm_print,经过第1行的声明,我们要给它提供两个参数:字符串的起始地址及长度。
特别强调一下,由于这里并不打算把c标准库也链接进来,所以在求字符串长度时,我们不能用string.h中的strlen函数。也就是说即使include 将其strlen的声明加进来,没有strlen实现本身所在的.o文件也是不行的。函数声明的作用是:一方面是告诉编译器该函数的参数所需要的栈空间大小及返回值,这样编译器能为其准备好执行环境,另一方面是如果该函数是在外部文件中定义的,一定要在链接阶段时将其对应的目标文件一块链接进来。所以这里第3~4行通过while循环求字符串的长度。字符串结尾必须是空字符’0’才行,否则while就是死循环了。这个字符串是代码C_with_S_S.S提供的,我们转过去看看。
1 section .data 2 str: db "asm_print says hello world!