1
0
Fork 0
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:
Michael Hope 2015-07-21 21:23:59 +02:00 committed by borisbstyle
parent 75c86825f7
commit 96306ff9ab
6 changed files with 36 additions and 61 deletions

View file

@ -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);
} }

View file

@ -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);

View file

@ -479,8 +479,7 @@ const struct serialPortVTable softSerialVTable[] = {
softSerialSetBaudRate, softSerialSetBaudRate,
isSoftSerialTransmitBufferEmpty, isSoftSerialTransmitBufferEmpty,
softSerialSetMode, softSerialSetMode,
.beginWrite = NULL, .writeBuf = NULL,
.endWrite = NULL,
} }
}; };

View file

@ -322,7 +322,6 @@ const struct serialPortVTable uartVTable[] = {
uartSetBaudRate, uartSetBaudRate,
isUartTransmitBufferEmpty, isUartTransmitBufferEmpty,
uartSetMode, uartSetMode,
.beginWrite = NULL, .writeBuf = NULL,
.endWrite = NULL,
} }
}; };

View file

@ -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)
{ {

View file

@ -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);