C语言程序号称能无缝移植到所有平台和系统,意思是你在Windows操作系统上写的C程序能毫无修改的直接在Linux系统上运行。那我们的操作系统也必须要支持标准的C语言程序移植,本章就来完成这个任务。
一、 Windows系统C程序
我们在Windows系统上写一个“hello world”C程序,取名cstdio.c,编译链接后成功运行:
二、 移植C程序
现在要将cstdio.c程序移植到我们的操作系统上。首先分析,该程序调用了两个标准函数:printf和exit。那我们的操作系统就需要制作出两个新的API,来提供给C语言应用程序调用。
(一)编写C程序标准函数
由于我们之前已经制作了字符串打印的API,因此这次制作就只需要建立在之前的基础上。抽取几个例子如下:
1.首先是C程序标准函数putchar,我们用程序putchar.c来实现:
void putchar(int c)
{
api_putchar(c);
return c;
}
这个逻辑比较简单,其实就是调用一次我们的API:api_putchar。
2.其次是C程序标准函数printf,我们用程序printf.c来实现:
#include <stdio.h>
#include <stdarg.h>
int printf(char *format,...)
{
va_list ap;
char s[1000];
int i;
va_start(ap,format);
i=vsprintf(s,format,ap);
api_putstr0();
va_end(ap);
return i;
}
这个相对来说就复杂些,这里不展开详解了,有兴趣的人可以自己研究。
3.最后再制作一个exit.c:
void exit(int status)
{
api_end();
}
(二)制作C程序标准函数库
我们仿照之前制作API库的方法,把所有的C程序标准函数程序放在一个目录cstdiolib下,然后将编译之后的目标文件全部包装到一个库文件中:cstdio.lib。最后把这些标准函数的申明写在一个取名stdio.h的头文件中。这样,C程序使用者就可以通过#include <stdio.h>来畅通无阻的调用了。
这个制作C程序标准函数库cstdio.lib的主体makefile:
OBJS_API = putchar.obj exit.obj printf.obj malloc.obj free.obj
default :
$(MAKE) cstdio.lib
cstdio.lib : Makefile $(OBJS_API)
$(GOLIB) $(OBJS_API) out:cstdio.lib
%.gas : %.c Makefile ../app_make.txt
$(CC1) -o $*.gas $*.c
%.nas : %.gas Makefile ../app_make.txt
$(GAS2NASK) $*.gas $*.nas
%.obj : %.nas Makefile
$(NASK) $*.nas $*.obj $*.lst
(三)验证支持C程序标准函数
将应用程序cstdio.c和我们的操作系统底层API库:apilib.lib(必须要,因为标准函数里面调用了底层API)以及刚制作的C程序标准函数库: cstdio.lib放在一起链接,最终生成可执行文件cstdio.hrb。
$(APP).bim : $(APP).obj $(APILIBPATH)apilib.lib $(CSTDIOPATH)cstdio.lib Makefile ../app_make.txt
$(OBJ2BIM) @$(RULEFILE) out:$(APP).bim map:$(APP).map stack:$(STACK)
$(APP).obj $(APILIBPATH)apilib.lib $(CSTDIOPATH)cstdio.lib
$(APP).hrb : $(APP).bim Makefile ../app_make.txt
$(BIM2HRB) $(APP).bim $(APP).hrb $(MALLOC)
所有的事情都准备好了,在窗口中启动运行cstdio.hrb能成功打印字符串,表明标准C程序成功移植到了我们的操作系统:
虽然,我们又回到了打印“hello world”的场景,但是已经时过境迁,第一天我们开始操作系统制作的时候,启动裸机打印出了“hello world”。通过一步步的努力,今天我们的操作系统已经可以支持标准C程序打印“hello world”了。如果有精力,把所有的C程序标准函数都来实现了,那这个操作系统就可以毫无阻拦的运行大型C程序了。