代码仓库在此,欢迎进行生物工程:
GCC - GNU Compiler Collection
- MinGW
- GCC 参数详解
- GCC 5 Release Series
- GCC - the GNU Compiler Collection
- mingw-w64 GCC for Windows 64 & 32 bits
- GCC Invocation
GNU 编译器套件 GNU Compiler Collection 包括 C、C++、Objective-C、Fortran、Java、Ada 和 Go 语言的前端,也包括了这些语言的库,如 libstdc++、libgcj 等等。GCC 的初衷是为GNU操作系统专门编写的一款编译器。GNU 系统是彻底的自由软件。此处,自由的含义是它尊重用户的自由。
此外,TCC - Tiny C Compiler 是一个小巧的编译器,用来研究编译原理是不错的目标。
MinGW 就是 GCC 的 Windows 移植版。
MinGW - Minimalist GNU on Windows 是将经典的开源 C/C++ 语言编译器 GCC 移植到了 Windows 平台下,并且包含了 Win32API ,因此可以将源代码编译为可在 Windows 中运行的可执行程序。而且还可以使用一些 Windows 不具备的,Linux 平台下的开发工具。
MinGW 包含 32-bit 和 64-bit 两种,MinGW-w64 可以编译生成 64-bit 或 32-bit 可执行程序,使用 -m32
选项。
正因为如此,MinGW 32-bit 版本现已被 MinGW-w64 所取代,且 MinGW 也早已停止了更新,内置的 GCC 停滞在了 4.8.1 版本,而 MinGW-w64 内置的 GCC 则持续更新。
使用 MinGW-w64 的优势:
- MinGW-w64 是开源软件,可以免费使用。
- MinGW-w64 由一个活跃的开源社区在持续维护,因此不会过时。
- MinGW-w64 支持最新的 C/C++ 语言标准。
- MinGW-w64 使用 Windows 的 C 语言运行库,因此,可以编译出无 DLL 依赖的 Windows 程序。
- 许多开源 IDE 集成 MinGW-w64,如 CodeBlocks,使它拥有友好的图形化界面。
MinGW-w64 是稳定可靠的、持续更新的 C/C++ 编译器,使用它可以免去很多麻烦,不用担心跟不上时代,也不用担心编译器本身有bug,可以放心的去编写程序。
GCC 有多个 Windows 移植版本,比较出名的就是 MinGW 和 TDM-GCC:
- MinGW:http://www.mingw.org/
- TDM-GCC: http://tdm-gcc.tdragon.net/download
- Cygwin:http://www.cygwin.com/
GCC 环境变量:
变量名 | 功能 |
---|---|
CPATH |
搜索目录列表,也可以使用命令选项,如 -I. -I/special/include |
C_INCLUDE_PATH |
搜索目录列表,分隔符号由 PATH_SEPARATOR 变量指定,通常是分号或冒号 |
CPLUS_INCLUDE_PATH |
搜索目录列表 |
OBJC_INCLUDE_PATH |
搜索目录列表 |
DEPENDENCIES_OUTPUT |
非系统依赖的输出,相当命令选项 -MM、-MT 和 -MF 结合 |
SUNPRO_DEPENDENCIES |
类似 DEPENDENCIES_OUTPUT,除了系统头文件不被忽略,相当 -M 选项 |
GCC 命令的常用选项:
选项 | 解释 |
---|---|
-ansi | 只支持 ANSI 标准的 C 语法。这一选项将禁止 GNU C 的某些特色, 例如 asm 或 typeof 关键词。 |
-c | 只编译并生成目标文件。 |
-DMACRO | 以字符串"1"定义 MACRO 宏。 |
-DMACRO | DEFN 以字符串"DEFN"定义 MACRO 宏。 |
-E | 只运行 C 预编译器。 |
-g | 生成调试信息。GNU 调试器可利用该信息。 |
-IDIRECTORY | 指定 DIRECTORY 为额外的头文件搜索路径。 |
-LDIRECTORY | 指定 DIRECTORY 为额外的函数库搜索路径。 |
-lLIBRARY | 连接时搜索指定的函数库LIBRARY。 |
-m486 | 针对 486 进行代码优化。 |
-o | FILE 生成指定的输出文件。用在生成可执行文件时。 |
-O0 | 不进行优化处理。 |
-O | 或 -O1 优化生成代码。 |
-O2 | 进一步优化。 |
-O3 | 比 -O2 更进一步优化,包括 inline 函数。 |
-shared | 生成共享目标文件。通常用在建立共享库时。 |
-static | 禁止使用共享连接。 |
-UMACRO | 取消对 MACRO 宏的定义。 |
-w | 不生成任何警告信息。 |
-Wall | 生成所有警告信息。 |
DLL with MinGW
在 Windows 下用 MinGW 编译 DLL:
/* add_basic.c
Demonstrates creating a DLL with an exported function, the inflexible way.
*/
__declspec(dllexport) int __cdecl Add(int a, int b)
{
return (a + b);
}
只需要添加 -shared
链接选项:
>gcc -c -o add_basic.o add_basic.c
>gcc -o add_basic.dll -s -shared add_basic.o -Wl,--subsystem,windows
以上分步演示了编译和链接两个过程,但是 GCC 可以一步执行:
gcc -o add_basic.dll -s -shared add_basic.c -Wl,--subsystem,windows
其中 -Wl,--subsystem,windows
不是必要的参数,因为不是编译窗口程序。注意 -s
选项,它清理导出的 DLL 符号,通过在发布 DLL 时使用。
对于动态链接库,用户在程序中使用时,为了程序能正确链接,就需要导入库
Import Library,即链接程序中使用的 .lib
文件。
下面,试着写一个程序来调用动态链接库的 Add(a, b) 方法:
/* addtest_basic.c
Demonstrates using the function imported from the DLL, the inelegant way.
*/
#include <stdlib.h>
#include <stdio.h>
/* Declare imported function so that we can actually use it. */
__declspec(dllimport) int __cdecl Add(int a, int b);
int main(int argc, char** argv)
{
printf("%d\n", Add(6, 23));
return EXIT_SUCCESS;
}
现在,执行编译链接:
>gcc -c -o addtest_basic.o addtest_basic.c
>gcc -o addtest_basic.exe -s addtest_basic.o -L. -ladd_basic
>addtest_basic.exe
其它 DLL 使用的高级知识点:
- Displaying functions exported from a DLL.
- The DllMain function.
- Using a module definition file.
- Exporting undecorated stdcall functions.
- Exporting C++ functions and variables from a DLL.
- Creating JNI DLLs.
- P/Invoking MinGW DLLs in .NET
- Setting the DLL base address.
- Loading and unloading DLLs at runtime.
Dll Information
使用 GNU binutils objdump 查看 DLL 导出函数符号:
>objdump -p AddLib.dll
There is an export table in .edata at 0x6da46000
The Export Tables (interpreted .edata section contents)
Export Flags 0
Time/Date stamp 4da9a500
Major/Minor 0/0
Name 00006046 AddLib.dll
Ordinal Base 1
Number in:
Export Address Table 00000003
[Name Pointer/Ordinal] Table 00000003
Table Addresses
Export Address Table 00006028
Name Pointer Table 00006034
Ordinal Table 00006040
Export Address Table -- Ordinal Base 1
[ 0] +base[ 1] 1280 Export RVA
[ 1] +base[ 2] 2004 Export RVA
[ 2] +base[ 3] 2000 Export RVA
[Ordinal/Name Pointer] Table
[ 0] Add
[ 1] bar
[ 2] foo
>dumpbin -exports AddLib.dll
Microsoft (R) COFF/PE Dumper Version 9.00.30729.01
Copyright (C) Microsoft Corporation. All rights reserved.
Dump of file AddLib.dll
File Type: DLL
Section contains the following exports for AddLib.dll
00000000 characteristics
4DA9A500 time date stamp Sat Apr 16 15:17:36 2011
0.00 version
1 ordinal base
3 number of functions
3 number of names
ordinal hint RVA name
1 0 00001280 Add
2 1 00002004 bar
3 2 00002000 foo
Summary
1000 .CRT
1000 .bss
1000 .data
1000 .edata
1000 .eh_fram
1000 .idata
1000 .rdata
1000 .reloc
1000 .rsrc
1000 .text
1000 .tls
The DllMain function.
DllMain 是 DLL 入口函数,在加载或卸载时被系统调用:
#include <windows.h>
BOOL WINAPI DllMain(HINSTANCE hinstDLL, DWORD fdwReason, LPVOID lpvReserved)
{
switch (fdwReason)
{
case DLL_PROCESS_ATTACH:
/* Code path executed when DLL is loaded into a process's address space. */
break;
case DLL_THREAD_ATTACH:
/* Code path executed when a new thread is created within the process. */
break;
case DLL_THREAD_DETACH:
/* Code path executed when a thread within the process has exited *cleanly*. */
break;
case DLL_PROCESS_DETACH:
/* Code path executed when DLL is unloaded from a process's address space. */
break;
}
return TRUE;
}
Using a module definition file.
除了 __declspec(dllexport)
标记一个导出函数,更方便的做法是使用模块定义文件 module definition file,它可以定义 DLL 中导出的变量、函数等等,如下 AddLib.def
:
LIBRARY AddLib.dll
EXPORTS
Add
foo
bar
对应的 C 文件头:
/* Define calling convention in one place, for convenience. */
#define ADDCALL __cdecl
/* Make sure functions are exported with C linkage under C++ compilers. */
#ifdef __cplusplus
extern "C"
{
#endif
/* Declare our Add function using the above definitions. */
int ADDCALL Add(int a, int b);
/* Exported variables. */
extern int foo;
extern int bar;
#ifdef __cplusplus
} // __cplusplus defined.
#endif
头文件中的导出变量、函数依然使用了 extern 关键字,确保它们在使用中能正确链接,函数实现代码如下:
#include "add.h"
int ADDCALL Add(int a, int b)
{
return (a + b);
}
/* Assign value to exported variables. */
int foo = 7;
int bar = 41;
在编译链接命令中使用模块定义文件 AddLib.def
:
>gcc -O3 -std=c99 -Wall -c add.c -o add.o
>gcc -o AddLib.dll add.o AddLib.def -shared -s -Wl,--subsystem,windows,--out-implib,libaddlib.a
Exporting Undecorated stdcall Functions
导出函数意味着 stdcall 调用转换,即 int Add(int, int)
这样的函数签名会导出变成 Add@8
类似格式,@
符号后面跟着的数字表示参数占据的空间,而 Microsoft’s Visual C++ 还会使用其它前缀,如下划线 _Add@8
。正因为 MSVC 和 MinGW 不同编译器之间的转换不一致,当开发出来的 DLL 被多用户使用时,他们使用什么编译器就受到约束了。
解决办法就是避免导出时,编译器对函数的重命名,传递 --kill-at
选项给链接程序,同时,需要重建导入库 import library,否则用户不能正确链接特殊处理过的导出函数。此时,--out-implib
创建的导入库无效,需要使用 dlltool.exe
工具,还有模块定义文件,它包含了函数正确的导出名字:
>gcc -o AddLib.dll add.o -shared -s -Wl,--subsystem,windows,--output-def,AddLib.def
>gcc -o AddLib.dll add.o -shared -s -Wl,--subsystem,windows,--kill-at
>dlltool --kill-at -d AddLib.def -D AddLib.dll -l libaddlib.a
上面的命令首先会创建修饰过函数名称的 DLL,使用了 --output-def,AddLib.def
链接参数生成模块定义文件,它包含了修饰过的函数名称。
第二步还是创建 DLL,但是传入了 --kill-at
链接参数,导出的函数名是未修饰过的,这一步不能创建模块定义文件。
最后,基于模块定义文件创建导入库,如果你关心不同编译器的表现,这一步会很有趣。事实上,Win32 API 函数都是以这种方式导出的,没有任何修饰。
Exporting C++ functions and variables
在 C++ DLL 的导出符号中,不同编译器之间是不通用的,甚至同一个编译器不同版本也不通用。因为 C++ 的复杂性,要处理异常、虚函数实现、或 STL 类型的不同内存模型等等。为了明确不兼容,编译器还会使用名称变形 name mangling
来处理导出符号。
导出全局符号,函数和变量,C/C++ 的做法都是一样的,不同的是 C 语言导出全局变量时,可以作为 C++ 对象实例导出,导出函数时可以重载。还可以导出 C++ 的类对象,这个导出的类对象所有静态方法和成员不区分 public、protected、private 访问修饰。
示例 Point 头文件:
#ifndef POINT_HPP
#define POINT_HPP
#ifdef POINT_EXPORTS
#define POINTAPI __declspec(dllexport)
#else
#define POINTAPI __declspec(dllimport)
#endif
#include <ostream>
using std::ostream;
class POINTAPI Point
{
public:
// Constructors.
Point();
Point(int x, int y);
// Getters and setters.
int getX() const;
int getY() const;
void setX(int x);
void setY(int y);
// Friend the overloaded operators, so they can access private Point data.
friend Point operator+(const Point& lhs, const Point& rhs);
friend ostream& operator<<(ostream& os, const Point& pt);
private:
int x, y;
};
// Overloaded operators.
POINTAPI Point operator+(const Point& lhs, const Point& rhs);
POINTAPI ostream& operator<<(ostream& os, const Point& pt);
extern POINTAPI Point foo;
extern POINTAPI Point bar;
#endif
实现代码:
#include "point.hpp"
Point::Point()
: x(0), y(0)
{ }
Point::Point(int x, int y)
: x(x), y(y)
{ }
int Point::getX() const { return this->x; }
int Point::getY() const { return this->y; }
void Point::setX(int x) { this->x = x; }
void Point::setY(int y) { this->y = y; }
Point operator+(const Point& lhs, const Point& rhs)
{
return Point(lhs.x + rhs.x, lhs.y + rhs.y);
}
ostream& operator<<(ostream& os, const Point& pt)
{
return os << "(" << pt.x << ", " << pt.y << ")";
}
Point foo(9, 6);
Point bar(3, 19);
编译生成 C++ 代码的 DLL,和 C 语言的 DLL 没有区别:
>g++ -c -o point.o point.cpp -D POINT_EXPORTS
>g++ -o point.dll point.o -s -shared -Wl,--subsystem,windows,--out-implib,libpoint.a
或者生成静态库,链接成无动态链接依赖的程序:
>gcc -c src\*.cpp -Iinclude
>ar rcs libpoint.a *.o
>gcc pointTest.cpp -I include/ -L lib/ -l point -o testPoint.exe
打包归档命令 ar 将所有 .o 文件打包为静态库,r 将文件插入静态库中,c 创建静态库,不管库是否存在,s 写入一个目标文件索引到库中,或者更新一个存在的目标文件索引。
这时创建了导入库 libpoint.a,这是可选的,因为除了链接程序,还有其它方法调用 DLL 中的 API。
使用 objdump -p
命令查看导出符号,可以我发现类似 _ZN5Point4setXEi
、_ZlsRSoRK5Point
这样的符号。使用 c++filt 这个 Demangle 工具可以将导出的 C++ 符号还原:
>c++filt -n _ZlsRSoRK5Point
operator<<(std::basic_ostream<char, std::char_traits<char> >&, Point const&)
>c++filt -n _ZN5Point4setXEi
Point::setX(int)
>c++filt -n _ZN5Point4setYEi
Point::setY(int)
>c++filt -n _ZN5PointC1Eii
Point::Point(int, int)
>c++filt -n _ZN5PointC1Ev
Point::Point()
>c++filt -n _ZN5PointC2Eii
Point::Point(int, int)
>c++filt -n _ZN5PointC2Ev
Point::Point()
>c++filt -n _ZNK5Point4getXEv
Point::getX() const
>c++filt -n _ZNK5Point4getYEv
Point::getY() const
>c++filt -n _ZlsRSoRK5Point
operator<<(std::basic_ostream<char, std::char_traits<char> >&, Point const&)
>c++filt -n _ZplRK5PointS1_
operator+(Point const&, Point const&)
创建 DLL 后,就可以写测试程序:
#include <iostream>
#include "point.hpp"
using namespace std;
int main(int argc, char** argv)
{
Point a;
Point b(2, 7);
Point c;
c.setX(85);
c.setY(24);
cout << "a = " << a << endl;
cout << "b = " << b << endl;
cout << "c = (" << c.getX() << ", " << c.getY() << ")\n";
cout << "foo + bar = " << foo << " + " << bar << " = " << (foo + bar) << endl;
return 0;
}
编译链接测试程序 testPoint.cpp:
>g++ -c -o pointtest.o pointtest.cpp
>g++ -o pointtest.exe -s pointtest.o -L. -lpoint
>pointtest.exe a = (0, 0)
b = (2, 7)
c = (85, 24)
foo + bar = (9, 6) + (3, 19) = (12, 25)
发布 DLL 时,需要注意避免破坏其它程序的正常运行,应该给 DLL 附加一个版本号后缀以区别,如下:
point-mingw-4.5.2.dll
point-msvc-2010.dll
这个工程目录结构:
─ ${application}
├── example
│ ├── CMakeLists.txt
│ └── testPoint.cpp
├── include
│ └── point.hpp
├── src
│ └── point.cpp
├── CMakeLists.txt
└── DllDemo.sublime-project
为了使用 CMake 自动化编译,在工程顶级目录设置 CMakeLists.txt:
cmake_minimum_required(VERSION 2.8)
project( dllDemo )
# set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -m64 -g -Wall -O2")
# set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -m64 -g -Wall -O2 -std=c++11")
set(CMAKE_CXX_FLAGS "-w" )
set(CMAKE_CXX_STANDARD 11)
set(CMAKE_CXX_STANDARD_REQUIRED True)
set(EXECUTABLE_OUTPUT_PATH ${PROJECT_SOURCE_DIR}/bin)
set(LIBRARY_OUTPUT_PATH ${PROJECT_SOURCE_DIR}/lib)
if (POLICY CMP0054)
cmake_policy(SET CMP0015 NEW)
endif()
include_directories( "./include/" )
# Static Libs
# set(CMAKE_EXE_LINKER_FLAGS "-static-libgcc -static-libstdc++ -static")
set(BUILD_SHARED_LIBS ON)
set(ENV{PATH} C:/CodeBlocks/MinGW/bin)
message($ENV{PATH})
execute_process(COMMAND where g++ )
execute_process(COMMAND where make )
aux_source_directory("src/" src)
add_library( point ${src} )
# message( ${CMAKE_INSTALL_LIBDIR} )
install(TARGETS point DESTINATION "/lib")
install(TARGETS point DESTINATION "${PROJECT_SOURCE_DIR}/bin")
add_subdirectory(example bin EXCLUDE_FROM_ALL)
# add_subdirectory(example bin)
在 example 子目录下设置 CMakeLists.txt:
cmake_minimum_required(VERSION 2.8)
include_directories("${PROJECT_SOURCE_DIR}/include/")
link_directories(
"${PROJECT_BINARY_DIR}/lib/"
"${PROJECT_SOURCE_DIR}/lib/"
)
# set(CMAKE_CXX_FLAGS "-Wl,-Bstatic" )
# set(CMAKE_FIND_LIBRARY_SUFFIXES ".a")
# link_libraries(point)
add_executable(PointTest pointTest.cpp)
# dynamic link
add_executable(PointTest pointTest.cpp)
target_link_libraries( PointTest point )
# static linke
# set_property(TARGET point PROPERTY IMPORTED_LOCATION libpoint.a)
# add_executable(PointTest pointTest.cpp)
# target_link_libraries( PointTest libpoint.a )
作为小巧、功能强大的 SublimeText,用它来编写 C++ 工程是组好的选择,工程文件配置如下,Ctrl-Shift-B 调用设置好的命令,先执行 CMake 生成 MinGW Makefiles 编译脚本,再执行 Make 或 Make install 生成动态链接库,然后生成 PointTest 程序:
{
"build_systems":
[
{
"env":{
"PATH":"c:/CodeBlocks/MinGW/bin/;%PATH%"
},
"encoding": "utf8",
"file_regex": "^ (.+)\\((\\d+)\\)(): ((?:fatal )?(?:error|warning) \\w+\\d\\d\\d\\d: .*) \\[.*$",
"name": "MinGW Build (Windows)",
"quiet": true,
"shell_cmd": "cmake --build .",
"variants":
[
{
"name": "Make help",
"shell_cmd": "make help"
}, {
"name": "MinGW Makefiles",
"shell_cmd": "cmake .. -G \"MinGW Makefiles\""
}, {
"name": "clean",
"shell_cmd": "make clean"
}, {
"name": "clear all",
"shell_cmd": "del * /s /q"
}, {
"name": "Make",
"shell_cmd": "make -j 4 all"
}, {
"name": "Make install",
"shell_cmd": "make install"
}, {
"name": "Make PointTest",
"shell_cmd": "make PointTest"
}
],
"working_dir": "${project_path}/build"
}
],
"folders":
[
{
"path": ".",
"binary_file_patterns":["*.noting"],
"name": "Dll Demo Project",
}
],
"settings":
{
"cmake":
{
"build_folder": "${project_path}/build"
}
}
}
Creating JNI DLLs
MinGW 创建的 DLL 可以和 Java Native Interface 一起使用,JNI 调用 Win32 函数使用 stdcall 调用约定,这种调用表示函数参数入栈顺序从右到左。
因为不同的语言想到交互时,需要有一致的函数调用和返回行为,C 语言作为一种历史悠久的编程语言,它的函数调用方式称为标准调用 stdcall,其它常见方式如下:
调用约定 | 清理堆栈 | 说明 |
---|---|---|
cdecl |
主调函数 | 参数从右到左 push 入栈 |
stdcall |
被调函数 | 同 cdecl
|
fastcall |
被调函数 | 参数从右到左 push 入栈,但优先使用寄存器传递,如 EAX、ECX、EDX |
thiscall |
被调函数 | 参数从右到左 push 入栈,this 指针通过 ECX 传递 |
declspec |
被调函数 | 用于 DLL 导出函数,如 __declspec(dllexport)
|
JVM 希望调用的 DLL 函数名是未修饰的,或者按 _[function name]@[size of arguments]
这样的格式修饰。错误的调用类似以下结果:
>java Hello
Exception in thread "main" java.lang.UnsatisfiedLinkError: Hello.add(II)I
at Hello.add(Native Method)
at Hello.main(Hello.java:5)
正确导出 JNI 调用的函数需要使用 --kill-at
编译选项,Java 示例如下:
public class Hello
{
public static void main(String[] args)
{
System.out.println("8 + 5 = " + Hello.add(8, 5));
}
static
{
System.loadLibrary("Hello");
}
public static native int add(int a, int b);
}
使用 System.loadLibrary()
加载 DLL,然后使用 Java 提供的命令编译并生成 C/C++ 头文件:
>javac Hello.java
>javah Hello
第二个命令生成 C/C++ 头文件类似如下:
/* DO NOT EDIT THIS FILE - it is machine generated */
#include <jni.h>
/* Header for class Hello */
#ifndef _Included_Hello
#define _Included_Hello
#ifdef __cplusplus
extern "C" {
#endif
/*
* Class: Hello
* Method: add
* Signature: (II)I
*/
JNIEXPORT jint JNICALL Java_Hello_add
(JNIEnv *, jclass, jint, jint);
#ifdef __cplusplus
}
#endif
#endif
接下来创建 C 代码文件,实现函数:
#include "Hello.h"
jint JNICALL Java_Hello_add(JNIEnv* env, jclass clazz, jint a, jint b)
{
return (a + b);
}
编译并运行测试它,--kill-at
别忘了,还有 JDK 版本和 MinGW 要统一为 32-bit 或 64-bit 版本:
>gcc -c -o Hello.o Hello.c -I "c:\Java\jdk\include\win32" -I "c:\Java\jdk\include"
>gcc -o Hello.dll -s -shared Hello.o -Wl,--subsystem,windows,--kill-at
>java Hello
8 + 5 = 13
在 CMake 编写脚本时,发现并不能正确使用 --kill-at,必须在 target_link_options 命令中使用 LINKER:
才能正确将参数传入链接程序:
target_link_options( hello PUBLIC --kill-at)
target_link_options( hello PUBLIC LINKER:--kill-at)
最后,注意,32-bit JVM 只能加载 32-bit DLL,64-bit JVM 也只能加载 64-bit DLLs,否则异常:
Can't load IA 32-bit .dll on a AMD 64-bit platform
P/Invoking MinGW DLLs in .NET
MinGW 编译的 DLL 与 .NET 一起使用要比 JNI 简单,因为不必按 JNI 规定格式进行设置。
C# 提供 P/Invoke 即 Platform Invoke 平台调用,调用非托管 DLL 中的函数,和关键字 DllImport 一起使用。 实际上,NET 基类库中定义的类型内部调用 Kernel32.dll、User32.dll、gdi32.dll 等非托管 DLL 中导出的函数。
使用 DllImport 将 DLL 导出的 stdcall 函数声明为 extern
即可:
using System;
using System.Runtime.InteropServices;
public class Hello
{
public static void Main(string[] args)
{
Console.WriteLine("8 + 5 = {0}", Hello.Add(8, 5));
}
[DllImport("AddLib.dll", CallingConvention = CallingConvention.Cdecl)]
extern public static int Add(int a, int b);
}
还可以指定调用约定方式,这就是语言更新换代带来的好处:
CallingConvention = CallingConvention.StdCall
.NET CLR 会尝试导入没有修饰的函数名,可以指定 MSVC 函数名修饰方式,如 _Add@8
:
ExactSpelling = true
当然,完全可以显式指定导入的函数名:
using System;
using System.Runtime.InteropServices;
public class Hello
{
public static void Main(string[] args)
{
Console.WriteLine("8 + 5 = {0}", Hello.Add(8, 5));
}
[DllImport("AddLib.dll", CallingConvention = CallingConvention.StdCall, EntryPoint = "Add@8", ExactSpelling = true)]
extern public static int Add(int a, int b);
}
注意,程序和 DLL 文件要 32-bit 或 64-bit 对应,否则异常:
>Hello.exe
未经处理的异常: System.BadImageFormatException: 试图加载格式不正确的程序。 (异常来自 HRESULT:0x8007000B)
在 Hello.Add(Int32 a, Int32 b)
在 Hello.Main(String[] args)
作为新式语言,C# 的编译器提供了平台链接选项:
>csc /platform:x86 /out:Hello.exe Hello.cs
Microsoft (R) Visual C# 2005 Compiler version 8.00.50727.4927
for Microsoft (R) Windows (R) 2005 Framework version 2.0.50727
Copyright (C) Microsoft Corporation 2001-2005. All rights reserved.
在 Visual Studio 中设置平台目标,在工程属性的 build 选项卡,这样就可以在 64-bit 系统编译 32 bit 目标程序,同样,可以指定 platform 为 x64。
Using MinGW DLLs with VB6 and VBA
MinGW 编译的 DLL 可以和 Visual Basic 6 或 VBA 一起使用,只要调用约定为 stdcall 方式,不支持 cdecl 或其它调用约定,并且使用 --kill-at
编译选项:
>gcc -o AddLib.dll add.o -shared -s -Wl,--subsystem,windows,--kill-at
然后,在 VB 代码中声明:
Private Declare Function MyAddFunction Lib "AddLib.dll" Alias "Add" (ByVal a As Long, ByVal b As Long) As Long
Sub Test()
Call MsgBox(MyAddFunction(4, 5))
End Sub
注意,VB 关键字 Alias
导出了 DLL 中的函数,并起了个别名。Visual Basic 只支持 ANSI 而不支持 Unicode。
如果在 VBA 中,还需要标记 PtrSafe,以确保可以在 64 bit 的 Microsoft Office 上运行,为了向后兼容 Office 2010,可以进行条件判断:
#If VBA7 Then
Private Declare PtrSafe Function MyAddFunction Lib "AddLib.dll" Alias "Add" (ByVal a As Long, ByVal b As Long) As Long
#Else
Private Declare Function MyAddFunction Lib "AddLib.dll" Alias "Add" (ByVal a As Long, ByVal b As Long) As Long
#End If
Sub Test()
Call MsgBox(MyAddFunction(4, 5))
End Sub
这些代码使用起来相当不舒服,VB 除了在 Office 中用得较多,几乎没什么用户了。
Setting the DLL base address
DLL 的基址 base address 是 Windows 系统加载 DLL 的默认地址,进程的内存空间是一个虚拟空间
virtual address space。程序中使用的 DLL 很多,当任意 DLL 的地址出现覆盖时,就不可能按 DLL 的基址去加载,而需要重定位 relocated 加载到不同的地址。这涉及到加载器的硬编码补丁操作,比较消耗资源。
默认 MinGW 链接程序基于 DLL 名字的哈希分散选择基址,这一般不会有什么问题。也可以通过 --image-base
链接参数设置基础:
>gcc -o AddLib.dll obj/add.o -shared -s ^
-Wl,--subsystem,windows,--out-implib,libaddlib.a,--image-base,0x10000000
然后再用 objdump 查看 ImageBase:
>objdump -p AddLib.dll
AddLib.dll: file format pei-i386
Characteristics 0x230e
executable
line numbers stripped
symbols stripped
32 bit words
debugging information removed
DLL
Time/Date Tue Apr 19 16:32:45 2011
Magic 010b (PE32)
MajorLinkerVersion 2
MinorLinkerVersion 21
SizeOfCode 00000c00
SizeOfInitializedData 00002200
SizeOfUninitializedData 00000200
AddressOfEntryPoint 000010c0
BaseOfCode 00001000
BaseOfData 00002000
ImageBase 10000000
SectionAlignment 00001000
FileAlignment 00000200
MajorOSystemVersion 4
MinorOSystemVersion 0
MajorImageVersion 1
MinorImageVersion 0
MajorSubsystemVersion 4
MinorSubsystemVersion 0
Win32Version 00000000
SizeOfImage 0000c000
SizeOfHeaders 00000400
CheckSum 0000383c
Subsystem 00000002 (Windows GUI)
DllCharacteristics 00000000
SizeOfStackReserve 00200000
SizeOfStackCommit 00001000
SizeOfHeapReserve 00100000
SizeOfHeapCommit 00001000
LoaderFlags 00000000
NumberOfRvaAndSizes 00000010
Loading and unloading DLLs at runtime
运行时加载 DLL 对于插件开发是非常有用的。
这里演示 void __cdecl DoPlugin();
导出函数,模拟插件的运行机制,程序中只需要调用 DoPlugin 就可以让插件运行起来。
需要用到 kernel32.dll 中的 Windows API LoadLibrary
,调用此函数将 DLL 加载到进程的地址空间中。Windows 系统自动对 DLL 的加载进行计数。加载成功计数增加一,返回一个模块句柄 HMODULE 也即是 DLL 加载到的内存地址信息。然后,通过 GetProcAddress
函数获取 DLL 导出函数的地址,继续使用 AddLib.dll
演示如何在运行时调用 Add
导出函数。
#include <windows.h>
#include <stdio.h>
#include <stdlib.h>
/* Function signature of the function exported from the DLL. */
typedef int (__cdecl *AddFunc)(int a, int b);
int main(int argc, char** argv)
{
HMODULE hAddLib;
AddFunc Add;
/* Attempt to load the DLL into the process's address space. */
if (! (hAddLib = LoadLibrary(TEXT("AddLib.dll"))))
{
fprintf(stderr, "Error loading \"AddLib.dll\".\n");
return EXIT_FAILURE;
}
/* Print the address that the DLL was loaded at. */
printf("Library is loaded at address %p.\n", hAddLib);
/* Attempt to get the memory address of the "Add()" function. */
if (! (Add = (AddFunc) GetProcAddress(hAddLib, "Add")))
{
fprintf(stderr, "Error locating \"Add\" function.\n");
return EXIT_FAILURE;
}
/* Print the address of the "Add()" function. */
printf("Add function is located at address %p.\n", Add);
/* Call the function and display the results. */
printf("7 + 41 = %d\n", Add(7, 41));
/* Unload the DLL. */
FreeLibrary(hAddLib);
return EXIT_SUCCESS;
}
程序有几点注意:
- LoadLibrary 和 GetProcAddress 返回 NULL 表示失败,应该进行检查。
- LoadLibrary 有 ANSI 和 Unicode 两个版本,GetProcAddress 总是使用 ANSI 字符串。
- 使用 C 语言类型定义的函数指针类型要和 DLL 导出函数完全匹配,否则会让程序崩溃。
- 最后,FreeLibrary 函数应该在不需要使用 DLL 时用来释放它,计数会减一,让系统知道何时回收内存。
运行程序测试:
>DynamicLoad.exe
Library is loaded at address 6DA40000.
Add function is located at address 6DA41280.
7 + 41 = 48