ardupilot/libraries/AP_HAL_SMACCM/Console.cpp
2013-01-10 16:29:46 +11:00

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));
}