altos: Add 'g' command to ublox GPS code.
[fw/altos] / src / core / ao.h
1 /*
2  * Copyright © 2009 Keith Packard <keithp@keithp.com>
3  *
4  * This program is free software; you can redistribute it and/or modify
5  * it under the terms of the GNU General Public License as published by
6  * the Free Software Foundation; version 2 of the License
7  *
8  * This program is distributed in the hope that it will be useful, but
9  * WITHOUT ANY WARRANTY; without even the implied warranty of
10  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
11  * General Public License for more details.
12  *
13  * You should have received a copy of the GNU General Public License along
14  * with this program; if not, write to the Free Software Foundation, Inc.,
15  * 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA.
16  */
17
18 #ifndef _AO_H_
19 #define _AO_H_
20
21 #include <stdint.h>
22 #include <stdio.h>
23 #include <string.h>
24 #include <stddef.h>
25 #include <ao_pins.h>
26 #include <ao_arch.h>
27
28 #define TRUE 1
29 #define FALSE 0
30
31 /* Convert a __data pointer into an __xdata pointer */
32 #ifndef DATA_TO_XDATA
33 #define DATA_TO_XDATA(a)        (a)
34 #endif
35 #ifndef PDATA_TO_XDATA
36 #define PDATA_TO_XDATA(a)       (a)
37 #endif
38 #ifndef CODE_TO_XDATA
39 #define CODE_TO_XDATA(a)        (a)
40 #endif
41
42 #ifndef HAS_TASK
43 #define HAS_TASK        1
44 #endif
45
46 #if HAS_TASK
47 #include <ao_task.h>
48 #else
49 #include <ao_notask.h>
50 #endif
51
52 /*
53  * ao_panic.c
54  */
55
56 #define AO_PANIC_NO_TASK        1       /* AO_NUM_TASKS is not large enough */
57 #define AO_PANIC_DMA            2       /* Attempt to start DMA while active */
58 #define AO_PANIC_MUTEX          3       /* Mis-using mutex API */
59 #define AO_PANIC_EE             4       /* Mis-using eeprom API */
60 #define AO_PANIC_LOG            5       /* Failing to read/write log data */
61 #define AO_PANIC_CMD            6       /* Too many command sets registered */
62 #define AO_PANIC_STDIO          7       /* Too many stdio handlers registered */
63 #define AO_PANIC_REBOOT         8       /* Reboot failed */
64 #define AO_PANIC_FLASH          9       /* Invalid flash part (or wrong blocksize) */
65 #define AO_PANIC_USB            10      /* Trying to send USB packet while busy */
66 #define AO_PANIC_BT             11      /* Communications with bluetooth device failed */
67 #define AO_PANIC_STACK          12      /* Stack overflow */
68 #define AO_PANIC_SPI            13      /* SPI communication failure */
69 #define AO_PANIC_CRASH          14      /* Processor crashed */
70 #define AO_PANIC_BUFIO          15      /* Mis-using bufio API */
71 #define AO_PANIC_SELF_TEST_CC1120       0x40 | 1        /* Self test failure */
72 #define AO_PANIC_SELF_TEST_HMC5883      0x40 | 2        /* Self test failure */
73 #define AO_PANIC_SELF_TEST_MPU6000      0x40 | 3        /* Self test failure */
74 #define AO_PANIC_SELF_TEST_MS5607       0x40 | 4        /* Self test failure */
75
76 /* Stop the operating system, beeping and blinking the reason */
77 void
78 ao_panic(uint8_t reason);
79
80 /*
81  * ao_timer.c
82  */
83
84 #ifndef AO_TICK_TYPE
85 #define AO_TICK_TYPE    uint16_t
86 #define AO_TICK_SIGNED  int16_t
87 #endif
88
89 extern volatile __data AO_TICK_TYPE ao_tick_count;
90
91 /* Our timer runs at 100Hz */
92 #ifndef AO_HERTZ
93 #define AO_HERTZ                100
94 #endif
95 #define AO_MS_TO_TICKS(ms)      ((ms) / (1000 / AO_HERTZ))
96 #define AO_SEC_TO_TICKS(s)      ((s) * AO_HERTZ)
97
98 /* Returns the current time in ticks */
99 AO_TICK_TYPE
100 ao_time(void);
101
102 /* Suspend the current task until ticks time has passed */
103 void
104 ao_delay(uint16_t ticks);
105
106 /* Set the ADC interval */
107 void
108 ao_timer_set_adc_interval(uint8_t interval);
109
110 /* Timer interrupt */
111 void
112 ao_timer_isr(void) ao_arch_interrupt(9);
113
114 /* Initialize the timer */
115 void
116 ao_timer_init(void);
117
118 /* Initialize the hardware clock. Must be called first */
119 void
120 ao_clock_init(void);
121
122 /*
123  * ao_mutex.c
124  */
125
126 #ifndef ao_mutex_get
127 void
128 ao_mutex_get(__xdata uint8_t *ao_mutex) __reentrant;
129
130 void
131 ao_mutex_put(__xdata uint8_t *ao_mutex) __reentrant;
132 #endif
133
134 /*
135  * ao_cmd.c
136  */
137
138 enum ao_cmd_status {
139         ao_cmd_success = 0,
140         ao_cmd_lex_error = 1,
141         ao_cmd_syntax_error = 2,
142 };
143
144 extern __pdata uint16_t ao_cmd_lex_i;
145 extern __pdata uint32_t ao_cmd_lex_u32;
146 extern __pdata char     ao_cmd_lex_c;
147 extern __pdata enum ao_cmd_status ao_cmd_status;
148
149 void
150 ao_put_string(__code char *s);
151
152 void
153 ao_cmd_lex(void);
154
155 void
156 ao_cmd_put8(uint8_t v);
157
158 void
159 ao_cmd_put16(uint16_t v);
160
161 uint8_t
162 ao_cmd_is_white(void);
163
164 void
165 ao_cmd_white(void);
166
167 int8_t
168 ao_cmd_hexchar(char c);
169
170 void
171 ao_cmd_hexbyte(void);
172
173 void
174 ao_cmd_hex(void);
175
176 void
177 ao_cmd_decimal(void);
178
179 /* Read a single hex nibble off stdin. */
180 uint8_t
181 ao_getnibble(void);
182
183 uint8_t
184 ao_match_word(__code char *word);
185
186 struct ao_cmds {
187         void            (*func)(void);
188         __code char     *help;
189 };
190
191 void
192 ao_cmd_register(const __code struct ao_cmds *cmds);
193
194 void
195 ao_cmd_init(void);
196
197 #if HAS_CMD_FILTER
198 /*
199  * Provided by an external module to filter raw command lines
200  */
201 uint8_t
202 ao_cmd_filter(void);
203 #endif
204
205 /*
206  * Various drivers
207  */
208 #if HAS_ADC
209 #include <ao_adc.h>
210 #endif
211
212 #if HAS_BEEP
213 #include <ao_beep.h>
214 #endif
215
216 #if LEDS_AVAILABLE
217 #include <ao_led.h>
218 #endif
219
220 #if HAS_USB
221 #include <ao_usb.h>
222 #endif
223
224 #if HAS_EEPROM
225 #include <ao_storage.h>
226 #endif
227
228 #if HAS_LOG
229 #include <ao_log.h>
230 #endif
231
232 #if HAS_FLIGHT
233 #include <ao_flight.h>
234 #include <ao_sample.h>
235 #endif
236
237 /*
238  * ao_report.c
239  */
240
241 #define AO_RDF_INTERVAL_TICKS   AO_SEC_TO_TICKS(5)
242 #define AO_RDF_LENGTH_MS        500
243 #define AO_RDF_CONTINUITY_MS    32
244 #define AO_RDF_CONTINUITY_PAUSE 96
245 #define AO_RDF_CONTINUITY_TOTAL ((AO_RDF_CONTINUITY_PAUSE + AO_RDF_CONTINUITY_MS) * 3 + AO_RDF_CONTINUITY_PAUSE)
246
247 /* This assumes that we're generating a 1kHz tone, which
248  * modulates the carrier at 2kbps, or 250kBps
249  */
250 #define AO_MS_TO_RDF_LEN(ms) ((ms) / 4)
251
252 #define AO_RADIO_RDF_LEN        AO_MS_TO_RDF_LEN(AO_RDF_LENGTH_MS)
253 #define AO_RADIO_CONT_TONE_LEN  AO_MS_TO_RDF_LEN(AO_RDF_CONTINUITY_MS)
254 #define AO_RADIO_CONT_PAUSE_LEN AO_MS_TO_RDF_LEN(AO_RDF_CONTINUITY_PAUSE)
255 #define AO_RADIO_CONT_TOTAL_LEN AO_MS_TO_RDF_LEN(AO_RDF_CONTINUITY_TOTAL)
256
257 /* returns a value 0-3 to indicate igniter continuity */
258 uint8_t
259 ao_report_igniter(void);
260
261 void
262 ao_report_init(void);
263
264 /*
265  * ao_convert.c
266  *
267  * Given raw data, convert to SI units
268  */
269
270 /* pressure from the sensor to altitude in meters */
271 int16_t
272 ao_pres_to_altitude(int16_t pres) __reentrant;
273
274 int16_t
275 ao_altitude_to_pres(int16_t alt) __reentrant;
276
277 int16_t
278 ao_temp_to_dC(int16_t temp) __reentrant;
279
280 /*
281  * ao_convert_pa.c
282  *
283  * Convert between pressure in Pa and altitude in meters
284  */
285
286 #include <ao_data.h>
287
288 alt_t
289 ao_pa_to_altitude(int32_t pa);
290
291 int32_t
292 ao_altitude_to_pa(alt_t alt);
293
294 #if HAS_DBG
295 #include <ao_dbg.h>
296 #endif
297
298 #if HAS_SERIAL_0 || HAS_SERIAL_1 || HAS_SERIAL_2 || HAS_SERIAL_3
299 #include <ao_serial.h>
300 #endif
301
302
303 /*
304  * ao_spi_slave.c
305  */
306
307 uint8_t
308 ao_spi_slave_recv(void *buf, uint16_t len);
309
310 void
311 ao_spi_slave_send(void *buf, uint16_t len);
312
313 void
314 ao_spi_slave_init(void);
315
316 /* This must be defined by the product; it will get called when chip
317  * select goes low, at which point it should use ao_spi_read and
318  * ao_spi_write to deal with the request
319  */
320
321 void
322 ao_spi_slave(void);
323
324 #include <ao_telemetry.h>
325 /*
326  * ao_gps.c
327  */
328
329 #define AO_GPS_NUM_SAT_MASK     (0xf << 0)
330 #define AO_GPS_NUM_SAT_SHIFT    (0)
331
332 #define AO_GPS_VALID            (1 << 4)
333 #define AO_GPS_RUNNING          (1 << 5)
334 #define AO_GPS_DATE_VALID       (1 << 6)
335 #define AO_GPS_COURSE_VALID     (1 << 7)
336
337 extern __pdata uint16_t ao_gps_tick;
338 extern __xdata uint8_t ao_gps_mutex;
339 extern __xdata struct ao_telemetry_location ao_gps_data;
340 extern __xdata struct ao_telemetry_satellite ao_gps_tracking_data;
341
342 struct ao_gps_orig {
343         uint8_t                 year;
344         uint8_t                 month;
345         uint8_t                 day;
346         uint8_t                 hour;
347         uint8_t                 minute;
348         uint8_t                 second;
349         uint8_t                 flags;
350         int32_t                 latitude;       /* degrees * 10⁷ */
351         int32_t                 longitude;      /* degrees * 10⁷ */
352         int16_t                 altitude;       /* m */
353         uint16_t                ground_speed;   /* cm/s */
354         uint8_t                 course;         /* degrees / 2 */
355         uint8_t                 hdop;           /* * 5 */
356         int16_t                 climb_rate;     /* cm/s */
357         uint16_t                h_error;        /* m */
358         uint16_t                v_error;        /* m */
359 };
360
361 struct ao_gps_sat_orig {
362         uint8_t         svid;
363         uint8_t         c_n_1;
364 };
365
366 #define AO_MAX_GPS_TRACKING     12
367
368 struct ao_gps_tracking_orig {
369         uint8_t                 channels;
370         struct ao_gps_sat_orig  sats[AO_MAX_GPS_TRACKING];
371 };
372
373 void
374 ao_gps(void);
375
376 void
377 ao_gps_print(__xdata struct ao_gps_orig *gps_data);
378
379 void
380 ao_gps_tracking_print(__xdata struct ao_gps_tracking_orig *gps_tracking_data);
381
382 void
383 ao_gps_show(void) __reentrant;
384
385 void
386 ao_gps_init(void);
387
388 /*
389  * ao_gps_report.c
390  */
391
392 void
393 ao_gps_report(void);
394
395 void
396 ao_gps_report_init(void);
397
398 /*
399  * ao_gps_report_mega.c
400  */
401
402 void
403 ao_gps_report_mega(void);
404
405 void
406 ao_gps_report_mega_init(void);
407
408 /*
409  * ao_telemetry_orig.c
410  */
411
412 #if LEGACY_MONITOR
413 struct ao_adc_orig {
414         uint16_t        tick;           /* tick when the sample was read */
415         int16_t         accel;          /* accelerometer */
416         int16_t         pres;           /* pressure sensor */
417         int16_t         temp;           /* temperature sensor */
418         int16_t         v_batt;         /* battery voltage */
419         int16_t         sense_d;        /* drogue continuity sense */
420         int16_t         sense_m;        /* main continuity sense */
421 };
422
423 struct ao_telemetry_orig {
424         uint16_t                serial;
425         uint16_t                flight;
426         uint8_t                 flight_state;
427         int16_t                 accel;
428         int16_t                 ground_accel;
429         union {
430                 struct {
431                         int16_t                 speed;
432                         int16_t                 unused;
433                 } k;
434                 int32_t         flight_vel;
435         } u;
436         int16_t                 height;
437         int16_t                 ground_pres;
438         int16_t                 accel_plus_g;
439         int16_t                 accel_minus_g;
440         struct ao_adc_orig      adc;
441         struct ao_gps_orig      gps;
442         char                    callsign[AO_MAX_CALLSIGN];
443         struct ao_gps_tracking_orig     gps_tracking;
444 };
445
446 struct ao_telemetry_tiny {
447         uint16_t                serial;
448         uint16_t                flight;
449         uint8_t                 flight_state;
450         int16_t                 height;         /* AGL in meters */
451         int16_t                 speed;          /* in m/s * 16 */
452         int16_t                 accel;          /* in m/s² * 16 */
453         int16_t                 ground_pres;    /* sensor units */
454         struct ao_adc           adc;            /* raw ADC readings */
455         char                    callsign[AO_MAX_CALLSIGN];
456 };
457
458 struct ao_telemetry_orig_recv {
459         struct ao_telemetry_orig        telemetry_orig;
460         int8_t                          rssi;
461         uint8_t                         status;
462 };
463
464 struct ao_telemetry_tiny_recv {
465         struct ao_telemetry_tiny        telemetry_tiny;
466         int8_t                          rssi;
467         uint8_t                         status;
468 };
469
470 #endif /* LEGACY_MONITOR */
471
472 /* Unfortunately, we've exposed the CC1111 rssi units as the 'usual' method
473  * for reporting RSSI. So, now we use these values everywhere
474  */
475 #define AO_RSSI_FROM_RADIO(radio)       ((int16_t) ((int8_t) (radio) >> 1) - 74)
476 #define AO_RADIO_FROM_RSSI(rssi)        (((int8_t) (rssi) + 74) << 1)
477
478 /*
479  * ao_radio_recv tacks on rssi and status bytes
480  */
481
482 struct ao_telemetry_raw_recv {
483         uint8_t                 packet[AO_MAX_TELEMETRY + 2];
484 };
485
486 /* Set delay between telemetry reports (0 to disable) */
487
488 #ifdef AO_SEND_ALL_BARO
489 #define AO_TELEMETRY_INTERVAL_PAD       AO_MS_TO_TICKS(100)
490 #define AO_TELEMETRY_INTERVAL_FLIGHT    AO_MS_TO_TICKS(100)
491 #define AO_TELEMETRY_INTERVAL_RECOVER   AO_MS_TO_TICKS(100)
492 #else
493 #define AO_TELEMETRY_INTERVAL_PAD       AO_MS_TO_TICKS(1000)
494 #define AO_TELEMETRY_INTERVAL_FLIGHT    AO_MS_TO_TICKS(100)
495 #define AO_TELEMETRY_INTERVAL_RECOVER   AO_MS_TO_TICKS(1000)
496 #endif
497
498 void
499 ao_telemetry_set_interval(uint16_t interval);
500
501 void
502 ao_rdf_set(uint8_t rdf);
503
504 void
505 ao_telemetry_init(void);
506
507 void
508 ao_telemetry_orig_init(void);
509
510 void
511 ao_telemetry_tiny_init(void);
512
513 /*
514  * ao_radio.c
515  */
516
517 extern __xdata uint8_t  ao_radio_dma;
518
519 extern __xdata int8_t   ao_radio_rssi;
520
521 #ifdef PKT_APPEND_STATUS_1_CRC_OK
522 #define AO_RADIO_STATUS_CRC_OK  PKT_APPEND_STATUS_1_CRC_OK
523 #else
524 #include <ao_fec.h>
525 #define AO_RADIO_STATUS_CRC_OK  AO_FEC_DECODE_CRC_OK
526 #endif
527
528 #ifndef HAS_RADIO_RECV
529 #define HAS_RADIO_RECV HAS_RADIO
530 #endif
531 #ifndef HAS_RADIO_XMIT
532 #define HAS_RADIO_XMIT HAS_RADIO
533 #endif
534
535 void
536 ao_radio_general_isr(void) ao_arch_interrupt(16);
537
538 #if HAS_RADIO_XMIT
539 void
540 ao_radio_send(const __xdata void *d, uint8_t size) __reentrant;
541 #endif
542
543 #if HAS_RADIO_RECV
544 uint8_t
545 ao_radio_recv(__xdata void *d, uint8_t size, uint8_t timeout) __reentrant;
546
547 void
548 ao_radio_recv_abort(void);
549 #endif
550
551 void
552 ao_radio_test(uint8_t on);
553
554 typedef int16_t (*ao_radio_fill_func)(uint8_t *buffer, int16_t len);
555
556 void
557 ao_radio_send_aprs(ao_radio_fill_func fill);
558
559 /*
560  * ao_radio_pa
561  */
562
563 #if HAS_RADIO_AMP
564 void
565 ao_radio_pa_on(void);
566
567 void
568 ao_radio_pa_off(void);
569
570 void
571 ao_radio_pa_init(void);
572 #else
573 #define ao_radio_pa_on()
574 #define ao_radio_pa_off()
575 #define ao_radio_pa_init()
576 #endif
577
578 /*
579  * Compute the packet length as follows:
580  *
581  * 2000 bps (for a 1kHz tone)
582  * so, for 'ms' milliseconds, we need
583  * 2 * ms bits, or ms / 4 bytes
584  */
585
586 void
587 ao_radio_rdf(void);
588
589 void
590 ao_radio_continuity(uint8_t c);
591
592 void
593 ao_radio_rdf_abort(void);
594
595 void
596 ao_radio_init(void);
597
598 /*
599  * ao_monitor.c
600  */
601
602 #if HAS_MONITOR
603
604 extern const char const * const ao_state_names[];
605
606 #define AO_MONITOR_RING 8
607
608 union ao_monitor {
609         struct ao_telemetry_raw_recv    raw;
610         struct ao_telemetry_all_recv    all;
611 #if LEGACY_MONITOR
612         struct ao_telemetry_orig_recv   orig;
613         struct ao_telemetry_tiny_recv   tiny;
614 #endif
615 };
616
617 extern __xdata union ao_monitor ao_monitor_ring[AO_MONITOR_RING];
618
619 #define ao_monitor_ring_next(n) (((n) + 1) & (AO_MONITOR_RING - 1))
620
621 extern __data uint8_t ao_monitoring;
622 extern __data uint8_t ao_monitor_head;
623
624 void
625 ao_monitor(void);
626
627 #define AO_MONITORING_OFF       0
628 #define AO_MONITORING_ORIG      1
629
630 void
631 ao_monitor_set(uint8_t monitoring);
632
633 void
634 ao_monitor_disable(void);
635
636 void
637 ao_monitor_enable(void);
638
639 void
640 ao_monitor_init(void) __reentrant;
641
642 #endif
643
644 /*
645  * ao_stdio.c
646  */
647
648 #define AO_READ_AGAIN   (-1)
649
650 struct ao_stdio {
651         int     (*_pollchar)(void);     /* Called with interrupts blocked */
652         void    (*putchar)(char c) __reentrant;
653         void    (*flush)(void);
654         uint8_t echo;
655 };
656
657 extern __xdata struct ao_stdio ao_stdios[];
658 extern __pdata int8_t ao_cur_stdio;
659 extern __pdata int8_t ao_num_stdios;
660
661 void
662 flush(void);
663
664 extern __xdata uint8_t ao_stdin_ready;
665
666 uint8_t
667 ao_echo(void);
668
669 int8_t
670 ao_add_stdio(int (*pollchar)(void),
671              void (*putchar)(char) __reentrant,
672              void (*flush)(void)) __reentrant;
673
674 /*
675  * ao_ignite.c
676  */
677
678 enum ao_igniter {
679         ao_igniter_drogue = 0,
680         ao_igniter_main = 1
681 };
682
683 void
684 ao_ignite(enum ao_igniter igniter);
685
686 enum ao_igniter_status {
687         ao_igniter_unknown,     /* unknown status (ambiguous voltage) */
688         ao_igniter_ready,       /* continuity detected */
689         ao_igniter_active,      /* igniter firing */
690         ao_igniter_open,        /* open circuit detected */
691 };
692
693 struct ao_ignition {
694         uint8_t request;
695         uint8_t fired;
696         uint8_t firing;
697 };
698
699 extern __xdata struct ao_ignition ao_ignition[2];
700
701 enum ao_igniter_status
702 ao_igniter_status(enum ao_igniter igniter);
703
704 extern __pdata uint8_t ao_igniter_present;
705
706 void
707 ao_ignite_set_pins(void);
708
709 void
710 ao_igniter_init(void);
711
712 /*
713  * ao_config.c
714  */
715
716 #if AO_PYRO_NUM
717 #include <ao_pyro.h>
718 #endif
719
720 #if HAS_FORCE_FREQ
721 /*
722  * Set this to force the frequency to 434.550MHz
723  */
724 extern __xdata uint8_t ao_force_freq;
725 #endif
726
727 #define AO_CONFIG_MAJOR 1
728 #define AO_CONFIG_MINOR 14
729
730 #define AO_AES_LEN 16
731
732 extern __xdata uint8_t ao_config_aes_seq;
733
734 struct ao_config {
735         uint8_t         major;
736         uint8_t         minor;
737         uint16_t        main_deploy;
738         int16_t         accel_plus_g;           /* changed for minor version 2 */
739         uint8_t         _legacy_radio_channel;
740         char            callsign[AO_MAX_CALLSIGN + 1];
741         uint8_t         apogee_delay;           /* minor version 1 */
742         int16_t         accel_minus_g;          /* minor version 2 */
743         uint32_t        radio_cal;              /* minor version 3 */
744         uint32_t        flight_log_max;         /* minor version 4 */
745         uint8_t         ignite_mode;            /* minor version 5 */
746         uint8_t         pad_orientation;        /* minor version 6 */
747         uint32_t        radio_setting;          /* minor version 7 */
748         uint8_t         radio_enable;           /* minor version 8 */
749         uint8_t         aes_key[AO_AES_LEN];    /* minor version 9 */
750         uint32_t        frequency;              /* minor version 10 */
751         uint16_t        apogee_lockout;         /* minor version 11 */
752 #if AO_PYRO_NUM
753         struct ao_pyro  pyro[AO_PYRO_NUM];      /* minor version 12 */
754 #endif
755         uint16_t        aprs_interval;          /* minor version 13 */
756 #if HAS_RADIO_POWER
757         uint8_t         radio_power;            /* minor version 14 */
758 #endif
759 #if HAS_RADIO_AMP
760         uint8_t         radio_amp;              /* minor version 14 */
761 #endif
762 };
763
764 #define AO_IGNITE_MODE_DUAL             0
765 #define AO_IGNITE_MODE_APOGEE           1
766 #define AO_IGNITE_MODE_MAIN             2
767
768 #define AO_RADIO_ENABLE_CORE            1
769 #define AO_RADIO_DISABLE_TELEMETRY      2
770 #define AO_RADIO_DISABLE_RDF            4
771
772 #define AO_PAD_ORIENTATION_ANTENNA_UP   0
773 #define AO_PAD_ORIENTATION_ANTENNA_DOWN 1
774
775 extern __xdata struct ao_config ao_config;
776
777 #define AO_CONFIG_MAX_SIZE      128
778
779 void
780 _ao_config_edit_start(void);
781
782 void
783 _ao_config_edit_finish(void);
784
785 void
786 ao_config_get(void);
787
788 void
789 ao_config_put(void);
790
791 void
792 ao_config_set_radio(void);
793
794 void
795 ao_config_init(void);
796
797 /*
798  * ao_rssi.c
799  */
800
801 void
802 ao_rssi_set(int rssi_value);
803
804 void
805 ao_rssi_init(uint8_t rssi_led);
806
807 /*
808  * ao_product.c
809  *
810  * values which need to be defined for
811  * each instance of a product
812  */
813
814 extern const char ao_version[];
815 extern const char ao_manufacturer[];
816 extern const char ao_product[];
817
818 /*
819  * Fifos
820  */
821
822 #define AO_FIFO_SIZE    32
823
824 struct ao_fifo {
825         uint8_t insert;
826         uint8_t remove;
827         char    fifo[AO_FIFO_SIZE];
828 };
829
830 #define ao_fifo_insert(f,c) do { \
831         (f).fifo[(f).insert] = (c); \
832         (f).insert = ((f).insert + 1) & (AO_FIFO_SIZE-1); \
833 } while(0)
834
835 #define ao_fifo_remove(f,c) do {\
836         c = (f).fifo[(f).remove]; \
837         (f).remove = ((f).remove + 1) & (AO_FIFO_SIZE-1); \
838 } while(0)
839
840 #define ao_fifo_full(f)         ((((f).insert + 1) & (AO_FIFO_SIZE-1)) == (f).remove)
841 #define ao_fifo_empty(f)        ((f).insert == (f).remove)
842
843 #if PACKET_HAS_MASTER || PACKET_HAS_SLAVE
844 #include <ao_packet.h>
845 #endif
846
847 #if HAS_BTM
848 #include <ao_btm.h>
849 #endif
850
851 #if HAS_COMPANION
852 #include <ao_companion.h>
853 #endif
854
855 #if HAS_LCD
856 #include <ao_lcd.h>
857 #endif
858
859 #if HAS_AES
860 #include <ao_aes.h>
861 #endif
862
863 /* ao_launch.c */
864
865 struct ao_launch_command {
866         uint16_t        tick;
867         uint16_t        serial;
868         uint8_t         cmd;
869         uint8_t         channel;
870         uint16_t        unused;
871 };
872
873 #define AO_LAUNCH_QUERY         1
874
875 struct ao_launch_query {
876         uint16_t        tick;
877         uint16_t        serial;
878         uint8_t         channel;
879         uint8_t         valid;
880         uint8_t         arm_status;
881         uint8_t         igniter_status;
882 };
883
884 #define AO_LAUNCH_ARM           2
885 #define AO_LAUNCH_FIRE          3
886
887 void
888 ao_launch_init(void);
889
890 /*
891  * ao_log_single.c
892  */
893
894 #define AO_LOG_TELESCIENCE_START        ((uint8_t) 's')
895 #define AO_LOG_TELESCIENCE_DATA         ((uint8_t) 'd')
896
897 #define AO_LOG_TELESCIENCE_NUM_ADC      12
898
899 struct ao_log_telescience {
900         uint8_t         type;
901         uint8_t         csum;
902         uint16_t        tick;
903         uint16_t        tm_tick;
904         uint8_t         tm_state;
905         uint8_t         unused;
906         uint16_t        adc[AO_LOG_TELESCIENCE_NUM_ADC];
907 };
908
909 #define AO_LOG_SINGLE_SIZE              32
910
911 union ao_log_single {
912         struct ao_log_telescience       telescience;
913         union ao_telemetry_all          telemetry;
914         uint8_t                         bytes[AO_LOG_SINGLE_SIZE];
915 };
916
917 extern __xdata union ao_log_single      ao_log_single_write_data;
918 extern __xdata union ao_log_single      ao_log_single_read_data;
919
920 void
921 ao_log_single_extra_query(void);
922
923 void
924 ao_log_single_list(void);
925
926 void
927 ao_log_single_main(void);
928
929 uint8_t
930 ao_log_single_write(void);
931
932 uint8_t
933 ao_log_single_read(uint32_t pos);
934
935 void
936 ao_log_single_start(void);
937
938 void
939 ao_log_single_stop(void);
940
941 void
942 ao_log_single_restart(void);
943
944 void
945 ao_log_single_set(void);
946
947 void
948 ao_log_single_delete(void);
949
950 void
951 ao_log_single_init(void);
952
953 void
954 ao_log_single(void);
955
956 /*
957  * ao_pyro_slave.c
958  */
959
960 #define AO_TELEPYRO_NUM_ADC     9
961
962 #ifndef ao_xmemcpy
963 #define ao_xmemcpy(d,s,c) memcpy(d,s,c)
964 #define ao_xmemset(d,v,c) memset(d,v,c)
965 #define ao_xmemcmp(d,s,c) memcmp(d,s,c)
966 #endif
967
968 /*
969  * ao_terraui.c
970  */
971
972 void
973 ao_terraui_init(void);
974
975 /*
976  * ao_battery.c
977  */
978
979 #ifdef BATTERY_PIN
980 void
981 ao_battery_isr(void) ao_arch_interrupt(1);
982
983 uint16_t
984 ao_battery_get(void);
985
986 void
987 ao_battery_init(void);
988 #endif /* BATTERY_PIN */
989
990 /*
991  * ao_sqrt.c
992  */
993
994 uint32_t
995 ao_sqrt(uint32_t op);
996
997 /*
998  * ao_freq.c
999  */
1000
1001 int32_t ao_freq_to_set(int32_t freq, int32_t cal) __reentrant;
1002
1003 /*
1004  * ao_ms5607.c
1005  */
1006
1007 void ao_ms5607_init(void);
1008
1009 #include <ao_arch_funcs.h>
1010
1011 #endif /* _AO_H_ */