]> www.fi.muni.cz Git - tinyboard.git/blobdiff - projects/step-up/pattern.c
Imported firmware from Project Bike Lights
[tinyboard.git] / projects / step-up / pattern.c
diff --git a/projects/step-up/pattern.c b/projects/step-up/pattern.c
new file mode 100644 (file)
index 0000000..55b64cc
--- /dev/null
@@ -0,0 +1,199 @@
+#include <avr/io.h>
+#include <stdlib.h> // for NULL
+
+#include "lights.h"
+
+static unsigned char led_counters[N_LEDS];
+static pattern_t *led_patterns[N_LEDS];
+
+static pattern_t boot_pattern[] = {
+       { 1, 0x6 },
+       { 0, 0x6 },
+       { 1, 0x3 },
+       { 0, 0x3 },
+       { 1, 0x2 },
+       { 0, 0x2 },
+       { 1, 0x1 },
+       { 0, 0x1 },
+       { 1, 0x1 },
+       { 0, 0x1 },
+       { 1, 0x1 },
+       { 0, 0x1 },
+       { 1, 0x1 },
+       { 0, 0x1 },
+       { 1, 0x10 },
+       { 0, 0x10 },
+       PATTERN_END
+};
+
+static pattern_t pattern_num[] = {
+       { 0, 0x5 },
+       { 1, 0x1 }, /* 10 */
+       { 0, 0x5 },
+       { 1, 0x1 }, /*  9 */
+       { 0, 0x5 },
+       { 1, 0x1 }, /*  8 */
+       { 0, 0x5 },
+       { 1, 0x1 }, /*  7 */
+       { 0, 0x5 },
+       { 1, 0x1 }, /*  6 */
+       { 0, 0x5 },
+       { 1, 0x1 }, /*  5 */
+       { 0, 0x5 },
+       { 1, 0x1 }, /*  4 */
+       { 0, 0x5 },
+       { 1, 0x1 }, /*  3 */
+       { 0, 0x5 },
+       { 1, 0x1 }, /*  2 */
+       { 0, 0x5 },
+       { 1, 0x1 }, /*  1 */
+       { 0, 0xF },
+       PATTERN_END
+};
+
+static pattern_t pattern_invnum[] = {
+       { 1, 0x5 },
+       { 0, 0x1 }, /* 10 */
+       { 1, 0x5 },
+       { 0, 0x1 }, /*  9 */
+       { 1, 0x5 },
+       { 0, 0x1 }, /*  8 */
+       { 1, 0x5 },
+       { 0, 0x1 }, /*  7 */
+       { 1, 0x5 },
+       { 0, 0x1 }, /*  6 */
+       { 1, 0x5 },
+       { 0, 0x1 }, /*  5 */
+       { 1, 0x5 },
+       { 0, 0x1 }, /*  4 */
+       { 1, 0x5 },
+       { 0, 0x1 }, /*  3 */
+       { 1, 0x5 },
+       { 0, 0x1 }, /*  2 */
+       { 1, 0x5 },
+       { 0, 0x1 }, /*  1 */
+       { 1, 0xF },
+       PATTERN_END
+};
+
+pattern_t off_pattern[] = {
+       { 0, 0x1 },
+       PATTERN_END
+};
+
+static void led_set_mode(unsigned char n, unsigned char mode)
+{
+       if (n < N_PWMLEDS) {
+               pwmled_set_mode(n, mode);
+       } else if (n < N_LEDS) {
+               gpio_set(n - N_PWMLEDS, mode);
+       }
+}
+
+void led_set_pattern(unsigned char n, pattern_t *pattern)
+{
+       if (!pattern)
+               pattern = off_pattern;
+
+       led_patterns[n] = pattern;
+
+       led_counters[n] = pattern->duration;
+       led_set_mode(n, pattern->mode);
+}
+
+void init_pattern()
+{
+       unsigned char i;
+
+       for (i = 0; i < N_LEDS; i++)
+               led_set_pattern(i, NULL);
+
+       led_set_pattern(N_PWMLEDS+1, boot_pattern);
+}
+
+pattern_t *number_pattern(unsigned char num, unsigned char inv)
+{
+       if (num >= 10)
+               num = 10;
+
+       if (inv) {
+               return pattern_invnum
+                       + sizeof(pattern_invnum)/sizeof(pattern_t)
+                       - 2 - 2*num;
+       } else {
+               return pattern_num
+                       + sizeof(pattern_num)/sizeof(pattern_t)
+                       - 2 - 2*num;
+       }
+}
+
+static pattern_t *pattern_select(unsigned char n)
+{
+       switch(n) {
+       case 0: return pwmled0_pattern_select();
+       case 1: return pwmled1_pattern_select();
+       case 2: return pwmled2_pattern_select();
+       case 3: return status_led_pattern_select();
+       case 4: return illumination_led_pattern_select();
+       case 6: return laser_pattern_select();
+       default: return NULL;
+       }
+}
+
+void pattern_reload()
+{
+       unsigned char i;
+
+       for (i = 0; i < N_LEDS; i++)
+               led_set_pattern(i, pattern_select(i));
+}
+
+static void inline pattern_finished(unsigned char n)
+{
+       unsigned char i;
+
+       led_patterns[n] = NULL;
+
+       if (n < N_PWMLEDS) {
+               for (i = 0; i < N_PWMLEDS; i++)
+                       if (led_patterns[i])
+                               return;
+
+               /* all pwmleds finished; restart them */
+               for (i = 0; i < N_PWMLEDS; i++)
+                       led_set_pattern(i, pattern_select(i));
+       } else if (n == 3) {
+               if (!led_patterns[4])
+                       led_set_pattern(4, pattern_select(4));
+       } else if (n == 4) {
+               if (!led_patterns[3])
+                       led_set_pattern(3, pattern_select(3));
+       } else {
+               led_set_pattern(n, pattern_select(n));
+       }
+}
+
+void patterns_next_tick()
+{
+       unsigned char i;
+
+       for (i = 0; i < N_LEDS; i++) {
+               if (!led_patterns[i]) {
+                       pattern_finished(i);
+                       continue;
+               }
+
+               if (--led_counters[i] == 0) {
+                       pattern_t *p = led_patterns[i];
+                       p++;
+                       if (p->duration == 0) { // END
+                               /* Keep the last state, wait for others */
+                               pattern_finished(i);
+                               continue;
+                       }
+                       led_set_pattern(i, p);
+               }
+
+       }
+}
+