Browse Source

robustify S.port and D telemetry detection (#4731)

sbg
Mark Whitehorn 9 years ago committed by Lorenz Meier
parent
commit
0a27d14f6c
  1. 94
      src/drivers/frsky_telemetry/frsky_telemetry.c

94
src/drivers/frsky_telemetry/frsky_telemetry.c

@ -67,8 +67,8 @@ @@ -67,8 +67,8 @@
static volatile bool thread_should_exit = false;
static volatile bool thread_running = false;
static int frsky_task;
typedef enum { IDLE, SPORT, DTYPE } frsky_state_t;
static frsky_state_t frsky_state = IDLE;
typedef enum { SCANNING, SPORT, DTYPE } frsky_state_t;
static frsky_state_t frsky_state = SCANNING;
/* functions */
static int sPort_open_uart(const char *uart_name, struct termios *uart_config, struct termios *uart_config_original);
@ -77,6 +77,7 @@ static void usage(void); @@ -77,6 +77,7 @@ static void usage(void);
static int frsky_telemetry_thread_main(int argc, char *argv[]);
__EXPORT int frsky_telemetry_main(int argc, char *argv[]);
#define DEBUG
/**
* Opens the UART device and sets all required serial parameters.
@ -106,7 +107,7 @@ static int sPort_open_uart(const char *uart_name, struct termios *uart_config, s @@ -106,7 +107,7 @@ static int sPort_open_uart(const char *uart_name, struct termios *uart_config, s
uart_config->c_oflag &= ~OPOST;
/* Set baud rate */
static const speed_t speed = B57600;
static const speed_t speed = B9600;
if (cfsetispeed(uart_config, speed) < 0 || cfsetospeed(uart_config, speed) < 0) {
warnx("ERR: %s: %d (cfsetispeed, cfsetospeed)\n", uart_name, termios_state);
@ -175,7 +176,7 @@ static int frsky_telemetry_thread_main(int argc, char *argv[]) @@ -175,7 +176,7 @@ static int frsky_telemetry_thread_main(int argc, char *argv[])
}
}
/* Open UART assuming SmartPort telemetry */
/* Open UART assuming D type telemetry */
struct termios uart_config_original;
struct termios uart_config;
const int uart = sPort_open_uart(device_name, &uart_config, &uart_config_original);
@ -194,9 +195,10 @@ static int frsky_telemetry_thread_main(int argc, char *argv[]) @@ -194,9 +195,10 @@ static int frsky_telemetry_thread_main(int argc, char *argv[])
/* Main thread loop */
char sbuf[20];
frsky_state = IDLE;
frsky_state = SCANNING;
frsky_state_t baudRate = DTYPE;
while (!thread_should_exit && frsky_state == IDLE) {
while (!thread_should_exit && frsky_state == SCANNING) {
/* 2 byte polling frames indicate SmartPort telemetry
* 11 byte packets indicate D type telemetry
*/
@ -204,18 +206,64 @@ static int frsky_telemetry_thread_main(int argc, char *argv[]) @@ -204,18 +206,64 @@ static int frsky_telemetry_thread_main(int argc, char *argv[])
if (status > 0) {
/* traffic on the port, D type is 11 bytes per frame, SmartPort is only 2
* allow a little time to receive the entire packet
* Wait long enough for 11 bytes at 9600 baud
*/
usleep(5000);
status = read(uart, &sbuf[0], sizeof(sbuf));
}
usleep(12000);
int nbytes = read(uart, &sbuf[0], sizeof(sbuf));
PX4_DEBUG("frsky input: %d bytes: %x %x, speed: %d", nbytes, sbuf[0], sbuf[1], baudRate);
// look for valid header byte
if (nbytes > 10) {
if (baudRate == DTYPE) {
// see if we got a valid D-type hostframe
struct adc_linkquality host_frame;
if (frsky_parse_host((uint8_t *)&sbuf[0], nbytes, &host_frame)) {
frsky_state = baudRate;
break;
}
/* received some data; check size of packet */
if (status > 0 && status < 3) {
frsky_state = SPORT;
} else {
// check for alternating S.port start bytes
int index = 0;
while (index < 2 && sbuf[index] != 0x7E) { index++; }
if (index < 2) {
int success = 1;
for (int i = index + 2; i < nbytes; i += 2) {
if (sbuf[i] != 0x7E) { success = 0; break; }
}
if (success) {
frsky_state = baudRate;
break;
}
}
}
}
// alternate between S.port and D-type baud rates
if (baudRate == SPORT) {
PX4_DEBUG("setting baud rate to %d", 9600);
set_uart_speed(uart, &uart_config, B9600);
baudRate = DTYPE;
} else {
PX4_DEBUG("setting baud rate to %d", 57600);
set_uart_speed(uart, &uart_config, B57600);
baudRate = SPORT;
}
// wait a second
usleep(1000000);
// flush buffer
read(uart, &sbuf[0], sizeof(sbuf));
} else if (status > 0) {
frsky_state = DTYPE;
}
}
@ -225,7 +273,7 @@ static int frsky_telemetry_thread_main(int argc, char *argv[]) @@ -225,7 +273,7 @@ static int frsky_telemetry_thread_main(int argc, char *argv[])
err(1, "could not allocate memory");
}
warnx("sending FrSky SmartPort telemetry");
PX4_INFO("sending FrSky SmartPort telemetry");
struct sensor_baro_s *sensor_baro = malloc(sizeof(struct sensor_baro_s));
@ -434,7 +482,7 @@ static int frsky_telemetry_thread_main(int argc, char *argv[]) @@ -434,7 +482,7 @@ static int frsky_telemetry_thread_main(int argc, char *argv[])
}
}
warnx("freeing sPort memory");
PX4_DEBUG("freeing sPort memory");
sPort_deinit();
free(sensor_baro);
@ -442,11 +490,11 @@ static int frsky_telemetry_thread_main(int argc, char *argv[]) @@ -442,11 +490,11 @@ static int frsky_telemetry_thread_main(int argc, char *argv[])
} else if (frsky_state == DTYPE) {
/* detected D type telemetry: reconfigure UART */
warnx("sending FrSky D type telemetry");
PX4_INFO("sending FrSky D type telemetry");
int status = set_uart_speed(uart, &uart_config, B9600);
if (status < 0) {
warnx("error setting speed for %s, quitting", device_name);
PX4_DEBUG("error setting speed for %s, quitting", device_name);
/* Reset the UART flags to original state */
tcsetattr(uart, TCSANOW, &uart_config_original);
close(uart);
@ -464,8 +512,6 @@ static int frsky_telemetry_thread_main(int argc, char *argv[]) @@ -464,8 +512,6 @@ static int frsky_telemetry_thread_main(int argc, char *argv[])
struct adc_linkquality host_frame;
// uint8_t dbuf[45];
/* send D8 mode telemetry */
while (!thread_should_exit) {
@ -506,7 +552,7 @@ static int frsky_telemetry_thread_main(int argc, char *argv[]) @@ -506,7 +552,7 @@ static int frsky_telemetry_thread_main(int argc, char *argv[])
// /* TODO: flush the input buffer if in full duplex mode */
// read(uart, &sbuf[0], sizeof(sbuf));
warnx("freeing frsky memory");
PX4_DEBUG("freeing frsky memory");
frsky_deinit();
}
@ -573,8 +619,8 @@ int frsky_telemetry_main(int argc, char *argv[]) @@ -573,8 +619,8 @@ int frsky_telemetry_main(int argc, char *argv[])
if (!strcmp(argv[1], "status")) {
if (thread_running) {
switch (frsky_state) {
case IDLE:
errx(0, "running: IDLE");
case SCANNING:
errx(0, "running: SCANNING");
break;
case SPORT:

Loading…
Cancel
Save