本文介绍如何为编译gmake编写Makefile文件
1 编写Makefile
根据前面所学内容编写Makefile如下,基本与前面所写的lua解释器的Makefile文件用法相同.注意生成文件为tcc-make.exe,除了名称有点差异,其他与前面生成make.exe完全相同。
c
# 编译gmake的Makefile
# make_name为生成程序的名称,可在批处理中进行设置
# 当没有设置make_name的值时,将make_name设置为tcc-make.exe
# 之所以不将将make_name设为make.exe,因为make.exe无法对make.exe进行处理
make_name ?= tcc-make.exe
#设置警告信息和包含路径
include = -Wall -I src -I lib -I src/w32/include
#设置编译常量
define = -D GMAKE_FOR_BUSYBOX -D WINDOWS32 -D HAVE_CONFIG_H -D HAVE_STDINT_H -D HAVE_STRNICMP -D HAVE_UMASK -D WIN32_LEAN_AND_MEAN= -D ftruncate=_chsize -D _cdecl=
#设置编译c代码路径,注意\*需写为\\*
src = lib\\*.c src\\*.c src\w32\\*.c src\w32\compat\\*.c src\w32\subproc\\*.c
#设置链接库
lib = -luser32 -ladvapi32
# 编译tcc-make.exe文件,默认指令,调用[link]编译指令
# 无论当前目录下是否存在tcc-make.exe文件,均可重新进行编译
build:link
@rem 注释
@echo 生成$(make_name)成功!
# 编译tcc-make.exe文件
# 若当前目录下有tcc-make.exe文件,则不会编译
$(make_name):
@tcc $(include) $(define) $(src) $(lib) -o $(make_name)
@echo 生成$(make_name)成功!
#只编译,生成obj文件
compile:
@rem 编译主要源码
@tcc -c $(include) $(define) src\\*.c
@rem 将main.o移动到lib目录下,主要为了后面编译静态库
move main.o lib\main.o
@rem 将misc.o重命名为misc1.o
@rem 这步很关键,因为后面src\w32\subproc\misc.c也会编译为misc.o,导致这个misc.o被覆盖从而无法编译
@move misc.o misc1.o
@rem 编译其他源码
@tcc -c $(include) $(define) lib\\*.c
@tcc -c $(include) $(define) src\w32\\*.c
@tcc -c $(include) $(define) src\w32\compat\\*.c
@tcc -c $(include) $(define) src\w32\subproc\\*.c
@echo 编译成功!
# 进行链接,会先调用[compile]指令
link:compile
@rem 对所有obj文件进行链接生成tcc-make.exe
@tcc lib\main.o *.o $(lib) -o $(make_name)
@echo 链接成功!
# 生成静态链接库并生成tcc-make.exe
lib:compile
@rem 生成静态链接库libmake.a
@tcc -r *.o -o libmake.a
@rem 使用静态链接库进行链接生成tcc-make.exe
@tcc lib\main.o libmake.a $(lib) -o $(make_name)
@echo 生成静态链接库并生成$(make_name)
# 生成动态链接库并生成tcc-make.exe
dll:compile
@rem 生成动态链接库make.dll
@rem 注意需包含lib\main.o
@tcc -shared -rdynamic lib\main.o *.o $(lib) -o make.dll
@rem 只需make.def,不需要lib\main.o
@tcc make.def -o $(make_name)
@echo 生成动态链接库并生成$(make_name)
# 显示tcc-make.exe的版本
v:$(make_name)
@.\$(make_name) -v
# 清除所有编译产生的文件
clean:
@if exist $(make_name) del $(make_name)
@if exist *.o del *.o
@if exist lib\main.o del lib\main.o
@if exist libmake.a del libmake.a
@if exist make.dll del make.dll
@if exist make.def del make.def
@echo 清除所有编译产生的文件成功
# 帮助
help:
@echo ------------------------------------
@echo 这是一个使用tcc编译$(make_name)的Makefile脚本,具体用法如下
@echo make-生成直接生成$(make_name)
@echo make build-生成直接生成$(make_name)
@echo make lua.exe-若没有$(make_name)生成$(make_name)
@echo make run-运行$(make_name)
@echo make compile-编译所有c代码
@echo make link-链接所有c代码
@echo make lib-生成静态链接库并生成$(make_name)
@echo make dll-生成动态链接库并生成$(make_name)
@echo make clean-清除所有编译文件
@echo make v-显示$(make_name)版本
@echo 中文用法见make 帮助
@echo ------------------------------------
# 中文化选项
# 实际上主要依靠执行依赖项build,编译指令为空指令
构建:build
@rem
运行:run
@rem
编译:compile
@rem
链接:link
@rem
静态:lib
@rem
动态:dll
@rem
清除:clean
@rem
版本:v
@rem
帮助:
@echo ------------------------------------
@echo 这是一个使用tcc编译$(make_name)的Makefile脚本
@echo 下面主要介绍中文指令用法,英文用法见make help
@echo make 构建-生成直接生成$(make_name)
@echo make 运行-运行$(make_name)
@echo make 编译-编译所有c代码
@echo make 链接-链接所有c代码
@echo make 静态-生成静态链接库并生成$(make_name)
@echo make 动态-生成动态链接库并生成$(make_name)
@echo make 清除-清除所有编译文件
@echo make 版本-显示lua版本
@echo ------------------------------------
2 特别注意事项
之所以生成文件为tcc-make.exe,是因为将生成文件名称设为make.exe会有很多问题,下面进行讲解,首先这个Makefile可以通过make_name变量控制输出文件名称,如
c
:: 将输出文件名称设置为gmake.exe
set make_name=gmake.exe
::编译生成gmake.exe
make
在批处理窗口输入上面内容,即可生成gamke.exe
若想生成make.exe,可以这么输入
c
:: 将输出文件名称设置为make.exe
set make_name=make.exe
::编译生成make.exe
make
上诉命令可以生成make.exe,但是重复输入make则无法重新生成make.exe,错误信息如下
即make.exe无法对当前目录下的make.exe文件进行删除,提示权限不足.暂时没有好的解决办法.在执行其他命令时也会有这个问题.
只能先生成其他名字再手动更改为make.exe,其次还有另外一种办法解决,就是就是使用其他make程序,我在C:\run\tcc目录下将原先的make.exe复制一份名称改为tcc-make,重新回到gmake文件夹,在批处理窗口输入
c
:: 将输出文件名称设置为make.exe
set make_name=make.exe
::使用tcc-make编译生成make.exe
tcc-make
::再次编译生成make.exe覆盖原来的make文件
tcc-make
可以看出tcc-make.exe程序可以正常生成和处理make.exe,但是tcc-make.exe也同样无法删除tcc-make.exe,即make程序无法删除同名文件.