2015-08-11 03:28:43 -03:00
|
|
|
#include <AP_HAL/AP_HAL.h>
|
2014-03-31 14:29:33 -03:00
|
|
|
|
|
|
|
#if CONFIG_HAL_BOARD == HAL_BOARD_VRBRAIN
|
|
|
|
#include "RCInput.h"
|
|
|
|
#include <drivers/drv_hrt.h>
|
|
|
|
#include <uORB/uORB.h>
|
|
|
|
|
|
|
|
using namespace VRBRAIN;
|
|
|
|
|
|
|
|
extern const AP_HAL::HAL& hal;
|
|
|
|
|
2015-12-02 11:14:20 -04:00
|
|
|
void VRBRAINRCInput::init()
|
2014-03-31 14:29:33 -03:00
|
|
|
{
|
|
|
|
_perf_rcin = perf_alloc(PC_ELAPSED, "APM_rcin");
|
|
|
|
_rc_sub = orb_subscribe(ORB_ID(input_rc));
|
|
|
|
if (_rc_sub == -1) {
|
2015-11-19 23:11:33 -04:00
|
|
|
AP_HAL::panic("Unable to subscribe to input_rc");
|
2014-03-31 14:29:33 -03:00
|
|
|
}
|
|
|
|
clear_overrides();
|
|
|
|
pthread_mutex_init(&rcin_mutex, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool VRBRAINRCInput::new_input()
|
|
|
|
{
|
|
|
|
pthread_mutex_lock(&rcin_mutex);
|
|
|
|
bool valid = _rcin.timestamp_last_signal != _last_read || _override_valid;
|
2015-02-08 18:57:16 -04:00
|
|
|
_last_read = _rcin.timestamp_last_signal;
|
|
|
|
_override_valid = false;
|
2014-03-31 14:29:33 -03:00
|
|
|
pthread_mutex_unlock(&rcin_mutex);
|
|
|
|
return valid;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint8_t VRBRAINRCInput::num_channels()
|
|
|
|
{
|
|
|
|
pthread_mutex_lock(&rcin_mutex);
|
|
|
|
uint8_t n = _rcin.channel_count;
|
|
|
|
pthread_mutex_unlock(&rcin_mutex);
|
|
|
|
return n;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint16_t VRBRAINRCInput::read(uint8_t ch)
|
|
|
|
{
|
|
|
|
if (ch >= RC_INPUT_MAX_CHANNELS) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
pthread_mutex_lock(&rcin_mutex);
|
|
|
|
if (_override[ch]) {
|
|
|
|
uint16_t v = _override[ch];
|
|
|
|
pthread_mutex_unlock(&rcin_mutex);
|
|
|
|
return v;
|
|
|
|
}
|
|
|
|
if (ch >= _rcin.channel_count) {
|
|
|
|
pthread_mutex_unlock(&rcin_mutex);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
uint16_t v = _rcin.values[ch];
|
|
|
|
pthread_mutex_unlock(&rcin_mutex);
|
|
|
|
return v;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint8_t VRBRAINRCInput::read(uint16_t* periods, uint8_t len)
|
|
|
|
{
|
|
|
|
if (len > RC_INPUT_MAX_CHANNELS) {
|
|
|
|
len = RC_INPUT_MAX_CHANNELS;
|
|
|
|
}
|
|
|
|
for (uint8_t i = 0; i < len; i++){
|
|
|
|
periods[i] = read(i);
|
|
|
|
}
|
|
|
|
return len;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool VRBRAINRCInput::set_overrides(int16_t *overrides, uint8_t len)
|
|
|
|
{
|
|
|
|
bool res = false;
|
|
|
|
for (uint8_t i = 0; i < len; i++) {
|
|
|
|
res |= set_override(i, overrides[i]);
|
|
|
|
}
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool VRBRAINRCInput::set_override(uint8_t channel, int16_t override) {
|
|
|
|
if (override < 0) {
|
|
|
|
return false; /* -1: no change. */
|
|
|
|
}
|
|
|
|
if (channel >= RC_INPUT_MAX_CHANNELS) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
_override[channel] = override;
|
|
|
|
if (override != 0) {
|
|
|
|
_override_valid = true;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void VRBRAINRCInput::clear_overrides()
|
|
|
|
{
|
|
|
|
for (uint8_t i = 0; i < RC_INPUT_MAX_CHANNELS; i++) {
|
|
|
|
set_override(i, 0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void VRBRAINRCInput::_timer_tick(void)
|
|
|
|
{
|
|
|
|
perf_begin(_perf_rcin);
|
|
|
|
bool rc_updated = false;
|
|
|
|
if (orb_check(_rc_sub, &rc_updated) == 0 && rc_updated) {
|
|
|
|
pthread_mutex_lock(&rcin_mutex);
|
|
|
|
orb_copy(ORB_ID(input_rc), _rc_sub, &_rcin);
|
|
|
|
pthread_mutex_unlock(&rcin_mutex);
|
|
|
|
}
|
|
|
|
// note, we rely on the vehicle code checking new_input()
|
|
|
|
// and a timeout for the last valid input to handle failsafe
|
|
|
|
perf_end(_perf_rcin);
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|