mirror of
https://github.com/hathach/tinyusb.git
synced 2025-04-09 18:44:31 +00:00
Lots of updates (especially error handling)
This commit is contained in:
parent
2aa10daf26
commit
fa5b5e4561
@ -107,13 +107,13 @@ uint8_t const * tud_hid_descriptor_report_cb(void)
|
|||||||
|
|
||||||
# define TUD_USBTMC_DESC_MAIN(_itfnum,_bNumEndpoints) \
|
# define TUD_USBTMC_DESC_MAIN(_itfnum,_bNumEndpoints) \
|
||||||
TUD_USBTMC_IF_DESCRIPTOR(_itfnum, _bNumEndpoints, /*_stridx = */ 4u, TUD_USBTMC_PROTOCOL_USB488), \
|
TUD_USBTMC_IF_DESCRIPTOR(_itfnum, _bNumEndpoints, /*_stridx = */ 4u, TUD_USBTMC_PROTOCOL_USB488), \
|
||||||
TUD_USBTMC_BULK_DESCRIPTORS(/* OUT = */0x03, /* IN = */ 0x83, /* packet size = */USBTMCD_MAX_PACKET_SIZE)
|
TUD_USBTMC_BULK_DESCRIPTORS(/* OUT = */0x01, /* IN = */ 0x81, /* packet size = */USBTMCD_MAX_PACKET_SIZE)
|
||||||
|
|
||||||
#if defined(CFG_TUD_USBTMC_ENABLE_INT_EP)
|
#if defined(CFG_TUD_USBTMC_ENABLE_INT_EP)
|
||||||
// Interrupt endpoint should be 2 bytes on a FS USB link
|
// Interrupt endpoint should be 2 bytes on a FS USB link
|
||||||
# define TUD_USBTMC_DESC(_itfnum) \
|
# define TUD_USBTMC_DESC(_itfnum) \
|
||||||
TUD_USBTMC_DESC_MAIN(_itfnum, /* _epCount = */ 3), \
|
TUD_USBTMC_DESC_MAIN(_itfnum, /* _epCount = */ 3), \
|
||||||
TUD_USBTMC_INT_DESCRIPTOR(/* INT ep # */ 0x84, /* epMaxSize = */ 2, /* bInterval = */16u )
|
TUD_USBTMC_INT_DESCRIPTOR(/* INT ep # */ 0x82, /* epMaxSize = */ 2, /* bInterval = */16u )
|
||||||
# define USBTMC_DESC_LEN (TUD_USBTMC_IF_DESCRIPTOR_LEN + TUD_USBTMC_BULK_DESCRIPTORS_LEN + TUD_USBTMC_INT_DESCRIPTOR_LEN)
|
# define USBTMC_DESC_LEN (TUD_USBTMC_IF_DESCRIPTOR_LEN + TUD_USBTMC_BULK_DESCRIPTORS_LEN + TUD_USBTMC_INT_DESCRIPTOR_LEN)
|
||||||
|
|
||||||
#else
|
#else
|
||||||
|
@ -84,6 +84,7 @@ static volatile uint32_t idnQuery;
|
|||||||
|
|
||||||
static uint32_t resp_delay = 125u; // Adjustable delay, to allow for better testing
|
static uint32_t resp_delay = 125u; // Adjustable delay, to allow for better testing
|
||||||
static size_t buffer_len;
|
static size_t buffer_len;
|
||||||
|
static size_t buffer_tx_ix; // for transmitting using multiple transfers
|
||||||
static uint8_t buffer[225]; // A few packets long should be enough.
|
static uint8_t buffer[225]; // A few packets long should be enough.
|
||||||
|
|
||||||
|
|
||||||
@ -108,6 +109,11 @@ bool tud_usbtmc_app_msgBulkOut_start_cb(uint8_t rhport, usbtmc_msg_request_dev_d
|
|||||||
(void)rhport;
|
(void)rhport;
|
||||||
(void)msgHeader;
|
(void)msgHeader;
|
||||||
buffer_len = 0;
|
buffer_len = 0;
|
||||||
|
if(msgHeader->TransferSize > sizeof(buffer))
|
||||||
|
{
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -122,6 +128,10 @@ bool tud_usbtmc_app_msg_data_cb(uint8_t rhport, void *data, size_t len, bool tra
|
|||||||
memcpy(&(buffer[buffer_len]), data, len);
|
memcpy(&(buffer[buffer_len]), data, len);
|
||||||
buffer_len += len;
|
buffer_len += len;
|
||||||
}
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
return false; // buffer overflow!
|
||||||
|
}
|
||||||
queryState = transfer_complete;
|
queryState = transfer_complete;
|
||||||
idnQuery = 0;
|
idnQuery = 0;
|
||||||
|
|
||||||
@ -145,8 +155,13 @@ bool tud_usbtmc_app_msg_data_cb(uint8_t rhport, void *data, size_t len, bool tra
|
|||||||
bool tud_usbtmc_app_msgBulkIn_complete_cb(uint8_t rhport)
|
bool tud_usbtmc_app_msgBulkIn_complete_cb(uint8_t rhport)
|
||||||
{
|
{
|
||||||
(void)rhport;
|
(void)rhport;
|
||||||
|
if((buffer_tx_ix == buffer_len) || idnQuery) // done
|
||||||
|
{
|
||||||
status &= (uint8_t)~(IEEE4882_STB_MAV); // clear MAV
|
status &= (uint8_t)~(IEEE4882_STB_MAV); // clear MAV
|
||||||
|
queryState = 0;
|
||||||
|
bulkInStarted = 0;
|
||||||
|
buffer_tx_ix = 0;
|
||||||
|
}
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
@ -161,15 +176,25 @@ bool tud_usbtmc_app_msgBulkIn_request_cb(uint8_t rhport, usbtmc_msg_request_dev_
|
|||||||
rspMsg.header.bTag = request->header.bTag,
|
rspMsg.header.bTag = request->header.bTag,
|
||||||
rspMsg.header.bTagInverse = request->header.bTagInverse;
|
rspMsg.header.bTagInverse = request->header.bTagInverse;
|
||||||
msgReqLen = request->TransferSize;
|
msgReqLen = request->TransferSize;
|
||||||
|
|
||||||
#ifdef xDEBUG
|
#ifdef xDEBUG
|
||||||
uart_tx_str_sync("MSG_IN_DATA: Requested!\r\n");
|
uart_tx_str_sync("MSG_IN_DATA: Requested!\r\n");
|
||||||
#endif
|
#endif
|
||||||
|
if(queryState == 0 || (buffer_tx_ix == 0))
|
||||||
|
{
|
||||||
TU_ASSERT(bulkInStarted == 0);
|
TU_ASSERT(bulkInStarted == 0);
|
||||||
bulkInStarted = 1;
|
bulkInStarted = 1;
|
||||||
|
|
||||||
// > If a USBTMC interface receives a Bulk-IN request prior to receiving a USBTMC command message
|
// > If a USBTMC interface receives a Bulk-IN request prior to receiving a USBTMC command message
|
||||||
// that expects a response, the device must NAK the request
|
// that expects a response, the device must NAK the request (*not stall*)
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
size_t txlen = tu_min32(buffer_len-buffer_tx_ix,msgReqLen);
|
||||||
|
usbtmcd_transmit_dev_msg_data(rhport, &buffer[buffer_tx_ix], txlen,
|
||||||
|
(buffer_tx_ix+txlen) == buffer_len, false);
|
||||||
|
buffer_tx_ix += txlen;
|
||||||
|
}
|
||||||
// Always return true indicating not to stall the EP.
|
// Always return true indicating not to stall the EP.
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
@ -197,17 +222,17 @@ void usbtmc_app_task_iter(void) {
|
|||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case 4: // time to transmit;
|
case 4: // time to transmit;
|
||||||
if(bulkInStarted) {
|
if(bulkInStarted && (buffer_tx_ix == 0)) {
|
||||||
queryState = 0;
|
|
||||||
bulkInStarted = 0;
|
|
||||||
|
|
||||||
if(idnQuery)
|
if(idnQuery)
|
||||||
{
|
{
|
||||||
usbtmcd_transmit_dev_msg_data(rhport, idn, tu_min32(sizeof(idn)-1,msgReqLen),false);
|
usbtmcd_transmit_dev_msg_data(rhport, idn, tu_min32(sizeof(idn)-1,msgReqLen),true,false);
|
||||||
|
queryState = 0;
|
||||||
|
bulkInStarted = 0;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
usbtmcd_transmit_dev_msg_data(rhport, buffer, tu_min32(buffer_len,msgReqLen),false);
|
buffer_tx_ix = tu_min32(buffer_len,msgReqLen);
|
||||||
|
usbtmcd_transmit_dev_msg_data(rhport, buffer, buffer_tx_ix, buffer_tx_ix == buffer_len, false);
|
||||||
}
|
}
|
||||||
// MAV is cleared in the transfer complete callback.
|
// MAV is cleared in the transfer complete callback.
|
||||||
}
|
}
|
||||||
|
@ -6,7 +6,8 @@ import sys
|
|||||||
|
|
||||||
def test_idn():
|
def test_idn():
|
||||||
idn = inst.query("*idn?");
|
idn = inst.query("*idn?");
|
||||||
assert idn == "TinyUSB,ModelNumber,SerialNumber,FirmwareVer123456\r\n"
|
assert (idn == "TinyUSB,ModelNumber,SerialNumber,FirmwareVer123456\r\n")
|
||||||
|
assert (inst.is_4882_compliant)
|
||||||
|
|
||||||
def test_echo(m,n):
|
def test_echo(m,n):
|
||||||
longstr = "0123456789abcdefghijklmnopqrstuvwxyz" * 50
|
longstr = "0123456789abcdefghijklmnopqrstuvwxyz" * 50
|
||||||
@ -34,6 +35,8 @@ def test_trig():
|
|||||||
|
|
||||||
|
|
||||||
def test_mav():
|
def test_mav():
|
||||||
|
inst.write("delay 50")
|
||||||
|
inst.read_stb() # clear STB
|
||||||
assert (inst.read_stb() == 0)
|
assert (inst.read_stb() == 0)
|
||||||
inst.write("123")
|
inst.write("123")
|
||||||
time.sleep(0.3)
|
time.sleep(0.3)
|
||||||
@ -61,8 +64,6 @@ def test_srq():
|
|||||||
rsp = inst.read()
|
rsp = inst.read()
|
||||||
assert(rsp == "123\r\n")
|
assert(rsp == "123\r\n")
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
def test_read_timeout():
|
def test_read_timeout():
|
||||||
inst.timeout = 500
|
inst.timeout = 500
|
||||||
# First read with no MAV
|
# First read with no MAV
|
||||||
@ -78,7 +79,53 @@ def test_read_timeout():
|
|||||||
t = time.time() - t0
|
t = time.time() - t0
|
||||||
assert ((t*1000.0) > (inst.timeout - 300))
|
assert ((t*1000.0) > (inst.timeout - 300))
|
||||||
assert ((t*1000.0) < (inst.timeout + 300))
|
assert ((t*1000.0) < (inst.timeout + 300))
|
||||||
print(f"Delay was {t}")
|
print(f"Delay was {t:0.3}")
|
||||||
|
# Response is still in queue, so send a clear (to be more helpful to the next test)
|
||||||
|
inst.clear()
|
||||||
|
time.sleep(0.3)
|
||||||
|
assert(0 == (inst.read_stb() & 0x10)), "MAV not reset after clear"
|
||||||
|
|
||||||
|
def test_abort_in():
|
||||||
|
inst.timeout = 200
|
||||||
|
# First read with no MAV
|
||||||
|
inst.read_stb()
|
||||||
|
assert (inst.read_stb() == 0)
|
||||||
|
inst.write("delay 500")
|
||||||
|
inst.write("xxx")
|
||||||
|
t0 = time.time()
|
||||||
|
try:
|
||||||
|
rsp = inst.read()
|
||||||
|
assert(false), "Read should have resulted in timeout"
|
||||||
|
except visa.VisaIOError:
|
||||||
|
print("Got expected exception")
|
||||||
|
t = time.time() - t0
|
||||||
|
assert ((t*1000.0) > (inst.timeout - 300))
|
||||||
|
assert ((t*1000.0) < (inst.timeout + 300))
|
||||||
|
print(f"Delay was {t:0.3}")
|
||||||
|
# Response is still in queue, so send a clear (to be more helpful to the next test)
|
||||||
|
inst.timeout = 800
|
||||||
|
y = inst.read()
|
||||||
|
assert(y == "xxx\r\n")
|
||||||
|
|
||||||
|
def test_indicate():
|
||||||
|
# perform indicator pulse
|
||||||
|
usb_iface = inst.get_visa_attribute(visa.constants.VI_ATTR_USB_INTFC_NUM)
|
||||||
|
retv = inst.control_in(request_type_bitmap_field=0xA1, request_id=64, request_value=0x0000, index=usb_iface, length=0x0001)
|
||||||
|
assert((retv[1] == visa.constants.StatusCode(0)) and (retv[0] == b'\x01')), f"indicator pulse failed: retv={retv}"
|
||||||
|
|
||||||
|
|
||||||
|
def test_multi_read():
|
||||||
|
old_chunk_size = inst.chunk_size
|
||||||
|
longstr = "0123456789abcdefghijklmnopqrstuvwxyz" * 10
|
||||||
|
timeout = 10
|
||||||
|
x = longstr[0:174]
|
||||||
|
inst.chunk_size = 50 # Seems chunk size only applies to read but not write
|
||||||
|
inst.write(x)
|
||||||
|
# I'm not sure how to request just the remaining bit using a max count... so just read it all.
|
||||||
|
y = inst.read()
|
||||||
|
assert (x + "\r\n" == y)
|
||||||
|
#inst.chunk_size = old_chunk_size
|
||||||
|
|
||||||
|
|
||||||
rm = visa.ResourceManager("/c/Windows/system32/visa64.dll")
|
rm = visa.ResourceManager("/c/Windows/system32/visa64.dll")
|
||||||
reslist = rm.list_resources("USB?::?*::INSTR")
|
reslist = rm.list_resources("USB?::?*::INSTR")
|
||||||
@ -89,12 +136,20 @@ if (len(reslist) == 0):
|
|||||||
|
|
||||||
inst = rm.open_resource(reslist[0]);
|
inst = rm.open_resource(reslist[0]);
|
||||||
inst.timeout = 3000
|
inst.timeout = 3000
|
||||||
|
|
||||||
inst.clear()
|
inst.clear()
|
||||||
|
|
||||||
print("+ IDN")
|
print("+ IDN")
|
||||||
test_idn()
|
test_idn()
|
||||||
|
|
||||||
inst.timeout = 3000
|
print("+test abort in")
|
||||||
|
test_abort_in()
|
||||||
|
|
||||||
|
|
||||||
|
inst.timeout = 2000
|
||||||
|
|
||||||
|
print("+ multi read")
|
||||||
|
test_multi_read()
|
||||||
|
|
||||||
|
|
||||||
print("+ echo delay=0")
|
print("+ echo delay=0")
|
||||||
@ -110,7 +165,7 @@ inst.write("delay 150")
|
|||||||
test_echo(53,76)
|
test_echo(53,76)
|
||||||
test_echo(165,170)
|
test_echo(165,170)
|
||||||
|
|
||||||
print("+ Read timeout (no MAV")
|
print("+ Read timeout (no MAV)")
|
||||||
test_read_timeout()
|
test_read_timeout()
|
||||||
|
|
||||||
print("+ MAV")
|
print("+ MAV")
|
||||||
@ -119,6 +174,9 @@ test_mav()
|
|||||||
print("+ SRQ")
|
print("+ SRQ")
|
||||||
test_srq()
|
test_srq()
|
||||||
|
|
||||||
|
print("+ indicate")
|
||||||
|
test_indicate()
|
||||||
|
|
||||||
print("+ TRIG")
|
print("+ TRIG")
|
||||||
test_trig()
|
test_trig()
|
||||||
|
|
||||||
|
@ -31,25 +31,31 @@
|
|||||||
* This file is part of the TinyUSB stack.
|
* This file is part of the TinyUSB stack.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
// Synchronization is needed in some spots.
|
/*
|
||||||
// These functions should NOT be called from interrupts.
|
* This library is not fully reentrant, though it is reentrant from the view
|
||||||
|
* of either the application layer or the USB stack. Due to its locking,
|
||||||
/* The library is designed that its functions can be called by any user task, with need for
|
* it is not safe to call its functions from interrupts.
|
||||||
* additional locking. In the case of "no OS", this task is never preempted other than by
|
|
||||||
* interrupts, and the USBTMC code isn't called by interrupts, so all is OK. In the case
|
|
||||||
* of an OS, this class driver uses the OSAL to perform locking. The code uses a single lock
|
|
||||||
* and does not call outside of this class with a lock held, so deadlocks won't happen.
|
|
||||||
*
|
*
|
||||||
* This module's application-facing functions are not reentrant. The application must
|
* The one exception is that its functions may not be called from the application
|
||||||
* only call them from a single thread (or implement its own locking).
|
* until the USB stack is initialized. This should not be a problem since the
|
||||||
|
* device shouldn't be sending messages until it receives a request from the
|
||||||
|
* host.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* In the case of single-CPU "no OS", this task is never preempted other than by
|
||||||
|
* interrupts, and the USBTMC code isn't called by interrupts, so all is OK. For "no OS",
|
||||||
|
* the mutex structure's main effect is to disable the USB interrupts.
|
||||||
|
* With an OS, this class driver uses the OSAL to perform locking. The code uses a single lock
|
||||||
|
* and does not call outside of this class with a lock held, so deadlocks won't happen.
|
||||||
|
*/
|
||||||
|
|
||||||
//Limitations:
|
//Limitations:
|
||||||
// "vendor-specific" commands are not handled.
|
// "vendor-specific" commands are not handled.
|
||||||
// Dealing with "termchar" must be handled by the application layer,
|
// Dealing with "termchar" must be handled by the application layer,
|
||||||
// though additional error checking is does in this module.
|
// though additional error checking is does in this module.
|
||||||
// talkOnly and listenOnly are NOT supported. They're no permitted
|
// talkOnly and listenOnly are NOT supported. They're not permitted
|
||||||
// in USB488, anyway.
|
// in USB488, anyway.
|
||||||
|
|
||||||
/* Supported:
|
/* Supported:
|
||||||
@ -64,10 +70,7 @@
|
|||||||
// TODO:
|
// TODO:
|
||||||
// USBTMC 3.2.2 error conditions not strictly followed
|
// USBTMC 3.2.2 error conditions not strictly followed
|
||||||
// No local lock-out, REN, or GTL.
|
// No local lock-out, REN, or GTL.
|
||||||
// Cannot handle clear.
|
|
||||||
// Clear message available status byte at the correct time? (488 4.3.1.3)
|
// Clear message available status byte at the correct time? (488 4.3.1.3)
|
||||||
// Abort bulk in/out
|
|
||||||
// No CLEAR_FEATURE/HALT no EP (yet)
|
|
||||||
|
|
||||||
|
|
||||||
#include "tusb_option.h"
|
#include "tusb_option.h"
|
||||||
@ -79,21 +82,24 @@
|
|||||||
#include "usbtmc_device.h"
|
#include "usbtmc_device.h"
|
||||||
#include "device/dcd.h"
|
#include "device/dcd.h"
|
||||||
#include "device/usbd.h"
|
#include "device/usbd.h"
|
||||||
|
#include "osal/osal.h"
|
||||||
|
|
||||||
|
// FIXME: I shouldn't need to include _pvt headers, but it is necessary for usbd_edpt_xfer, _stall, and _busy
|
||||||
|
#include "device/usbd_pvt.h"
|
||||||
|
|
||||||
#ifdef xDEBUG
|
#ifdef xDEBUG
|
||||||
#include "uart_util.h"
|
#include "uart_util.h"
|
||||||
static char logMsg[150];
|
static char logMsg[150];
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
/*
|
||||||
// FIXME: I shouldn't need to include _pvt headers.
|
* The state machine does not allow simultaneous reading and writing. This is
|
||||||
#include "device/usbd_pvt.h"
|
* consistent with USBTMC.
|
||||||
|
*/
|
||||||
static uint8_t termChar;
|
|
||||||
static uint8_t termCharRequested = false;
|
|
||||||
|
|
||||||
typedef enum
|
typedef enum
|
||||||
{
|
{
|
||||||
|
STATE_CLOSED,
|
||||||
STATE_IDLE,
|
STATE_IDLE,
|
||||||
STATE_RCV,
|
STATE_RCV,
|
||||||
STATE_TX_REQUESTED,
|
STATE_TX_REQUESTED,
|
||||||
@ -126,16 +132,12 @@ typedef struct
|
|||||||
uint8_t lastBulkOutTag; // used for aborts (mostly)
|
uint8_t lastBulkOutTag; // used for aborts (mostly)
|
||||||
uint8_t lastBulkInTag; // used for aborts (mostly)
|
uint8_t lastBulkInTag; // used for aborts (mostly)
|
||||||
|
|
||||||
uint8_t const * devInBuffer;
|
uint8_t const * devInBuffer; // pointer to application-layer used for transmissions
|
||||||
} usbtmc_interface_state_t;
|
} usbtmc_interface_state_t;
|
||||||
|
|
||||||
static usbtmc_interface_state_t usbtmc_state =
|
static usbtmc_interface_state_t usbtmc_state =
|
||||||
{
|
{
|
||||||
.state = STATE_IDLE,
|
|
||||||
.itf_id = 0xFF,
|
.itf_id = 0xFF,
|
||||||
.ep_bulk_in = 0,
|
|
||||||
.ep_bulk_out = 0,
|
|
||||||
.ep_int_in = 0
|
|
||||||
};
|
};
|
||||||
|
|
||||||
// We need all headers to fit in a single packet in this implementation.
|
// We need all headers to fit in a single packet in this implementation.
|
||||||
@ -147,6 +149,9 @@ TU_VERIFY_STATIC(
|
|||||||
static bool handle_devMsgOutStart(uint8_t rhport, void *data, size_t len);
|
static bool handle_devMsgOutStart(uint8_t rhport, void *data, size_t len);
|
||||||
static bool handle_devMsgOut(uint8_t rhport, void *data, size_t len, size_t packetLen);
|
static bool handle_devMsgOut(uint8_t rhport, void *data, size_t len, size_t packetLen);
|
||||||
|
|
||||||
|
static uint8_t termChar;
|
||||||
|
static uint8_t termCharRequested = false;
|
||||||
|
|
||||||
|
|
||||||
osal_mutex_def_t usbtmcLockBuffer;
|
osal_mutex_def_t usbtmcLockBuffer;
|
||||||
static osal_mutex_t usbtmcLock;
|
static osal_mutex_t usbtmcLock;
|
||||||
@ -155,6 +160,23 @@ static osal_mutex_t usbtmcLock;
|
|||||||
#define criticalEnter() do {osal_mutex_lock(usbtmcLock,OSAL_TIMEOUT_WAIT_FOREVER); } while (0)
|
#define criticalEnter() do {osal_mutex_lock(usbtmcLock,OSAL_TIMEOUT_WAIT_FOREVER); } while (0)
|
||||||
#define criticalLeave() do {osal_mutex_unlock(usbtmcLock); } while (0)
|
#define criticalLeave() do {osal_mutex_unlock(usbtmcLock); } while (0)
|
||||||
|
|
||||||
|
bool atomicChangeState(usbtmcd_state_enum expectedState, usbtmcd_state_enum newState)
|
||||||
|
{
|
||||||
|
bool ret = true;
|
||||||
|
criticalEnter();
|
||||||
|
usbtmcd_state_enum oldState = usbtmc_state.state;
|
||||||
|
if (oldState == expectedState)
|
||||||
|
{
|
||||||
|
usbtmc_state.state = newState;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
ret = false;
|
||||||
|
}
|
||||||
|
criticalLeave();
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
// called from app
|
// called from app
|
||||||
// We keep a reference to the buffer, so it MUST not change until the app is
|
// We keep a reference to the buffer, so it MUST not change until the app is
|
||||||
// notified that the transfer is complete.
|
// notified that the transfer is complete.
|
||||||
@ -165,6 +187,7 @@ static osal_mutex_t usbtmcLock;
|
|||||||
bool usbtmcd_transmit_dev_msg_data(
|
bool usbtmcd_transmit_dev_msg_data(
|
||||||
uint8_t rhport,
|
uint8_t rhport,
|
||||||
const void * data, size_t len,
|
const void * data, size_t len,
|
||||||
|
bool endOfMessage,
|
||||||
bool usingTermChar)
|
bool usingTermChar)
|
||||||
{
|
{
|
||||||
const unsigned int txBufLen = sizeof(usbtmc_state.ep_bulk_in_buf);
|
const unsigned int txBufLen = sizeof(usbtmc_state.ep_bulk_in_buf);
|
||||||
@ -172,54 +195,39 @@ bool usbtmcd_transmit_dev_msg_data(
|
|||||||
#ifndef NDEBUG
|
#ifndef NDEBUG
|
||||||
TU_ASSERT(len > 0u);
|
TU_ASSERT(len > 0u);
|
||||||
TU_ASSERT(len <= usbtmc_state.transfer_size_remaining);
|
TU_ASSERT(len <= usbtmc_state.transfer_size_remaining);
|
||||||
|
TU_ASSERT(usbtmc_state.transfer_size_sent == 0u);
|
||||||
if(usingTermChar)
|
if(usingTermChar)
|
||||||
{
|
{
|
||||||
TU_ASSERT(tud_usbtmc_app_capabilities.bmDevCapabilities.canEndBulkInOnTermChar);
|
TU_ASSERT(tud_usbtmc_app_capabilities.bmDevCapabilities.canEndBulkInOnTermChar);
|
||||||
TU_ASSERT(termCharRequested);
|
TU_ASSERT(termCharRequested);
|
||||||
TU_ASSERT(((uint8_t*)data)[len-1] == termChar);
|
TU_ASSERT(((uint8_t*)data)[len-1u] == termChar);
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
TU_VERIFY(usbtmc_state.state == STATE_TX_REQUESTED);
|
TU_VERIFY(usbtmc_state.state == STATE_TX_REQUESTED);
|
||||||
usbtmc_msg_dev_dep_msg_in_header_t *hdr = (usbtmc_msg_dev_dep_msg_in_header_t*)usbtmc_state.ep_bulk_in_buf;
|
usbtmc_msg_dev_dep_msg_in_header_t *hdr = (usbtmc_msg_dev_dep_msg_in_header_t*)usbtmc_state.ep_bulk_in_buf;
|
||||||
memset(hdr, 0x00, sizeof(*hdr));
|
tu_varclr(hdr);
|
||||||
hdr->header.MsgID = USBTMC_MSGID_DEV_DEP_MSG_IN;
|
hdr->header.MsgID = USBTMC_MSGID_DEV_DEP_MSG_IN;
|
||||||
hdr->header.bTag = usbtmc_state.lastBulkInTag;
|
hdr->header.bTag = usbtmc_state.lastBulkInTag;
|
||||||
hdr->header.bTagInverse = (uint8_t)~(usbtmc_state.lastBulkInTag);
|
hdr->header.bTagInverse = (uint8_t)~(usbtmc_state.lastBulkInTag);
|
||||||
hdr->TransferSize = len;
|
hdr->TransferSize = len;
|
||||||
hdr->bmTransferAttributes.EOM = 1u;
|
hdr->bmTransferAttributes.EOM = endOfMessage;
|
||||||
hdr->bmTransferAttributes.UsingTermChar = usingTermChar;
|
hdr->bmTransferAttributes.UsingTermChar = usingTermChar;
|
||||||
|
|
||||||
// Copy in the header
|
// Copy in the header
|
||||||
size_t packetLen = sizeof(*hdr);
|
const size_t headerLen = sizeof(*hdr);
|
||||||
|
const size_t dataLen = ((headerLen + hdr->TransferSize) <= txBufLen) ?
|
||||||
|
len : (txBufLen - headerLen);
|
||||||
|
const size_t packetLen = headerLen + dataLen;
|
||||||
|
|
||||||
// If it fits in a single trasnmission:
|
memcpy((uint8_t*)(usbtmc_state.ep_bulk_in_buf) + headerLen, data, dataLen);
|
||||||
if((packetLen + hdr->TransferSize) <= txBufLen)
|
usbtmc_state.transfer_size_remaining = len - dataLen;
|
||||||
{
|
usbtmc_state.transfer_size_sent = dataLen;
|
||||||
memcpy((uint8_t*)(usbtmc_state.ep_bulk_in_buf) + packetLen, data, hdr->TransferSize);
|
usbtmc_state.devInBuffer = (uint8_t*)data + (dataLen);
|
||||||
packetLen = (uint16_t)(packetLen + hdr->TransferSize);
|
|
||||||
usbtmc_state.transfer_size_remaining = 0;
|
|
||||||
usbtmc_state.transfer_size_sent = len;
|
|
||||||
usbtmc_state.devInBuffer = NULL;
|
|
||||||
}
|
|
||||||
else /* partial packet */
|
|
||||||
{
|
|
||||||
memcpy((uint8_t*)(usbtmc_state.ep_bulk_in_buf) + packetLen, data, txBufLen - packetLen);
|
|
||||||
usbtmc_state.devInBuffer = (uint8_t*)data + (txBufLen - packetLen);
|
|
||||||
usbtmc_state.transfer_size_remaining = len - (txBufLen - packetLen);
|
|
||||||
usbtmc_state.transfer_size_sent = txBufLen - packetLen;
|
|
||||||
packetLen = txBufLen;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
criticalEnter();
|
|
||||||
{
|
|
||||||
TU_VERIFY(usbtmc_state.state == STATE_TX_REQUESTED);
|
|
||||||
// We used packetlen as a max, not the buffer size, so this is OK here, no need for modulus
|
|
||||||
usbtmc_state.state = (packetLen >= txBufLen) ? STATE_TX_INITIATED : STATE_TX_SHORTED;
|
|
||||||
}
|
|
||||||
criticalLeave();
|
|
||||||
|
|
||||||
|
bool stateChanged =
|
||||||
|
atomicChangeState(STATE_TX_REQUESTED, (packetLen >= txBufLen) ? STATE_TX_INITIATED : STATE_TX_SHORTED);
|
||||||
|
TU_VERIFY(stateChanged);
|
||||||
TU_VERIFY(usbd_edpt_xfer(rhport, usbtmc_state.ep_bulk_in, usbtmc_state.ep_bulk_in_buf, (uint16_t)packetLen));
|
TU_VERIFY(usbd_edpt_xfer(rhport, usbtmc_state.ep_bulk_in, usbtmc_state.ep_bulk_in_buf, (uint16_t)packetLen));
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
@ -244,20 +252,16 @@ void usbtmcd_init_cb(void)
|
|||||||
bool usbtmcd_open_cb(uint8_t rhport, tusb_desc_interface_t const * itf_desc, uint16_t *p_length)
|
bool usbtmcd_open_cb(uint8_t rhport, tusb_desc_interface_t const * itf_desc, uint16_t *p_length)
|
||||||
{
|
{
|
||||||
(void)rhport;
|
(void)rhport;
|
||||||
|
TU_ASSERT(usbtmc_state.state == STATE_CLOSED);
|
||||||
uint8_t const * p_desc;
|
uint8_t const * p_desc;
|
||||||
uint8_t found_endpoints = 0;
|
uint8_t found_endpoints = 0;
|
||||||
|
|
||||||
|
#ifndef NDEBUG
|
||||||
usbtmcd_reset_cb(rhport);
|
TU_ASSERT(itf_desc->bInterfaceClass == TUD_USBTMC_APP_CLASS);
|
||||||
|
TU_ASSERT(itf_desc->bInterfaceSubClass == TUD_USBTMC_APP_SUBCLASS);
|
||||||
// Perhaps there are other application specific class drivers, so don't assert here.
|
|
||||||
if( itf_desc->bInterfaceClass != TUD_USBTMC_APP_CLASS)
|
|
||||||
return false;
|
|
||||||
if( itf_desc->bInterfaceSubClass != TUD_USBTMC_APP_SUBCLASS)
|
|
||||||
return false;
|
|
||||||
|
|
||||||
// Only 2 or 3 endpoints are allowed for USBTMC.
|
// Only 2 or 3 endpoints are allowed for USBTMC.
|
||||||
TU_ASSERT((itf_desc->bNumEndpoints == 2) || (itf_desc->bNumEndpoints ==3));
|
TU_ASSERT((itf_desc->bNumEndpoints == 2) || (itf_desc->bNumEndpoints ==3));
|
||||||
|
#endif
|
||||||
|
|
||||||
// Interface
|
// Interface
|
||||||
(*p_length) = 0u;
|
(*p_length) = 0u;
|
||||||
@ -318,28 +322,27 @@ bool usbtmcd_open_cb(uint8_t rhport, tusb_desc_interface_t const * itf_desc, uin
|
|||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
#endif
|
#endif
|
||||||
|
usbtmc_state.state = STATE_IDLE;
|
||||||
TU_VERIFY( usbd_edpt_xfer(rhport, usbtmc_state.ep_bulk_out, usbtmc_state.ep_bulk_out_buf, 64));
|
TU_VERIFY( usbd_edpt_xfer(rhport, usbtmc_state.ep_bulk_out, usbtmc_state.ep_bulk_out_buf, 64));
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
void usbtmcd_reset_cb(uint8_t rhport)
|
void usbtmcd_reset_cb(uint8_t rhport)
|
||||||
{
|
{
|
||||||
// FIXME: Do endpoints need to be closed here?
|
|
||||||
usbtmc_state.state = STATE_IDLE;
|
|
||||||
usbtmc_state.itf_id = 0xFF;
|
|
||||||
usbtmc_state.ep_bulk_in = 0;
|
|
||||||
usbtmc_state.ep_bulk_out = 0;
|
|
||||||
usbtmc_state.ep_int_in = 0;
|
|
||||||
usbtmc_state.lastBulkInTag = 0;
|
|
||||||
usbtmc_state.lastBulkOutTag = 0;
|
|
||||||
|
|
||||||
(void)rhport;
|
(void)rhport;
|
||||||
|
|
||||||
|
criticalEnter();
|
||||||
|
tu_varclr(&usbtmc_state);
|
||||||
|
usbtmc_state.itf_id = 0xFFu;
|
||||||
|
criticalLeave();
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool handle_devMsgOutStart(uint8_t rhport, void *data, size_t len)
|
static bool handle_devMsgOutStart(uint8_t rhport, void *data, size_t len)
|
||||||
{
|
{
|
||||||
(void)rhport;
|
(void)rhport;
|
||||||
TU_VERIFY(usbtmc_state.state == STATE_IDLE);
|
bool stateChanged = atomicChangeState(STATE_IDLE, STATE_RCV);
|
||||||
|
TU_VERIFY(stateChanged);
|
||||||
|
|
||||||
// must be a header, should have been confirmed before calling here.
|
// must be a header, should have been confirmed before calling here.
|
||||||
usbtmc_msg_request_dev_dep_out *msg = (usbtmc_msg_request_dev_dep_out*)data;
|
usbtmc_msg_request_dev_dep_out *msg = (usbtmc_msg_request_dev_dep_out*)data;
|
||||||
usbtmc_state.transfer_size_remaining = msg->TransferSize;
|
usbtmc_state.transfer_size_remaining = msg->TransferSize;
|
||||||
@ -352,6 +355,9 @@ static bool handle_devMsgOutStart(uint8_t rhport, void *data, size_t len)
|
|||||||
static bool handle_devMsgOut(uint8_t rhport, void *data, size_t len, size_t packetLen)
|
static bool handle_devMsgOut(uint8_t rhport, void *data, size_t len, size_t packetLen)
|
||||||
{
|
{
|
||||||
(void)rhport;
|
(void)rhport;
|
||||||
|
|
||||||
|
TU_VERIFY(usbtmc_state.state == STATE_RCV);
|
||||||
|
|
||||||
bool shortPacket = (packetLen < USBTMCD_MAX_PACKET_SIZE);
|
bool shortPacket = (packetLen < USBTMCD_MAX_PACKET_SIZE);
|
||||||
|
|
||||||
// Packet is to be considered complete when we get enough data or at a short packet.
|
// Packet is to be considered complete when we get enough data or at a short packet.
|
||||||
@ -360,18 +366,14 @@ static bool handle_devMsgOut(uint8_t rhport, void *data, size_t len, size_t pack
|
|||||||
atEnd = true;
|
atEnd = true;
|
||||||
if(len > usbtmc_state.transfer_size_remaining)
|
if(len > usbtmc_state.transfer_size_remaining)
|
||||||
len = usbtmc_state.transfer_size_remaining;
|
len = usbtmc_state.transfer_size_remaining;
|
||||||
tud_usbtmc_app_msg_data_cb(rhport,data, len, atEnd);
|
TU_VERIFY(tud_usbtmc_app_msg_data_cb(rhport,data, len, atEnd));
|
||||||
|
// TODO: Go to an error state upon failure other than just stalling the EP?
|
||||||
|
|
||||||
usbtmc_state.transfer_size_remaining -= len;
|
usbtmc_state.transfer_size_remaining -= len;
|
||||||
usbtmc_state.transfer_size_sent += len;
|
usbtmc_state.transfer_size_sent += len;
|
||||||
if(atEnd)
|
bool stateChanged = atomicChangeState(STATE_RCV, atEnd ? STATE_IDLE : STATE_RCV);
|
||||||
{
|
TU_VERIFY(stateChanged);
|
||||||
usbtmc_state.state = STATE_IDLE;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
usbtmc_state.state = STATE_RCV;
|
|
||||||
}
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -379,16 +381,11 @@ static bool handle_devMsgIn(uint8_t rhport, void *data, size_t len)
|
|||||||
{
|
{
|
||||||
TU_VERIFY(len == sizeof(usbtmc_msg_request_dev_dep_in));
|
TU_VERIFY(len == sizeof(usbtmc_msg_request_dev_dep_in));
|
||||||
usbtmc_msg_request_dev_dep_in *msg = (usbtmc_msg_request_dev_dep_in*)data;
|
usbtmc_msg_request_dev_dep_in *msg = (usbtmc_msg_request_dev_dep_in*)data;
|
||||||
|
bool stateChanged = atomicChangeState(STATE_IDLE, STATE_TX_REQUESTED);
|
||||||
criticalEnter();
|
TU_VERIFY(stateChanged);
|
||||||
{
|
|
||||||
TU_VERIFY(usbtmc_state.state == STATE_IDLE);
|
|
||||||
usbtmc_state.state = STATE_TX_REQUESTED;
|
|
||||||
usbtmc_state.lastBulkInTag = msg->header.bTag;
|
usbtmc_state.lastBulkInTag = msg->header.bTag;
|
||||||
usbtmc_state.transfer_size_remaining = msg->TransferSize;
|
usbtmc_state.transfer_size_remaining = msg->TransferSize;
|
||||||
usbtmc_state.transfer_size_sent = 0u;
|
usbtmc_state.transfer_size_sent = 0u;
|
||||||
}
|
|
||||||
criticalLeave();
|
|
||||||
|
|
||||||
termCharRequested = msg->bmTransferAttributes.TermCharEnabled;
|
termCharRequested = msg->bmTransferAttributes.TermCharEnabled;
|
||||||
termChar = msg->TermChar;
|
termChar = msg->TermChar;
|
||||||
@ -403,7 +400,7 @@ static bool handle_devMsgIn(uint8_t rhport, void *data, size_t len)
|
|||||||
bool usbtmcd_xfer_cb(uint8_t rhport, uint8_t ep_addr, xfer_result_t result, uint32_t xferred_bytes)
|
bool usbtmcd_xfer_cb(uint8_t rhport, uint8_t ep_addr, xfer_result_t result, uint32_t xferred_bytes)
|
||||||
{
|
{
|
||||||
TU_VERIFY(result == XFER_RESULT_SUCCESS);
|
TU_VERIFY(result == XFER_RESULT_SUCCESS);
|
||||||
|
//uart_tx_str_sync("TMC XFER CB\r\n");
|
||||||
if(usbtmc_state.state == STATE_CLEARING) {
|
if(usbtmc_state.state == STATE_CLEARING) {
|
||||||
return true; /* I think we can ignore everything here */
|
return true; /* I think we can ignore everything here */
|
||||||
}
|
}
|
||||||
@ -457,7 +454,7 @@ bool usbtmcd_xfer_cb(uint8_t rhport, uint8_t ep_addr, xfer_result_t result, uint
|
|||||||
|
|
||||||
case STATE_ABORTING_BULK_OUT:
|
case STATE_ABORTING_BULK_OUT:
|
||||||
TU_VERIFY(false);
|
TU_VERIFY(false);
|
||||||
return false; // Should be stalled by now...
|
return false; // Should be stalled by now, shouldn't have received a packet.
|
||||||
case STATE_TX_REQUESTED:
|
case STATE_TX_REQUESTED:
|
||||||
case STATE_TX_INITIATED:
|
case STATE_TX_INITIATED:
|
||||||
case STATE_ABORTING_BULK_IN:
|
case STATE_ABORTING_BULK_IN:
|
||||||
@ -534,19 +531,26 @@ bool usbtmcd_control_request_cb(uint8_t rhport, tusb_control_request_t const * r
|
|||||||
(request->bRequest == TUSB_REQ_CLEAR_FEATURE) &&
|
(request->bRequest == TUSB_REQ_CLEAR_FEATURE) &&
|
||||||
(request->wValue == TUSB_REQ_FEATURE_EDPT_HALT))
|
(request->wValue == TUSB_REQ_FEATURE_EDPT_HALT))
|
||||||
{
|
{
|
||||||
if((request->wIndex) == usbtmc_state.ep_bulk_out)
|
uint32_t ep_addr = (request->wIndex);
|
||||||
|
|
||||||
|
if(ep_addr == usbtmc_state.ep_bulk_out)
|
||||||
{
|
{
|
||||||
usmtmcd_app_bulkOut_clearFeature_cb(rhport);
|
usmtmcd_app_bulkOut_clearFeature_cb(rhport);
|
||||||
|
// And start a new OUT xfer request now that things are clear
|
||||||
|
TU_ASSERT( usbd_edpt_xfer(rhport, usbtmc_state.ep_bulk_out, usbtmc_state.ep_bulk_out_buf, 64));
|
||||||
}
|
}
|
||||||
else if ((request->wIndex) == usbtmc_state.ep_bulk_in)
|
else if (ep_addr == usbtmc_state.ep_bulk_in)
|
||||||
{
|
{
|
||||||
usmtmcd_app_bulkIn_clearFeature_cb(rhport);
|
usmtmcd_app_bulkIn_clearFeature_cb(rhport);
|
||||||
}
|
}
|
||||||
return false; // We want USBD core to handle sending the status response, and clear the stall condition
|
else
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
// We only handle class requests, IN direction.
|
// Otherwise, we only handle class requests.
|
||||||
// (for now)
|
|
||||||
if(request->bmRequestType_bit.type != TUSB_REQ_TYPE_CLASS)
|
if(request->bmRequestType_bit.type != TUSB_REQ_TYPE_CLASS)
|
||||||
{
|
{
|
||||||
return false;
|
return false;
|
||||||
@ -571,7 +575,7 @@ bool usbtmcd_control_request_cb(uint8_t rhport, tusb_control_request_t const * r
|
|||||||
{
|
{
|
||||||
rsp.USBTMC_status = USBTMC_STATUS_FAILED;
|
rsp.USBTMC_status = USBTMC_STATUS_FAILED;
|
||||||
}
|
}
|
||||||
else if(usbtmc_state.lastBulkOutTag == (request->wValue & 0xf7u))
|
else if(usbtmc_state.lastBulkOutTag == (request->wValue & 0x7Fu))
|
||||||
{
|
{
|
||||||
rsp.USBTMC_status = USBTMC_STATUS_TRANSFER_NOT_IN_PROGRESS;
|
rsp.USBTMC_status = USBTMC_STATUS_TRANSFER_NOT_IN_PROGRESS;
|
||||||
}
|
}
|
||||||
@ -579,7 +583,9 @@ bool usbtmcd_control_request_cb(uint8_t rhport, tusb_control_request_t const * r
|
|||||||
{
|
{
|
||||||
rsp.USBTMC_status = USBTMC_STATUS_SUCCESS;
|
rsp.USBTMC_status = USBTMC_STATUS_SUCCESS;
|
||||||
// Check if we've queued a short packet
|
// Check if we've queued a short packet
|
||||||
|
criticalEnter();
|
||||||
usbtmc_state.state = STATE_ABORTING_BULK_OUT;
|
usbtmc_state.state = STATE_ABORTING_BULK_OUT;
|
||||||
|
criticalLeave();
|
||||||
TU_VERIFY(tud_usbtmc_app_initiate_abort_bulk_out_cb(rhport, &(rsp.USBTMC_status)));
|
TU_VERIFY(tud_usbtmc_app_initiate_abort_bulk_out_cb(rhport, &(rsp.USBTMC_status)));
|
||||||
usbd_edpt_stall(rhport, usbtmc_state.ep_bulk_out);
|
usbd_edpt_stall(rhport, usbtmc_state.ep_bulk_out);
|
||||||
}
|
}
|
||||||
@ -610,13 +616,15 @@ bool usbtmcd_control_request_cb(uint8_t rhport, tusb_control_request_t const * r
|
|||||||
TU_VERIFY(request->wIndex == usbtmc_state.ep_bulk_in);
|
TU_VERIFY(request->wIndex == usbtmc_state.ep_bulk_in);
|
||||||
// wValue is the requested bTag to abort
|
// wValue is the requested bTag to abort
|
||||||
if((usbtmc_state.state == STATE_TX_REQUESTED || usbtmc_state.state == STATE_TX_INITIATED) &&
|
if((usbtmc_state.state == STATE_TX_REQUESTED || usbtmc_state.state == STATE_TX_INITIATED) &&
|
||||||
usbtmc_state.lastBulkInTag == (request->wValue & 0xf7u))
|
usbtmc_state.lastBulkInTag == (request->wValue & 0x7Fu))
|
||||||
{
|
{
|
||||||
rsp.USBTMC_status = USBTMC_STATUS_SUCCESS;
|
rsp.USBTMC_status = USBTMC_STATUS_SUCCESS;
|
||||||
usbtmc_state.transfer_size_remaining = 0;
|
usbtmc_state.transfer_size_remaining = 0u;
|
||||||
// Check if we've queued a short packet
|
// Check if we've queued a short packet
|
||||||
|
criticalEnter();
|
||||||
usbtmc_state.state = ((usbtmc_state.transfer_size_sent % USBTMCD_MAX_PACKET_SIZE) == 0) ?
|
usbtmc_state.state = ((usbtmc_state.transfer_size_sent % USBTMCD_MAX_PACKET_SIZE) == 0) ?
|
||||||
STATE_ABORTING_BULK_IN : STATE_ABORTING_BULK_IN_SHORTED;
|
STATE_ABORTING_BULK_IN : STATE_ABORTING_BULK_IN_SHORTED;
|
||||||
|
criticalLeave();
|
||||||
if(usbtmc_state.transfer_size_sent == 0)
|
if(usbtmc_state.transfer_size_sent == 0)
|
||||||
{
|
{
|
||||||
// Send short packet, nothing is in the buffer yet
|
// Send short packet, nothing is in the buffer yet
|
||||||
@ -652,6 +660,7 @@ bool usbtmcd_control_request_cb(uint8_t rhport, tusb_control_request_t const * r
|
|||||||
.NBYTES_RXD_TXD = usbtmc_state.transfer_size_sent,
|
.NBYTES_RXD_TXD = usbtmc_state.transfer_size_sent,
|
||||||
};
|
};
|
||||||
TU_VERIFY(tud_usbtmc_app_check_abort_bulk_in_cb(rhport, &rsp));
|
TU_VERIFY(tud_usbtmc_app_check_abort_bulk_in_cb(rhport, &rsp));
|
||||||
|
criticalEnter();
|
||||||
switch(usbtmc_state.state)
|
switch(usbtmc_state.state)
|
||||||
{
|
{
|
||||||
case STATE_ABORTING_BULK_IN_ABORTED:
|
case STATE_ABORTING_BULK_IN_ABORTED:
|
||||||
@ -665,6 +674,7 @@ bool usbtmcd_control_request_cb(uint8_t rhport, tusb_control_request_t const * r
|
|||||||
default:
|
default:
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
criticalLeave();
|
||||||
TU_VERIFY(tud_control_xfer(rhport, request, (void*)&rsp,sizeof(rsp)));
|
TU_VERIFY(tud_control_xfer(rhport, request, (void*)&rsp,sizeof(rsp)));
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
@ -678,7 +688,9 @@ bool usbtmcd_control_request_cb(uint8_t rhport, tusb_control_request_t const * r
|
|||||||
// control endpoint response shown in Table 31, and clear all input buffers and output buffers.
|
// control endpoint response shown in Table 31, and clear all input buffers and output buffers.
|
||||||
usbd_edpt_stall(rhport, usbtmc_state.ep_bulk_out);
|
usbd_edpt_stall(rhport, usbtmc_state.ep_bulk_out);
|
||||||
usbtmc_state.transfer_size_remaining = 0;
|
usbtmc_state.transfer_size_remaining = 0;
|
||||||
|
criticalEnter();
|
||||||
usbtmc_state.state = STATE_CLEARING;
|
usbtmc_state.state = STATE_CLEARING;
|
||||||
|
criticalLeave();
|
||||||
TU_VERIFY(tud_usbtmc_app_initiate_clear_cb(rhport, &tmcStatusCode));
|
TU_VERIFY(tud_usbtmc_app_initiate_clear_cb(rhport, &tmcStatusCode));
|
||||||
TU_VERIFY(tud_control_xfer(rhport, request, (void*)&tmcStatusCode,sizeof(tmcStatusCode)));
|
TU_VERIFY(tud_control_xfer(rhport, request, (void*)&tmcStatusCode,sizeof(tmcStatusCode)));
|
||||||
return true;
|
return true;
|
||||||
@ -702,7 +714,11 @@ bool usbtmcd_control_request_cb(uint8_t rhport, tusb_control_request_t const * r
|
|||||||
TU_VERIFY(tud_usbtmc_app_check_clear_cb(rhport, &clearStatusRsp));
|
TU_VERIFY(tud_usbtmc_app_check_clear_cb(rhport, &clearStatusRsp));
|
||||||
}
|
}
|
||||||
if(clearStatusRsp.USBTMC_status == USBTMC_STATUS_SUCCESS)
|
if(clearStatusRsp.USBTMC_status == USBTMC_STATUS_SUCCESS)
|
||||||
|
{
|
||||||
|
criticalEnter();
|
||||||
usbtmc_state.state = STATE_IDLE;
|
usbtmc_state.state = STATE_IDLE;
|
||||||
|
criticalLeave();
|
||||||
|
}
|
||||||
TU_VERIFY(tud_control_xfer(rhport, request, (void*)&clearStatusRsp,sizeof(clearStatusRsp)));
|
TU_VERIFY(tud_control_xfer(rhport, request, (void*)&clearStatusRsp,sizeof(clearStatusRsp)));
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
@ -785,7 +801,7 @@ bool usbtmcd_control_complete_cb(uint8_t rhport, tusb_control_request_t const *
|
|||||||
{
|
{
|
||||||
(void)rhport;
|
(void)rhport;
|
||||||
//------------- Class Specific Request -------------//
|
//------------- Class Specific Request -------------//
|
||||||
TU_VERIFY (request->bmRequestType_bit.type == TUSB_REQ_TYPE_CLASS);
|
TU_ASSERT (request->bmRequestType_bit.type == TUSB_REQ_TYPE_CLASS);
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
@ -91,7 +91,7 @@ TU_ATTR_WEAK bool tud_usbtmc_app_msg_trigger_cb(uint8_t rhport, usbtmc_msg_gener
|
|||||||
bool usbtmcd_transmit_dev_msg_data(
|
bool usbtmcd_transmit_dev_msg_data(
|
||||||
uint8_t rhport,
|
uint8_t rhport,
|
||||||
const void * data, size_t len,
|
const void * data, size_t len,
|
||||||
bool usingTermChar);
|
bool endOfMessage, bool usingTermChar);
|
||||||
|
|
||||||
|
|
||||||
/* "callbacks" from USB device core */
|
/* "callbacks" from USB device core */
|
||||||
|
Loading…
x
Reference in New Issue
Block a user