【问题标题】:How do I pass a class object to a ctypes.windll.LoadLibrary(DLL) function?如何将类对象传递给 ctypes.windll.LoadLibrary(DLL) 函数?
【发布时间】:2012-10-10 22:49:33
【问题描述】:

我正在使用 Python 并尝试使用 WinUSB(来自 Microsoft)与这个简单的 USB 设备通信。我遇到了一个问题,因为为了找到设备,调用 setupAPI 函数 SetupDiEnumDeviceInterfaces 需要一个结构对象,我将其定义为一个类:

class _SP_DEVINFO_DATA:
    def __init__(self, ClassGUID, DevInst = ""):        
        '''flags = SPINT_DEFAULT, SPINT_REMOVED, or SPINT_ACTIVE'''
        self._ClassGUID = None
        self._DevInst = None
        self._Reserved = None
        self._cbSize = None

    ###Getters:
    def getClassGUID(self):
        return self._ClassGUID
    def getDevInst(self):
        return self._DevInst
    def getReserved(self):
        return self._Reserved
    def getcbSize(self):
        return self._cbSize

    ###Setters:
    def setClassGUID(self, value):
        self._ClassGUID = value
    def setDevInst(self, value):
        self._DevInst = value
    def setReserved(self, value):
        self._Reserved = value
    def setcbSize(self):
        self._cbSize = sys.getsizeinfo(self)       

    ClassGUID = property(getClassGUID, setClassGUID, None, "Class GUID")
    DevInst = property(getDevInst, setDevInst, None, "Device Instance")
    Reserved = property(getReserved, setReserved, None, "RESERVED: DO NOT USE")
    cbSize = property(getcbSize, setcbSize, None, "CB Size. Set automatically")

我尝试使用该属性,因为它给了我错误:

<type 'exceptions.TypeError'>: Don't know how to convert parameter

否则,我读到像这样定义参数可以解决问题,但事实并非如此,而且我不确定在这里做什么。

我想使用 WinUSB,因为我只需要从设备读取并写入设备,仅此而已,而且 WinUSB 似乎有我需要的东西,但在我解决这个问题之前,我有点卡住了

有什么建议吗?如何将类对象传递给使用 ctypes.windll.LoadLibrary(DLL) 加载的 DLL 函数?

如果有更简单的方法可以做到这一点,我也完全赞成。

谢谢。

【问题讨论】:

    标签: python class dll ctypes windows-7-x64


    【解决方案1】:

    正如@Roland 所说,您必须从ctypes.Structure 派生。这是一个工作版本:

    import ctypes
    from ctypes import wintypes
    import uuid
    
    class _SP_DEVINFO_DATA(ctypes.Structure):
        _fields_ = [("cbSize", wintypes.DWORD),
                    ("ClassGuid", ctypes.c_char * 16),
                    ("DevInst", wintypes.DWORD),
                    ("Reserved", wintypes.LPVOID)]
    
        def __init__(self, guid, inst):
            self.cbSize = ctypes.sizeof(_SP_DEVINFO_DATA)
            self.ClassGuid = uuid.UUID(guid).get_bytes()
            self.DevInst = (ctypes.c_ulong)(inst)
            self.Reserved = None
    
        def __repr__(self):
            return "_SP_DEV_INFO_DATA(cbsize={},ClassGuid={},DevInst={})".format(
                self.cbSize,uuid.UUID(bytes=self.ClassGuid),hex(self.DevInst))
    
    sp = _SP_DEVINFO_DATA('08751880-13bb-11e2-96f0-402cf4ca5e51',0x12345678)
    print sp
    

    输出:

    _SP_DEV_INFO_DATA(cbsize=28,ClassGuid=08751880-13bb-11e2-96f0-402cf4ca5e51,DevInst=0x12345678L)
    

    【讨论】:

    • 好的,谢谢。不过,我遇到了另一个问题,因为当我尝试运行 sAPI.SetupDiGetClassDevsExW(receiverGUID, None, None, "DIGCF_ALLCLASSES", None, None, None) 时,它成功了 (Kernel32.LastError = 0),但是,我得到的都是返回是一个整数值,我不知道如何处理。是否有可以浏览的好资源来帮助我利用该价值?
    • 另外,我如何确定要放入 DevInst 参数的内容?我看到您输入了一个十六进制值,尽管是任意值,但这就是那里的内容吗?我从哪里得到这个价值?谢谢
    • SetupDiGetClassDevs 为属于您使用 GUID 指定的设备类的所有设备返回 设备信息集 的句柄。然后使用SetupDiEnumDeviceInfo 查询设备集。您无需填写 _SP_DEVINFO_DATA 结构,SetupDiEnumDeviceInfo 会填写。请参阅 Microsoft 文档。 C中有例子。msdn.microsoft.com/en-us/library/windows/hardware/…
    • 非常感谢您的意见,这对您有很大帮助。如果您可以提供更多信息,那么当我调用 SetupDiEnumDeviceInterfaces 时,我得到的错误值为 259。我用谷歌搜索它,它说这意味着 ERROR_NO_MORE_ITEMS。当我调用该函数时,我应该将类 guid (HIDClass) 还是设备 guid 传递给它?另外,作为参考,SetupDiGetClassDevsWX(此WinUSB迭代中不存在SetupDiGetClassDevs)返回的值只是整数值2429888。仅传递它就足够了吗?再次感谢。
    • 另外,由于某种原因,传入对象的 guid 被截断。当我在对象外部的字符串上执行 uuid.UUID().get_bytes 时,我得到 'tZ\x17\xa0t\xd3\x11\xd0\xb6\xfe\x00\xa0\xc9\x0fW\xda' ,但在对象中我只看到'tZ\x17\xa0t\xd3\x11\xd0\xb6\xfe'。我读到我必须将初始化的 _SP_DEV_INFO_DATA 对象传递给 SetupDiEnumDeviceInfo 函数。我应该用什么值来初始化这个对象?
    【解决方案2】:

    你的类应该派生自ctypes.Structure

    import ctypes
    
    class _SP_DEVINFO_DATA(ctypes.Structure):
    
        _fields_ = [("cbSize", ctypes.c_ulong),
                    ("ClassGuid", ctypes.c_ubyte * 12),
                    ("DevInst", ctypes.c_ulong),
                    ("Reserved", ctypes.c_voidp)]
    
        def __init__(self, guid, inst):
            self.cbSize = 24
            # GUID doesn't work like this... Needs proper conversion
            #self.ClassGuid = (ctypes.c_ubyte * 12)(bytearray(guid))
            self.DevInst = (ctypes.c_ulong)(inst)
            self.Reserved = None
    

    【讨论】:

    • 我现在收到关于这一行的错误:self.ClassGuid = (ctypes.c_ubyte * 12)(bytearray(guid)) 它告诉我需要一个整数。如何将 GUID 转换为整数值?我应该使用 uuid 模块吗?我认为这会奏效,谢谢您的建议。我只需要弄清楚如何正确处理 GUID。
    • 我在想我想使用 uuid.UUID() 函数从文字值中获取一个实际的 GUID 类型对象,但是我该去哪里呢?
    • uuid.UUID(&lt;guid_string&gt;).get_bytes()。 GUID 是 16 个字节而不是 12 个字节。
    猜你喜欢
    • 2015-02-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2014-06-04
    • 2015-07-07
    • 1970-01-01
    相关资源
    最近更新 更多