mirror of
https://github.com/betaflight/betaflight.git
synced 2025-07-25 01:05:27 +03:00
Add uartWriteBuf() to improve performance Optimised transmit buffer space check Tidy up group duration calculations Add uartBeginWrite, uartEndWrite and serialWriteBufNoFlush Remove OSD grouping and check on the fly. Implement multi-pass artificial horizon rendering. Fix rendering of camera frame Fix stick overlay background rendering Fix channel rendering Fix ESC information rendering Make Spec Prearm Display deterministic Co-authored-by: Petr Ledvina <ledvinap@gmail.com>
157 lines
5.7 KiB
C
157 lines
5.7 KiB
C
/*
|
|
* This file is part of Cleanflight and Betaflight.
|
|
*
|
|
* Cleanflight and Betaflight are free software. You can redistribute
|
|
* this software and/or modify this software under the terms of the
|
|
* GNU General Public License as published by the Free Software
|
|
* Foundation, either version 3 of the License, or (at your option)
|
|
* any later version.
|
|
*
|
|
* Cleanflight and Betaflight are distributed in the hope that they
|
|
* will be useful, but WITHOUT ANY WARRANTY; without even the implied
|
|
* warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
|
* See the GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this software.
|
|
*
|
|
* If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#pragma once
|
|
|
|
#include "drivers/io.h"
|
|
#include "drivers/io_types.h"
|
|
#include "drivers/resource.h"
|
|
// TODO(hertz@): uncomment and use UARTDevice_e::MAX_UARTDEV
|
|
// #include "drivers/serial_uart.h"
|
|
|
|
#include "pg/pg.h"
|
|
|
|
typedef enum {
|
|
MODE_RX = 1 << 0,
|
|
MODE_TX = 1 << 1,
|
|
MODE_RXTX = MODE_RX | MODE_TX
|
|
} portMode_e;
|
|
|
|
typedef enum {
|
|
SERIAL_NOT_INVERTED = 0 << 0,
|
|
SERIAL_INVERTED = 1 << 0,
|
|
SERIAL_STOPBITS_1 = 0 << 1,
|
|
SERIAL_STOPBITS_2 = 1 << 1,
|
|
SERIAL_PARITY_NO = 0 << 2,
|
|
SERIAL_PARITY_EVEN = 1 << 2,
|
|
SERIAL_UNIDIR = 0 << 3,
|
|
SERIAL_BIDIR = 1 << 3,
|
|
|
|
/*
|
|
* Note on SERIAL_BIDIR_PP
|
|
* With SERIAL_BIDIR_PP, the very first start bit of back-to-back bytes
|
|
* is lost and the first data byte will be lost by a framing error.
|
|
* To ensure the first start bit to be sent, prepend a zero byte (0x00)
|
|
* to actual data bytes.
|
|
*/
|
|
SERIAL_BIDIR_OD = 0 << 4,
|
|
SERIAL_BIDIR_PP = 1 << 4,
|
|
SERIAL_BIDIR_NOPULL = 1 << 5, // disable pulls in BIDIR RX mode
|
|
SERIAL_BIDIR_PP_PD = 1 << 6, // PP mode, normall inverted, but with PullDowns, to fix SA after bidir issue fixed (#10220)
|
|
|
|
// If this option is set then switch the TX line to input when not in use to detect it being pulled low
|
|
SERIAL_CHECK_TX = 1 << 7,
|
|
} portOptions_e;
|
|
|
|
// Define known line control states which may be passed up by underlying serial driver callback
|
|
#define CTRL_LINE_STATE_DTR (1 << 0)
|
|
#define CTRL_LINE_STATE_RTS (1 << 1)
|
|
|
|
typedef void (*serialReceiveCallbackPtr)(uint16_t data, void *rxCallbackData); // used by serial drivers to return frames to app
|
|
typedef void (*serialIdleCallbackPtr)();
|
|
|
|
typedef struct serialPort_s {
|
|
|
|
const struct serialPortVTable *vTable;
|
|
|
|
portMode_e mode;
|
|
portOptions_e options;
|
|
|
|
uint32_t baudRate;
|
|
|
|
uint32_t rxBufferSize;
|
|
uint32_t txBufferSize;
|
|
volatile uint8_t *rxBuffer;
|
|
volatile uint8_t *txBuffer;
|
|
uint32_t rxBufferHead;
|
|
uint32_t rxBufferTail;
|
|
uint32_t txBufferHead;
|
|
uint32_t txBufferTail;
|
|
|
|
serialReceiveCallbackPtr rxCallback;
|
|
void *rxCallbackData;
|
|
|
|
serialIdleCallbackPtr idleCallback;
|
|
|
|
uint8_t identifier;
|
|
} serialPort_t;
|
|
|
|
#define SERIAL_PORT_MAX_INDEX 11
|
|
|
|
typedef struct serialPinConfig_s {
|
|
ioTag_t ioTagTx[SERIAL_PORT_MAX_INDEX];
|
|
ioTag_t ioTagRx[SERIAL_PORT_MAX_INDEX];
|
|
ioTag_t ioTagInverter[SERIAL_PORT_MAX_INDEX];
|
|
} serialPinConfig_t;
|
|
|
|
PG_DECLARE(serialPinConfig_t, serialPinConfig);
|
|
|
|
#if defined(USE_SOFTSERIAL)
|
|
#define SOFTSERIAL_COUNT 2
|
|
|
|
typedef struct softSerialPinConfig_s {
|
|
ioTag_t ioTagTx[SOFTSERIAL_COUNT];
|
|
ioTag_t ioTagRx[SOFTSERIAL_COUNT];
|
|
} softSerialPinConfig_t;
|
|
|
|
PG_DECLARE(softSerialPinConfig_t, softSerialPinConfig);
|
|
#endif
|
|
|
|
struct serialPortVTable {
|
|
void (*serialWrite)(serialPort_t *instance, uint8_t ch);
|
|
|
|
uint32_t (*serialTotalRxWaiting)(const serialPort_t *instance);
|
|
uint32_t (*serialTotalTxFree)(const serialPort_t *instance);
|
|
|
|
uint8_t (*serialRead)(serialPort_t *instance);
|
|
|
|
// Specified baud rate may not be allowed by an implementation, use serialGetBaudRate to determine actual baud rate in use.
|
|
void (*serialSetBaudRate)(serialPort_t *instance, uint32_t baudRate);
|
|
|
|
bool (*isSerialTransmitBufferEmpty)(const serialPort_t *instance);
|
|
|
|
void (*setMode)(serialPort_t *instance, portMode_e mode);
|
|
void (*setCtrlLineStateCb)(serialPort_t *instance, void (*cb)(void *instance, uint16_t ctrlLineState), void *context);
|
|
void (*setBaudRateCb)(serialPort_t *instance, void (*cb)(serialPort_t *context, uint32_t baud), serialPort_t *context);
|
|
|
|
void (*writeBuf)(serialPort_t *instance, const void *data, int count);
|
|
// Optional functions used to buffer large writes.
|
|
void (*beginWrite)(serialPort_t *instance);
|
|
void (*endWrite)(serialPort_t *instance);
|
|
};
|
|
|
|
void serialWrite(serialPort_t *instance, uint8_t ch);
|
|
uint32_t serialRxBytesWaiting(const serialPort_t *instance);
|
|
uint32_t serialTxBytesFree(const serialPort_t *instance);
|
|
void serialWriteBuf(serialPort_t *instance, const uint8_t *data, int count);
|
|
void serialWriteBufNoFlush(serialPort_t *instance, const uint8_t *data, int count);
|
|
uint8_t serialRead(serialPort_t *instance);
|
|
void serialSetBaudRate(serialPort_t *instance, uint32_t baudRate);
|
|
void serialSetMode(serialPort_t *instance, portMode_e mode);
|
|
void serialSetCtrlLineStateCb(serialPort_t *instance, void (*cb)(void *context, uint16_t ctrlLineState), void *context);
|
|
void serialSetBaudRateCb(serialPort_t *instance, void (*cb)(serialPort_t *context, uint32_t baud), serialPort_t *context);
|
|
bool isSerialTransmitBufferEmpty(const serialPort_t *instance);
|
|
void serialPrint(serialPort_t *instance, const char *str);
|
|
uint32_t serialGetBaudRate(serialPort_t *instance);
|
|
|
|
// A shim that adapts the bufWriter API to the serialWriteBuf() API.
|
|
void serialWriteBufShim(void *instance, const uint8_t *data, int count);
|
|
void serialBeginWrite(serialPort_t *instance);
|
|
void serialEndWrite(serialPort_t *instance);
|