- store a stable copy of ARM binaries for production use
- cp src/chaoskey-v1.0/{*.elf,*.ihx,*.bin} \
- src/easymega-v1.0/{*.elf,*.ihx} \
- src/easymega-v2.0/{*.elf,*.ihx} \
- src/easymini-v1.0/{*.elf,*.ihx} \
- src/easymini-v2.0/{*.elf,*.ihx} \
- src/easytimer-v1/{*.elf,*.ihx} \
- src/telebt-v3.0/{*.elf,*.ihx} \
- src/telebt-v4.0/{*.elf,*.ihx} \
- src/teledongle-v3.0/{*.elf,*.ihx} \
- src/telegps-v1.0/{*.elf,*.ihx} \
- src/telegps-v2.0/{*.elf,*.ihx} \
- src/telemega-v[1-4].0/{*.elf,*.ihx} \
- src/telemetrum-v2.0/{*.elf,*.ihx} \
- src/telemetrum-v3.0/{*.elf,*.ihx} \
- src/telemini-v3.0/{*.elf,*.ihx} \
- src/telelco-v2.0/{*.elf,*.ihx} \
- src/telefireeight-v1.0/{*.elf,*.ihx} \
- src/telefireeight-v2.0/{*.elf,*.ihx} \
+ cp src/chaoskey-v1.0/{*.elf,*.ihx,*.bin,*.map} \
+ src/easymega-v1.0/{*.elf,*.ihx,*.map} \
+ src/easymega-v2.0/{*.elf,*.ihx,*.map} \
+ src/easymini-v1.0/{*.elf,*.ihx,*.map} \
+ src/easymini-v2.0/{*.elf,*.ihx,*.map} \
+ src/easytimer-v1/{*.elf,*.ihx,*.map} \
+ src/telebt-v3.0/{*.elf,*.ihx,*.map} \
+ src/telebt-v4.0/{*.elf,*.ihx,*.map} \
+ src/teledongle-v3.0/{*.elf,*.ihx,*.map} \
+ src/telegps-v1.0/{*.elf,*.ihx,*.map} \
+ src/telegps-v2.0/{*.elf,*.ihx,*.map} \
+ src/telemega-v[1-4].0/{*.elf,*.ihx,*.map} \
+ src/telemetrum-v2.0/{*.elf,*.ihx,*.map} \
+ src/telemetrum-v3.0/{*.elf,*.ihx,*.map} \
+ src/telemini-v3.0/{*.elf,*.ihx,*.map} \
+ src/telelco-v2.0/{*.elf,*.ihx,*.map} \
+ src/telefireeight-v1.0/{*.elf,*.ihx,*.map} \
+ src/telefireeight-v2.0/{*.elf,*.ihx,*.map} \
~/altusmetrumllc/Binaries/
- cp src/chaoskey-v1.0/flash-loader/{*.elf,*.bin} \
+ cp src/chaoskey-v1.0/flash-loader/{*.elf,*.bin,*.map} \
src/easymega-v1.0/flash-loader/*.elf \
src/easymega-v2.0/flash-loader/*.elf \
src/easymini-v1.0/flash-loader/*.elf \
- src/easymini-v2.0/flash-loader/{*.elf,*.bin} \
- src/easytimer-v1/flash-loader/{*.elf,*.bin} \
+ src/easymini-v2.0/flash-loader/{*.elf,*.bin,*.map} \
+ src/easytimer-v1/flash-loader/{*.elf,*.bin,*.map} \
src/telebt-v3.0/flash-loader/*.elf \
- src/telebt-v4.0/flash-loader/{*.elf,*.bin} \
+ src/telebt-v4.0/flash-loader/{*.elf,*.bin,*.map} \
src/teledongle-v3.0/flash-loader/*.elf \
src/telegps-v1.0/flash-loader/*.elf \
- src/telegps-v2.0/flash-loader/{*.elf,*.bin} \
+ src/telegps-v2.0/flash-loader/{*.elf,*.bin,*.map} \
src/telemega-v[1-4].0/flash-loader/*.elf \
src/telemetrum-v2.0/flash-loader/*.elf \
src/telemetrum-v3.0/flash-loader/*.elf \
- src/telemini-v3.0/flash-loader/{*.elf,*.bin} \
+ src/telemini-v3.0/flash-loader/{*.elf,*.bin,*.map} \
src/telelco-v2.0/flash-loader/*.elf \
src/telefireeight-v1.0/flash-loader/*.elf \
src/telefireeight-v2.0/flash-loader/*.elf \
"Dual Deploy",
"Redundant Apogee",
"Redundant Main",
+ "Separation & Apogee",
};
static String[] aprs_interval_values = {
SetOutPath $INSTDIR
File "../src/telemetrum-v2.0/telemetrum-v2.0-${VERSION}.ihx"
-; File "../src/telemetrum-v3.0/telemetrum-v3.0-${VERSION}.ihx"
+ File "../src/telemetrum-v3.0/telemetrum-v3.0-${VERSION}.ihx"
File "../src/telemini-v3.0/telemini-v3.0-${VERSION}.ihx"
File "../src/telegps-v1.0/telegps-v1.0-${VERSION}.ihx"
File "../src/telegps-v2.0/telegps-v2.0-${VERSION}.ihx"
is fired first, followed after a two second
delay by the 'main' channel.
+ Separation & Apogee::
+ This fires the 'main' channel when the first motor
+ burns out and fires the 'apogee' charge at apogee.
+
ifdef::telemetrum,telemega,easymega,easytimer[]
==== Pad Orientation
#define ao_arch_task_globals uint8_t ao_cpu_sleep_disable;
-#define ao_arch_init_stack(task, start) \
+#define ao_arch_init_stack(task, sp, start) \
do { \
- uint8_t *sp = task->stack8 + AO_STACK_SIZE - 1; \
uint16_t a = (uint16_t) start; \
int i; \
\
ao_igniter_main = 1
};
-void
-ao_ignite(enum ao_igniter igniter);
-
enum ao_igniter_status {
ao_igniter_unknown, /* unknown status (ambiguous voltage) */
ao_igniter_ready, /* continuity detected */
ao_config_log_set, ao_config_log_show },
#endif
#if HAS_IGNITE
- { "i <0 dual, 1 apogee, 2 main>\0Igniter mode",
+ { "i <0 dual, 1 apogee, 2 main, 3 booster>\0Igniter mode",
ao_config_ignite_mode_set, ao_config_ignite_mode_show },
#endif
#if HAS_AES
#define AO_IGNITE_MODE_DUAL 0
#define AO_IGNITE_MODE_APOGEE 1
#define AO_IGNITE_MODE_MAIN 2
+#define AO_IGNITE_MODE_BOOSTER 3
#define AO_RADIO_ENABLE_CORE 1
#define AO_RADIO_DISABLE_TELEMETRY 2
)
{
ao_flight_state = ao_flight_boost;
+ ao_wakeup(&ao_flight_state);
+
ao_launch_tick = ao_boost_tick = ao_sample_tick;
/* start logging data */
ao_gps_new = AO_GPS_NEW_DATA | AO_GPS_NEW_TRACKING;
ao_wakeup(&ao_gps_new);
#endif
-
- ao_wakeup(&ao_flight_state);
}
break;
case ao_flight_boost:
#else
ao_flight_state = ao_flight_coast;
#endif
- ++ao_motor_number;
ao_wakeup(&ao_flight_state);
+ ++ao_motor_number;
}
break;
#if HAS_ACCEL && HAS_BARO
#endif
)
{
-#if HAS_IGNITE
- /* ignite the drogue charge */
- ao_ignite(ao_igniter_drogue);
-#endif
-
+ /* enter drogue state */
+ ao_flight_state = ao_flight_drogue;
+ ao_wakeup(&ao_flight_state);
#if HAS_TELEMETRY
/* slow down the telemetry system */
ao_telemetry_set_interval(AO_TELEMETRY_INTERVAL_RECOVER);
/* Turn the RDF beacon back on */
ao_rdf_set(1);
#endif
-
- /* and enter drogue state */
- ao_flight_state = ao_flight_drogue;
- ao_wakeup(&ao_flight_state);
}
else
#else /* not HAS_BARO */
ao_interval_max_accel_through - ao_interval_min_accel_through <= ao_data_accel_to_sample(MAX_QUIET_ACCEL))
{
ao_flight_state = ao_flight_landed;
+ ao_wakeup(&ao_flight_state);
#if HAS_ADC
/* turn off the ADC capture */
ao_timer_set_adc_interval(0);
#endif
-
- ao_wakeup(&ao_flight_state);
}
/* Reset interval values */
*/
if (ao_height <= ao_config.main_deploy)
{
-#if HAS_IGNITE
- ao_ignite(ao_igniter_main);
-#endif
+ ao_flight_state = ao_flight_main;
+ ao_wakeup(&ao_flight_state);
/*
* Start recording min/max height
ao_interval_end = ao_sample_tick + AO_INTERVAL_TICKS;
ao_interval_min_height = ao_interval_max_height = ao_avg_height;
-
- ao_flight_state = ao_flight_main;
- ao_wakeup(&ao_flight_state);
}
break;
if (ao_interval_max_height - ao_interval_min_height <= AO_M_TO_HEIGHT(4))
{
ao_flight_state = ao_flight_landed;
-
+ ao_wakeup(&ao_flight_state);
#if HAS_ADC
/* turn off the ADC capture */
ao_timer_set_adc_interval(0);
#endif
-
- ao_wakeup(&ao_flight_state);
}
ao_interval_min_height = ao_interval_max_height = ao_avg_height;
ao_interval_end = ao_sample_tick + AO_INTERVAL_TICKS;
#if HAS_IGNITE
struct ao_ignition ao_ignition[2];
-void
-ao_ignite(enum ao_igniter igniter)
-{
- ao_arch_block_interrupts();
- ao_ignition[igniter].request = 1;
- ao_wakeup(&ao_ignition);
- ao_arch_release_interrupts();
-}
-
#ifndef AO_SENSE_DROGUE
#define AO_SENSE_DROGUE(p) ((p)->adc.sense_d)
#define AO_SENSE_MAIN(p) ((p)->adc.sense_m)
static void
ao_igniter_fire(enum ao_igniter igniter)
{
- ao_ignition[igniter].firing = 1;
- switch(ao_config.ignite_mode) {
- case AO_IGNITE_MODE_DUAL:
+ if (!ao_ignition[igniter].fired) {
+ ao_ignition[igniter].firing = 1;
+ ao_ignition[igniter].fired = 1;
switch (igniter) {
case ao_igniter_drogue:
AO_IGNITER_SET_DROGUE(1);
AO_IGNITER_SET_MAIN(0);
break;
}
- break;
- case AO_IGNITE_MODE_APOGEE:
- switch (igniter) {
- case ao_igniter_drogue:
- AO_IGNITER_SET_DROGUE(1);
- ao_delay(AO_IGNITER_FIRE_TIME);
- AO_IGNITER_SET_DROGUE(0);
- ao_delay(AO_IGNITER_CHARGE_TIME);
- AO_IGNITER_SET_MAIN(1);
- ao_delay(AO_IGNITER_FIRE_TIME);
- AO_IGNITER_SET_MAIN(0);
- break;
- default:
- break;
- }
- break;
- case AO_IGNITE_MODE_MAIN:
- switch (igniter) {
- case ao_igniter_main:
- AO_IGNITER_SET_DROGUE(1);
- ao_delay(AO_IGNITER_FIRE_TIME);
- AO_IGNITER_SET_DROGUE(0);
- ao_delay(AO_IGNITER_CHARGE_TIME);
- AO_IGNITER_SET_MAIN(1);
- ao_delay(AO_IGNITER_FIRE_TIME);
- AO_IGNITER_SET_MAIN(0);
- break;
- default:
- break;
- }
- break;
+ ao_ignition[igniter].firing = 0;
+ ao_delay(AO_IGNITER_CHARGE_TIME);
}
- ao_ignition[igniter].firing = 0;
}
static void
ao_igniter(void)
{
- enum ao_igniter igniter;
-
ao_config_get();
for (;;) {
- ao_sleep(&ao_ignition);
- for (igniter = ao_igniter_drogue; igniter <= ao_igniter_main; igniter++) {
- if (ao_ignition[igniter].request && !ao_ignition[igniter].fired) {
- if (igniter == ao_igniter_drogue && ao_config.apogee_delay)
- ao_delay(AO_SEC_TO_TICKS(ao_config.apogee_delay));
-
- ao_igniter_fire(igniter);
- ao_delay(AO_IGNITER_CHARGE_TIME);
- ao_ignition[igniter].fired = 1;
+ /* Wait for flight state change */
+ ao_sleep(&ao_flight_state);
+
+ /* Fire any igniters that are supposed to be triggered
+ * in this new state
+ */
+ switch(ao_config.ignite_mode) {
+ case AO_IGNITE_MODE_DUAL:
+ if (ao_flight_drogue <= ao_flight_state && ao_flight_state < ao_flight_landed)
+ ao_igniter_fire(ao_igniter_drogue);
+ if (ao_flight_main <= ao_flight_state && ao_flight_state < ao_flight_landed)
+ ao_igniter_fire(ao_igniter_main);
+ break;
+ case AO_IGNITE_MODE_APOGEE:
+ if (ao_flight_drogue <= ao_flight_state && ao_flight_state < ao_flight_landed) {
+ ao_igniter_fire(ao_igniter_drogue);
+ ao_igniter_fire(ao_igniter_main);
}
+ break;
+ case AO_IGNITE_MODE_MAIN:
+ if (ao_flight_main <= ao_flight_state && ao_flight_state < ao_flight_landed) {
+ ao_igniter_fire(ao_igniter_drogue);
+ ao_igniter_fire(ao_igniter_main);
+ }
+ break;
+ case AO_IGNITE_MODE_BOOSTER:
+ if (ao_flight_fast <= ao_flight_state && ao_flight_state < ao_flight_landed)
+ ao_igniter_fire(ao_igniter_main);
+ if (ao_flight_drogue <= ao_flight_state && ao_flight_state < ao_flight_landed)
+ ao_igniter_fire(ao_igniter_drogue);
+ break;
}
}
}
#if HAS_STACK_GUARD
#include <ao_mpu.h>
#endif
+#include <picotls.h>
#define DEBUG 0
#endif /* HAS_TASK_QUEUE */
+static inline void *
+ao_stack_top(struct ao_task *task)
+{
+ uint8_t *top = &task->stack8[AO_STACK_SIZE];
+
+ /* Subtract off the TLS space, but keep the resulting
+ * stack 8-byte aligned
+ */
+#if USE_TLS
+ return top - ((_tls_size() + 7) & ~3);
+#else
+ return top;
+#endif
+}
+
void
ao_add_task(struct ao_task * task, void (*task_func)(void), const char *name)
{
* Construct a stack frame so that it will 'return'
* to the start of the task
*/
- ao_arch_init_stack(task, task_func);
+ uint32_t *sp = ao_stack_top(task);
+#if USE_TLS
+ _init_tls(sp);
+#endif
+ ao_arch_init_stack(task, sp, task_func);
ao_arch_critical(
#if HAS_TASK_QUEUE
ao_task_init_queue(task);
#endif
#if AO_CHECK_STACK
in_yield = 0;
+#endif
+#if USE_TLS
+ _set_tls(ao_stack_top(ao_cur_task));
#endif
ao_arch_restore_stack();
}
#if HAS_TASK
static inline void
-ao_arch_init_stack(struct ao_task *task, void *start)
+ao_arch_init_stack(struct ao_task *task, uint32_t *sp, void *start)
{
- uint32_t *sp = &task->stack32[AO_STACK_SIZE >> 2];
uint32_t a = (uint32_t) start;
int i;
#if HAS_TASK
static inline void
-ao_arch_init_stack(struct ao_task *task, void *start)
+ao_arch_init_stack(struct ao_task *task, uint32_t *sp, void *start)
{
- uint32_t *sp = &task->stack32[AO_STACK_SIZE>>2];
uint32_t a = (uint32_t) start;
int i;
#if HAS_TASK
static inline void
-ao_arch_init_stack(struct ao_task *task, void *start)
+ao_arch_init_stack(struct ao_task *task, uint32_t *sp, void *start)
{
- uint32_t *sp = &task->stack32[AO_STACK_SIZE>>2];
uint32_t a = (uint32_t) start;
int i;
#if HAS_TASK
static inline void
-ao_arch_init_stack(struct ao_task *task, void *start)
+ao_arch_init_stack(struct ao_task *task, uint32_t *sp, void *start)
{
- uint32_t *sp = &task->stack32[AO_STACK_SIZE >> 2];
uint32_t a = (uint32_t) start;
int i;