当前位置: 首页 > 工具软件 > PyUSB > 使用案例 >

python libusb_使用python开发usb的两种方式(windriver与pyusb)

郗浩言
2023-12-01

背景

最近在给一个FPGA板子做上位机界面,上位机与下位机的通信采用USB方案,

驱动采用WinDriver。

但在实际调试过程中,发现WinDriver不同版本之间兼容性差,并且在win10

上表现不佳。实际的数据传输流程如下:

python usb dll(through ctypes) windriver usb device

由于dll文件是在win7机器上编译的,故仅能在win7上使用,在win10机器上出错。

使用python的项目都应该是简洁而优雅地,遂研究了在python操作usb device的两种方式。

驱动无关的调试软件使用bus hound

WinDriver

WinDriver经常与Jungo connectivity联系在一起,安装了WinDriver驱动的usb device在

设备管理器中也显示为Jungo devices。

完整的WinDriver开发流程应该从驱动开始,使用C/C++调用WinDriver提供的库与usb device

通信,将此程序编译为dll供其他程序调用。

将usb device连接上电脑,使用WinDriver给设备安装驱动。

在python中使用ctypes调用上文中的dll,完成调用过程。

PyUsb

pyusb是一个python库,可以方便地使用python操作usb设备。pyusb的数据传输流程如下:

python pyusb pyusb backend usb device

很明显可以看出省略了dll,大大减少了代码量。

具体使用过程:

1. 下载并安装pyusb backend

2. 连接usb device,使用pyusb backend安装驱动,我选择libusb,一般可以正常使用。不行就换其他的。

3. 编写python脚本,可以参考官方教程

缺点:

1. windriver有一个可视化的调试工具,可以单独发送接收数据以确定usb device是否正常,pyusb暂时没有找到。但

找到了一个非官方的基于tk的pywinusb hid调试工具

pyusb demo

我认为官方教程中的操作有些复杂,可以做如下简化:

1. 官方例程中使用get_active_configuration(), usb.util.find_descriptor()找设备描述符,我没有调试出来且繁杂,不如在 dev.set_configuration()之后直接dev.write(),前提是

已经知道设备描述符,这个可以通过一个简单的python脚本查询.

#!/usr/bin/python

# 此代码仅供示例,未测试

# 原始链接 http://www.bubuko.com/infodetail-2281652.html

import sys

import usb.core

# find USB devices

dev = usb.core.find(find_all=True)

# loop through devices, printing vendor and product ids in decimal and hex

for cfg in dev:

sys.stdout.write('Decimal VendorID=' + str(cfg.idVendor) + ' & ProductID=' + str(cfg.idProduct) + '\n')

sys.stdout.write('Hexadecimal VendorID=' + hex(cfg.idVendor) + ' & ProductID=' + hex(cfg.idProduct) + '\n\n')

一个demo代码如下,本代码同时采用了windriver和pyusb的方式。

由于完整运行该代码需要dll库文件、FPGA下位机配合,所以本代码仅供示例,大概率无法复现。

#!/bin/bash

import ctypes

from ctypes import *

# prepared to be called from the same class directory python scripts

import sys

import os

import usb.core

# now our hardware is vid = 0x03fd, pid = 0x0100)

class hardware_usb():

def __init__(self, vid, pid, read_length = 512, backend='libusb'):

'''

vid: vendor id

pid: product id

read_length : buffer length for reading

backend: select one from ['libusb', 'windriver']

'''

self.read_length = read_length

self.backend = backend

if backend == 'libusb':

usb_dev = usb.core.find(idVendor=vid, idProduct=pid)

if usb_dev != None :

usb_dev.set_configuration()

self.read_addr = 0x82

self.write_addr = 0x03

self.dev = usb_dev

self.init_status = True

else:

self.init_status = False

elif backend == 'windriver':

try:

dll = ctypes.cdll.LoadLibrary( "msdev_dll.dll")

except Exception as e :

print("Error occurs when init usb,%s:%s" %(str(e.__class__), str(e.args)))

self.init_status = False

else:

# input : void

# output : int

# assum 0 to be successful

self.init_driver = getattr(dll, '?InitDriver@@YA_NXZ')

# input : PVOID pBuffer, DWORD &dwSize

# output: bool

self._read = getattr(dll, '?Read_USB@@YA_NPAXAAK@Z')

# input : PVOID pBuffer, DWORD dwSize

# output: bool

self._write = getattr(dll, '?Write_USB@@YA_NPAXK@Z')

if 1 == self.init_driver():

self.init_status = True

else:

self.init_status = False

def write(self, data):

'''

Write data to usb

Note that data must bytes type

Return True if success

'''

if self.backend == 'windriver':

# BUG 此处使用c_char_p, 遇到\x00就被截断,可能有bug

buffer = ctypes.c_char_p()

buffer.value = data

leng = ctypes.c_ulong(len(data))

try:

status = self._write(buffer,leng)

except Exception as e :

print("Error occurs when write usb,%s:%s" %(str(e.__class__), str(e.args)))

status = 0

if status == 1:

ret = True

else :

ret = False

elif self.backend == 'libusb':

write_len = self.dev.write(self.write_addr, data)

# Test if data is written

if write_len == len(data):

ret = True

else:

ret = False

return ret

def read(self):

'''

Read data from usb

Note we will return bytes like data

Return None if error occurs

'''

if self.backend == 'windriver':

buffer_class = c_ubyte * 512

buffer = buffer_class()

leng = ctypes.c_ulong()

try :

status = self._read(buffer, ctypes.byref(leng) )

except Exception as e :

print("Error occurs when read from usb,%s:%s" %(str(e.__class__), str(e.args)))

status = 0

if status == 1:

ret = bytes(buffer[:leng.value])

else:

ret = b''

elif self.backend == 'libusb':

try:

ret = self.dev.read(self.read_addr, self.read_length)

ret = bytes(ret)

except Exception as e:

print("Error occurs when read from usb,%s:%s" %(str(e.__class__), str(e.args)))

ret = b''

return ret

if __name__ == "__main__":

usb_ins = hardware_usb(vid=0x03fd, pid=0x0100)

loop_num = 1

# 在测试中发现有一定概率写入出错

while True:

print('============================')

print('Loop num is', loop_num)

print('test write function')

data = b'\x7e\x7f\x00\x66'

try:

write_ret = usb_ins.write(data)

read_ret = usb_ins.read()

except:

print('Error occurs, return')

break

print('write function returns', write_ret)

print('test read function')

print(' read function returns', read_ret)

loop_num = loop_num + 1

 类似资料: