You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
186 lines
4.9 KiB
186 lines
4.9 KiB
/* |
|
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/>. |
|
*/ |
|
|
|
/* |
|
SBUS output support |
|
*/ |
|
|
|
#include "ch.h" |
|
#include "hal.h" |
|
#include "iofirmware.h" |
|
#include "rc.h" |
|
#include <AP_HAL/AP_HAL.h> |
|
#include <AP_Math/AP_Math.h> |
|
#include <AP_SBusOut/AP_SBusOut.h> |
|
|
|
extern const AP_HAL::HAL& hal; |
|
|
|
// usart3 is for SBUS input and output |
|
static const SerialConfig uart3_cfg = { |
|
100000, // speed |
|
USART_CR1_PCE | USART_CR1_M, // cr1, enable even parity |
|
USART_CR2_STOP_1, // cr2, two stop bits |
|
0, // cr3 |
|
nullptr, // irq_cb |
|
nullptr, // ctx |
|
}; |
|
|
|
// listen for parity errors on sd3 input |
|
static event_listener_t sd3_listener; |
|
|
|
void sbus_out_write(uint16_t *channels, uint8_t nchannels) |
|
{ |
|
uint8_t buffer[25]; |
|
AP_SBusOut::sbus_format_frame(channels, nchannels, buffer); |
|
chnWrite(&SD3, buffer, sizeof(buffer)); |
|
} |
|
|
|
// usart1 is for DSM input and (optionally) debug to FMU |
|
static const SerialConfig uart1_cfg = { |
|
115200, // speed |
|
0, // cr1 |
|
0, // cr2 |
|
0, // cr3 |
|
nullptr, // irq_cb |
|
nullptr, // ctx |
|
}; |
|
|
|
/* |
|
init rcin on DSM USART1 |
|
*/ |
|
void AP_IOMCU_FW::rcin_serial_init(void) |
|
{ |
|
sdStart(&SD1, &uart1_cfg); |
|
sdStart(&SD3, &uart3_cfg); |
|
chEvtRegisterMaskWithFlags(chnGetEventSource(&SD3), |
|
&sd3_listener, |
|
EVENT_MASK(1), |
|
SD_PARITY_ERROR); |
|
rcprotocol = AP_RCProtocol::get_singleton(); |
|
|
|
// disable input for SBUS with pulses, we will use the UART for |
|
// SBUS. |
|
rcprotocol->disable_for_pulses(AP_RCProtocol::SBUS); |
|
rcprotocol->disable_for_pulses(AP_RCProtocol::SBUS_NI); |
|
} |
|
|
|
static struct { |
|
uint32_t num_dsm_bytes; |
|
uint32_t num_sbus_bytes; |
|
uint32_t num_sbus_errors; |
|
eventflags_t sbus_error; |
|
} rc_stats; |
|
|
|
/* |
|
check for data on DSM RX uart |
|
*/ |
|
void AP_IOMCU_FW::rcin_serial_update(void) |
|
{ |
|
uint8_t b[16]; |
|
uint32_t n; |
|
|
|
// read from DSM port |
|
if ((n = chnReadTimeout(&SD1, b, sizeof(b), TIME_IMMEDIATE)) > 0) { |
|
n = MIN(n, sizeof(b)); |
|
rc_stats.num_dsm_bytes += n; |
|
for (uint8_t i=0; i<n; i++) { |
|
rcprotocol->process_byte(b[i], 115200); |
|
} |
|
//BLUE_TOGGLE(); |
|
} |
|
|
|
// read from SBUS port |
|
if ((n = chnReadTimeout(&SD3, b, sizeof(b), TIME_IMMEDIATE)) > 0) { |
|
eventflags_t flags; |
|
if ((flags = chEvtGetAndClearFlags(&sd3_listener)) & SD_PARITY_ERROR) { |
|
rc_stats.sbus_error = flags; |
|
rc_stats.num_sbus_errors++; |
|
} else { |
|
n = MIN(n, sizeof(b)); |
|
rc_stats.num_sbus_bytes += n; |
|
for (uint8_t i=0; i<n; i++) { |
|
rcprotocol->process_byte(b[i], 100000); |
|
} |
|
} |
|
} |
|
} |
|
|
|
/* |
|
sleep for 1ms using a busy loop |
|
*/ |
|
static void delay_one_ms(uint32_t &now) |
|
{ |
|
while (now == AP_HAL::millis()) ; |
|
now = AP_HAL::millis(); |
|
} |
|
|
|
/* |
|
perform a DSM bind operation |
|
*/ |
|
void AP_IOMCU_FW::dsm_bind_step(void) |
|
{ |
|
uint32_t now = last_ms; |
|
switch (dsm_bind_state) { |
|
case 1: |
|
palSetLineMode(HAL_GPIO_PIN_SPEKTRUM_PWR_EN, PAL_MODE_OUTPUT_PUSHPULL); |
|
SPEKTRUM_POWER(0); |
|
palSetLineMode(HAL_GPIO_PIN_SPEKTRUM_OUT, PAL_MODE_OUTPUT_PUSHPULL); |
|
SPEKTRUM_SET(1); |
|
dsm_bind_state = 2; |
|
last_dsm_bind_ms = now; |
|
break; |
|
|
|
case 2: |
|
if (now - last_dsm_bind_ms >= 500) { |
|
SPEKTRUM_POWER(1); |
|
dsm_bind_state = 3; |
|
last_dsm_bind_ms = now; |
|
} |
|
break; |
|
|
|
case 3: { |
|
if (now - last_dsm_bind_ms >= 72) { |
|
// 9 pulses works with all satellite receivers, and supports the highest |
|
// available protocol |
|
delay_one_ms(now); |
|
const uint8_t num_pulses = 9; |
|
for (uint8_t i=0; i<num_pulses; i++) { |
|
// the delay should be 120us, but we are running our |
|
// clock at 1kHz, and 1ms works fine |
|
delay_one_ms(now); |
|
SPEKTRUM_SET(0); |
|
delay_one_ms(now); |
|
SPEKTRUM_SET(1); |
|
} |
|
last_dsm_bind_ms = now; |
|
dsm_bind_state = 4; |
|
} |
|
break; |
|
} |
|
|
|
case 4: |
|
if (now - last_dsm_bind_ms >= 50) { |
|
// set back as input pin |
|
palSetLineMode(HAL_GPIO_PIN_SPEKTRUM_OUT, PAL_MODE_INPUT); |
|
dsm_bind_state = 0; |
|
} |
|
break; |
|
|
|
default: |
|
dsm_bind_state = 0; |
|
break; |
|
} |
|
} |
|
|
|
|