ao-tools/ao-dumplog: Dump entire eeprom contents
[fw/altos] / ao-tools / ao-dumplog / ao-dumplog.c
index b930f0e5769ad4c1f2818663ada6a25df9e560c8..41d0ac102d27d86cb24e433bdde9b9441bb6f740 100644 (file)
@@ -20,6 +20,7 @@
 #include <stdlib.h>
 #include <unistd.h>
 #include <getopt.h>
+#include <string.h>
 #include "cc-usb.h"
 #include "cc.h"
 
 static const struct option options[] = {
        { .name = "tty", .has_arg = 1, .val = 'T' },
        { .name = "device", .has_arg = 1, .val = 'D' },
+       { .name = "remote", .has_arg = 0, .val = 'R' },
+       { .name = "channel", .has_arg = 1, .val = 'C' },
        { 0, 0, 0, 0},
 };
 
 static void usage(char *program)
 {
-       fprintf(stderr, "usage: %s [--tty <tty-name>] [--device <device-name>\n", program);
+       fprintf(stderr, "usage: %s [--tty <tty-name>] [--device <device-name>] [--remote] [--channel <radio-channel>]\n", program);
        exit(1);
 }
 
+static uint8_t
+log_checksum(int d[8])
+{
+       uint8_t sum = 0x5a;
+       int     i;
+
+       for (i = 0; i < 8; i++)
+               sum += (uint8_t) d[i];
+       return -sum;
+}
+
+static const char *state_names[] = {
+       "startup",
+       "idle",
+       "pad",
+       "boost",
+       "fast",
+       "coast",
+       "drogue",
+       "main",
+       "landed",
+       "invalid"
+};
+
+
 int
 main (int argc, char **argv)
 {
@@ -47,11 +75,25 @@ main (int argc, char **argv)
        char            line[8192];
        FILE            *out;
        char            *filename;
-       int             serial_number;
+       int             serial_number = 0;
+       int             channel = 0;
+       int             flight = 0;
        char            cmd;
        int             tick, a, b;
+       int             block;
+       int             addr;
+       int             received_addr;
+       int             data[8];
+       int             done;
+       int             column;
+       int             remote = 0;
+       int             any_valid;
+       int             invalid;
+       char            serial_line[8192];
+       unsigned        storage_size;
+       int             blocks;
 
-       while ((c = getopt_long(argc, argv, "T:D:", options, NULL)) != -1) {
+       while ((c = getopt_long(argc, argv, "T:D:C:R", options, NULL)) != -1) {
                switch (c) {
                case 'T':
                        tty = optarg;
@@ -59,13 +101,23 @@ main (int argc, char **argv)
                case 'D':
                        device = optarg;
                        break;
+               case 'R':
+                       remote = 1;
+                       break;
+               case 'C':
+                       channel = atoi(optarg);
+                       break;
                default:
                        usage(argv[0]);
                        break;
                }
        }
-       if (!tty)
-               tty = cc_usbdevs_find_by_arg(device, "TeleMetrum");
+       if (!tty) {
+               if (remote)
+                       tty = cc_usbdevs_find_by_arg(device, "TeleDongle");
+               else
+                       tty = cc_usbdevs_find_by_arg(device, "TeleMetrum");
+       }
        if (!tty)
                tty = getenv("ALTOS_TTY");
        if (!tty)
@@ -73,30 +125,104 @@ main (int argc, char **argv)
        cc = cc_usb_open(tty);
        if (!cc)
                exit(1);
+       if (remote)
+               cc_usb_open_remote(cc, channel);
        /* send a 'version' command followed by a 'log' command */
-       cc_usb_printf(cc, "v\nl\n");
+       cc_usb_printf(cc, "v\n");
        out = NULL;
        for (;;) {
                cc_usb_getline(cc, line, sizeof (line));
-               if (!strcmp (line, "end"))
+               if (sscanf(line, "serial-number %u", &serial_number) == 1)
+                       strcpy(serial_line, line);
+               if (!strncmp(line, "software-version", 16))
                        break;
-               if (sscanf(line, "serial-number %u", &serial_number) == 1) {
-                       filename = cc_make_filename(serial_number, "eeprom");
-                       out = fopen (filename, "w");
-                       if (!out) {
-                               perror(filename);
-                       }
-                       fprintf (out, "%s\n", line);
-               } else if (sscanf(line, "%c %x %x %x", &cmd, &tick, &a, &b) == 4) {
-                       if (out) {
-                               fprintf(out, "%s\n", line);
-                               if (cmd == 'S' && a == 8) {
-                                       fclose(out);
-                                       out = NULL;
+       }
+       if (!serial_number) {
+               fprintf(stderr, "no serial number found\n");
+               cc_usb_close(cc);
+               exit(1);
+       }
+       cc_usb_printf(cc, "f\n");
+       storage_size = 0;
+       for (;;) {
+               cc_usb_getline(cc, line, sizeof(line));
+               if (sscanf(line, "Storage size: %u", &storage_size) == 1)
+                       break;
+       }
+       printf ("Serial number: %d Storage size: %u\n", serial_number, storage_size);
+       if (storage_size)
+               blocks = storage_size / 256;
+       else
+               blocks = 511;
+       done = 0;
+       column = 0;
+       for (block = 0; !done && block < blocks; block++) {
+               cc_usb_printf(cc, "e %x\n", block);
+               if (column == 64) {
+                       putchar('\n');
+                       column = 0;
+               }
+               putchar('.'); fflush(stdout); column++;
+               any_valid = 0;
+               for (addr = 0; addr < 0x100;) {
+                       cc_usb_getline(cc, line, sizeof (line));
+                       if (sscanf(line, "00%x %x %x %x %x %x %x %x %x",
+                                         &received_addr,
+                                         &data[0], &data[1], &data[2], &data[3],
+                                         &data[4], &data[5], &data[6], &data[7]) == 9)
+                       {
+                               if (received_addr != addr)
+                                       fprintf(stderr, "data out of sync at 0x%x\n",
+                                               block * 256 + received_addr);
+
+                               if (log_checksum(data) != 0)
+                                       fprintf (stderr, "invalid checksum at 0x%x\n",
+                                                block * 256 + received_addr);
+                               else
+                                       any_valid = 1;
+
+                               cmd = data[0];
+                               tick = data[2] + (data[3] << 8);
+                               a = data[4] + (data[5] << 8);
+                               b = data[6] + (data[7] << 8);
+                               if (cmd == 'F') {
+                                       flight = b;
+                                       filename = cc_make_filename(serial_number, flight, "eeprom");
+                                       printf ("Flight:       %d\n", flight);
+                                       printf ("File name:     %s\n", filename);
+                                       out = fopen (filename, "w");
+                                       if (!out) {
+                                               perror(filename);
+                                               exit(1);
+                                       }
+                                       fprintf(out, "%s\n", serial_line);
+                               }
+
+                               if (cmd == 'S' && a <= 8) {
+                                       if (column) putchar('\n');
+                                       printf("%s\n", state_names[a]);
+                                       column = 0;
+                               }
+                               if (out) {
+                                       fprintf(out, "%c %4x %4x %4x\n",
+                                               cmd, tick, a, b);
+                                       if (cmd == 'S' && a == 8) {
+                                               fclose(out);
+                                               out = NULL;
+                                               done = 1;
+                                       }
                                }
+                               addr += 8;
                        }
                }
+               if (!any_valid) {
+                       fclose(out);
+                       out = NULL;
+                       done = 1;
+               }
        }
+       if (column)
+               putchar('\n');
        if (out)
                fclose (out);
        cc_usb_close(cc);