mirror of
https://github.com/ArduPilot/ardupilot
synced 2025-01-03 22:48:29 -04:00
213 lines
3.7 KiB
C++
213 lines
3.7 KiB
C++
/*
|
|
* Console.cpp --- AP_HAL_SMACCM console driver.
|
|
*
|
|
* Copyright (C) 2012, Galois, Inc.
|
|
* All Rights Reserved.
|
|
*
|
|
* This software is released under the "BSD3" license. Read the file
|
|
* "LICENSE" for more information.
|
|
*/
|
|
|
|
#include <stdarg.h>
|
|
#include <stdlib.h>
|
|
|
|
#include "Console.h"
|
|
|
|
using namespace SMACCM;
|
|
|
|
SMACCMConsoleDriver::SMACCMConsoleDriver(AP_HAL::BetterStream* delegate) :
|
|
_d(delegate), _user_backend(false)
|
|
{
|
|
}
|
|
|
|
void SMACCMConsoleDriver::init(void *args)
|
|
{
|
|
}
|
|
|
|
void SMACCMConsoleDriver::backend_open()
|
|
{
|
|
_txbuf.allocate(128);
|
|
_rxbuf.allocate(16);
|
|
_user_backend = true;
|
|
}
|
|
|
|
void SMACCMConsoleDriver::backend_close()
|
|
{
|
|
_user_backend = false;
|
|
}
|
|
|
|
size_t SMACCMConsoleDriver::backend_read(uint8_t *data, size_t len)
|
|
{
|
|
for (size_t i = 0; i < len; i++) {
|
|
int16_t b = _txbuf.pop();
|
|
if (b != -1) {
|
|
data[i] = (uint8_t) b;
|
|
} else {
|
|
return i;
|
|
}
|
|
}
|
|
|
|
return len;
|
|
}
|
|
|
|
size_t SMACCMConsoleDriver::backend_write(const uint8_t *data, size_t len)
|
|
{
|
|
for (size_t i = 0; i < len; i++) {
|
|
bool valid = _rxbuf.push(data[i]);
|
|
if (!valid) {
|
|
return i;
|
|
}
|
|
}
|
|
|
|
return len;
|
|
}
|
|
|
|
void SMACCMConsoleDriver::print_P(const prog_char_t *pstr)
|
|
{
|
|
_d->print_P(pstr);
|
|
}
|
|
|
|
void SMACCMConsoleDriver::println_P(const prog_char_t *pstr)
|
|
{
|
|
_d->println_P(pstr);
|
|
}
|
|
|
|
void SMACCMConsoleDriver::printf(const char *pstr, ...)
|
|
{
|
|
va_list ap;
|
|
va_start(ap, pstr);
|
|
_d->vprintf(pstr, ap);
|
|
va_end(ap);
|
|
}
|
|
|
|
void SMACCMConsoleDriver::_printf_P(const prog_char *pstr, ...)
|
|
{
|
|
va_list ap;
|
|
va_start(ap, pstr);
|
|
_d->vprintf_P(pstr, ap);
|
|
va_end(ap);
|
|
}
|
|
|
|
void SMACCMConsoleDriver::vprintf(const char *pstr, va_list ap)
|
|
{
|
|
_d->vprintf(pstr, ap);
|
|
}
|
|
|
|
void SMACCMConsoleDriver::vprintf_P(const prog_char *pstr, va_list ap)
|
|
{
|
|
_d->vprintf(pstr, ap);
|
|
}
|
|
|
|
int16_t SMACCMConsoleDriver::available()
|
|
{
|
|
if (_user_backend) {
|
|
return _rxbuf.bytes_used();
|
|
} else {
|
|
return _d->available();
|
|
}
|
|
}
|
|
|
|
int16_t SMACCMConsoleDriver::txspace()
|
|
{
|
|
if (_user_backend) {
|
|
return _txbuf.bytes_free();
|
|
} else {
|
|
return _d->txspace();
|
|
}
|
|
}
|
|
|
|
int16_t SMACCMConsoleDriver::read()
|
|
{
|
|
if (_user_backend) {
|
|
return _rxbuf.pop();
|
|
} else {
|
|
return _d->read();
|
|
}
|
|
}
|
|
|
|
int16_t SMACCMConsoleDriver::peek()
|
|
{
|
|
if (_user_backend) {
|
|
return _rxbuf.peek();
|
|
} else {
|
|
return _d->peek();
|
|
}
|
|
}
|
|
|
|
size_t SMACCMConsoleDriver::write(uint8_t c)
|
|
{
|
|
if (_user_backend) {
|
|
return (_txbuf.push(c) ? 1 : 0);
|
|
} else {
|
|
return _d->write(c);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* SMACCMConsoleDriver::Buffer implementation.
|
|
* A synchronous nonblocking ring buffer, based on the AVRUARTDriver::Buffer
|
|
*/
|
|
|
|
bool SMACCMConsoleDriver::Buffer::allocate(uint16_t size)
|
|
{
|
|
_head = 0;
|
|
_tail = 0;
|
|
uint8_t shift;
|
|
/* Hardcoded max size of 1024. sue me. */
|
|
for ( shift = 1;
|
|
( 1U << shift ) < 1024 && ( 1U << shift) < size;
|
|
shift++
|
|
) ;
|
|
uint16_t tmpmask = (1U << shift) - 1;
|
|
|
|
if ( _bytes != NULL ) {
|
|
if ( _mask == tmpmask ) {
|
|
return true;
|
|
}
|
|
free(_bytes);
|
|
}
|
|
_mask = tmpmask;
|
|
_bytes = (uint8_t*) malloc(_mask+1);
|
|
return (_bytes != NULL);
|
|
}
|
|
|
|
bool SMACCMConsoleDriver::Buffer::push(uint8_t b)
|
|
{
|
|
uint16_t next = (_head + 1) & _mask;
|
|
if ( next == _tail ) {
|
|
return false;
|
|
}
|
|
_bytes[_head] = b;
|
|
_head = next;
|
|
return true;
|
|
}
|
|
|
|
int16_t SMACCMConsoleDriver::Buffer::pop()
|
|
{
|
|
if ( _tail == _head ) {
|
|
return -1;
|
|
}
|
|
uint8_t b = _bytes[_tail];
|
|
_tail = ( _tail + 1 ) & _mask;
|
|
return (int16_t) b;
|
|
}
|
|
|
|
int16_t SMACCMConsoleDriver::Buffer::peek()
|
|
{
|
|
if ( _tail == _head ) {
|
|
return -1;
|
|
}
|
|
uint8_t b = _bytes[_tail];
|
|
return (int16_t) b;
|
|
}
|
|
|
|
uint16_t SMACCMConsoleDriver::Buffer::bytes_used()
|
|
{
|
|
return ((_head - _tail) & _mask);
|
|
}
|
|
|
|
uint16_t SMACCMConsoleDriver::Buffer::bytes_free()
|
|
{
|
|
return ((_mask+1) - ((_head - _tail) & _mask));
|
|
}
|