ardupilot/libraries/AP_Scripting/AP_Scripting_SerialDevice.cpp
Thomas Watson a077e4a3ed AP_Scripting: introduce serial device simulation support
Allows a script to simulate a device attached via any serial protocol.
The script can read and write data and have it handled according to the
protocol as if exchanged over a serial port. The script can then do
protocol translation, data filtering and validation,
hardware-in-the-loop simulation, experimentation, etc., especially in
combination with the scripting protocol which lets the script itself
handle an attached device and so interpose any communication.
2024-06-27 12:00:18 +10:00

143 lines
3.4 KiB
C++

/*
port for a script to access from a device perspective
*/
#include "AP_Scripting_config.h"
#include "AP_Scripting.h"
#if AP_SCRIPTING_ENABLED && AP_SCRIPTING_SERIALDEVICE_ENABLED
#include <AP_Math/AP_Math.h>
#include <AP_BoardConfig/AP_BoardConfig.h>
#ifndef AP_SCRIPTING_SERIALDEVICE_MIN_TXSIZE
#define AP_SCRIPTING_SERIALDEVICE_MIN_TXSIZE 2048
#endif
#ifndef AP_SCRIPTING_SERIALDEVICE_MIN_RXSIZE
#define AP_SCRIPTING_SERIALDEVICE_MIN_RXSIZE 2048
#endif
/*
initialise scripting serial ports
*/
void AP_Scripting_SerialDevice::init(void)
{
if (enable == 0) {
return;
}
for (uint8_t i=0; i<ARRAY_SIZE(ports); i++) {
auto &p = ports[i];
p.state.idx = AP_SERIALMANAGER_SCR_PORT_1 + i;
p.init();
AP::serialmanager().register_port(&p);
}
}
/*
initialise port
*/
void AP_Scripting_SerialDevice::Port::init(void)
{
begin(1000000, 0, 0); // assume 1MBaud rate even though it's a bit meaningless
}
size_t AP_Scripting_SerialDevice::Port::device_write(const uint8_t *buffer, size_t size)
{
WITH_SEMAPHORE(sem);
if (readbuffer) {
return readbuffer->write(buffer, size);
}
return 0;
}
ssize_t AP_Scripting_SerialDevice::Port::device_read(uint8_t *buffer, uint16_t count)
{
WITH_SEMAPHORE(sem);
if (writebuffer) {
return writebuffer->read(buffer, count);
}
return 0;
}
uint32_t AP_Scripting_SerialDevice::Port::device_available(void)
{
WITH_SEMAPHORE(sem);
if (writebuffer) {
return writebuffer->available();
}
return 0;
}
/*
available space in outgoing buffer
*/
uint32_t AP_Scripting_SerialDevice::Port::txspace(void)
{
WITH_SEMAPHORE(sem);
return writebuffer != nullptr ? writebuffer->space() : 0;
}
void AP_Scripting_SerialDevice::Port::_begin(uint32_t b, uint16_t rxS, uint16_t txS)
{
rxS = MAX(rxS, AP_SCRIPTING_SERIALDEVICE_MIN_RXSIZE);
txS = MAX(txS, AP_SCRIPTING_SERIALDEVICE_MIN_TXSIZE);
init_buffers(rxS, txS);
}
size_t AP_Scripting_SerialDevice::Port::_write(const uint8_t *buffer, size_t size)
{
WITH_SEMAPHORE(sem);
return writebuffer != nullptr ? writebuffer->write(buffer, size) : 0;
}
ssize_t AP_Scripting_SerialDevice::Port::_read(uint8_t *buffer, uint16_t count)
{
WITH_SEMAPHORE(sem);
return readbuffer != nullptr ? readbuffer->read(buffer, count) : -1;
}
uint32_t AP_Scripting_SerialDevice::Port::_available()
{
WITH_SEMAPHORE(sem);
return readbuffer != nullptr ? readbuffer->available() : 0;
}
bool AP_Scripting_SerialDevice::Port::_discard_input()
{
WITH_SEMAPHORE(sem);
if (readbuffer != nullptr) {
readbuffer->clear();
}
return true;
}
/*
initialise read/write buffers
*/
bool AP_Scripting_SerialDevice::Port::init_buffers(const uint32_t size_rx, const uint32_t size_tx)
{
if (size_tx == last_size_tx &&
size_rx == last_size_rx) {
return true;
}
WITH_SEMAPHORE(sem);
if (readbuffer == nullptr) {
readbuffer = NEW_NOTHROW ByteBuffer(size_rx);
} else {
readbuffer->set_size_best(size_rx);
}
if (writebuffer == nullptr) {
writebuffer = NEW_NOTHROW ByteBuffer(size_tx);
} else {
writebuffer->set_size_best(size_tx);
}
last_size_rx = size_rx;
last_size_tx = size_tx;
return readbuffer != nullptr && writebuffer != nullptr;
}
#endif // AP_SCRIPTING_ENABLED && AP_SCRIPTING_SERIALDEVICE_ENABLED