mirror of
https://github.com/betaflight/betaflight.git
synced 2025-07-19 14:25:20 +03:00
serial: add writeBuf() and implement for USB.
This lets USB send up to 32 bytes in a frame instead of 32 one byte frames. Add a fallback for drivers that don't implement writeBuf(). Signed-off-by: Michael Hope <mlhx@google.com>
This commit is contained in:
parent
75c86825f7
commit
96306ff9ab
6 changed files with 36 additions and 61 deletions
|
@ -40,6 +40,17 @@ void serialWrite(serialPort_t *instance, uint8_t ch)
|
||||||
instance->vTable->serialWrite(instance, ch);
|
instance->vTable->serialWrite(instance, ch);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void serialWriteBuf(serialPort_t *instance, void *data, int count)
|
||||||
|
{
|
||||||
|
if (instance->vTable->writeBuf) {
|
||||||
|
instance->vTable->writeBuf(instance, data, count);
|
||||||
|
} else {
|
||||||
|
for (uint8_t *p = data; count > 0; count--, p++) {
|
||||||
|
serialWrite(instance, *p);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
uint8_t serialRxBytesWaiting(serialPort_t *instance)
|
uint8_t serialRxBytesWaiting(serialPort_t *instance)
|
||||||
{
|
{
|
||||||
return instance->vTable->serialTotalRxWaiting(instance);
|
return instance->vTable->serialTotalRxWaiting(instance);
|
||||||
|
@ -70,14 +81,7 @@ void serialSetMode(serialPort_t *instance, portMode_t mode)
|
||||||
instance->vTable->setMode(instance, mode);
|
instance->vTable->setMode(instance, mode);
|
||||||
}
|
}
|
||||||
|
|
||||||
void serialBeginWrite(serialPort_t *instance)
|
void serialWriteBufShim(void *instance, void *data, int count)
|
||||||
{
|
{
|
||||||
if (instance->vTable->beginWrite)
|
serialWriteBuf((serialPort_t *)instance, data, count);
|
||||||
instance->vTable->beginWrite(instance);
|
|
||||||
}
|
|
||||||
|
|
||||||
void serialEndWrite(serialPort_t *instance)
|
|
||||||
{
|
|
||||||
if (instance->vTable->endWrite)
|
|
||||||
instance->vTable->endWrite(instance);
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -74,14 +74,13 @@ struct serialPortVTable {
|
||||||
|
|
||||||
void (*setMode)(serialPort_t *instance, portMode_t mode);
|
void (*setMode)(serialPort_t *instance, portMode_t mode);
|
||||||
|
|
||||||
// Optional functions used to buffer large writes.
|
void (*writeBuf)(serialPort_t *instance, void *data, int count);
|
||||||
void (*beginWrite)(serialPort_t *instance);
|
|
||||||
void (*endWrite)(serialPort_t *instance);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
void serialWrite(serialPort_t *instance, uint8_t ch);
|
void serialWrite(serialPort_t *instance, uint8_t ch);
|
||||||
uint8_t serialRxBytesWaiting(serialPort_t *instance);
|
uint8_t serialRxBytesWaiting(serialPort_t *instance);
|
||||||
uint8_t serialTxBytesFree(serialPort_t *instance);
|
uint8_t serialTxBytesFree(serialPort_t *instance);
|
||||||
|
void serialWriteBuf(serialPort_t *instance, void *data, int count);
|
||||||
uint8_t serialRead(serialPort_t *instance);
|
uint8_t serialRead(serialPort_t *instance);
|
||||||
void serialSetBaudRate(serialPort_t *instance, uint32_t baudRate);
|
void serialSetBaudRate(serialPort_t *instance, uint32_t baudRate);
|
||||||
void serialSetMode(serialPort_t *instance, portMode_t mode);
|
void serialSetMode(serialPort_t *instance, portMode_t mode);
|
||||||
|
@ -89,5 +88,5 @@ bool isSerialTransmitBufferEmpty(serialPort_t *instance);
|
||||||
void serialPrint(serialPort_t *instance, const char *str);
|
void serialPrint(serialPort_t *instance, const char *str);
|
||||||
uint32_t serialGetBaudRate(serialPort_t *instance);
|
uint32_t serialGetBaudRate(serialPort_t *instance);
|
||||||
|
|
||||||
void serialBeginWrite(serialPort_t *instance);
|
// A shim that adapts the bufWriter API to the serialWriteBuf() API.
|
||||||
void serialEndWrite(serialPort_t *instance);
|
void serialWriteBufShim(void *instance, void *data, int count);
|
||||||
|
|
|
@ -479,8 +479,7 @@ const struct serialPortVTable softSerialVTable[] = {
|
||||||
softSerialSetBaudRate,
|
softSerialSetBaudRate,
|
||||||
isSoftSerialTransmitBufferEmpty,
|
isSoftSerialTransmitBufferEmpty,
|
||||||
softSerialSetMode,
|
softSerialSetMode,
|
||||||
.beginWrite = NULL,
|
.writeBuf = NULL,
|
||||||
.endWrite = NULL,
|
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -322,7 +322,6 @@ const struct serialPortVTable uartVTable[] = {
|
||||||
uartSetBaudRate,
|
uartSetBaudRate,
|
||||||
isUartTransmitBufferEmpty,
|
isUartTransmitBufferEmpty,
|
||||||
uartSetMode,
|
uartSetMode,
|
||||||
.beginWrite = NULL,
|
.writeBuf = NULL,
|
||||||
.endWrite = NULL,
|
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
|
@ -27,7 +27,6 @@
|
||||||
#include "usb_core.h"
|
#include "usb_core.h"
|
||||||
#include "usb_init.h"
|
#include "usb_init.h"
|
||||||
#include "hw_config.h"
|
#include "hw_config.h"
|
||||||
#include "common/utils.h"
|
|
||||||
|
|
||||||
#include "drivers/system.h"
|
#include "drivers/system.h"
|
||||||
|
|
||||||
|
@ -83,49 +82,30 @@ static uint8_t usbVcpRead(serialPort_t *instance)
|
||||||
return buf[0];
|
return buf[0];
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool usbVcpFlush(vcpPort_t *port)
|
static void usbVcpWriteBuf(serialPort_t *instance, void *data, int count)
|
||||||
{
|
{
|
||||||
uint8_t count = port->txAt;
|
UNUSED(instance);
|
||||||
port->txAt = 0;
|
|
||||||
|
|
||||||
if (count == 0) {
|
|
||||||
return true;
|
if (!(usbIsConnected() && usbIsConfigured())) {
|
||||||
}
|
return;
|
||||||
if (!usbIsConnected() || !usbIsConfigured()) {
|
|
||||||
return false;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
uint32_t txed;
|
|
||||||
uint32_t start = millis();
|
uint32_t start = millis();
|
||||||
|
for (uint8_t *p = data; count > 0; ) {
|
||||||
|
uint32_t txed = CDC_Send_DATA(p, count);
|
||||||
|
count -= txed;
|
||||||
|
p += txed;
|
||||||
|
|
||||||
do {
|
if (millis() - start > USB_TIMEOUT) {
|
||||||
txed = CDC_Send_DATA(port->txBuf, count);
|
break;
|
||||||
} while (txed != count && (millis() - start < USB_TIMEOUT));
|
}
|
||||||
|
}
|
||||||
return txed == count;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void usbVcpWrite(serialPort_t *instance, uint8_t c)
|
static void usbVcpWrite(serialPort_t *instance, uint8_t c)
|
||||||
{
|
{
|
||||||
vcpPort_t *port = container_of(instance, vcpPort_t, port);
|
usbVcpWriteBuf(instance, &c, sizeof(c));
|
||||||
|
|
||||||
port->txBuf[port->txAt++] = c;
|
|
||||||
if (!port->buffering || port->txAt >= ARRAYLEN(port->txBuf)) {
|
|
||||||
usbVcpFlush(port);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static void usbVcpBeginWrite(serialPort_t *instance)
|
|
||||||
{
|
|
||||||
vcpPort_t *port = container_of(instance, vcpPort_t, port);
|
|
||||||
port->buffering = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void usbVcpEndWrite(serialPort_t *instance)
|
|
||||||
{
|
|
||||||
vcpPort_t *port = container_of(instance, vcpPort_t, port);
|
|
||||||
port->buffering = false;
|
|
||||||
usbVcpFlush(port);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
uint8_t usbTxBytesFree() {
|
uint8_t usbTxBytesFree() {
|
||||||
|
@ -133,7 +113,7 @@ uint8_t usbTxBytesFree() {
|
||||||
return 255;
|
return 255;
|
||||||
}
|
}
|
||||||
|
|
||||||
const struct serialPortVTable usbVTable[] = { { usbVcpWrite, usbVcpAvailable, usbTxBytesFree, usbVcpRead, usbVcpSetBaudRate, isUsbVcpTransmitBufferEmpty, usbVcpSetMode } };
|
const struct serialPortVTable usbVTable[] = { { usbVcpWrite, usbVcpAvailable, usbTxBytesFree, usbVcpRead, usbVcpSetBaudRate, isUsbVcpTransmitBufferEmpty, usbVcpSetMode, .writeBuf = usbVcpWriteBuf } };
|
||||||
|
|
||||||
serialPort_t *usbVcpOpen(void)
|
serialPort_t *usbVcpOpen(void)
|
||||||
{
|
{
|
||||||
|
|
|
@ -21,12 +21,6 @@
|
||||||
|
|
||||||
typedef struct {
|
typedef struct {
|
||||||
serialPort_t port;
|
serialPort_t port;
|
||||||
|
|
||||||
// Buffer used during bulk writes.
|
|
||||||
uint8_t txBuf[20];
|
|
||||||
uint8_t txAt;
|
|
||||||
// Set if the port is in bulk write mode and can buffer.
|
|
||||||
bool buffering;
|
|
||||||
} vcpPort_t;
|
} vcpPort_t;
|
||||||
|
|
||||||
serialPort_t *usbVcpOpen(void);
|
serialPort_t *usbVcpOpen(void);
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue