发布时间:2010-2-12 11:44
分类名称:Private
从网上找了一篇文章: http://www.mvps.org/user32/nocrt.html,
还可以从一些地方参考 http://wiki.wxwidgets.org/Reducing_Executable_Size
http://www.hailstorm.net/papers/smallwin32.htm#buffer
Introduction
This document describes some methods on how to minimize the size of images - be they dll's or exe's. The methods include elimination of the c-runtime stubs, and compiler & linker settings. The compiler and linker I concentrate on is MSVC 6 - most of the tips apply also to MSVC 5. While many of the concepts presented here apply to other development enviroments the actual command line parameters and #pragmas will obviously be different - check your enviroments documentation.
Doing without the C-Runtime
The c-runtime is a library of functions that do stuff for you the programmer. These functions are platform independent, and act as an abstraction layer between your program and the operating system. This has some drawbacks:
Bugs. While most c-runtimes are well tested, each layer you add to an
application introduces the possibility of more bugs It takes up space.
You application must either include the co
Stop using c-runtime functions. Some functions you can use (the string,
and memory manipulation functions) as they are available in intrinsic
form, or the OS API provides direct
equilavents. More work might be required to replace other runtime calls,
especially where the OS
does not provide an equilavent service. Implement a couple of runtime
functions that the C++ compiler assumes exists. new, delete and
_purecall may be necessary for c++ modules. You will need to provide the
application with an entrypoint. Some compiler switches may need to be
changed to make the generated object files link properly in a no-crt
enviroment. Change the linker settings to prevent the linker from
including the runtime libraries NB: Note that the c-runtime startup
co
Functions you can use
The following functions are available in intrinsic form by the compiler. Beware though that the instrinsic forms of these may not be as optimised as the library form.
memcmp memcpy memset strcmp strcpy strlen strcat strset
Required Functions The c++ compiler absolutely requires
that you implement __purecall,new and delete. If C++ exception ahndling
is enabled more are required that I have no clue how to write. Either
don't use C++ exceptions, or find the .obj files that implement these
functions and link them into your project.
A simple implementation of these functions:
void* __cdecl operator new(unsigned int cb) {
return HeapAlloc(GetProcessHeap(),0,cb);
}
void __cdecl operator delete(void* pv) {
HeapFree(GetProcessHead(),0,pv);
}
extern “C” int _cdecl _purecall(void) {
return 0;
}
In addition to the C++ functions listed above you will need to provide your application with a new entry point. Typically an application starts at a functions called main, WinMain, or DllMain. These functions are actually called from the c-runtimes real entry point. Here are the prototypes and real names of an applications entry point:
EXTERN_C int WINAPI mainCRTStartup();
EXTERN_C int WINAPI WinMainCRTStartup();
EXTERN_C BOOL WINAPI _DllMainCRTStartup(
HINSTANCE hInstDll, // handle to the DLL module
DWORD fdwReason, // reason for calling function
LPVOID lpvReserved, // reserved
);
Application Termination
Normally, when the co
Example Implementation
As a debugging aid it is usefull to implement the entrypoint function
in a manner similar to the c-runtimes.
EXTERN_C int WINAPI WinMainCRTStartup() {
HINSTANCE hInstance = GetModuleHandle(NULL);
LPSTR lpszCmdLine = GetCommandLine();
int r = WinMain(hInstance,NULL,lpszCmdLine,SW_SHOWDEFAULT);
ExitProcess(r);
return r; // this will never be reached.
}
Compiler Switches
The following table describes the compiler switches that should be set
to ensure successful compilation using MSVC++ 6
Switch Act
delete /GX
This switch enables C++ exception handling which requires a number of
functions related to unwinding the stack.
delete /GZ
This switch enables some advanced debugging features. With these
features enabled the linker will look for a function called _chkstk.
add /Oi
Add this switch to ensure that intrinsic functions are enabled.
add /Zl
Usually the compiler embeds a “defaultlib” refrence to the c-runtime in
the .obj file. This switch (do not confuse with the /ZI switch) ensures
that defaultlib entries are not written to the generated OBJs.
Linker Switches
The following switches are optional if the compiler is set correctly.
However, if just on
Switch Act
add /nodefaultlib
This switch is not really necessary if you compile with /Zl, as there
should be no default libraries to ignore. If however you use a 3rd party
library, or any old obj files that do include a defaultlib entry, then
the linker will silently ignore your custom entry point, unless you use
the following switch
add /entry:function
Use this if you wish to use a non-standard name for your entry point.
This is a good idea if you are linking to a 3rd party library or object
co
add /opt:nowin98 The MSVC 6 linker defaults to a 4Kb
section padding in PE files as an optimizaton to speed load times on
Windows 98. Very small projects will benefit with a file size saving of
about 16Kb.
More MSVC 6 Linker settings
Microsofts linkers prior to 6.00 produced PE iamges with file level
section alignments of 512 bytes. Link 6.00 aligns sections in a file on
4Kb boundries to optimise loading of the image under 98. For
compatability reasons 98 must load files with the old alignment (but
will do so with a performance hit), and, if you are targetting NT you
can use the old 512 byte padding without any performance deficit at all.
The linker switch you must add to the link line is: (the second line
details how the command might be embedded as a linker option in a .C
file.
// linker options can be embedded directly in .cpp co
#if defined(_MSC_VER) && _MSC_VER >= 1200
#pragma comment(linker, ”/OPT:NOWIN98” )
#endif