From 7fc0e7e349693ec2594aeeda01f3242cce1c5e42 Mon Sep 17 00:00:00 2001
From: Jimmy van den Bergh <jimmy.vandenbergh@admesy.com>
Date: Thu, 21 Nov 2024 12:54:47 +0100
Subject: [PATCH 1/4] updated to support partial usbtmc message reads from
 device to support IEEE 488.2 arbitrary block reads fixes the requested data
 to be send in the header instead of recv_chunk.

---
 CHANGES                       |  2 ++
 pyvisa_py/protocols/usbtmc.py | 21 ++++++++++-----------
 2 files changed, 12 insertions(+), 11 deletions(-)

diff --git a/CHANGES b/CHANGES
index 823fa5a6..2b38adaf 100644
--- a/CHANGES
+++ b/CHANGES
@@ -10,6 +10,8 @@ PyVISA-py Changelog
   has been read (see specification), and only expects a header on the first packet received.
 - fix usbtmc implementation to properly discard the alignment bytes
   ensuring only the actual data (`transfer_size`) is retained in the message PR #465 
+- Implemented partial USBTMC message functionality that allows reading the amount of bytes 
+  specified by host.
 - add support for VI_ATTR_SUPPRESS_END_EN for USB resources PR #449
 
 0.7.2 (07/03/2024)
diff --git a/pyvisa_py/protocols/usbtmc.py b/pyvisa_py/protocols/usbtmc.py
index 8bbac953..6e7b3659 100644
--- a/pyvisa_py/protocols/usbtmc.py
+++ b/pyvisa_py/protocols/usbtmc.py
@@ -455,13 +455,6 @@ def write(self, data):
         return size
 
     def read(self, size):
-        header_size = 12
-        max_padding = 511
-        recv_chunk = self.usb_recv_ep.wMaxPacketSize - header_size
-
-        if size > 0 and size < recv_chunk:
-            recv_chunk = size
-
         eom = False
 
         raw_read = super(USBTMC, self).read
@@ -473,12 +466,12 @@ def read(self, size):
             received_transfer = bytearray()
             self._btag = (self._btag % 255) + 1
 
-            req = BulkInMessage.build_array(self._btag, recv_chunk, None)
+            req = BulkInMessage.build_array(self._btag, size, None)
 
             raw_write(req)
 
             try:
-                resp = raw_read(recv_chunk + header_size + max_padding)
+                resp = raw_read(self.usb_recv_ep.wMaxPacketSize)
                 response = BulkInMessage.from_bytes(resp)
                 received_transfer.extend(response.data)
                 while (
@@ -491,15 +484,21 @@ def read(self, size):
                     # is sent (one whose length is less than wMaxPacketSize)
                     # wMaxPacketSize may be incorrectly reported by certain drivers.
                     # Therefore, continue reading until the transfer_size is reached.
-                    resp = raw_read(recv_chunk + header_size + max_padding)
+                    resp = raw_read(self.usb_recv_ep.wMaxPacketSize)
                     received_transfer.extend(resp)
 
                 # Detect EOM only when device sends all expected bytes.
                 if len(received_transfer) >= response.transfer_size:
                     eom = response.transfer_attributes & 1
+                if not eom and len(received_transfer) >= size:
+                    # Read asking for 'size' bytes from the device.
+                    # This may be less then the device wants to send back in a message
+                    # Therefore the request does not mean that we must receive a EOM.
+                    # Multiple `transfers` will be required to retrieve the remaining bytes.
+                    eom = True
                     # Truncate data to the specified length (discard padding)
                     # USBTMC header (12 bytes) has already truncated
-                    received_message.extend(received_transfer[: response.transfer_size])
+                received_message.extend(received_transfer[: response.transfer_size])
             except (usb.core.USBError, ValueError):
                 # Abort failed Bulk-IN operation.
                 self._abort_bulk_in(self._btag)

From ca8df6c9e0d7b76f3d56c6c1bb125b5fa26a7eab Mon Sep 17 00:00:00 2001
From: Jimmy van den Bergh <jimmy.vandenbergh@admesy.com>
Date: Mon, 2 Dec 2024 11:06:28 +0100
Subject: [PATCH 2/4] Speedoptimazation for read, we now use large chuck sizes.

---
 pyvisa_py/protocols/usbtmc.py | 56 +++++++++++++++++++++++++----------
 1 file changed, 40 insertions(+), 16 deletions(-)

diff --git a/pyvisa_py/protocols/usbtmc.py b/pyvisa_py/protocols/usbtmc.py
index 6e7b3659..ddcad088 100644
--- a/pyvisa_py/protocols/usbtmc.py
+++ b/pyvisa_py/protocols/usbtmc.py
@@ -12,6 +12,7 @@
 """
 
 import enum
+import math
 import struct
 import time
 import warnings
@@ -455,6 +456,7 @@ def write(self, data):
         return size
 
     def read(self, size):
+        usbtmc_header_size = 12
         eom = False
 
         raw_read = super(USBTMC, self).read
@@ -467,25 +469,23 @@ def read(self, size):
             self._btag = (self._btag % 255) + 1
 
             req = BulkInMessage.build_array(self._btag, size, None)
-
             raw_write(req)
 
             try:
-                resp = raw_read(self.usb_recv_ep.wMaxPacketSize)
+                # make sure the data request is in multitudes of wMaxPacketSize.
+                # + 1 * wMaxPacketSize for message sizes that equals wMaxPacketSize == size + usbtmc_header_size.
+                # This to be able to retrieve a short package to end communication
+                # (see USB 2.0 Section 5.8.3 and USBTMC Section 3.3)
+                chunck_size = (
+                    math.floor(
+                        (size + usbtmc_header_size) / self.usb_recv_ep.wMaxPacketSize
+                    )
+                    + 1
+                ) * self.usb_recv_ep.wMaxPacketSize
+                resp = raw_read(chunck_size)
+
                 response = BulkInMessage.from_bytes(resp)
                 received_transfer.extend(response.data)
-                while (
-                    len(resp) == self.usb_recv_ep.wMaxPacketSize
-                    or len(received_transfer) < response.transfer_size
-                ):
-                    # USBTMC Section 3.3 specifies that the first usb packet
-                    # must contain the header. the remaining packets do not need
-                    # the header the message is finished when a "short packet"
-                    # is sent (one whose length is less than wMaxPacketSize)
-                    # wMaxPacketSize may be incorrectly reported by certain drivers.
-                    # Therefore, continue reading until the transfer_size is reached.
-                    resp = raw_read(self.usb_recv_ep.wMaxPacketSize)
-                    received_transfer.extend(resp)
 
                 # Detect EOM only when device sends all expected bytes.
                 if len(received_transfer) >= response.transfer_size:
@@ -496,8 +496,32 @@ def read(self, size):
                     # Therefore the request does not mean that we must receive a EOM.
                     # Multiple `transfers` will be required to retrieve the remaining bytes.
                     eom = True
-                    # Truncate data to the specified length (discard padding)
-                    # USBTMC header (12 bytes) has already truncated
+                else:
+                    while (
+                        (len(resp) % self.usb_recv_ep.wMaxPacketSize) == 0
+                        or len(received_transfer) < response.transfer_size
+                    ) and not eom:
+                        # USBTMC Section 3.3 specifies that the first usb packet
+                        # must contain the header. the remaining packets do not need
+                        # the header the message is finished when a "short packet"
+                        # is sent (one whose length is less than wMaxPacketSize)
+                        # wMaxPacketSize may be incorrectly reported by certain drivers.
+                        # Therefore, continue reading until the transfer_size is reached.
+                        chunck_size = (
+                            math.floor(
+                                (size - len(received_transfer))
+                                / self.usb_recv_ep.wMaxPacketSize
+                            )
+                            + 1
+                        ) * self.usb_recv_ep.wMaxPacketSize
+                        resp = raw_read(chunck_size)
+                        received_transfer.extend(resp)
+                    if len(received_transfer) >= response.transfer_size:
+                        eom = response.transfer_attributes & 1
+                    if not eom and len(received_transfer) >= size:
+                        eom = True
+                # Truncate data to the specified length (discard padding)
+                # USBTMC header (12 bytes) has already truncated
                 received_message.extend(received_transfer[: response.transfer_size])
             except (usb.core.USBError, ValueError):
                 # Abort failed Bulk-IN operation.

From 4286e5647cd95c352e484f42b836869caa056c02 Mon Sep 17 00:00:00 2001
From: Jimmy van den Bergh <93945571+Jimmyvandenbergh@users.noreply.github.com>
Date: Sat, 7 Dec 2024 23:29:58 +0100
Subject: [PATCH 3/4] Fix typo in chunck VS chunk

---
 pyvisa_py/protocols/usbtmc.py | 8 ++++----
 1 file changed, 4 insertions(+), 4 deletions(-)

diff --git a/pyvisa_py/protocols/usbtmc.py b/pyvisa_py/protocols/usbtmc.py
index ddcad088..da0cc888 100644
--- a/pyvisa_py/protocols/usbtmc.py
+++ b/pyvisa_py/protocols/usbtmc.py
@@ -476,13 +476,13 @@ def read(self, size):
                 # + 1 * wMaxPacketSize for message sizes that equals wMaxPacketSize == size + usbtmc_header_size.
                 # This to be able to retrieve a short package to end communication
                 # (see USB 2.0 Section 5.8.3 and USBTMC Section 3.3)
-                chunck_size = (
+                chunk_size = (
                     math.floor(
                         (size + usbtmc_header_size) / self.usb_recv_ep.wMaxPacketSize
                     )
                     + 1
                 ) * self.usb_recv_ep.wMaxPacketSize
-                resp = raw_read(chunck_size)
+                resp = raw_read(chunk_size)
 
                 response = BulkInMessage.from_bytes(resp)
                 received_transfer.extend(response.data)
@@ -507,14 +507,14 @@ def read(self, size):
                         # is sent (one whose length is less than wMaxPacketSize)
                         # wMaxPacketSize may be incorrectly reported by certain drivers.
                         # Therefore, continue reading until the transfer_size is reached.
-                        chunck_size = (
+                        chunk_size = (
                             math.floor(
                                 (size - len(received_transfer))
                                 / self.usb_recv_ep.wMaxPacketSize
                             )
                             + 1
                         ) * self.usb_recv_ep.wMaxPacketSize
-                        resp = raw_read(chunck_size)
+                        resp = raw_read(chunk_size)
                         received_transfer.extend(resp)
                     if len(received_transfer) >= response.transfer_size:
                         eom = response.transfer_attributes & 1

From 6a0c720ea44d0bd45620c113051de35a0966023f Mon Sep 17 00:00:00 2001
From: Matthieu Dartiailh <marul@laposte.net>
Date: Tue, 17 Dec 2024 08:50:28 +0100
Subject: [PATCH 4/4] Update release notes

---
 CHANGES | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/CHANGES b/CHANGES
index 2b38adaf..91133c39 100644
--- a/CHANGES
+++ b/CHANGES
@@ -11,7 +11,7 @@ PyVISA-py Changelog
 - fix usbtmc implementation to properly discard the alignment bytes
   ensuring only the actual data (`transfer_size`) is retained in the message PR #465 
 - Implemented partial USBTMC message functionality that allows reading the amount of bytes 
-  specified by host.
+  specified by host PR #470
 - add support for VI_ATTR_SUPPRESS_END_EN for USB resources PR #449
 
 0.7.2 (07/03/2024)