NMake的原理及使用.doc_第1页
NMake的原理及使用.doc_第2页
NMake的原理及使用.doc_第3页
NMake的原理及使用.doc_第4页
NMake的原理及使用.doc_第5页
已阅读5页,还剩3页未读 继续免费阅读

下载本文档

版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领

文档简介

NMake的原理及使用1 makefile入门Windows CE的构建系统大量使用了Nmake工具和makfile。在大多数微软的软件和驱动开发包中都会包含Nmake工具。因此,这里有必要介绍一下makefile和Nmake工具。1.1 makefile简介对于许多Windows下的程序员来说,makefile可能还是个陌生的名词。因为Windows下的许多集成开发环境(例如:Microsoft Visual Studio和Borland C+ Builder等)可以帮助开发人员完成makefile需要完成的功能。通常只需要在集成开发环境中按个按钮,工具就可自动帮助我们编译、链接整个项目。想象如果没有了集成开发环境,那么就需要有另外一种方式来管理对项目的构建。简单的来说,makefile负责帮助开发人员简化代码的编译、链接等构建工作。对于只包含几个文件的简单的项目,开发人员完全可以通过手动控制编译器、链接器来完成对项目的构建。但是想象一下对于一个拥有几百个、甚至几千个文件的大型项目,如果每次构建都是通过手动完成,那消耗的工作量和复杂程度是不可想象的。在这种情况下,makefile就有了它的用武之地。makefile关系到了整个工程的编译规则。一个工程中的源文件不计数,其按类型、功能、模块分别放在若干个目录中,makefile定义了一系列的规则来指定,哪些文件需要先编译,哪些文件需要后编译,哪些文件需要重新编译,甚至于进行更复杂的功能操作,因为makefile就像一个自动化脚本一样,其中也可以执行操作系统的命令。makefile带来的最大的好处是“自动化构建”。写好makefile之后,在编译的时候只需要一个命令,整个工程完全自动编译、链接,极大的提高了软件开发的效率。makefile本质上只是一个文本文件,本身并不能运行。在运行makefile的时候还是需要外部程序来对makefile进行解释执行。NMake.exe就是用来解析并执行makefile的工具。当用户输入nmake命令后,首先nmake会读取makefile,然后解析makefile,并根据makefile的规则来确定要编译哪些代码。然后nmake会调用编译器、链接器等一些开发工具,完成对代码的编译链接。最终会生成可执行文件。 图:makefile的工作流程值得一提的是makefile既不是Windows CE特有的工具也不是微软的发明创造。makefile是一种通用的自动化构建手段。在 UNIX/Linux平台下有着广泛而众多的应用。许多开发工具都会提供NMake类似的工具。比如:Delphi的make,Visual C+的 nmake,Linux下GNU的make等等。1.2 makefile的编写规则makefile是由一个个推导和规则构成的,在NMake中这也被叫做描述块(Description Blocks)。一个最基本的推导规则的语法如下。targets. : dependents. commands.targets也就是一个目标文件,可以是Object File,也可以是可执行文件。还可以是一个标签(Label)。targets必须在一行的顶格写,前面不能有空格。dependents就是要生成target所需要的文件或是目标依赖项。dependents与targets之间用冒号间隔。一个targets可以有多个dependents。command也就是NMake需要执行的命令。其中commands可以是任意的Windows命令行命令。这是一个文件的依赖关系,也就是说,target这一个或多个的目标文件依赖于dependents中的文件,其生成规则定义在command中。也就是说,dependents中如果有一个以上的文件比targets文件要新的话,command所定义的命令就会被执行。这就是makefile的规则。也就是Makefile中最核心的内容。掌握了makefile最核心的内容,就可以尝试编写第一个makefile了。但是仅仅知道了这一点还远远不够。makefile还有很多细节的内容,下面会一点一点地介绍。在此之前,先看一个可以实际运行的makefile。以便读者对makefile 有个感性的认识。1.3 一个实际可以运行的makefile在%_WINCEROOT%PBWorkspacesMyPlatform下新建立一个目录hello,然后在hello目录下创建hello.cpp,内容如下:#include int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPTSTR lpCmdLine, int nCmdShow) MessageBox(NULL, LHello, Lbb, 0);本部分内容的目的就是使用NMake工具来把Hello.cpp构建为Hello.exe可执行文件。为此,在hello目录下再创建一个文本文件,名为makefile(没有扩展名)。然后用文本编辑器在makefile中输入如下内容:#This is a demo makefilehello.exe: hello.obj echo linking. link -MACHINE:x86 -NODEFAULTLIB -subsystem:windowsce,5.00 -entry:WinMainCRTStartup -LIBPATH:E:WINCE500PBWorkspacesMyPlatformprojrootcesysgensdklibx86retail hello.obj coredll.lib corelibc.libhello.obj: hello.cpp echo compiling. cl -nologo -c -I. -IE:WINCE500publiccommonsdkinc -DUNICODE -D_UNICODE -DUNDER_CE=500 -D_WIN32_WCE=500 -DWIN32 -DSTRICT -Dx86 -D_X86_ -DINTERNATIONAL -DL0804 -DINTLMSG_CODEPAGE=1252 hello.cppclean: del hello.obj, hello.exe第一行是注释,在makefile中,用#表示该行内容是注释。按照上文介绍的推导规则,这段makefile定义了两个依赖规则。hello.exe依赖于hello.obj,hello.obj又依赖于hello.cpp。要从hello.cpp生成hello.obj,需要经过编译过程,执行编译命令。上文makefile代码中定义了两个命令,一个是操作系统的内部命令 echo,作用是输出一个字符串,另外一个是调用C+编译器cl.exe,并用-c指示它只进行编译工作而不进行链接。对于从hello.obj生成 hello.exe的过程也是一样的。中间调用了链接器link.exe把几个库文件链接成hello.exe。为了简单起见,用到的路径都直接采用了写死的绝对路径。从“开始” “程序” “Microsoft Windows CE 5.0” 菜单打开Windows CE的控制台。然后cd到hello目录,输入nmake命令,控制台的输出结果如下:E:WINCE500PBWorkspacesMyPlatformhellonmakecompiling.Windows CE Version (Release) (Built on Mar 1 2004 21:46:39)cl -nologo -c -I. -IE:WINCE500publiccommonsdkinc -DUNICODE -D_UNICODE -DUNDER_CE=500 -D_WIN32_WCE=500 -DWIN32 -DSTRICT -Dx86 -D_X86_ -DINTERNATIONAL -DL0804 -DINTLMSG_CODEPAGE=1252 hello.cpphello.cpplinking.link -MACHINE:x86 -NODEFAULTLIB -subsystem:windowsce,5.00 -entry:WinMainCRTStartup -LIBPATH:E:WINCE500PBWorkspacesMyPlatformprojrootcesysgensdklibx86retail hello.obj coredll.lib corelibc.libMicrosoft (R) Incremental Linker Version 7.10.4017Copyright (C) Microsoft Corporation. All rights reserved.这是使用dir命令查看hello目录,可以看到多了hello.obj和hello.cpp两个文件。这说明nmake已经帮我们生成了目标文件。对于这个makefile,还有最后要介绍的一点。makefile中的clean是一个标签,通常所有的makefile中都会有clean这样一个标签,用来清理生成的文件,以便重新编译。为了调用这个标签可以在命令行下输入如下指令nmake clean这样,nmake就会帮我们调用系统的del命令,删除构建时生成的hello.obj和hello.exe文件了。1.4 使用变量让我们再回头来看看上一节中的示例makefile。如果我们希望把生成的文件由hello.exe改变为nihao.exe,那么仅仅这一丁点改动,在 makefile中需要修改的地方就多达五处。这对于makefile的维护非常不方便。如果能有一种类似于C语言中宏或者变量的机制,就可以解决这个问题了。为了makefile的易维护,在makefile中我们可以使用变量。makefile的变量也就是一个字符串,理解成C语言中的宏可能会更好。比如,我们声明一个变量,叫TARGETNAME,在makefile一开始就这样定义: TARGETNAME = hello于是,我们就可以很方便地在我们的makefile中以“(TARGETNAME)”的方式来使用这个变量了,于是我们的改良版makefile就变成下面这个样子:TARGETNAME = helloSOURCES = hello.cppTARGETLIBS = (TARGETNAME).obj coredll.lib corelibc.libLINK = linkCPPFLAGS = -nologo -c -I. -IE:WINCE500publiccommonsdkinc -DUNICODE -D_UNICODE -DUNDER_CE=500 -D_WIN32_WCE=500 -DWIN32 -DSTRICT -Dx86 -D_X86_ -DINTERNATIONAL -DL0804 -DINTLMSG_CODEPAGE=1252LFLAGS = -MACHINE:x86 -NODEFAULTLIB -subsystem:windowsce,5.00 -entry:WinMainCRTStartup -LIBPATH:E:WINCE500PBWorkspacesMyPlatformprojrootcesysgensdklibx86retail(TARGETNAME).exe: (TARGETNAME).obj echo linking. (LINK) (LFLAGS) (TARGETLIBS)(TARGETNAME).obj: (SOURCES) echo compiling. (CPP) (CPPFLAGS) (SOURCES)clean: del *.obj, *.exe在这一版本的makefile中,我们在makefile的最开始定义了六个变量。TARGETNAME表示要生成的文件名,SOURCES表示源代码列表。TARGETLIBS表示要链接的库列表。LINK表示链接器的名称。CPPFLAGS和LFLAGS分别表示编译器和链接器的命令行参数。有了这些变量定义之后,编译和链接的命令就可以写的非常简洁了,例如编译源代码的命令就可以写成:(CPP) (CPPFLAGS) (SOURCES)。nmake工具会自动用变量替换这些标记,最终的结果与第一个版本还是一样的。如果读者细心的话,可以发现我们在makefile中并没有定义CPP这个变量,但是在makefile中我们依然使用了CPP变量。这又是为什么呢?其实 nmake工具默认会设置一些变量的值,对于C+编译器,nmake会默认定义CPP变量,并把它的值赋为cl,因此,在使用的时候,makefile 代码中就不需要重复定义了。1.5 使用预处理经过上一节的改动,namefile已经有了很大的灵活性。但是,依然达不到尽善尽美的地步。如果我们要把EXE文件的入口点从WinMainCRTStartup()函数修改到WinMain(),那么依然需要修改makefile。使用预处理可以很好的解决上面的问题。在makefile中,NMake工具允许使用预处理机制来完成如下功能:l 按条件处理makefilel 显示错误信息l 包含其它的makefilel 打开/关闭某些nmake工具的命令行开关预处理指令以“!”开头,必须出现在每行的最开始。最常用的预处理指令是条件处理。Nmake支持如下的条件预处理指令:!IF !IFDEF!IFNDEF!ELSE!ELSEIF!ELSEIFDEF!ELSEIFNDEF!ENDIF它们的用法与C语言的预处理宏类似,相信读者可以很容易的理解它们的含义。使用预处理机制来修改上一个版本的makefile,得到的新makefile如下所示:TARGETNAME = helloSOURCES = hello.cppEXEENTRY = WinMainTARGETLIBS = (TARGETNAME).obj coredll.lib corelibc.libLINK = link!IFDEF EXEENTRY! MESSAGE EXEENTRY: (EXEENTRY)EXEENTRYOPTION=-entry:(EXEENTRY)!ELSEEXEENTRYOPTION=-entry:WinMainCRTStartup!ENDIFCPPFLAGS = -nologo -c -I. -IE:WINCE500publiccommonsdkinc -DUNICODE -D_UNICODE -DUNDER_CE=500 -D_WIN32_WCE=500 -DWIN32 -DSTRICT -Dx86 -D_X86_ -DINTERNATIONAL -DL0804 -DINTLMSG_CODEPAGE=1252LFLAGS = (EXEENTRYOPTION) -MACHINE:x86 -NODEFAULTLIB -subsystem:windowsce,5.00 -LIBPATH:E:WINCE500 PBWorkspacesMyPlatformprojrootcesysgensdklibx86retail(TARGETNAME).exe: (TARGETNAME).obj echo linking. (LINK) (LFLAGS) (TARGETLIBS)(TARGETNAME).obj: (SOURCES) echo compiling. (CPP) (CPPFLAGS) (SOURCES)clean: del *.obj, *.exe在这个版本中,主要的改动是新增加了一个变量EXEENTRY,并且增加了对于这个变量的预处理判断。如果用户定义了EXEENTRY变量,则把变量 EXEENTRYOPTION的值设置成-entry:EXEENTRY,否则就设置成默认的CRT入口函数 -entry:WinMainCRTStartup。修改相应的LFLAGS,把EXEENTRYOPTION加到LFLAGS中,修改就生效了。这样,其实新增加的EXEENTRY是一个可选的变量,如果用户没有定义这个变量的值,构建也不会出错。使用预处理技术对于维护makefile,保持它的向下兼容非常有效。注意,代码中出现的!MESSAGE也是一个宏,用来向标准输出stdout输出一个字符串。1.6 包含其它文件经过上面的修改,makefile中的有些模块已经非常通用了,对于每个项目都建立一个makefile也是比较复杂的。为了增强代码的重用性,可以考虑把makefile代码中通用的部分抽取出来,放在一个独立的文件中。以便在多个项目中公用。预处理的另外一个作用是包含其它makefile文件。语法是:! INCLUDE 使用这个功能,可以实现把makefile拆分的目的。这次,我们把makefile拆分成三个文件,名字分别叫:sources、makefile和makefile.def,都放在hello目录中。三个文件的内容分别如下:sources文件的内容:# This is a demo sources fileTARGETNAME = helloSOURCES = hello.cppEXEENTRY = WinMainTARGETLIBS = coredll.lib corelibc.libmakefile文件的内容:! INCLUDE makefile.defmakefile.inc文件的内容:! INCLUDE .sourcesTARGETLIBS = (TARGETLIBS) (TARGETNAME).objLINK = link!IFDEF EXEENTRY! MESSAGE EXEENTRY: (EXEENTRY)EXEENTRYOPTION=-entry:(EXEENTRY)!ELSEEXEENTRYOPTION=-entry:WinMainCRTStartup!ENDIFCPPFLAGS = -nologo -c -I. -IE:WINCE500publiccommonsdkinc -DUNICODE -D_UNICODE

温馨提示

  • 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
  • 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
  • 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
  • 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
  • 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
  • 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
  • 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

评论

0/150

提交评论