|
|
@ -4,11 +4,11 @@ |
|
|
|
// TinyFrame integration
|
|
|
|
// TinyFrame integration
|
|
|
|
//
|
|
|
|
//
|
|
|
|
|
|
|
|
|
|
|
|
#include <USB/usb_device.h> |
|
|
|
|
|
|
|
#include "platform.h" |
|
|
|
#include "platform.h" |
|
|
|
#include "task_main.h" |
|
|
|
#include "task_main.h" |
|
|
|
|
|
|
|
|
|
|
|
#include "USB/usbd_cdc_if.h" |
|
|
|
#include "USB/usbd_cdc_if.h" |
|
|
|
|
|
|
|
#include "USB/usb_device.h" |
|
|
|
#include "TinyFrame.h" |
|
|
|
#include "TinyFrame.h" |
|
|
|
|
|
|
|
|
|
|
|
extern osSemaphoreId semVcomTxReadyHandle; |
|
|
|
extern osSemaphoreId semVcomTxReadyHandle; |
|
|
@ -16,6 +16,30 @@ extern osMutexId mutTinyFrameTxHandle; |
|
|
|
|
|
|
|
|
|
|
|
void TF_WriteImpl(TinyFrame *tf, const uint8_t *buff, uint32_t len) |
|
|
|
void TF_WriteImpl(TinyFrame *tf, const uint8_t *buff, uint32_t len) |
|
|
|
{ |
|
|
|
{ |
|
|
|
|
|
|
|
#if 1 |
|
|
|
|
|
|
|
const uint32_t real_size = len; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Padding to a multiple of 64 bytes - this is supposed to maximize the bulk transfer speed
|
|
|
|
|
|
|
|
if (len&0x3F) { |
|
|
|
|
|
|
|
uint32_t pad = (64 - (len&0x3F)); |
|
|
|
|
|
|
|
memset((void *) (buff + len), 0, pad); |
|
|
|
|
|
|
|
len += pad; // padding to a multiple of 64 (size of the endpoint)
|
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// We bypass the USBD driver library's overhead by using the HAL function directly
|
|
|
|
|
|
|
|
assert_param(HAL_OK == HAL_PCD_EP_Transmit(hUsbDeviceFS.pData, CDC_IN_EP, (uint8_t *) buff, len)); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// The buffer is the TF transmit buffer, we can't leave it to work asynchronously because
|
|
|
|
|
|
|
|
// the next call could modify it before it's been transmitted (in the case of a chunked / multi-part frame)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// the assumption here is that all until the last chunk use the full buffer capacity
|
|
|
|
|
|
|
|
if (real_size == TF_SENDBUF_LEN) { |
|
|
|
|
|
|
|
if (pdTRUE != xSemaphoreTake(semVcomTxReadyHandle, 100)) { |
|
|
|
|
|
|
|
TF_Error("Tx stalled in WriteImpl"); |
|
|
|
|
|
|
|
return; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
#else |
|
|
|
(void) tf; |
|
|
|
(void) tf; |
|
|
|
#define CHUNK 64 // same as TF_SENDBUF_LEN, so we should always have only one run of the loop
|
|
|
|
#define CHUNK 64 // same as TF_SENDBUF_LEN, so we should always have only one run of the loop
|
|
|
|
int32_t total = (int32_t) len; |
|
|
|
int32_t total = (int32_t) len; |
|
|
@ -37,16 +61,26 @@ void TF_WriteImpl(TinyFrame *tf, const uint8_t *buff, uint32_t len) |
|
|
|
buff += chunksize; |
|
|
|
buff += chunksize; |
|
|
|
total -= chunksize; |
|
|
|
total -= chunksize; |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
#endif |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
/** Claim the TX interface before composing and sending a frame */ |
|
|
|
/** Claim the TX interface before composing and sending a frame */ |
|
|
|
bool TF_ClaimTx(TinyFrame *tf) |
|
|
|
bool TF_ClaimTx(TinyFrame *tf) |
|
|
|
{ |
|
|
|
{ |
|
|
|
(void) tf; |
|
|
|
(void) tf; |
|
|
|
// assert_param(osThreadGetId() != tskMainHandle);
|
|
|
|
// assert_param(!inIRQ()); // useless delay
|
|
|
|
assert_param(!inIRQ()); |
|
|
|
assert_param(pdTRUE == xSemaphoreTake(mutTinyFrameTxHandle, 5000)); // trips the wd
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// The last chunk from some previous frame may still be being transmitted,
|
|
|
|
|
|
|
|
// wait for it to finish (the semaphore is given in the CDC tx done handler)
|
|
|
|
|
|
|
|
if (pdTRUE != xSemaphoreTake(semVcomTxReadyHandle, 100)) { |
|
|
|
|
|
|
|
TF_Error("Tx stalled in Claim"); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// release the guarding mutex again
|
|
|
|
|
|
|
|
assert_param(pdTRUE == xSemaphoreGive(mutTinyFrameTxHandle)); |
|
|
|
|
|
|
|
return false; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
assert_param(osOK == osMutexWait(mutTinyFrameTxHandle, 5000)); |
|
|
|
|
|
|
|
return true; |
|
|
|
return true; |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
@ -54,5 +88,7 @@ bool TF_ClaimTx(TinyFrame *tf) |
|
|
|
void TF_ReleaseTx(TinyFrame *tf) |
|
|
|
void TF_ReleaseTx(TinyFrame *tf) |
|
|
|
{ |
|
|
|
{ |
|
|
|
(void) tf; |
|
|
|
(void) tf; |
|
|
|
assert_param(osOK == osMutexRelease(mutTinyFrameTxHandle)); |
|
|
|
assert_param(pdTRUE == xSemaphoreGive(mutTinyFrameTxHandle)); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// the last payload is sent asynchronously
|
|
|
|
} |
|
|
|
} |
|
|
|