Shofel2_T124_python/venv/lib/python3.10/site-packages/usb1/testUSB1.py

333 lines
12 KiB
Python
Raw Normal View History

2024-05-25 16:45:07 +00:00
# Copyright (C) 2010-2021 Vincent Pelletier <plr.vincent@gmail.com>
#
# This library is free software; you can redistribute it and/or
# modify it under the terms of the GNU Lesser General Public
# License as published by the Free Software Foundation; either
# version 2.1 of the License, or (at your option) any later version.
#
# This library is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
# Lesser General Public License for more details.
#
# You should have received a copy of the GNU Lesser General Public
# License along with this library; if not, write to the Free Software
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
# pylint: disable=invalid-name, missing-docstring, too-many-public-methods
from ctypes import pointer, sizeof
import functools
import gc
import itertools
import sys
import unittest
import weakref
import usb1
from . import libusb1
buff_len = 1024
buffer_base = [x % 256 for x in range(buff_len)]
buff = bytes(buffer_base)
other_buff = bytes(reversed(buffer_base))
bytearray_buff = bytearray(buffer_base)
class USBContext(usb1.USBContext):
def open(self):
try:
return super().open()
except usb1.USBError:
raise unittest.SkipTest(
'usb1.USBContext() fails - no USB bus on system ?'
)
def checkTransferAllocCount(func):
@functools.wraps(func)
def wrapper(self, *args, **kw):
before = self.transfer_alloc_count
libusb_free_transfer = libusb1.libusb_free_transfer
libusb_alloc_transfer = libusb1.libusb_alloc_transfer
try:
libusb1.libusb_free_transfer = self._fakeFreeTransfer
libusb1.libusb_alloc_transfer = self._fakeAllocTransfer
result = func(self, *args, **kw)
finally:
libusb1.libusb_free_transfer = libusb_free_transfer
libusb1.libusb_alloc_transfer = libusb_alloc_transfer
gc.collect()
self.assertEqual(self.transfer_alloc_count, before)
return result
return wrapper
class USBTransferTests(unittest.TestCase):
def __init__(self, *args, **kw):
super().__init__(*args, **kw)
usb1.loadLibrary()
self.transfer_alloc_count = 0
def _fakeFreeTransfer(self, _):
self.transfer_alloc_count -= 1
def _fakeAllocTransfer(self, isochronous_count):
self.transfer_alloc_count += 1
buffer = bytearray(
sizeof(
libusb1.libusb_transfer,
) + sizeof(
libusb1.libusb_iso_packet_descriptor,
) * max(0, isochronous_count - 1),
)
transfer = libusb1.libusb_transfer.from_buffer(buffer)
# Keep a reference (in the finalizer itself) to the buffer for as long
# as transfer is alive.
weakref.finalize(transfer, lambda _: None, buffer)
return pointer(transfer)
@staticmethod
def getTransfer(iso_packets=0):
# Dummy handle
return usb1.USBTransfer(
context=None,
handle=None,
iso_packets=iso_packets,
before_submit=lambda x: None,
after_completion=lambda x: None,
registerFinalizer=lambda handle, finalizer: None,
unregisterFinalizer=lambda handle: None,
)
@staticmethod
def testGetVersion():
"""
Just testing getVersion doesn't raise...
"""
usb1.getVersion()
@staticmethod
def testHasCapability():
"""
Just testing hasCapability doesn't raise...
"""
usb1.hasCapability(usb1.CAP_HAS_CAPABILITY)
@checkTransferAllocCount
def testSetControl(self):
"""
Simplest test: feed some data, must not raise.
"""
transfer = self.getTransfer()
request_type = usb1.TYPE_STANDARD
request = usb1.REQUEST_GET_STATUS
value = 0
index = 0
def callback(transfer):
pass
user_data = []
timeout = 1000
# All provided, buffer variant
transfer.setControl(
request_type, request, value, index, buff,
callback=callback, user_data=user_data, timeout=timeout)
self.assertEqual(buff, transfer.getBuffer())
self.assertRaises(ValueError, transfer.setBuffer, buff)
# All provided, buffer length variant
transfer.setControl(
request_type, request, value, index, buff_len,
callback=callback, user_data=user_data, timeout=timeout)
# No timeout
transfer.setControl(
request_type, request, value, index, buff,
callback=callback, user_data=user_data)
# No user data
transfer.setControl(
request_type, request, value, index, buff, callback=callback)
# No callback
transfer.setControl(request_type, request, value, index, buff)
def _testTransferSetter(self, transfer, setter_id):
endpoint = 0x81
def callback(transfer):
pass
user_data = []
timeout = 1000
setter = getattr(transfer, setter_id)
# All provided, buffer variant
setter(
endpoint, buff, callback=callback, user_data=user_data,
timeout=timeout)
self.assertEqual(buff, transfer.getBuffer())
transfer.setBuffer(other_buff)
self.assertEqual(other_buff, transfer.getBuffer())
transfer.setBuffer(bytearray_buff)
self.assertEqual(bytearray_buff, transfer.getBuffer())
transfer.setBuffer(buff_len)
self.assertEqual(buff_len, len(transfer.getBuffer()))
# All provided, buffer length variant
setter(
endpoint, buff_len, callback=callback, user_data=user_data,
timeout=timeout)
# No timeout
setter(endpoint, buff, callback=callback, user_data=user_data)
# No user data
setter(endpoint, buff, callback=callback)
# No callback
setter(endpoint, buff)
@checkTransferAllocCount
def testSetBulk(self):
"""
Simplest test: feed some data, must not raise.
Also, test setBuffer/getBuffer.
"""
self._testTransferSetter(self.getTransfer(), 'setBulk')
@checkTransferAllocCount
def testSetInterrupt(self):
"""
Simplest test: feed some data, must not raise.
Also, test setBuffer/getBuffer.
"""
self._testTransferSetter(self.getTransfer(), 'setInterrupt')
@checkTransferAllocCount
def testSetIsochronous(self):
"""
Simplest test: feed some data, must not raise.
Also, test setBuffer/getBuffer/getISOBufferList/iterISO.
"""
iso_transfer_count = 16
transfer = self.getTransfer(iso_transfer_count)
self._testTransferSetter(transfer, 'setIsochronous')
# Returns whole buffers
self.assertEqual(
bytearray(itertools.chain(*transfer.getISOBufferList())),
buff,
)
# Returns actually transfered data, so here nothing
self.assertEqual(bytearray(
itertools.chain(*[x for _, x in transfer.iterISO()])),
bytearray(),
)
# Fake reception of whole transfers
c_transfer = getattr(
transfer,
'_' + transfer.__class__.__name__ + '__transfer'
)
for iso_metadata in libusb1.get_iso_packet_list(c_transfer):
iso_metadata.actual_length = iso_metadata.length
# Now iterISO returns everythig
self.assertEqual(bytearray(
itertools.chain(*[x for _, x in transfer.iterISO()])),
buff,
)
@checkTransferAllocCount
def testSetGetCallback(self):
transfer = self.getTransfer()
def callback(transfer):
pass
transfer.setCallback(callback)
got_callback = transfer.getCallback()
self.assertEqual(callback, got_callback)
def _testDescriptors(self, get_extra=False):
"""
Test descriptor walk.
Needs any usb device, which won't be opened.
"""
with USBContext() as context:
device_list = context.getDeviceList(skip_on_error=True)
found = False
seen_extra = False
for device in device_list:
device.getBusNumber()
device.getPortNumber()
device.getPortNumberList()
device.getDeviceAddress()
for settings in device.iterSettings():
for endpoint in settings:
pass
for configuration in device.iterConfigurations():
if get_extra and len(configuration.getExtra()) > 0:
seen_extra = True
for interface in configuration:
for settings in interface:
if get_extra and len(settings.getExtra()) > 0:
seen_extra = True
for endpoint in settings:
if get_extra and len(endpoint.getExtra()) > 0:
seen_extra = True
found = True
if not found:
raise unittest.SkipTest('descriptor walk test did not complete')
if get_extra and not seen_extra:
raise unittest.SkipTest('did not see any extra descriptors')
def testDescriptors(self):
self._testDescriptors()
def testDescriptorsWithExtra(self):
self._testDescriptors(get_extra=True)
def testDefaultEnumScope(self):
"""
Enum instances must only affect the scope they are created in.
"""
ENUM_NAME = 'THE_ANSWER'
ENUM_VALUE = 42
local_dict = locals()
global_dict = globals()
self.assertEqual(local_dict.get(ENUM_NAME), None)
self.assertEqual(global_dict.get(ENUM_NAME), None)
self.assertEqual(getattr(libusb1, ENUM_NAME, None), None)
# pylint: disable=unused-variable
TEST_ENUM = libusb1.Enum({ENUM_NAME: ENUM_VALUE})
# pylint: enable=unused-variable
self.assertEqual(local_dict.get(ENUM_NAME), ENUM_VALUE)
self.assertEqual(global_dict.get(ENUM_NAME), None)
self.assertEqual(getattr(libusb1, ENUM_NAME, None), None)
def testExplicitEnumScope(self):
"""
Enum instances must only affect the scope they are created in.
"""
ENUM_NAME = 'THE_ANSWER'
ENUM_VALUE = 42
local_dict = locals()
global_dict = globals()
self.assertEqual(local_dict.get(ENUM_NAME), None)
self.assertEqual(global_dict.get(ENUM_NAME), None)
self.assertEqual(getattr(libusb1, ENUM_NAME, None), None)
# pylint: disable=unused-variable
TEST_ENUM = libusb1.Enum({ENUM_NAME: ENUM_VALUE}, global_dict)
# pylint: enable=unused-variable
try:
self.assertEqual(local_dict.get(ENUM_NAME), None)
self.assertEqual(global_dict.get(ENUM_NAME), ENUM_VALUE)
self.assertEqual(getattr(libusb1, ENUM_NAME, None), None)
finally:
del global_dict[ENUM_NAME]
def testImplicitUSBContextOpening(self):
"""
Test pre-1.5 API backward compatibility.
First method call which needs a context succeeds.
Further calls return None.
"""
context = USBContext() # Deprecated
try:
fd_list = context.getPollFDList()
except NotImplementedError:
raise unittest.SkipTest('libusb without file descriptor events')
self.assertNotEqual(fd_list, None)
context.exit() # Deprecated
self.assertEqual(context.getPollFDList(), None)
def testHasVersion(self):
# Property is present and non-empty
self.assertTrue(usb1.__version__)
if __name__ == '__main__':
unittest.main()