285 lines
6.3 KiB
C
285 lines
6.3 KiB
C
/*
|
|
* Copyright © 2014 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; either version 2 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, write to the Free Software Foundation, Inc.,
|
|
* 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA.
|
|
*/
|
|
|
|
#include <err.h>
|
|
#include <fcntl.h>
|
|
#include <gelf.h>
|
|
#include <stdio.h>
|
|
#include <stdint.h>
|
|
#include <stdlib.h>
|
|
#include <sysexits.h>
|
|
#include <unistd.h>
|
|
#include <getopt.h>
|
|
#include <string.h>
|
|
#include <stdbool.h>
|
|
#include <ctype.h>
|
|
#include "ao-elf.h"
|
|
#include "ccdbg.h"
|
|
#include "cc-usb.h"
|
|
#include "cc.h"
|
|
#include "ao-verbose.h"
|
|
|
|
static const struct option options[] = {
|
|
{ .name = "tty", .has_arg = 1, .val = 'T' },
|
|
{ .name = "device", .has_arg = 1, .val = 'D' },
|
|
{ .name = "raw", .has_arg = 0, .val = 'r' },
|
|
{ .name = "verbose", .has_arg = 1, .val = 'v' },
|
|
{ .name = "rplus", .has_arg = 1, .val = 'a' },
|
|
{ .name = "rminus", .has_arg = 1, .val = 'b' },
|
|
{ .name = "adcmax", .has_arg = 1, .val = 'm' },
|
|
{ 0, 0, 0, 0},
|
|
};
|
|
|
|
static void usage(char *program)
|
|
{
|
|
fprintf(stderr, "usage: %s [--verbose=<verbose>] [--device=<device>] [--tty=<tty>] [--rplus=val] [--rminus=val] [--adcmax=val] main|drogue\n", program);
|
|
exit(1);
|
|
}
|
|
|
|
static void
|
|
done(struct cc_usb *cc, int code)
|
|
{
|
|
/* cc_usb_printf(cc, "a\n"); */
|
|
cc_usb_close(cc);
|
|
exit (code);
|
|
}
|
|
|
|
struct igniter {
|
|
struct igniter *next;
|
|
char name[512];
|
|
char status[512];
|
|
int adc;
|
|
};
|
|
|
|
static bool
|
|
map_igniter_name(char *adc_name, char *igniter_name)
|
|
{
|
|
char *colon = strchr(adc_name, ':');
|
|
if (!colon)
|
|
return false;
|
|
*colon = '\0';
|
|
if (strlen(adc_name) == 1 && isupper(adc_name[0])) {
|
|
igniter_name[0] = '0' + adc_name[0] - 'A';
|
|
igniter_name[1] = '\0';
|
|
} else {
|
|
strcpy(igniter_name, adc_name);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static const char *
|
|
other_igniter_name(const char *name)
|
|
{
|
|
if (!strcmp(name, "drogue"))
|
|
return "apogee";
|
|
if (!strcmp(name, "apogee"))
|
|
return "drogue";
|
|
return name;
|
|
}
|
|
|
|
static struct igniter *
|
|
igniters(struct cc_usb *usb)
|
|
{
|
|
struct igniter *head = NULL, **tail = &head;
|
|
cc_usb_printf(usb, "t\na\nv\n");
|
|
for (;;) {
|
|
char line[512];
|
|
char name[512];
|
|
char status[512];
|
|
char adc_name[512];
|
|
char igniter_name[512];
|
|
|
|
cc_usb_getline(usb, line, sizeof (line));
|
|
if (strstr(line, "software-version"))
|
|
break;
|
|
if (sscanf(line, "Igniter: %s Status: %s", name, status) == 2) {
|
|
struct igniter *i = calloc (1, sizeof (struct igniter));
|
|
strcpy(i->name, name);
|
|
strcpy(i->status, status);
|
|
i->next = NULL;
|
|
*tail = i;
|
|
tail = &i->next;
|
|
}
|
|
if (strncmp(line, "tick:", 5) == 0) {
|
|
char *tok;
|
|
char *l = line;
|
|
bool found_igniter = false;
|
|
|
|
while ((tok = strtok(l, " ")) != NULL) {
|
|
l = NULL;
|
|
if (found_igniter) {
|
|
struct igniter *i;
|
|
for (i = head; i; i = i->next)
|
|
if (!strcmp(i->name, igniter_name) ||
|
|
!strcmp(i->name, other_igniter_name(igniter_name)))
|
|
{
|
|
i->adc = atoi(tok);
|
|
break;
|
|
}
|
|
found_igniter = false;
|
|
} else {
|
|
strcpy(adc_name, tok);
|
|
found_igniter = map_igniter_name(adc_name, igniter_name);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return head;
|
|
}
|
|
|
|
static void
|
|
free_igniters(struct igniter *i) {
|
|
struct igniter *n;
|
|
|
|
while (i) {
|
|
n = i->next;
|
|
free(i);
|
|
i = n;
|
|
}
|
|
}
|
|
|
|
static struct igniter *
|
|
find_igniter(struct igniter *i, char *name)
|
|
{
|
|
for (; i; i = i->next)
|
|
if (strcmp(i->name, name) == 0) {
|
|
printf("igniter %s adc %d\n", i->name, i->adc);
|
|
return i;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
static const double ref_volts = 3.3;
|
|
|
|
static double
|
|
compute_voltage(int adc, double rplus, double rminus, int adc_max)
|
|
{
|
|
return (double) adc / (double) adc_max * ref_volts * (rplus + rminus) / rminus;
|
|
}
|
|
|
|
static int
|
|
do_igniter(struct cc_usb *usb, char *name, double rplus, double rminus, int adc_max)
|
|
{
|
|
struct igniter *all = igniters(usb);
|
|
struct igniter *this = find_igniter(all, name);
|
|
double volts = -1;
|
|
if (!this) {
|
|
struct igniter *i;
|
|
printf("no igniter %s found in", name);
|
|
for (i = all; i; i = i->next)
|
|
printf(" %s", i->name);
|
|
printf("\n");
|
|
free_igniters(all);
|
|
return 0;
|
|
}
|
|
if (rplus && rminus && adc_max) {
|
|
volts = compute_voltage(this->adc, rplus, rminus, adc_max);
|
|
if (volts < 1 || volts > 4) {
|
|
printf("igniter %s voltage is %f, not in range of 1-4 volts\n", this->name, volts);
|
|
free_igniters(all);
|
|
return 0;
|
|
}
|
|
}
|
|
if (strcmp(this->status, "open") == 0) {
|
|
printf("igniter %s status is %s\n", this->name, this->status);
|
|
free_igniters(all);
|
|
return 0;
|
|
}
|
|
cc_usb_printf(usb, "i DoIt %s\n", this->name);
|
|
cc_usb_sync(usb);
|
|
free_igniters(all);
|
|
usleep(200000);
|
|
return 1;
|
|
}
|
|
|
|
int
|
|
main (int argc, char **argv)
|
|
{
|
|
char *device = NULL;
|
|
int i;
|
|
int c;
|
|
struct cc_usb *cc = NULL;
|
|
char *tty = NULL;
|
|
int verbose = 0;
|
|
int ret = 0;
|
|
double rplus = 0.0;
|
|
double rminus = 0.0;
|
|
int adcmax = 0;
|
|
|
|
while ((c = getopt_long(argc, argv, "rT:D:c:s:v:a:b:m:", options, NULL)) != -1) {
|
|
switch (c) {
|
|
case 'T':
|
|
tty = optarg;
|
|
break;
|
|
case 'D':
|
|
device = optarg;
|
|
break;
|
|
case 'v':
|
|
verbose++;
|
|
break;
|
|
case 'a':
|
|
rplus = strtod(optarg, NULL);
|
|
break;
|
|
case 'b':
|
|
rminus = strtod(optarg, NULL);
|
|
break;
|
|
case 'm':
|
|
adcmax = strtol(optarg, NULL, 0);
|
|
break;
|
|
default:
|
|
usage(argv[0]);
|
|
break;
|
|
}
|
|
}
|
|
|
|
ao_verbose = verbose;
|
|
|
|
if (verbose > 1)
|
|
ccdbg_add_debug(CC_DEBUG_BITBANG);
|
|
|
|
if (!tty)
|
|
tty = cc_usbdevs_find_by_arg(device, "TeleMega-v1.0");
|
|
if (!tty)
|
|
tty = cc_usbdevs_find_by_arg(device, "TeleMetrum-v2.0");
|
|
if (!tty)
|
|
tty = cc_usbdevs_find_by_arg(device, "TeleMini-v2.0");
|
|
if (!tty)
|
|
tty = cc_usbdevs_find_by_arg(device, "EasyMega-v1.0");
|
|
if (!tty)
|
|
tty = cc_usbdevs_find_by_arg(device, "EasyMetrum-v1.0");
|
|
if (!tty)
|
|
tty = cc_usbdevs_find_by_arg(device, "EasyMini-v1.0");
|
|
if (!tty)
|
|
tty = getenv("ALTOS_TTY");
|
|
if (!tty)
|
|
tty="/dev/ttyACM0";
|
|
|
|
cc = cc_usb_open(tty);
|
|
|
|
if (!cc)
|
|
exit(1);
|
|
|
|
for (i = optind; i < argc; i++) {
|
|
char *name = argv[i];
|
|
|
|
if (!do_igniter(cc, name, rplus, rminus, adcmax))
|
|
ret++;
|
|
}
|
|
done(cc, ret);
|
|
}
|