2015-08-11 03:28:43 -03:00
|
|
|
#include <AP_HAL/AP_HAL.h>
|
2013-01-04 07:25:36 -04:00
|
|
|
|
|
|
|
#if CONFIG_HAL_BOARD == HAL_BOARD_PX4
|
|
|
|
#include "RCOutput.h"
|
|
|
|
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
|
|
|
|
#include <drivers/drv_pwm_output.h>
|
2014-11-20 03:31:41 -04:00
|
|
|
#include <drivers/drv_hrt.h>
|
2016-04-15 03:38:58 -03:00
|
|
|
#include <drivers/drv_pwm_output.h>
|
|
|
|
#include <drivers/drv_sbus.h>
|
2013-01-04 07:25:36 -04:00
|
|
|
|
2017-04-02 11:55:10 -03:00
|
|
|
#include <AP_BoardConfig/AP_BoardConfig.h>
|
|
|
|
|
2013-01-04 07:25:36 -04:00
|
|
|
extern const AP_HAL::HAL& hal;
|
|
|
|
|
|
|
|
using namespace PX4;
|
|
|
|
|
2016-04-13 03:28:50 -03:00
|
|
|
/*
|
|
|
|
enable RCOUT_DEBUG_LATENCY to measure output latency using a logic
|
|
|
|
analyser. AUX6 will go high during the cork/push output.
|
|
|
|
*/
|
|
|
|
#define RCOUT_DEBUG_LATENCY 0
|
|
|
|
|
2015-12-02 11:14:20 -04:00
|
|
|
void PX4RCOutput::init()
|
2013-01-04 07:25:36 -04:00
|
|
|
{
|
2013-01-25 04:16:28 -04:00
|
|
|
_perf_rcout = perf_alloc(PC_ELAPSED, "APM_rcout");
|
2015-02-13 05:21:03 -04:00
|
|
|
_pwm_fd = open(PWM_OUTPUT0_DEVICE_PATH, O_RDWR);
|
2013-01-04 07:25:36 -04:00
|
|
|
if (_pwm_fd == -1) {
|
2015-11-19 23:11:07 -04:00
|
|
|
AP_HAL::panic("Unable to open " PWM_OUTPUT0_DEVICE_PATH);
|
2013-01-04 07:25:36 -04:00
|
|
|
}
|
2013-01-27 00:05:28 -04:00
|
|
|
if (ioctl(_pwm_fd, PWM_SERVO_ARM, 0) != 0) {
|
2013-01-26 21:52:03 -04:00
|
|
|
hal.console->printf("RCOutput: Unable to setup IO arming\n");
|
|
|
|
}
|
2013-05-16 03:26:57 -03:00
|
|
|
if (ioctl(_pwm_fd, PWM_SERVO_SET_ARM_OK, 0) != 0) {
|
|
|
|
hal.console->printf("RCOutput: Unable to setup IO arming OK\n");
|
|
|
|
}
|
2016-04-15 03:38:58 -03:00
|
|
|
|
2017-04-07 02:07:17 -03:00
|
|
|
_rate_mask_main = 0;
|
|
|
|
_rate_mask_alt = 0;
|
2013-04-25 06:59:30 -03:00
|
|
|
_alt_fd = -1;
|
|
|
|
_servo_count = 0;
|
|
|
|
_alt_servo_count = 0;
|
|
|
|
|
|
|
|
if (ioctl(_pwm_fd, PWM_SERVO_GET_COUNT, (unsigned long)&_servo_count) != 0) {
|
|
|
|
hal.console->printf("RCOutput: Unable to get servo count\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-06-03 22:00:34 -03:00
|
|
|
for (uint8_t i=0; i<ORB_MULTI_MAX_INSTANCES; i++) {
|
|
|
|
_outputs[i].pwm_sub = orb_subscribe_multi(ORB_ID(actuator_outputs), i);
|
|
|
|
}
|
2014-11-05 06:38:12 -04:00
|
|
|
|
2015-11-25 03:10:30 -04:00
|
|
|
#if !defined(CONFIG_ARCH_BOARD_PX4FMU_V4)
|
2016-10-23 03:15:32 -03:00
|
|
|
struct stat st;
|
|
|
|
// don't try and open px4fmu unless there is a px4io. Otherwise we
|
|
|
|
// can end up opening the same device twice
|
|
|
|
if (stat("/dev/px4io", &st) == 0) {
|
|
|
|
_alt_fd = open("/dev/px4fmu", O_RDWR);
|
|
|
|
if (_alt_fd == -1) {
|
|
|
|
hal.console->printf("RCOutput: failed to open /dev/px4fmu");
|
|
|
|
}
|
2013-04-25 06:59:30 -03:00
|
|
|
}
|
2015-11-25 03:10:30 -04:00
|
|
|
#endif
|
2014-08-27 06:52:07 -03:00
|
|
|
|
|
|
|
// ensure not to write zeros to disabled channels
|
2016-04-15 03:38:58 -03:00
|
|
|
for (uint8_t i=0; i < PX4_NUM_OUTPUT_CHANNELS; i++) {
|
2014-08-27 06:52:07 -03:00
|
|
|
_period[i] = PWM_IGNORE_THIS_CHANNEL;
|
2017-05-16 00:55:35 -03:00
|
|
|
_last_sent[i] = PWM_IGNORE_THIS_CHANNEL;
|
2014-08-27 06:52:07 -03:00
|
|
|
}
|
2014-01-19 21:56:30 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void PX4RCOutput::_init_alt_channels(void)
|
|
|
|
{
|
|
|
|
if (_alt_fd == -1) {
|
|
|
|
return;
|
|
|
|
}
|
2013-04-25 06:59:30 -03:00
|
|
|
if (ioctl(_alt_fd, PWM_SERVO_ARM, 0) != 0) {
|
|
|
|
hal.console->printf("RCOutput: Unable to setup alt IO arming\n");
|
2014-01-19 21:56:30 -04:00
|
|
|
return;
|
2013-04-25 06:59:30 -03:00
|
|
|
}
|
2013-05-16 03:26:57 -03:00
|
|
|
if (ioctl(_alt_fd, PWM_SERVO_SET_ARM_OK, 0) != 0) {
|
2014-01-15 07:25:50 -04:00
|
|
|
hal.console->printf("RCOutput: Unable to setup alt IO arming OK\n");
|
2014-01-19 21:56:30 -04:00
|
|
|
return;
|
2013-05-16 03:26:57 -03:00
|
|
|
}
|
2013-04-25 06:59:30 -03:00
|
|
|
if (ioctl(_alt_fd, PWM_SERVO_GET_COUNT, (unsigned long)&_alt_servo_count) != 0) {
|
|
|
|
hal.console->printf("RCOutput: Unable to get servo count\n");
|
|
|
|
}
|
2013-01-04 07:25:36 -04:00
|
|
|
}
|
|
|
|
|
2016-01-04 16:17:44 -04:00
|
|
|
/*
|
|
|
|
set output frequency on outputs associated with fd
|
|
|
|
*/
|
2017-04-07 02:07:17 -03:00
|
|
|
void PX4RCOutput::set_freq_fd(int fd, uint32_t chmask, uint16_t freq_hz, uint32_t &rate_mask)
|
2013-01-04 07:25:36 -04:00
|
|
|
{
|
2017-06-16 08:23:37 -03:00
|
|
|
if (_output_mode == MODE_PWM_BRUSHED) {
|
2017-05-24 23:08:52 -03:00
|
|
|
freq_hz /= 8; // divide by 8 for 8MHz clock
|
|
|
|
// remember max period
|
|
|
|
_period_max = 1000000UL/freq_hz;
|
2016-11-29 06:10:27 -04:00
|
|
|
}
|
|
|
|
|
2016-01-04 16:17:44 -04:00
|
|
|
// we can't set this per channel
|
2016-04-22 00:24:24 -03:00
|
|
|
if (freq_hz > 50 || freq_hz == 1) {
|
2013-04-25 06:59:30 -03:00
|
|
|
// we're being asked to set the alt rate
|
2018-04-03 05:10:49 -03:00
|
|
|
if (_output_mode == MODE_PWM_ONESHOT || _output_mode == MODE_PWM_ONESHOT125) {
|
2016-04-15 18:23:49 -03:00
|
|
|
/*
|
|
|
|
set a 1Hz update for oneshot. This periodic output will
|
|
|
|
never actually trigger, instead we will directly trigger
|
|
|
|
the pulse after each push()
|
|
|
|
*/
|
|
|
|
freq_hz = 1;
|
|
|
|
}
|
2016-11-29 06:10:27 -04:00
|
|
|
//::printf("SET_UPDATE_RATE %d %u\n", fd, (unsigned)freq_hz);
|
2016-01-04 16:17:44 -04:00
|
|
|
if (ioctl(fd, PWM_SERVO_SET_UPDATE_RATE, (unsigned long)freq_hz) != 0) {
|
2013-04-25 06:59:30 -03:00
|
|
|
hal.console->printf("RCOutput: Unable to set alt rate to %uHz\n", (unsigned)freq_hz);
|
|
|
|
return;
|
|
|
|
}
|
2013-01-05 07:45:17 -04:00
|
|
|
_freq_hz = freq_hz;
|
|
|
|
}
|
2013-04-25 06:59:30 -03:00
|
|
|
|
2016-01-04 16:17:44 -04:00
|
|
|
/* work out the new rate mask. The outputs have 3 groups of servos.
|
2013-04-25 06:59:30 -03:00
|
|
|
|
|
|
|
Group 0: channels 0 1
|
|
|
|
Group 1: channels 4 5 6 7
|
|
|
|
Group 2: channels 2 3
|
|
|
|
|
|
|
|
Channels within a group must be set to the same rate.
|
|
|
|
|
|
|
|
For the moment we never set the channels above 8 to more than
|
|
|
|
50Hz
|
|
|
|
*/
|
2016-04-22 00:24:24 -03:00
|
|
|
if (freq_hz > 50 || freq_hz == 1) {
|
2013-04-25 06:59:30 -03:00
|
|
|
// we are setting high rates on the given channels
|
2017-04-07 02:07:17 -03:00
|
|
|
rate_mask |= chmask & 0xFF;
|
|
|
|
if (rate_mask & 0x3) {
|
|
|
|
rate_mask |= 0x3;
|
2013-04-25 06:59:30 -03:00
|
|
|
}
|
2017-04-07 02:07:17 -03:00
|
|
|
if (rate_mask & 0xc) {
|
|
|
|
rate_mask |= 0xc;
|
2013-04-25 06:59:30 -03:00
|
|
|
}
|
2017-04-07 02:07:17 -03:00
|
|
|
if (rate_mask & 0xF0) {
|
|
|
|
rate_mask |= 0xF0;
|
2013-04-25 06:59:30 -03:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// we are setting low rates on the given channels
|
|
|
|
if (chmask & 0x3) {
|
2017-04-07 02:07:17 -03:00
|
|
|
rate_mask &= ~0x3;
|
2013-04-25 06:59:30 -03:00
|
|
|
}
|
|
|
|
if (chmask & 0xc) {
|
2017-04-07 02:07:17 -03:00
|
|
|
rate_mask &= ~0xc;
|
2013-04-25 06:59:30 -03:00
|
|
|
}
|
|
|
|
if (chmask & 0xf0) {
|
2017-04-07 02:07:17 -03:00
|
|
|
rate_mask &= ~0xf0;
|
2013-04-25 06:59:30 -03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-04-07 02:07:17 -03:00
|
|
|
if (ioctl(fd, PWM_SERVO_SET_SELECT_UPDATE_RATE, rate_mask) != 0) {
|
|
|
|
hal.console->printf("RCOutput: Unable to set alt rate mask to 0x%x\n", (unsigned)rate_mask);
|
2013-04-25 06:59:30 -03:00
|
|
|
}
|
2016-11-29 06:10:27 -04:00
|
|
|
|
2017-06-16 08:23:37 -03:00
|
|
|
if (_output_mode == MODE_PWM_BRUSHED) {
|
2016-11-29 06:10:27 -04:00
|
|
|
ioctl(fd, PWM_SERVO_SET_UPDATE_CLOCK, 8);
|
|
|
|
}
|
2013-01-04 07:25:36 -04:00
|
|
|
}
|
|
|
|
|
2016-01-04 16:17:44 -04:00
|
|
|
/*
|
|
|
|
set output frequency
|
|
|
|
*/
|
|
|
|
void PX4RCOutput::set_freq(uint32_t chmask, uint16_t freq_hz)
|
|
|
|
{
|
2018-04-03 05:10:49 -03:00
|
|
|
if (freq_hz > 50 && (_output_mode == MODE_PWM_ONESHOT || _output_mode == MODE_PWM_ONESHOT125)) {
|
2016-05-25 23:18:40 -03:00
|
|
|
// rate is irrelevent in oneshot
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-05-11 00:01:15 -03:00
|
|
|
// re-fetch servo count as it might have changed due to a change in BRD_PWM_COUNT
|
|
|
|
if (_pwm_fd != -1 && ioctl(_pwm_fd, PWM_SERVO_GET_COUNT, (unsigned long)&_servo_count) != 0) {
|
|
|
|
hal.console->printf("RCOutput: Unable to get servo count\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-01-04 16:17:44 -04:00
|
|
|
// greater than 400 doesn't give enough room at higher periods for
|
|
|
|
// the down pulse
|
2017-06-16 08:23:37 -03:00
|
|
|
if (freq_hz > 400 && _output_mode != MODE_PWM_BRUSHED) {
|
2016-01-04 16:17:44 -04:00
|
|
|
freq_hz = 400;
|
|
|
|
}
|
|
|
|
uint32_t primary_mask = chmask & ((1UL<<_servo_count)-1);
|
|
|
|
uint32_t alt_mask = chmask >> _servo_count;
|
|
|
|
if (primary_mask && _pwm_fd != -1) {
|
2017-04-07 02:07:17 -03:00
|
|
|
set_freq_fd(_pwm_fd, primary_mask, freq_hz, _rate_mask_main);
|
2016-01-04 16:17:44 -04:00
|
|
|
}
|
|
|
|
if (alt_mask && _alt_fd != -1) {
|
2017-04-07 02:07:17 -03:00
|
|
|
set_freq_fd(_alt_fd, alt_mask, freq_hz, _rate_mask_alt);
|
2016-01-04 16:17:44 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-01-04 07:25:36 -04:00
|
|
|
uint16_t PX4RCOutput::get_freq(uint8_t ch)
|
|
|
|
{
|
2017-04-07 02:07:17 -03:00
|
|
|
if (ch < _servo_count) {
|
|
|
|
if (_rate_mask_main & (1U<<ch)) {
|
|
|
|
return _freq_hz;
|
|
|
|
}
|
|
|
|
} else if (_alt_fd != -1) {
|
|
|
|
if (_rate_mask_alt & (1U<<(ch-_servo_count))) {
|
|
|
|
return _freq_hz;
|
|
|
|
}
|
2013-04-25 06:59:30 -03:00
|
|
|
}
|
|
|
|
return 50;
|
2013-01-04 07:25:36 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void PX4RCOutput::enable_ch(uint8_t ch)
|
|
|
|
{
|
2014-08-27 06:52:07 -03:00
|
|
|
if (ch >= PX4_NUM_OUTPUT_CHANNELS) {
|
|
|
|
return;
|
|
|
|
}
|
2014-11-13 20:58:09 -04:00
|
|
|
if (ch >= 8 && !(_enabled_channels & (1U<<ch))) {
|
2016-05-12 14:01:14 -03:00
|
|
|
// this is the first enable of an auxiliary channel - setup
|
2014-01-19 21:56:30 -04:00
|
|
|
// aux channels now. This delayed setup makes it possible to
|
|
|
|
// use BRD_PWM_COUNT to setup the number of PWM channels.
|
|
|
|
_init_alt_channels();
|
|
|
|
}
|
|
|
|
_enabled_channels |= (1U<<ch);
|
2014-08-27 13:28:30 -03:00
|
|
|
if (_period[ch] == PWM_IGNORE_THIS_CHANNEL) {
|
|
|
|
_period[ch] = 0;
|
2017-05-16 00:55:35 -03:00
|
|
|
_last_sent[ch] = 0;
|
2014-08-27 13:28:30 -03:00
|
|
|
}
|
2013-01-04 07:25:36 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void PX4RCOutput::disable_ch(uint8_t ch)
|
|
|
|
{
|
2014-08-27 06:52:07 -03:00
|
|
|
if (ch >= PX4_NUM_OUTPUT_CHANNELS) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2014-01-19 21:56:30 -04:00
|
|
|
_enabled_channels &= ~(1U<<ch);
|
2014-08-27 06:52:07 -03:00
|
|
|
_period[ch] = PWM_IGNORE_THIS_CHANNEL;
|
2013-01-04 07:25:36 -04:00
|
|
|
}
|
|
|
|
|
2014-01-15 07:25:50 -04:00
|
|
|
void PX4RCOutput::set_safety_pwm(uint32_t chmask, uint16_t period_us)
|
|
|
|
{
|
|
|
|
struct pwm_output_values pwm_values;
|
|
|
|
memset(&pwm_values, 0, sizeof(pwm_values));
|
|
|
|
for (uint8_t i=0; i<_servo_count; i++) {
|
|
|
|
if ((1UL<<i) & chmask) {
|
|
|
|
pwm_values.values[i] = period_us;
|
|
|
|
}
|
|
|
|
pwm_values.channel_count++;
|
|
|
|
}
|
|
|
|
int ret = ioctl(_pwm_fd, PWM_SERVO_SET_DISARMED_PWM, (long unsigned int)&pwm_values);
|
|
|
|
if (ret != OK) {
|
|
|
|
hal.console->printf("Failed to setup disarmed PWM for 0x%08x to %u\n", (unsigned)chmask, period_us);
|
2014-04-20 19:36:52 -03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void PX4RCOutput::set_failsafe_pwm(uint32_t chmask, uint16_t period_us)
|
|
|
|
{
|
|
|
|
struct pwm_output_values pwm_values;
|
|
|
|
memset(&pwm_values, 0, sizeof(pwm_values));
|
|
|
|
for (uint8_t i=0; i<_servo_count; i++) {
|
|
|
|
if ((1UL<<i) & chmask) {
|
|
|
|
pwm_values.values[i] = period_us;
|
|
|
|
}
|
|
|
|
pwm_values.channel_count++;
|
|
|
|
}
|
|
|
|
int ret = ioctl(_pwm_fd, PWM_SERVO_SET_FAILSAFE_PWM, (long unsigned int)&pwm_values);
|
|
|
|
if (ret != OK) {
|
|
|
|
hal.console->printf("Failed to setup failsafe PWM for 0x%08x to %u\n", (unsigned)chmask, period_us);
|
2014-01-15 07:25:50 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-14 05:23:58 -03:00
|
|
|
bool PX4RCOutput::force_safety_on(void)
|
|
|
|
{
|
2016-05-27 03:18:05 -03:00
|
|
|
_safety_state_request = AP_HAL::Util::SAFETY_DISARMED;
|
2016-05-27 14:23:57 -03:00
|
|
|
_safety_state_request_last_ms = 1;
|
2016-05-27 03:18:05 -03:00
|
|
|
return true;
|
2014-09-14 05:23:58 -03:00
|
|
|
}
|
|
|
|
|
2014-02-11 00:56:44 -04:00
|
|
|
void PX4RCOutput::force_safety_off(void)
|
|
|
|
{
|
2016-05-27 03:18:05 -03:00
|
|
|
_safety_state_request = AP_HAL::Util::SAFETY_ARMED;
|
2016-05-27 14:23:57 -03:00
|
|
|
_safety_state_request_last_ms = 1;
|
2016-05-27 03:18:05 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
void PX4RCOutput::force_safety_pending_requests(void)
|
|
|
|
{
|
|
|
|
// check if there is a pending saftey_state change. If so (timer != 0) then set it.
|
2016-08-11 01:09:33 -03:00
|
|
|
uint32_t now = AP_HAL::millis();
|
2016-05-27 03:18:05 -03:00
|
|
|
if (_safety_state_request_last_ms != 0 &&
|
2016-08-11 01:09:33 -03:00
|
|
|
now - _safety_state_request_last_ms >= 100) {
|
|
|
|
if (hal.util->safety_switch_state() == _safety_state_request &&
|
|
|
|
_safety_state_request_last_ms != 1) {
|
2016-05-27 03:18:05 -03:00
|
|
|
_safety_state_request_last_ms = 0;
|
|
|
|
} else if (_safety_state_request == AP_HAL::Util::SAFETY_DISARMED) {
|
|
|
|
// current != requested, set it
|
|
|
|
ioctl(_pwm_fd, PWM_SERVO_SET_FORCE_SAFETY_ON, 0);
|
2016-08-11 01:09:33 -03:00
|
|
|
_safety_state_request_last_ms = now;
|
2016-05-27 03:18:05 -03:00
|
|
|
} else if (_safety_state_request == AP_HAL::Util::SAFETY_ARMED) {
|
|
|
|
// current != requested, set it
|
|
|
|
ioctl(_pwm_fd, PWM_SERVO_SET_FORCE_SAFETY_OFF, 0);
|
2016-08-11 01:09:33 -03:00
|
|
|
_safety_state_request_last_ms = now;
|
2016-05-27 03:18:05 -03:00
|
|
|
}
|
|
|
|
}
|
2018-04-13 02:56:56 -03:00
|
|
|
// also update safety button options if needed
|
|
|
|
if (now - _last_safety_options_check_ms > 1000) {
|
|
|
|
_last_safety_options_check_ms = now;
|
|
|
|
AP_BoardConfig *boardconfig = AP_BoardConfig::get_instance();
|
|
|
|
if (boardconfig) {
|
|
|
|
uint16_t desired_options = 0;
|
|
|
|
uint16_t options = boardconfig->get_safety_button_options();
|
|
|
|
if (!(options & AP_BoardConfig::BOARD_SAFETY_OPTION_BUTTON_ACTIVE_SAFETY_OFF)) {
|
|
|
|
desired_options |= PWM_SERVO_SET_SAFETY_OPTION_DISABLE_BUTTON_OFF;
|
|
|
|
}
|
|
|
|
if (!(options & AP_BoardConfig::BOARD_SAFETY_OPTION_BUTTON_ACTIVE_SAFETY_ON)) {
|
|
|
|
desired_options |= PWM_SERVO_SET_SAFETY_OPTION_DISABLE_BUTTON_ON;
|
|
|
|
}
|
|
|
|
if (!(options & AP_BoardConfig::BOARD_SAFETY_OPTION_BUTTON_ACTIVE_ARMED) && hal.util->get_soft_armed()) {
|
|
|
|
desired_options |= PWM_SERVO_SET_SAFETY_OPTION_DISABLE_BUTTON_OFF | PWM_SERVO_SET_SAFETY_OPTION_DISABLE_BUTTON_ON;
|
|
|
|
}
|
|
|
|
if (_last_safety_options != desired_options) {
|
|
|
|
if (ioctl(_pwm_fd, PWM_SERVO_SET_SAFETY_OPTIONS, desired_options) == OK) {
|
|
|
|
_last_safety_options = desired_options;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2016-05-27 03:18:05 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
void PX4RCOutput::force_safety_no_wait(void)
|
|
|
|
{
|
|
|
|
if (_safety_state_request_last_ms != 0) {
|
|
|
|
_safety_state_request_last_ms = 1;
|
|
|
|
force_safety_pending_requests();
|
2014-02-11 00:56:44 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-01-04 07:25:36 -04:00
|
|
|
void PX4RCOutput::write(uint8_t ch, uint16_t period_us)
|
|
|
|
{
|
2016-04-13 05:24:31 -03:00
|
|
|
if (ch >= PX4_NUM_OUTPUT_CHANNELS) {
|
2013-01-22 16:29:59 -04:00
|
|
|
return;
|
|
|
|
}
|
2014-01-19 21:56:30 -04:00
|
|
|
if (!(_enabled_channels & (1U<<ch))) {
|
|
|
|
// not enabled
|
|
|
|
return;
|
|
|
|
}
|
2013-02-23 19:04:17 -04:00
|
|
|
if (ch >= _max_channel) {
|
|
|
|
_max_channel = ch + 1;
|
2013-01-25 04:16:28 -04:00
|
|
|
}
|
2016-11-29 06:10:27 -04:00
|
|
|
|
2018-04-03 05:10:49 -03:00
|
|
|
if (_output_mode == MODE_PWM_ONESHOT125) {
|
2018-05-11 03:05:01 -03:00
|
|
|
if (((ch < _servo_count) && ((1U<<ch) & _rate_mask_main)) ||
|
|
|
|
((ch >= _servo_count) && ((1U<<(ch-_servo_count)) & _rate_mask_alt))) {
|
|
|
|
// we treat oneshot125 very simply on HAL_PX4, with 1us
|
|
|
|
// resolution. Correctly handling it would use a 125 nsec
|
|
|
|
// step size, to give the full 1000 steps
|
|
|
|
period_us /= 8;
|
|
|
|
}
|
2018-04-03 05:10:49 -03:00
|
|
|
}
|
|
|
|
|
2017-06-20 22:40:12 -03:00
|
|
|
// keep unscaled value
|
|
|
|
_last_sent[ch] = period_us;
|
|
|
|
|
2017-06-16 08:23:37 -03:00
|
|
|
if (_output_mode == MODE_PWM_BRUSHED) {
|
2016-11-29 19:21:48 -04:00
|
|
|
// map from the PWM range to 0 t0 100% duty cycle. For 16kHz
|
|
|
|
// this ends up being 0 to 500 pulse width in units of
|
|
|
|
// 125usec.
|
|
|
|
if (period_us <= _esc_pwm_min) {
|
2016-11-29 06:10:27 -04:00
|
|
|
period_us = 0;
|
2016-11-29 19:21:48 -04:00
|
|
|
} else if (period_us >= _esc_pwm_max) {
|
2017-05-24 23:08:52 -03:00
|
|
|
period_us = _period_max;
|
2016-11-29 06:10:27 -04:00
|
|
|
} else {
|
2016-11-29 19:21:48 -04:00
|
|
|
uint32_t pdiff = period_us - _esc_pwm_min;
|
2017-05-24 23:08:52 -03:00
|
|
|
period_us = pdiff*_period_max/(_esc_pwm_max - _esc_pwm_min);
|
2016-11-29 06:10:27 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-04-15 18:23:49 -03:00
|
|
|
/*
|
|
|
|
only mark an update is needed if there has been a change, or we
|
|
|
|
are in oneshot mode. In oneshot mode we always need to send the
|
|
|
|
output
|
|
|
|
*/
|
2016-04-15 03:38:58 -03:00
|
|
|
if (period_us != _period[ch] ||
|
2018-04-03 05:10:49 -03:00
|
|
|
_output_mode == MODE_PWM_ONESHOT ||
|
|
|
|
_output_mode == MODE_PWM_ONESHOT125) {
|
2013-01-22 16:29:59 -04:00
|
|
|
_period[ch] = period_us;
|
2013-01-25 04:16:28 -04:00
|
|
|
_need_update = true;
|
2013-01-22 16:29:59 -04:00
|
|
|
}
|
2013-01-04 07:25:36 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
uint16_t PX4RCOutput::read(uint8_t ch)
|
|
|
|
{
|
2013-01-22 16:29:59 -04:00
|
|
|
if (ch >= PX4_NUM_OUTPUT_CHANNELS) {
|
|
|
|
return 0;
|
2013-01-04 07:25:36 -04:00
|
|
|
}
|
2014-11-05 06:38:12 -04:00
|
|
|
// if px4io has given us a value for this channel use that,
|
|
|
|
// otherwise use the value we last sent. This makes it easier to
|
|
|
|
// observe the behaviour of failsafe in px4io
|
2015-06-03 22:00:34 -03:00
|
|
|
for (uint8_t i=0; i<ORB_MULTI_MAX_INSTANCES; i++) {
|
|
|
|
if (_outputs[i].pwm_sub >= 0 &&
|
|
|
|
ch < _outputs[i].outputs.noutputs &&
|
|
|
|
_outputs[i].outputs.output[ch] > 0) {
|
|
|
|
return _outputs[i].outputs.output[ch];
|
|
|
|
}
|
2014-11-05 06:38:12 -04:00
|
|
|
}
|
2013-01-22 16:29:59 -04:00
|
|
|
return _period[ch];
|
2013-01-04 07:25:36 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void PX4RCOutput::read(uint16_t* period_us, uint8_t len)
|
|
|
|
{
|
|
|
|
for (uint8_t i=0; i<len; i++) {
|
|
|
|
period_us[i] = read(i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-03-10 17:55:58 -04:00
|
|
|
uint16_t PX4RCOutput::read_last_sent(uint8_t ch)
|
|
|
|
{
|
|
|
|
if (ch >= PX4_NUM_OUTPUT_CHANNELS) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-05-16 00:55:35 -03:00
|
|
|
return _last_sent[ch];
|
2016-03-10 17:55:58 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void PX4RCOutput::read_last_sent(uint16_t* period_us, uint8_t len)
|
|
|
|
{
|
|
|
|
for (uint8_t i=0; i<len; i++) {
|
|
|
|
period_us[i] = read_last_sent(i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-11-20 03:31:41 -04:00
|
|
|
/*
|
|
|
|
update actuator armed state
|
|
|
|
*/
|
|
|
|
void PX4RCOutput::_arm_actuators(bool arm)
|
|
|
|
{
|
|
|
|
if (_armed.armed == arm) {
|
|
|
|
// already armed;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
_armed.timestamp = hrt_absolute_time();
|
|
|
|
_armed.armed = arm;
|
2016-10-23 23:35:39 -03:00
|
|
|
if (arm) {
|
|
|
|
// this latches ready_to_arm to true once set once. Otherwise
|
|
|
|
// we have a race condition with px4io safety switch update
|
|
|
|
_armed.ready_to_arm = true;
|
|
|
|
}
|
2014-11-20 03:31:41 -04:00
|
|
|
_armed.lockdown = false;
|
|
|
|
_armed.force_failsafe = false;
|
2017-04-23 22:32:33 -03:00
|
|
|
|
|
|
|
if (_actuator_armed_pub == nullptr) {
|
|
|
|
_actuator_armed_pub = orb_advertise(ORB_ID(actuator_armed), &_armed);
|
|
|
|
} else {
|
|
|
|
orb_publish(ORB_ID(actuator_armed), _actuator_armed_pub, &_armed);
|
|
|
|
}
|
2014-11-20 03:31:41 -04:00
|
|
|
}
|
|
|
|
|
2016-04-13 03:28:50 -03:00
|
|
|
void PX4RCOutput::_send_outputs(void)
|
2013-01-25 04:16:28 -04:00
|
|
|
{
|
2015-11-19 23:11:07 -04:00
|
|
|
uint32_t now = AP_HAL::micros();
|
2013-01-25 04:16:28 -04:00
|
|
|
|
2014-11-06 02:29:14 -04:00
|
|
|
if ((_enabled_channels & ((1U<<_servo_count)-1)) == 0) {
|
|
|
|
// no channels enabled
|
2014-11-20 03:31:41 -04:00
|
|
|
_arm_actuators(false);
|
2014-11-06 02:29:14 -04:00
|
|
|
goto update_pwm;
|
|
|
|
}
|
|
|
|
|
2013-01-25 04:16:28 -04:00
|
|
|
// always send at least at 20Hz, otherwise the IO board may think
|
|
|
|
// we are dead
|
|
|
|
if (now - _last_output > 50000) {
|
|
|
|
_need_update = true;
|
|
|
|
}
|
|
|
|
|
2015-12-05 17:31:45 -04:00
|
|
|
// check for PWM count changing. This can happen then the user changes BRD_PWM_COUNT
|
|
|
|
if (now - _last_config_us > 1000000) {
|
|
|
|
if (_pwm_fd != -1) {
|
|
|
|
ioctl(_pwm_fd, PWM_SERVO_GET_COUNT, (unsigned long)&_servo_count);
|
|
|
|
}
|
|
|
|
if (_alt_fd != -1) {
|
|
|
|
ioctl(_alt_fd, PWM_SERVO_GET_COUNT, (unsigned long)&_alt_servo_count);
|
|
|
|
}
|
|
|
|
_last_config_us = now;
|
|
|
|
}
|
|
|
|
|
2013-01-25 04:16:28 -04:00
|
|
|
if (_need_update && _pwm_fd != -1) {
|
|
|
|
_need_update = false;
|
|
|
|
perf_begin(_perf_rcout);
|
2016-04-15 03:38:58 -03:00
|
|
|
uint8_t to_send = _max_channel<_servo_count?_max_channel:_servo_count;
|
|
|
|
if (_sbus_enabled) {
|
|
|
|
to_send = _max_channel;
|
|
|
|
}
|
|
|
|
if (to_send > 0) {
|
|
|
|
for (int i=to_send-1; i >= 0; i--) {
|
2017-02-01 05:43:23 -04:00
|
|
|
if (_period[i] == PWM_IGNORE_THIS_CHANNEL) {
|
2016-04-15 03:38:58 -03:00
|
|
|
to_send = i;
|
2016-04-21 22:51:08 -03:00
|
|
|
} else {
|
|
|
|
break;
|
2016-04-15 03:38:58 -03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (to_send > 0) {
|
2017-04-23 22:32:33 -03:00
|
|
|
_arm_actuators(true);
|
2017-04-18 15:25:29 -03:00
|
|
|
|
2016-04-15 03:38:58 -03:00
|
|
|
::write(_pwm_fd, _period, to_send*sizeof(_period[0]));
|
|
|
|
}
|
2016-03-12 19:03:56 -04:00
|
|
|
if (_max_channel > _servo_count) {
|
|
|
|
// maybe send updates to alt_fd
|
2013-04-25 06:59:30 -03:00
|
|
|
if (_alt_fd != -1 && _alt_servo_count > 0) {
|
|
|
|
uint8_t n = _max_channel - _servo_count;
|
|
|
|
if (n > _alt_servo_count) {
|
|
|
|
n = _alt_servo_count;
|
|
|
|
}
|
2016-04-13 05:24:31 -03:00
|
|
|
if (n > 0) {
|
|
|
|
::write(_alt_fd, &_period[_servo_count], n*sizeof(_period[0]));
|
|
|
|
}
|
2013-04-25 06:59:30 -03:00
|
|
|
}
|
|
|
|
}
|
2018-03-09 04:30:54 -04:00
|
|
|
|
2013-01-25 04:16:28 -04:00
|
|
|
perf_end(_perf_rcout);
|
|
|
|
_last_output = now;
|
|
|
|
}
|
2014-11-05 06:38:12 -04:00
|
|
|
|
|
|
|
update_pwm:
|
2015-06-03 22:00:34 -03:00
|
|
|
for (uint8_t i=0; i<ORB_MULTI_MAX_INSTANCES; i++) {
|
|
|
|
bool rc_updated = false;
|
|
|
|
if (_outputs[i].pwm_sub >= 0 &&
|
|
|
|
orb_check(_outputs[i].pwm_sub, &rc_updated) == 0 &&
|
|
|
|
rc_updated) {
|
|
|
|
orb_copy(ORB_ID(actuator_outputs), _outputs[i].pwm_sub, &_outputs[i].outputs);
|
|
|
|
}
|
|
|
|
}
|
2014-11-05 06:38:12 -04:00
|
|
|
|
2013-01-25 04:16:28 -04:00
|
|
|
}
|
|
|
|
|
2016-04-13 03:28:50 -03:00
|
|
|
void PX4RCOutput::cork()
|
|
|
|
{
|
|
|
|
#if RCOUT_DEBUG_LATENCY
|
|
|
|
hal.gpio->pinMode(55, HAL_GPIO_OUTPUT);
|
|
|
|
hal.gpio->write(55, 1);
|
|
|
|
#endif
|
|
|
|
_corking = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void PX4RCOutput::push()
|
|
|
|
{
|
|
|
|
#if RCOUT_DEBUG_LATENCY
|
|
|
|
hal.gpio->pinMode(55, HAL_GPIO_OUTPUT);
|
|
|
|
hal.gpio->write(55, 0);
|
|
|
|
#endif
|
2017-04-17 21:02:11 -03:00
|
|
|
if (_corking) {
|
|
|
|
_corking = false;
|
2018-04-03 05:10:49 -03:00
|
|
|
if (_output_mode == MODE_PWM_ONESHOT ||
|
|
|
|
_output_mode == MODE_PWM_ONESHOT125) {
|
2017-04-17 21:02:11 -03:00
|
|
|
// run timer immediately in oneshot mode
|
|
|
|
_send_outputs();
|
|
|
|
}
|
2016-04-13 03:28:50 -03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-02-07 23:55:10 -04:00
|
|
|
void PX4RCOutput::timer_tick(void)
|
2016-04-13 03:28:50 -03:00
|
|
|
{
|
2018-04-03 05:10:49 -03:00
|
|
|
if (_output_mode != MODE_PWM_ONESHOT && _output_mode != MODE_PWM_ONESHOT125 && !_corking) {
|
2016-04-15 18:23:49 -03:00
|
|
|
/* in oneshot mode the timer does nothing as the outputs are
|
|
|
|
* sent from push() */
|
2016-04-13 03:28:50 -03:00
|
|
|
_send_outputs();
|
|
|
|
}
|
2016-05-27 03:18:05 -03:00
|
|
|
|
|
|
|
force_safety_pending_requests();
|
2016-04-13 03:28:50 -03:00
|
|
|
}
|
|
|
|
|
2016-04-15 03:38:58 -03:00
|
|
|
/*
|
|
|
|
enable sbus output
|
|
|
|
*/
|
2017-11-22 13:39:05 -04:00
|
|
|
bool PX4RCOutput::enable_px4io_sbus_out(uint16_t rate_hz)
|
2016-04-15 03:38:58 -03:00
|
|
|
{
|
|
|
|
int fd = open("/dev/px4io", 0);
|
|
|
|
if (fd == -1) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
for (uint8_t tries=0; tries<10; tries++) {
|
|
|
|
if (ioctl(fd, SBUS_SET_PROTO_VERSION, 1) != 0) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (ioctl(fd, PWM_SERVO_SET_SBUS_RATE, rate_hz) != 0) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
close(fd);
|
|
|
|
_sbus_enabled = true;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
close(fd);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
setup output mode
|
|
|
|
*/
|
2018-03-14 03:05:23 -03:00
|
|
|
void PX4RCOutput::set_output_mode(uint16_t mask, enum output_mode mode)
|
2016-04-15 03:38:58 -03:00
|
|
|
{
|
2016-04-16 05:52:31 -03:00
|
|
|
if (_output_mode == mode) {
|
|
|
|
// no change
|
|
|
|
return;
|
|
|
|
}
|
2018-04-03 05:10:49 -03:00
|
|
|
if (mode == MODE_PWM_ONESHOT || mode == MODE_PWM_ONESHOT125) {
|
2016-04-22 00:50:05 -03:00
|
|
|
// when using oneshot we don't want the regular pulses. The
|
|
|
|
// best we can do with the current PX4Firmware code is ask for
|
|
|
|
// 1Hz. This does still produce pulses, but the trigger calls
|
|
|
|
// mean the timer is constantly reset, so no pulses are
|
|
|
|
// produced except when triggered by push() when the main loop
|
|
|
|
// is running
|
2017-04-07 02:07:17 -03:00
|
|
|
set_freq_fd(_pwm_fd, _rate_mask_main, 1, _rate_mask_main);
|
|
|
|
if (_alt_fd != -1) {
|
|
|
|
set_freq_fd(_alt_fd, _rate_mask_alt, 1, _rate_mask_alt);
|
|
|
|
}
|
2016-04-22 00:24:24 -03:00
|
|
|
}
|
2016-04-15 03:38:58 -03:00
|
|
|
_output_mode = mode;
|
2016-11-29 06:10:27 -04:00
|
|
|
switch (_output_mode) {
|
|
|
|
case MODE_PWM_ONESHOT:
|
2018-04-03 05:10:49 -03:00
|
|
|
case MODE_PWM_ONESHOT125:
|
2016-04-15 03:38:58 -03:00
|
|
|
ioctl(_pwm_fd, PWM_SERVO_SET_ONESHOT, 1);
|
2016-04-15 18:23:49 -03:00
|
|
|
if (_alt_fd != -1) {
|
|
|
|
ioctl(_alt_fd, PWM_SERVO_SET_ONESHOT, 1);
|
|
|
|
}
|
2016-11-29 06:10:27 -04:00
|
|
|
break;
|
2018-03-15 17:28:38 -03:00
|
|
|
case MODE_PWM_DSHOT150:
|
|
|
|
case MODE_PWM_DSHOT300:
|
|
|
|
case MODE_PWM_DSHOT600:
|
|
|
|
case MODE_PWM_DSHOT1200:
|
2018-03-16 18:49:24 -03:00
|
|
|
case MODE_PWM_NONE:
|
2018-03-15 17:28:38 -03:00
|
|
|
// treat as normal PWM for now
|
|
|
|
hal.console->printf("DShot not supported\n");
|
|
|
|
FALLTHROUGH;
|
2016-11-29 06:10:27 -04:00
|
|
|
case MODE_PWM_NORMAL:
|
2016-04-15 03:38:58 -03:00
|
|
|
ioctl(_pwm_fd, PWM_SERVO_SET_ONESHOT, 0);
|
2016-04-15 18:23:49 -03:00
|
|
|
if (_alt_fd != -1) {
|
|
|
|
ioctl(_alt_fd, PWM_SERVO_SET_ONESHOT, 0);
|
|
|
|
}
|
2016-11-29 06:10:27 -04:00
|
|
|
break;
|
2017-06-16 08:23:37 -03:00
|
|
|
case MODE_PWM_BRUSHED:
|
2016-11-29 06:10:27 -04:00
|
|
|
// setup an 8MHz clock. This has the effect of scaling all outputs by 8x
|
|
|
|
ioctl(_pwm_fd, PWM_SERVO_SET_UPDATE_CLOCK, 8);
|
|
|
|
if (_alt_fd != -1) {
|
|
|
|
ioctl(_alt_fd, PWM_SERVO_SET_UPDATE_CLOCK, 8);
|
|
|
|
}
|
|
|
|
break;
|
2016-04-15 03:38:58 -03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-07-16 20:19:00 -03:00
|
|
|
// set default output update rate
|
|
|
|
void PX4RCOutput::set_default_rate(uint16_t rate_hz)
|
|
|
|
{
|
2017-07-25 00:13:40 -03:00
|
|
|
if (rate_hz != _default_rate_hz) {
|
|
|
|
// set servo update rate for first 8 pwm channels
|
|
|
|
ioctl(_pwm_fd, PWM_SERVO_SET_DEFAULT_UPDATE_RATE, rate_hz);
|
|
|
|
if (_alt_fd != -1) {
|
|
|
|
// set servo update rate for auxiliary channels
|
|
|
|
ioctl(_alt_fd, PWM_SERVO_SET_DEFAULT_UPDATE_RATE, rate_hz);
|
|
|
|
}
|
|
|
|
_default_rate_hz = rate_hz;
|
|
|
|
}
|
2017-07-16 20:19:00 -03:00
|
|
|
}
|
|
|
|
|
2013-01-04 07:25:36 -04:00
|
|
|
#endif // CONFIG_HAL_BOARD
|