/* * Copyright © 2009 Keith Packard * * 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); }