如何显式调用CRT启动函数

时间:2017-09-20 21:14:14

标签: c++ windows microsoft-runtime-library

有没有什么方法可以在我的代码中明确调用mainCRTStartup函数中通常调用的函数?我特别感兴趣的是 initterm initterm_e 函数。甚至认为我的程序与crt链接(它必须是,对吧?)我不能只调用initterm - 我试图声明并且只是调用而且我得到了未定义的引用。

有什么办法吗?

1 个答案:

答案 0 :(得分:1)

_initterm_initterm_e(msdn _initterm_e中的注释声明错误 - 实际上它使用_PIFV类型)由不同的CRT dll导出。它也在lib文件中声明。并且这个功能的实现非常简单:

typedef void (__cdecl *_PVFV)();

void _initterm(const _PVFV *ppfn, const _PVFV *end)
{
    do 
    {
        if (_PVFV pfn = *++ppfn)
        {
            pfn();
        }
    } while (ppfn < end);
}

typedef int  (__cdecl *_PIFV)();

int _initterm_e(const _PIFV *ppfn, const _PIFV *end)
{
    do 
    {
        if (_PIFV pfn = *++ppfn)
        {
            if (int err = pfn()) return err;
        }
    } while (ppfn < end);

    return 0;
}

所以你可以导入它使用自己的实现。 这里更快的问题 - 什么用作输入参数?

c ++ CL 编译器如何最小化)使用( ".CRT$XCA", ".CRT$XCZ" )部分来放置 c ++ 全局初始化。所以我们可以做下一步:

extern "C"
{
#pragma const_seg(".CRT$XCA")
  const _PVFV __xc_a = 0;
#pragma const_seg(".CRT$XCZ")
  const _PVFV __xc_z = 0;

#pragma const_seg(".CRT$XIA")
  const _PVFV __xi_a = 0;
#pragma const_seg(".CRT$XIZ")
  const _PVFV __xi_z = 0;
}

    /*
     * do initializations
     */
    initret = _initterm_e( &__xi_a, &__xi_z );
    if ( initret != 0 )
        return initret;

    /*
     * do C++ initializations
     */
    _initterm( &__xc_a, &__xc_z );

但这里需要非常好的理解 - 我们做什么以及为什么