Initial Commit - Copy from Altus Metrum AltOS
This commit is contained in:
1
ao-tools/ao-test-igniter/.gitignore
vendored
Normal file
1
ao-tools/ao-test-igniter/.gitignore
vendored
Normal file
@@ -0,0 +1 @@
|
||||
ao-test-igniter
|
11
ao-tools/ao-test-igniter/Makefile.am
Normal file
11
ao-tools/ao-test-igniter/Makefile.am
Normal file
@@ -0,0 +1,11 @@
|
||||
bin_PROGRAMS=ao-test-igniter
|
||||
|
||||
AM_CFLAGS=$(WARN_CFLAGS) -I$(top_srcdir)/ao-tools/lib $(LIBUSB_CFLAGS)
|
||||
|
||||
ao_test_igniter_DEPENDENCIES = $(top_builddir)/ao-tools/lib/libao-tools.a
|
||||
|
||||
ao_test_igniter_LDADD=$(top_builddir)/ao-tools/lib/libao-tools.a $(LIBUSB_LIBS)
|
||||
|
||||
ao_test_igniter_SOURCES=ao-test-igniter.c
|
||||
|
||||
man_MANS = ao-test-igniter.1
|
59
ao-tools/ao-test-igniter/ao-test-igniter.1
Normal file
59
ao-tools/ao-test-igniter/ao-test-igniter.1
Normal file
@@ -0,0 +1,59 @@
|
||||
.\"
|
||||
.\" 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; 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.
|
||||
.\"
|
||||
.\"
|
||||
.TH AO-LOAD 1 "ao-test-igniter" ""
|
||||
.SH NAME
|
||||
ao-test-igniter \- test AltOS flight computer igniters
|
||||
.SH SYNOPSIS
|
||||
.B "ao-test-igniter"
|
||||
[\-T \fItty-device\fP]
|
||||
[\--tty \fItty-device\fP]
|
||||
[\-D \fIaltos-device\fP]
|
||||
[\--device \fIaltos-device\fP]
|
||||
\fIigniter-name...\fP
|
||||
.SH DESCRIPTION
|
||||
.I ao-test-igniter
|
||||
test the status and then fires the specified igniters on the target device
|
||||
.SH OPTIONS
|
||||
.TP
|
||||
\-T tty-device | --tty tty-device
|
||||
This selects which tty device the debugger uses to communicate with
|
||||
the target device. The special name 'BITBANG' directs ao-dbg to use
|
||||
the cp2103 connection, otherwise this should be a usb serial port
|
||||
connected to a suitable cc1111 debug node.
|
||||
.TP
|
||||
\-D AltOS-device | --device AltOS-device
|
||||
Search for a connected device. This requires an argument of one of the
|
||||
following forms:
|
||||
.IP
|
||||
TeleMega:2
|
||||
.br
|
||||
TeleMega
|
||||
.br
|
||||
2
|
||||
.IP
|
||||
Leaving out the product name will cause the tool to select a suitable
|
||||
product, leaving out the serial number will cause the tool to match
|
||||
one of the available devices.
|
||||
.SH USAGE
|
||||
.I ao-test-igniter
|
||||
opens the target device, lists the available igniter and then fires
|
||||
the specified ones. And error occurrs if any of the specified igniters
|
||||
is not present or not ready.
|
||||
.SH AUTHOR
|
||||
Keith Packard
|
284
ao-tools/ao-test-igniter/ao-test-igniter.c
Normal file
284
ao-tools/ao-test-igniter/ao-test-igniter.c
Normal file
@@ -0,0 +1,284 @@
|
||||
/*
|
||||
* 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);
|
||||
}
|
Reference in New Issue
Block a user