如何在 go 中调用 user32.dll 的 EnumDisplaySettings

How to call user32.dll's EnumDisplaySettingsA in go

我正在尝试通过 Win32 API 获取显示信息。到目前为止,我已经很好地管理了 EnumDisplayDevicesA,但是 EnumDisplaySettingsA 给我带来了麻烦。

无论我如何设置前两个变量,函数 returns 为零(表示失败),没有关于 为什么 它失败的额外信息。

这是我的代码的简化版本,仅包含相关功能;

package main

import (
    "fmt"
    "syscall"
    "unsafe"
)

var (
    dll                 = syscall.MustLoadDLL("user32.dll")
    enumDisplaySettings = dll.MustFindProc("EnumDisplaySettingsA")
)

type devMode struct {
    dmDeviceName       [32]uint16
    dmSpecVersion      uint16
    dmDriverVersion    uint16
    dmSize             uint16
    dmDriverExtra      uint16
    dmFields           uint32
    dmOrientation      int16
    dmPaperSize        int16
    dmPaperLength      int16
    dmPaperWidth       int16
    dmScale            int16
    dmCopies           int16
    dmDefaultSource    int16
    dmPrintQuality     int16
    dmColor            int16
    dmDuplex           int16
    dmYResolution      int16
    dmTTOption         int16
    dmCollate          int16
    dmFormName         [32]uint16
    dmLogPixels        uint16
    dmBitsPerPel       uint32
    dmPelsWidth        uint32
    dmPelsHeight       uint32
    dmDisplayFlags     uint32
    dmDisplayFrequency uint32
    dmICMMethod        uint32
    dmICMIntent        uint32
    dmMediaType        uint32
    dmDitherType       uint32
    dmReserved1        uint32
    dmReserved2        uint32
    dmPanningWidth     uint32
    dmPanningHeight    uint32
}

func queryDisplaySettings() devMode {
    out := devMode{}
    out.dmSize = uint16(unsafe.Sizeof(out))
    outptr := uintptr(unsafe.Pointer(&out))

    namePtr := uintptr(unsafe.Pointer(nil))

    var iModeNum uint32 = 4294967295
    enumCurrentSettings := uintptr(unsafe.Pointer(&iModeNum))

    ret, _, _ := enumDisplaySettings.Call(namePtr, enumCurrentSettings, outptr)
    if ret == 0 {
        fmt.Printf("Failed EnumDisplaySettings")
    }
    return out
}

func main() {
    settings := queryDisplaySettings()

    fmt.Printf("\n%v\n", settings.dmPelsWidth)
    fmt.Printf("%v\n", settings.dmPelsHeight)
    fmt.Printf("%v\n\n", settings.dmDisplayFrequency)
}

我的消息来源:

此处的代码存在多个问题。

首先,您对 devMode 的类型定义是针对 DEVMODEW 的,但您正在调用 EnumDisplaySettingsA。但是,您首先不应该调用它(它是 ANSI 版本),因此请改用 EnumDisplaySettingsW (UNICODE)。

接下来,EnumDisplaySettingsA/EnumDisplaySettingsW 的第二个参数是 DWORD (uint32),但是您传递的不是值,而是地址。

所以替换

var iModeNum uint32 = 4294967295
enumCurrentSettings := uintptr(unsafe.Pointer(&iModeNum))

只有

iModeNum := uintptr(4294967295)

它应该一切正常。