4

根据标题,我如何检查用户安装了哪个版本的 DirectX?检查 FeatureLevel 是不够的,因为我的应用程序可以在功能级别 10.0 上运行,但需要安装 DirectX 11.1。

为什么这不是重复的:

  • 如何在 C# 中编写代码以在我的机器上获取直接 X 版本?
    • 这个问题的第一个答案是“如果 Windows 7,DirectX = 11,如果 Windows Vista,DirectX = 10”。这是错误的,因为 Vista 同时支持 DirectX 10 和 11,而 Windows 7 支持 DirectX 11 和 11.1。
    • 第二个答案引用了仅适用于 DirectX 9 及更低版本的注册表项。即使在安装了 DirectX 11.1 的 Windows 7 系统上,此注册表项也永远不会指示安装的版本高于 9.0c
  • .NET 如何检测是否支持 DirectX 10?
    • 这个问题的答案再次引用了 DirectX 9 和更低版本的相同注册表项。

我需要一个适用于 DirectX 10 安装及更高版本的答案。这意味着确定它们的版本是 10、10.1、11 还是 11.1。

4

2 回答 2

3

编辑:删除注册表检查方法,因为它仅适用于 Dx <=9 (thx @Telanor)

这种方法非常非常慢,但我发现只有一种方法是 100% 准确的

private static int checkdxversion_dxdiag()
{
    Process.Start("dxdiag", "/x dxv.xml");
    while (!File.Exists("dxv.xml"))
        Thread.Sleep(1000);
    XmlDocument doc = new XmlDocument();
    doc.Load("dxv.xml");
    XmlNode dxd = doc.SelectSingleNode("//DxDiag");
    XmlNode dxv = dxd.SelectSingleNode("//DirectXVersion");

    return Convert.ToInt32(dxv.InnerText.Split(' ')[1]);
}
于 2013-06-16T09:28:36.983 回答
3

另一种可能性是IDxDiagProvider直接使用 COM 对象并浏览IDxDiagContainer它产生的层次结构 - 这就是dxdiag.exe内部所做的。它也需要一点时间才能完成,因此它也不是一个快速的解决方案,但至少您不需要创建或解析原始文件。

显然,这个功能以前被封装在托管的 DirectX 程序集中Microsoft.DirectX.Diagnostics,因为它们与 COM 对象提供的接口非常相似,但是这些程序集已经过时并且在 .NET Core 中不起作用,所以让我们自己封装这些 COM 对象!有关方法的文档,您仍然可以参考上面链接的文档。

首先,您需要将IDxDiagProvider接口和DxDiagProvidercoclass 连同DXDIAG_INIT_PARAMS传递给提供者:

[ComImport]
[Guid("A65B8071-3BFE-4213-9A5B-491DA4461CA7")]
public class DxDiagProvider { }

[Guid("9C6B4CB0-23F8-49CC-A3ED-45A55000A6D2")]
[InterfaceType(ComInterfaceType.InterfaceIsIUnknown)]
public interface IDxDiagProvider
{
    void Initialize(ref DXDIAG_INIT_PARAMS pParams);
    void GetRootContainer(out IDxDiagContainer ppInstance);
}

[StructLayout(LayoutKind.Sequential)]
public struct DXDIAG_INIT_PARAMS
{
    public int dwSize;
    public uint dwDxDiagHeaderVersion;
    public bool bAllowWHQLChecks;
    public IntPtr pReserved;
};

您还需要包装IDxDiagContainer类:

[Guid("7D0F462F-4064-4862-BC7F-933E5058C10F")]
[InterfaceType(ComInterfaceType.InterfaceIsIUnknown)]
public interface IDxDiagContainer
{
    void EnumChildContainerNames(uint dwIndex, string pwszContainer, uint cchContainer);
    void EnumPropNames(uint dwIndex, string pwszPropName, uint cchPropName);
    void GetChildContainer(string pwszContainer, out IDxDiagContainer ppInstance);
    void GetNumberOfChildContainers(out uint pdwCount);
    void GetNumberOfProps(out uint pdwCount);
    void GetProp(string pwszPropName, out object pvarProp);
}

现在我们开始使用我们的包装器并且必须执行以下操作来检索版本信息:

  • 通过创建 coclass 并将其转换为接口来实例化提供程序。
  • 使用初始化参数初始化提供程序。
  • 获取根容器。
  • 获取DxDiag_SystemInfo子容器。
  • 阅读 DirectX 版本属性。

正确清理 COM 资源的代码如下所示:

IDxDiagProvider provider = null;
IDxDiagContainer rootContainer = null;
IDxDiagContainer systemInfoContainer = null;
try
{
    // Instantiate and initialize the provider.
    provider = (IDxDiagProvider)new DxDiagProvider();
    DXDIAG_INIT_PARAMS initParams = new DXDIAG_INIT_PARAMS
    {
        dwSize = Marshal.SizeOf<DXDIAG_INIT_PARAMS>(),
        dwDxDiagHeaderVersion = 111
    };
    provider.Initialize(ref initParams);

    // Get the Root\SystemInfo container.
    provider.GetRootContainer(out rootContainer);
    rootContainer.GetChildContainer("DxDiag_SystemInfo", out systemInfoContainer);

    // Read the DirectX version info.
    int versionMajor = GetProperty<int>(container, "dwDirectXVersionMajor");
    int versionMinor = GetProperty<int>(container, "dwDirectXVersionMinor");
    string versionLetter = GetProperty<string>(container, "szDirectXVersionLetter");
    bool isDebug = GetProperty<bool>(container, "bDebug");
}
finally
{
    if (provider != null)
        Marshal.ReleaseComObject(provider);
    if (rootContainer != null)
        Marshal.ReleaseComObject(rootContainer);
    if (systemInfoContainer != null)
        Marshal.ReleaseComObject(systemInfoContainer);
}

如您所见,我创建了一个小实用程序方法来从COM 接口返回GetProperty的值中检索正确类型的属性:VARIANT

private static T GetProperty<T>(IDxDiagContainer container, string propName)
{
    container.GetProp(propName, out object variant);
    return (T)Convert.ChangeType(variant, typeof(T));
}
于 2019-02-22T22:06:09.120 回答