Initial Commit - Copy from Altus Metrum AltOS
This commit is contained in:
1
ao-tools/ao-cal-freq/.gitignore
vendored
Normal file
1
ao-tools/ao-cal-freq/.gitignore
vendored
Normal file
@@ -0,0 +1 @@
|
||||
ao-cal-freq
|
11
ao-tools/ao-cal-freq/Makefile.am
Normal file
11
ao-tools/ao-cal-freq/Makefile.am
Normal file
@@ -0,0 +1,11 @@
|
||||
bin_PROGRAMS=ao-cal-freq
|
||||
|
||||
AM_CFLAGS=$(WARN_CFLAGS) -I$(top_srcdir)/ao-tools/lib $(LIBUSB_CFLAGS)
|
||||
|
||||
ao_cal_freq_DEPENDENCIES = $(top_builddir)/ao-tools/lib/libao-tools.a
|
||||
|
||||
ao_cal_freq_LDADD=$(top_builddir)/ao-tools/lib/libao-tools.a $(LIBUSB_LIBS) -lm
|
||||
|
||||
ao_cal_freq_SOURCES=ao-cal-freq.c
|
||||
|
||||
man_MANS = ao-cal-freq.1
|
74
ao-tools/ao-cal-freq/ao-cal-freq.1
Normal file
74
ao-tools/ao-cal-freq/ao-cal-freq.1
Normal file
@@ -0,0 +1,74 @@
|
||||
.\"
|
||||
.\" 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-cal-freq" ""
|
||||
.SH NAME
|
||||
ao-cal-freq \- Calibrate AltOS flight computer frequency
|
||||
.SH SYNOPSIS
|
||||
.B "ao-cal-freq"
|
||||
[\-T \fItty-device\fP]
|
||||
[\--tty \fItty-device\fP]
|
||||
[\-D \fIaltos-device\fP]
|
||||
[\--device \fIaltos-device\fP]
|
||||
[\-v] [\--verbose]
|
||||
[\-n] [\--nosave]
|
||||
[\-o \fIcal-value-output-file\fP]
|
||||
[\--output \fIcal-value-output-file\fP]
|
||||
.SH DESCRIPTION
|
||||
.I ao-cal-freq
|
||||
drives the frequency calibration process and saves the result.
|
||||
.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.
|
||||
.TP
|
||||
\-o cal-value-output-file | --output cal-value-output-file
|
||||
Writes the resulting calibration value to the specified file.
|
||||
.TP
|
||||
\-n | --nosave
|
||||
Inhibits saving the calibration value on the device. Necessary for
|
||||
devices which don't have on-board configuration storage.
|
||||
.TP
|
||||
\-v | --verbose
|
||||
Makes
|
||||
.I ao-cal-freq
|
||||
chatty about communication with the target device.
|
||||
.SH USAGE
|
||||
.I ao-cal-freq
|
||||
opens the target device, interactively calibrates the frequency, then
|
||||
shows the resulting calibration values and (optionally) saves them to
|
||||
configuration memory and/or a file.
|
||||
.SH AUTHOR
|
||||
Keith Packard
|
296
ao-tools/ao-cal-freq/ao-cal-freq.c
Normal file
296
ao-tools/ao-cal-freq/ao-cal-freq.c
Normal file
@@ -0,0 +1,296 @@
|
||||
/*
|
||||
* 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 <termios.h>
|
||||
#include <math.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 = "verbose", .has_arg = 0, .val = 'v' },
|
||||
{ .name = "output", .has_arg = 1, .val = 'o' },
|
||||
{ .name = "nosave", .has_arg = 0, .val = 'n' },
|
||||
{ 0, 0, 0, 0},
|
||||
};
|
||||
|
||||
static void usage(char *program)
|
||||
{
|
||||
fprintf(stderr, "usage: %s [--verbose] [--nosave] [--device=<device>] [-tty=<tty>] [--output=<cal-value-file>]\n", program);
|
||||
exit(1);
|
||||
}
|
||||
|
||||
static char **
|
||||
tok(char *line) {
|
||||
char **strs = malloc (sizeof (char *)), *str;
|
||||
int n = 0;
|
||||
|
||||
while ((str = strtok(line, " \t"))) {
|
||||
line = NULL;
|
||||
strs = realloc(strs, (n + 2) * sizeof (char *));
|
||||
strs[n] = strdup(str);
|
||||
n++;
|
||||
}
|
||||
strs[n] = '\0';
|
||||
return strs;
|
||||
}
|
||||
|
||||
struct flash {
|
||||
struct flash *next;
|
||||
char line[512];
|
||||
char **strs;
|
||||
};
|
||||
|
||||
static struct flash *
|
||||
flash(struct cc_usb *usb)
|
||||
{
|
||||
struct flash *head = NULL, **tail = &head;
|
||||
cc_usb_printf(usb, "c s\nv\n");
|
||||
for (;;) {
|
||||
char line[512];
|
||||
struct flash *b;
|
||||
|
||||
cc_usb_getline(usb, line, sizeof (line));
|
||||
b = malloc (sizeof (struct flash));
|
||||
strcpy(b->line, line);
|
||||
b->strs = tok(line);
|
||||
b->next = NULL;
|
||||
*tail = b;
|
||||
tail = &b->next;
|
||||
if (strstr(line, "software-version"))
|
||||
break;
|
||||
}
|
||||
return head;
|
||||
}
|
||||
|
||||
static char **
|
||||
find_flash(struct flash *b, char *word0) {
|
||||
for (;b; b = b->next) {
|
||||
if (strstr(b->line, word0))
|
||||
return b->strs;
|
||||
}
|
||||
return NULL;
|
||||
}
|
||||
|
||||
static int
|
||||
do_save(struct cc_usb *usb)
|
||||
{
|
||||
int ret = 0;
|
||||
|
||||
printf("Saving calibration to device\n");
|
||||
cc_usb_printf(usb, "c w\nv\n");
|
||||
for (;;) {
|
||||
char line[512];
|
||||
|
||||
cc_usb_getline(usb, line, sizeof (line));
|
||||
if (strstr(line, "Nothing to save"))
|
||||
ret = 1;
|
||||
if (strstr(line, "Saved"))
|
||||
ret = 1;
|
||||
if (strstr(line, "software-version"))
|
||||
break;
|
||||
}
|
||||
if (!ret) {
|
||||
printf("Calibration save failed\n");
|
||||
}
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int
|
||||
do_output(char *output, int cur_cal)
|
||||
{
|
||||
printf ("Saving calibration value to file \"%s\"\n", output);
|
||||
|
||||
FILE *out = fopen(output, "w");
|
||||
int ret = 1;
|
||||
|
||||
if (!out) {
|
||||
perror(output);
|
||||
return 0;
|
||||
}
|
||||
|
||||
if (fprintf(out, "%d\n", cur_cal) < 0) {
|
||||
perror("fprintf");
|
||||
ret = 0;
|
||||
}
|
||||
if (fflush(out) != 0) {
|
||||
perror("fflush");
|
||||
ret = 0;
|
||||
}
|
||||
if (fclose(out) != 0) {
|
||||
perror("fclose");
|
||||
ret = 0;
|
||||
}
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int
|
||||
do_cal(char *tty, int save, char *output)
|
||||
{
|
||||
struct cc_usb *usb = NULL;
|
||||
struct flash *b;
|
||||
char line[1024];
|
||||
double measured_freq;
|
||||
char **cur_freq_words;
|
||||
char **cur_cal_words;
|
||||
char *line_end;
|
||||
int cur_freq;
|
||||
int cur_cal;
|
||||
int new_cal;
|
||||
int ret = 1;
|
||||
int changed = 0;
|
||||
|
||||
for(;;) {
|
||||
usb = cc_usb_open(tty);
|
||||
|
||||
if (!usb) {
|
||||
fprintf(stderr, "failed to open device\n");
|
||||
ret = 0;
|
||||
break;
|
||||
}
|
||||
|
||||
cc_usb_printf(usb, "E 0\n");
|
||||
|
||||
b = flash(usb);
|
||||
|
||||
cur_cal_words = find_flash(b, "Radio cal:");
|
||||
cur_freq_words = find_flash(b, "Frequency:");
|
||||
|
||||
if (!cur_cal_words || !cur_freq_words) {
|
||||
fprintf(stderr, "no response\n");
|
||||
ret = 0;
|
||||
break;
|
||||
}
|
||||
|
||||
cur_cal = atoi(cur_cal_words[2]);
|
||||
cur_freq = atoi(cur_freq_words[1]);
|
||||
|
||||
printf ("Current radio calibration %d\n", cur_cal);
|
||||
printf ("Current radio frequency: %7.3f\n", cur_freq / 1000.0);
|
||||
|
||||
cc_usb_sync(usb);
|
||||
cc_usb_printf(usb, "C 1\n");
|
||||
cc_usb_sync(usb);
|
||||
|
||||
printf("Generating RF carrier. Please enter measured frequency [enter for done]: ");
|
||||
fflush(stdout);
|
||||
fgets(line, sizeof (line) - 1, stdin);
|
||||
cc_usb_printf(usb, "C 0\n");
|
||||
cc_usb_sync(usb);
|
||||
|
||||
measured_freq = strtod(line, &line_end);
|
||||
if (line_end == line)
|
||||
break;
|
||||
|
||||
new_cal = floor ((((double) cur_freq / 1000.0) / measured_freq) * cur_cal + 0.5);
|
||||
|
||||
if (new_cal == cur_cal) {
|
||||
printf("Calibration value %d unchanged\n", cur_cal);
|
||||
} else {
|
||||
printf ("Setting cal value %d\n", new_cal);
|
||||
|
||||
cc_usb_printf (usb, "c f %d\n", new_cal);
|
||||
changed = 1;
|
||||
cc_usb_sync(usb);
|
||||
}
|
||||
cc_usb_close(usb);
|
||||
}
|
||||
if (usb) {
|
||||
if (ret && save) {
|
||||
if (changed) {
|
||||
if (!do_save(usb))
|
||||
ret = 0;
|
||||
} else {
|
||||
printf("Calibration unchanged, not saving\n");
|
||||
}
|
||||
}
|
||||
if (ret && output) {
|
||||
if (!do_output(output, cur_cal))
|
||||
ret = 0;
|
||||
}
|
||||
cc_usb_close(usb);
|
||||
}
|
||||
return ret;
|
||||
}
|
||||
|
||||
int
|
||||
main (int argc, char **argv)
|
||||
{
|
||||
char *device = NULL;
|
||||
int c;
|
||||
char *tty = NULL;
|
||||
int verbose = 0;
|
||||
int save = 1;
|
||||
int ret = 0;
|
||||
char *output = NULL;
|
||||
|
||||
while ((c = getopt_long(argc, argv, "vnT:D:o:", options, NULL)) != -1) {
|
||||
switch (c) {
|
||||
case 'T':
|
||||
tty = optarg;
|
||||
break;
|
||||
case 'D':
|
||||
device = optarg;
|
||||
break;
|
||||
case 'v':
|
||||
verbose++;
|
||||
break;
|
||||
case 'n':
|
||||
save = 0;
|
||||
break;
|
||||
case 'o':
|
||||
output = optarg;
|
||||
break;
|
||||
default:
|
||||
usage(argv[0]);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
ao_verbose = verbose;
|
||||
if (verbose)
|
||||
ccdbg_add_debug(CC_DEBUG_BITBANG);
|
||||
|
||||
if (!tty)
|
||||
tty = cc_usbdevs_find_by_arg(device, "AltosFlash");
|
||||
if (!tty)
|
||||
tty = cc_usbdevs_find_by_arg(device, "TeleMega");
|
||||
if (!tty)
|
||||
tty = getenv("ALTOS_TTY");
|
||||
if (!tty)
|
||||
tty="/dev/ttyACM0";
|
||||
|
||||
if (!do_cal(tty, save, output))
|
||||
ret = 1;
|
||||
return ret;
|
||||
}
|
Reference in New Issue
Block a user