ardupilot/libraries/AP_HAL_QURT/UARTDriver.cpp
Andrew Tridgell 0816937ab1 HAL_QURT: initial implementation
this provides RC onput, RC output, scheduling, storage, UARTs and all
necessary support routines to fly ArduPilot on QURT
2015-12-27 16:12:28 +11:00

298 lines
6.5 KiB
C++

/*
This program 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 program 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/AP_HAL.h>
#if CONFIG_HAL_BOARD == HAL_BOARD_QURT
#include <stdlib.h>
#include <unistd.h>
#include "UARTDriver.h"
#include <sys/types.h>
#include <sys/stat.h>
#include <stdio.h>
#include <fcntl.h>
#include <dev_fs_lib_serial.h>
#include <AP_HAL/utility/RingBuffer.h>
using namespace QURT;
extern const AP_HAL::HAL& hal;
UARTDriver::UARTDriver(const char *name) :
devname(name)
{
}
void UARTDriver::begin(uint32_t b)
{
begin(b, 16384, 16384);
}
static const struct {
uint32_t baudrate;
enum DSPAL_SERIAL_BITRATES arg;
} baudrate_table[] = {
{ 9600, DSPAL_SIO_BITRATE_9600 },
{ 14400, DSPAL_SIO_BITRATE_14400 },
{ 19200, DSPAL_SIO_BITRATE_19200 },
{ 38400, DSPAL_SIO_BITRATE_38400 },
{ 57600, DSPAL_SIO_BITRATE_57600 },
{ 76800, DSPAL_SIO_BITRATE_76800 },
{ 115200, DSPAL_SIO_BITRATE_115200 },
{ 230400, DSPAL_SIO_BITRATE_230400 },
{ 250000, DSPAL_SIO_BITRATE_250000 },
{ 460800, DSPAL_SIO_BITRATE_460800 },
{ 921600, DSPAL_SIO_BITRATE_921600 },
{ 2000000, DSPAL_SIO_BITRATE_2000000 },
};
extern "C" {
static void read_callback_trampoline(void *, char *, size_t );
}
static void read_callback_trampoline(void *ctx, char *buf, size_t size)
{
((UARTDriver *)ctx)->_read_callback(buf, size);
}
/*
callback for incoming data
*/
void UARTDriver::_read_callback(char *buf, size_t size)
{
if (readbuf == nullptr) {
return;
}
uint32_t n = readbuf->write((const uint8_t *)buf, size);
if (n != size) {
HAP_PRINTF("read_callback lost %u %u", n, size);
}
}
void UARTDriver::begin(uint32_t b, uint16_t rxS, uint16_t txS)
{
if (rxS < 4096) {
rxS = 4096;
}
if (txS < 4096) {
txS = 4096;
}
if (fd == -1) {
fd = open(devname, O_RDWR | O_NONBLOCK);
if (fd == -1) {
AP_HAL::panic("Unable to open %s", devname);
}
}
/*
allocate the read buffer
we allocate buffers before we successfully open the device as we
want to allocate in the early stages of boot, and cause minimum
thrashing of the heap once we are up. The ttyACM0 driver may not
connect for some time after boot
*/
if (rxS != 0 && (readbuf == nullptr || rxS != readbuf->get_size())) {
initialised = false;
if (readbuf != nullptr) {
delete readbuf;
}
readbuf = new ByteBuffer(rxS);
}
/*
allocate the write buffer
*/
if (txS != 0 && (writebuf == nullptr || txS != writebuf->get_size())) {
initialised = false;
if (writebuf != nullptr) {
delete writebuf;
}
writebuf = new ByteBuffer(txS);
}
struct dspal_serial_ioctl_receive_data_callback callback;
callback.context = this;
callback.rx_data_callback_func_ptr = read_callback_trampoline;
int ret = ioctl(fd, SERIAL_IOCTL_SET_RECEIVE_DATA_CALLBACK, (void *)&callback);
if (b != 0) {
for (uint8_t i=0; i<ARRAY_SIZE(baudrate_table); i++) {
if (b <= baudrate_table[i].baudrate) {
struct dspal_serial_ioctl_data_rate rate;
rate.bit_rate = baudrate_table[i].arg;
ret = ioctl(fd, SERIAL_IOCTL_SET_DATA_RATE, (void *)&rate);
break;
}
}
}
if (readbuf && writebuf && fd != -1) {
initialised = true;
}
}
void UARTDriver::end()
{
initialised = false;
if (fd != -1) {
::close(fd);
fd = -1;
}
if (readbuf) {
delete readbuf;
readbuf = nullptr;
}
if (writebuf) {
delete writebuf;
writebuf = nullptr;
}
}
void UARTDriver::flush()
{
}
bool UARTDriver::is_initialized()
{
return fd != -1 && initialised;
}
void UARTDriver::set_blocking_writes(bool blocking)
{
nonblocking_writes = !blocking;
}
bool UARTDriver::tx_pending()
{
return false;
}
/* QURT implementations of Stream virtual methods */
int16_t UARTDriver::available()
{
if (!initialised) {
return 0;
}
return readbuf->available();
}
int16_t UARTDriver::txspace()
{
if (!initialised) {
return 0;
}
return writebuf->space();
}
int16_t UARTDriver::read()
{
uint8_t c;
if (!initialised) {
return -1;
}
if (!lock.take(0)) {
return 0;
}
if (readbuf->empty()) {
lock.give();
return -1;
}
readbuf->read(&c, 1);
lock.give();
return c;
}
/* QURT implementations of Print virtual methods */
size_t UARTDriver::write(uint8_t c)
{
if (!initialised) {
return 0;
}
if (!lock.take(0)) {
return 0;
}
while (writebuf->space() == 0) {
if (nonblocking_writes) {
lock.give();
return 0;
}
hal.scheduler->delay_microseconds(1000);
}
writebuf->write(&c, 1);
lock.give();
return 1;
}
size_t UARTDriver::write(const uint8_t *buffer, size_t size)
{
if (!initialised) {
return 0;
}
if (!nonblocking_writes) {
/*
use the per-byte delay loop in write() above for blocking writes
*/
size_t ret = 0;
while (size--) {
if (write(*buffer++) != 1) break;
ret++;
}
return ret;
}
if (!lock.take(0)) {
return 0;
}
size_t ret = writebuf->write(buffer, size);
lock.give();
return ret;
}
/*
push any pending bytes to/from the serial port. This is called at
1kHz in the timer thread
*/
void UARTDriver::timer_tick(void)
{
uint16_t n;
if (fd == -1 || !initialised || !lock.take_nonblocking()) {
return;
}
in_timer = true;
// write any pending bytes
n = writebuf->available();
if (n == 0) {
in_timer = false;
lock.give();
return;
}
if (n > 64) {
n = 64;
}
uint8_t buf[n];
writebuf->read(buf, n);
::write(fd, buf, n);
lock.give();
in_timer = false;
}
#endif