diff options
Diffstat (limited to 'aoview/aoview_monitor.c')
-rw-r--r-- | aoview/aoview_monitor.c | 144 |
1 files changed, 144 insertions, 0 deletions
diff --git a/aoview/aoview_monitor.c b/aoview/aoview_monitor.c new file mode 100644 index 00000000..1c9b7189 --- /dev/null +++ b/aoview/aoview_monitor.c @@ -0,0 +1,144 @@ +/* + * Copyright © 2009 Keith Packard <keithp@keithp.com> + * + * 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; version 2 of the License. + * + * 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, write to the Free Software Foundation, Inc., + * 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA. + */ + +#include "aoview.h" + +static struct aoview_serial *monitor_serial; + +#define MONITOR_LEN 1024 + +static char monitor_line[MONITOR_LEN + 1]; +static int monitor_pos; + +void +aoview_monitor_disconnect(void) +{ + if (monitor_serial) { + aoview_serial_close(monitor_serial); + monitor_serial = NULL; + } +} + +static void +aoview_parse_string(char *target, int len, char *source) +{ + strncpy(target, source, len-1); + target[len-1] = '\0'; +} + +static void +aoview_parse_int(int *target, char *source) +{ + *target = strtol(source, NULL, 0); +} + +static void +aoview_parse_pos(double *target, char *source) +{ + int deg; + double min; + char dir; + double r; + + if (sscanf(source, "%d°%lf'%c", °, &min, &dir) != 3) { + *target = 0; + return; + } + r = deg + min / 60.0; + if (dir == 'S' || dir == 'W') + r = -r; + *target = r; +} + +static void +aoview_monitor_parse(char *line) +{ + char *saveptr; + char *words[64]; + int nword; + struct aostate state; + + printf ("%s\n", line); + for (nword = 0; nword < 64; nword++) { + words[nword] = strtok_r(line, " \t\n", &saveptr); + line = NULL; + if (words[nword] == NULL) + break; + } + if (nword < 26) + return; + if (strcmp(words[0], "CALL") != 0) + return; + aoview_parse_string(state.callsign, sizeof (state.callsign), words[1]); + aoview_parse_int(&state.serial, words[3]); + aoview_parse_int(&state.rssi, words[5]); + aoview_parse_string(state.state, sizeof (state.state), words[9]); + aoview_parse_int(&state.tick, words[10]); + aoview_parse_int(&state.accel, words[12]); + aoview_parse_int(&state.pres, words[14]); + aoview_parse_int(&state.temp, words[16]); + aoview_parse_int(&state.batt, words[18]); + aoview_parse_int(&state.drogue, words[20]); + aoview_parse_int(&state.main, words[22]); + if (strcmp (words[26], "unlocked") != 0 && nword >= 29) { + sscanf(words[26], "%d:%d:%d", &state.gps_time.hour, &state.gps_time.minute, &state.gps_time.second); + aoview_parse_pos(&state.lat, words[27]); + aoview_parse_pos(&state.lon, words[28]); + sscanf(words[29], "%dm", &state.alt); + } else { + state.gps_time.hour = state.gps_time.minute = state.gps_time.second = 0; + state.lat = state.lon = 0; + state.alt = 0; + } + aoview_state_notify(&state); +} + +static gboolean +aoview_monitor_callback(void *user_data) +{ + int c; + + if (!monitor_serial) + return FALSE; + + for (;;) { + c = aoview_serial_getc(monitor_serial); + if (c == -1) + break; + if (c == '\r') + continue; + if (c == '\n') { + monitor_line[monitor_pos] = '\0'; + if (monitor_pos) + aoview_monitor_parse(monitor_line); + monitor_pos = 0; + } else if (monitor_pos < MONITOR_LEN) + monitor_line[monitor_pos++] = c; + } + return TRUE; +} + +void +aoview_monitor_connect(char *tty) +{ + aoview_monitor_disconnect(); + monitor_serial = aoview_serial_open(tty); + aoview_serial_set_callback(monitor_serial, + aoview_monitor_callback, + monitor_serial, + NULL); +} |