223 lines
5.4 KiB
C++
223 lines
5.4 KiB
C++
/*
|
|
* This file is free software: you can redistribute it and/or modify it
|
|
* 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.
|
|
*
|
|
* This file is distributed in the hope that it 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 program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include <AP_HAL_ESP32/UARTDriver.h>
|
|
#include <AP_Math/AP_Math.h>
|
|
|
|
#include "esp_log.h"
|
|
|
|
extern const AP_HAL::HAL& hal;
|
|
|
|
namespace ESP32
|
|
{
|
|
|
|
UARTDesc uart_desc[] = {HAL_ESP32_UART_DEVICES};
|
|
|
|
void UARTDriver::vprintf(const char *fmt, va_list ap)
|
|
{
|
|
|
|
uart_port_t p = uart_desc[uart_num].port;
|
|
if (p == 0) {
|
|
esp_log_writev(ESP_LOG_INFO, "", fmt, ap);
|
|
} else {
|
|
AP_HAL::UARTDriver::vprintf(fmt, ap);
|
|
}
|
|
}
|
|
|
|
void UARTDriver::_begin(uint32_t b, uint16_t rxS, uint16_t txS)
|
|
{
|
|
if (uart_num < ARRAY_SIZE(uart_desc)) {
|
|
uart_port_t p = uart_desc[uart_num].port;
|
|
if (!_initialized) {
|
|
|
|
uart_config_t config = {
|
|
.baud_rate = (int)b,
|
|
.data_bits = UART_DATA_8_BITS,
|
|
.parity = UART_PARITY_DISABLE,
|
|
.stop_bits = UART_STOP_BITS_1,
|
|
.flow_ctrl = UART_HW_FLOWCTRL_DISABLE,
|
|
};
|
|
uart_param_config(p, &config);
|
|
uart_set_pin(p,
|
|
uart_desc[uart_num].tx,
|
|
uart_desc[uart_num].rx,
|
|
UART_PIN_NO_CHANGE, UART_PIN_NO_CHANGE);
|
|
//uart_driver_install(p, 2*UART_FIFO_LEN, 0, 0, nullptr, 0);
|
|
uart_driver_install(p, 2*UART_FIFO_LEN, 0, 0, nullptr, 0);
|
|
_readbuf.set_size(RX_BUF_SIZE);
|
|
_writebuf.set_size(TX_BUF_SIZE);
|
|
|
|
_initialized = true;
|
|
} else {
|
|
flush();
|
|
uart_set_baudrate(p, b);
|
|
|
|
}
|
|
}
|
|
_baudrate = b;
|
|
}
|
|
|
|
void UARTDriver::_end()
|
|
{
|
|
if (_initialized) {
|
|
uart_driver_delete(uart_desc[uart_num].port);
|
|
_readbuf.set_size(0);
|
|
_writebuf.set_size(0);
|
|
}
|
|
_initialized = false;
|
|
}
|
|
|
|
void UARTDriver::_flush()
|
|
{
|
|
uart_port_t p = uart_desc[uart_num].port;
|
|
uart_flush(p);
|
|
}
|
|
|
|
bool UARTDriver::is_initialized()
|
|
{
|
|
return _initialized;
|
|
}
|
|
|
|
bool UARTDriver::tx_pending()
|
|
{
|
|
return (_writebuf.available() > 0);
|
|
}
|
|
|
|
|
|
uint32_t UARTDriver::_available()
|
|
{
|
|
if (!_initialized) {
|
|
return 0;
|
|
}
|
|
return _readbuf.available();
|
|
}
|
|
|
|
uint32_t UARTDriver::txspace()
|
|
{
|
|
if (!_initialized) {
|
|
return 0;
|
|
}
|
|
int result = _writebuf.space();
|
|
result -= TX_BUF_SIZE / 4;
|
|
return MAX(result, 0);
|
|
|
|
}
|
|
|
|
ssize_t IRAM_ATTR UARTDriver::_read(uint8_t *buffer, uint16_t count)
|
|
{
|
|
if (!_initialized) {
|
|
return -1;
|
|
}
|
|
|
|
const uint32_t ret = _readbuf.read(buffer, count);
|
|
if (ret == 0) {
|
|
return 0;
|
|
}
|
|
|
|
|
|
_receive_timestamp_update();
|
|
|
|
return ret;
|
|
}
|
|
|
|
void IRAM_ATTR UARTDriver::_timer_tick(void)
|
|
{
|
|
if (!_initialized) {
|
|
return;
|
|
}
|
|
read_data();
|
|
write_data();
|
|
}
|
|
|
|
void IRAM_ATTR UARTDriver::read_data()
|
|
{
|
|
uart_port_t p = uart_desc[uart_num].port;
|
|
int count = 0;
|
|
do {
|
|
count = uart_read_bytes(p, _buffer, sizeof(_buffer), 0);
|
|
if (count > 0) {
|
|
_readbuf.write(_buffer, count);
|
|
}
|
|
} while (count > 0);
|
|
}
|
|
|
|
void IRAM_ATTR UARTDriver::write_data()
|
|
{
|
|
uart_port_t p = uart_desc[uart_num].port;
|
|
int count = 0;
|
|
_write_mutex.take_blocking();
|
|
do {
|
|
count = _writebuf.peekbytes(_buffer, sizeof(_buffer));
|
|
if (count > 0) {
|
|
count = uart_tx_chars(p, (const char*) _buffer, count);
|
|
_writebuf.advance(count);
|
|
}
|
|
} while (count > 0);
|
|
_write_mutex.give();
|
|
}
|
|
|
|
size_t IRAM_ATTR UARTDriver::_write(const uint8_t *buffer, size_t size)
|
|
{
|
|
if (!_initialized) {
|
|
return 0;
|
|
}
|
|
|
|
_write_mutex.take_blocking();
|
|
|
|
|
|
size_t ret = _writebuf.write(buffer, size);
|
|
_write_mutex.give();
|
|
return ret;
|
|
}
|
|
|
|
bool UARTDriver::_discard_input()
|
|
{
|
|
//uart_port_t p = uart_desc[uart_num].port;
|
|
//return uart_flush_input(p) == ESP_OK;
|
|
return false;
|
|
}
|
|
|
|
// record timestamp of new incoming data
|
|
void IRAM_ATTR UARTDriver::_receive_timestamp_update(void)
|
|
{
|
|
_receive_timestamp[_receive_timestamp_idx^1] = AP_HAL::micros64();
|
|
_receive_timestamp_idx ^= 1;
|
|
}
|
|
|
|
|
|
/*
|
|
return timestamp estimate in microseconds for when the start of
|
|
a nbytes packet arrived on the uart. This should be treated as a
|
|
time constraint, not an exact time. It is guaranteed that the
|
|
packet did not start being received after this time, but it
|
|
could have been in a system buffer before the returned time.
|
|
This takes account of the baudrate of the link. For transports
|
|
that have no baudrate (such as USB) the time estimate may be
|
|
less accurate.
|
|
A return value of zero means the HAL does not support this API
|
|
*/
|
|
uint64_t UARTDriver::receive_time_constraint_us(uint16_t nbytes)
|
|
{
|
|
uint64_t last_receive_us = _receive_timestamp[_receive_timestamp_idx];
|
|
if (_baudrate > 0) {
|
|
// assume 10 bits per byte. For USB we assume zero transport delay
|
|
uint32_t transport_time_us = (1000000UL * 10UL / _baudrate) * (nbytes + available());
|
|
last_receive_us -= transport_time_us;
|
|
}
|
|
return last_receive_us;
|
|
}
|
|
|
|
}
|