12

更新:添加了RoInitializeroapi.h 中的内容

我正在编写一个纯 C++11 WinRT 库。我不使用 WRL 或 C++/CX(显然,如果我想要纯 C++11)。

我让我的代码在 MSVC 上编译和运行,但我想看看我是否可以让代码在 Mingw Gcc 上编译和运行。具体来说,我使用的是从 nuwen.net 获得的 GCC 4.7.2。

此时我需要的是一种调用 Windows API 函数RoInitialize,和函数, RoUnitialize,的方法。RoGetActivationFactoryHSTRINGWindowsCreateStringWindowsDuplicateStringWindowsDeleteString

我尝试在 G++ 中编译这个程序,但得到了错误

extern "C"{
__declspec(dllimport)int __stdcall RoInitialize(int);
}
int main(){
    RoInitialize(1);
}

我试图编译但得到:

c:\Users\jrb\Desktop>g++ gccwinrt.cpp
C:\Users\jrb\AppData\Local\Temp\ccy7y1V9.o:gccwinrt.cpp:(.text+0x1e): undefined
reference to `_imp__RoInitialize@4'
collect2.exe: error: ld returned 1 exit status

如果有人能指出我如何声明这些函数以及我需要链接到哪些库的正确方向,我将不胜感激。即使它需要LoadLibrary/GetProcAddress我仍然可以接受

更新:这是RoInitialize标题 roapi.h 中的内容

ROAPI
_Check_return_
HRESULT
WINAPI
RoInitialize(
    _In_ RO_INIT_TYPE initType
);

ROAPI is just a define for __declspec(dllimport)
_Check_return_ is part of SAL (Secure Annotations Language?)
HRESULT maps to int32
WINAPI is a define for __stdcall
RO_INIT_TYPE is an enumeration so int should cover it
4

4 回答 4

16

这些函数的导入库是runtimeobject.lib(MSDN 文档没有提到)。它可以在 Windows 8 的 Windows SDK 中找到。

于 2013-05-24T13:32:15.030 回答
6

您需要链接的库是windowsapp.lib(仅此库,删除所有其他库)。

本主题列出了作为通用 Windows 平台 (UWP) 的一部分并由所有 Windows 10 设备实现的 Win32 API。为方便起见,Microsoft Windows 软件开发工具包 (SDK) 中提供了一个名为 WindowsApp.lib 的总括库,该工具包提供了这组 Win32 API 的导出。将您的应用与 WindowsApp.lib(而不是其他库)链接以访问这些 API。

所有 Windows 10 设备上存在的 API - Microsoft Docs

有关实现的功能的完整列表,请参见该页面;它包括RoInitialize, RoUnitialize,RoGetActivationFactoryWindowsCreateString, WindowsDuplicateString, & WindowsDeleteString(还有很多很多)。如果您需要其他功能,您可能还需要链接扩展 API

类似的指导最初是在 https://msdn.microsoft.com/en-gb/windows/uwp/get-started/universal-application-platform-guide#writing-code但不再存在:

Windowsapp.lib 是一个“伞形”库,为 UWP API 提供导出。链接到 Windowsapp.lib 将添加到您的应用对所有 Windows 10 设备系列上存在的 dll 的依赖关系。

对于延迟加载,您将需要加载api-ms-win-core-winrt-l1-1-0.dll,这是一个API 集,是 Windows 可以加载的 API 集合,无论实际 DLL 位于何处。此特定集被列为 Windows 8.1 API 集,但是如果您查看RoInitialize的文档,它会说支持的最低客户端是 Windows 8。假设您使用LoadLibraryand GetProcAddress,这无关紧要。

实现该方法的实际 DLL 是combase.dll,但它们使用这些 API 集 DLL 作为间接级别,以便将来可以自由更改或更新该原始 DLL。

为便于将来参考,Windows 10 (UWP) 的 API 集列在与 Windows 8(和 8.1)的 API 集不同的页面上。存根 DLL(用于延迟加载)是相同的。https://msdn.microsoft.com/library/windows/desktop/mt186421

许多其他 API 也在他们自己的文档中列出了相应的 API 集。例如,MSDN 以AllowSetForeground为例。

于 2016-12-14T11:20:20.613 回答
0

运行时链接它是这样的:

#include <roapi.h>

namespace
{
    FARPROC LoadComBaseFunction(const char* function_name)
    {
        static HMODULE const handle = ::LoadLibraryA("combase.dll");
        return handle ? ::GetProcAddress(handle, function_name) : nullptr;
    }

    decltype(&::RoInitialize) GetRoInitializeFunction()
    {
        static decltype(&::RoInitialize) const function = reinterpret_cast<decltype(&::RoInitialize)>(LoadComBaseFunction("RoInitialize"));
        return function;
    }
}


HRESULT RoInitialize(RO_INIT_TYPE init_type)
{
    auto ro_initialize_func = GetRoInitializeFunction();
    if (!ro_initialize_func)
        return E_FAIL;

    return ro_initialize_func(init_type);
}

资源

于 2019-04-03T08:32:34.403 回答
-1

如果您没有包含 RoInitialize 的导入库,则需要使用 LoadLibrary/GetProcAddress 来解析类型。

于 2013-05-13T04:56:34.260 回答