日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程语言 > c/c++ >内容正文

c/c++

GCC和MSVC的INITIALIZER的实现

發(fā)布時間:2025/3/19 c/c++ 25 豆豆
生活随笔 收集整理的這篇文章主要介紹了 GCC和MSVC的INITIALIZER的实现 小編覺得挺不錯的,現(xiàn)在分享給大家,幫大家做個參考.

熟悉linux內(nèi)核編程的應該知道內(nèi)核模塊有一個宏叫 module_init,當內(nèi)核模塊被靜態(tài)編譯到內(nèi)核后,會在內(nèi)核init階段調(diào)用每個被module_init聲明過的函數(shù)。這是如何實現(xiàn)的呢?其實是用到了鏈接器的特性。具體可參考

https://blog.csdn.net/lu_embedded/article/details/51432616

大致就是告訴連接器將函數(shù)指針放到一個特定的程序段,然后在需要的時候遍歷這個程序段,拿到每個函數(shù)指針,然后調(diào)用。

那么MSVC有沒有這樣的特性呢,搜索了一番,答案是有的,具體可參考

https://docs.microsoft.com/en-us/cpp/c-runtime-library/crt-initialization?redirectedfrom=MSDN&view=msvc-160

原理基本上是一樣的,只不過是調(diào)用時機是VC runtime。

網(wǎng)上還找到了利用gcc runtime 和 vc runtime實現(xiàn)的跨平臺INITIALIZER

// Initializer/finalizer sample for MSVC and GCC/Clang.// 2010-2016 Joe Lowe. Released into the public domain. #include <stdio.h> #include <stdlib.h>#ifdef __cplusplus#define INITIALIZER(f) \static void f(void); \struct f##_t_ { f##_t_(void) { f(); } }; static f##_t_ f##_; \static void f(void) #elif defined(_MSC_VER)#pragma section(".CRT$XCU",read)#define INITIALIZER2_(f,p) \static void f(void); \__declspec(allocate(".CRT$XCU")) void (*f##_)(void) = f; \__pragma(comment(linker,"/include:" p #f "_")) \static void f(void)#ifdef _WIN64#define INITIALIZER(f) INITIALIZER2_(f,"")#else#define INITIALIZER(f) INITIALIZER2_(f,"_")#endif #else#define INITIALIZER(f) \static void f(void) __attribute__((constructor)); \static void f(void) #endifstatic void finalize(void) {printf( "finalize\n"); }INITIALIZER( initialize) {printf( "initialize\n");atexit( finalize); }int main( int argc, char** argv) {printf( "main\n");return 0; }

另外,我也實現(xiàn)了一個簡單的MSVC版本,如下。

#include <windows.h>//以下內(nèi)容可放到公共頭文件 #pragma section(".MYINIT$A", read) #pragma section(".MYINIT$P", read) #pragma section(".MYINIT$Z", read) typedef void (*PFN_INIT)(void);#define _INIT_FN(fn, pre) \static void fn(void); \__declspec(allocate(".MYINIT$P")) PFN_INIT fn##_ = fn; \__pragma(comment(linker,"/include:" pre #fn "_")) \static void fn(void) #ifdef _WIN64 #define INIT_FN(fn) _INIT_FN(fn, "") #else #define INIT_FN(fn) _INIT_FN(fn, "_") #endif//以下內(nèi)容放到 main 文件 __declspec(allocate(".MYINIT$A")) int __myinit_a = 0; __declspec(allocate(".MYINIT$Z")) int __myinit_z = 0;void CallInitFn(void) {PFN_INIT *start = (PFN_INIT *)&__myinit_a;PFN_INIT *end = (PFN_INIT *)&__myinit_z;while(start < end){if(*start){(*start)();}start++;} }int main(int argc, char *argv[]) {CallInitFn();Sleep(1000); }//以下INIT函數(shù)可分散放到不同的源文件 INIT_FN(init3) {OutputDebugStringA("init3\n"); }INIT_FN(init4) {OutputDebugStringA("init4\n"); }

?

總結

以上是生活随笔為你收集整理的GCC和MSVC的INITIALIZER的实现的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網(wǎng)站內(nèi)容還不錯,歡迎將生活随笔推薦給好友。