213 lines
6.2 KiB
C
213 lines
6.2 KiB
C
/*
|
|
Copyright 2022, Savanni D'Gerinel <savanni@luminescent-dreams.com>
|
|
|
|
This file is part of Savanni's AVR library.
|
|
|
|
This AVR library 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 3 of the License, or (at your option) any later version.
|
|
|
|
This AVR library 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 AVR library. If not, see <https://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include <avr/interrupt.h>
|
|
#include <avr/sleep.h>
|
|
#include <dio.h>
|
|
#include <display.h>
|
|
#include "lantern.h"
|
|
#include <rfm.h>
|
|
#include <sk9822.h>
|
|
#include <stdbool.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
|
|
#define FPS 15
|
|
#define TICKS_PER_SECOND 7812
|
|
#define FPS_TIMEOUT TICKS_PER_SECOND / FPS
|
|
#define FPS_MS 1000 / FPS
|
|
#define PULSE_OFF_COUNT FPS * 5
|
|
#define PULSE_ON_COUNT 5
|
|
|
|
typedef enum {
|
|
lme_off,
|
|
lme_normal,
|
|
lme_spooky,
|
|
lme_eerie,
|
|
} lantern_msg_e;
|
|
|
|
typedef struct {
|
|
bool frame_timeout;
|
|
} status_flags_t;
|
|
|
|
status_flags_t status_flags;
|
|
|
|
/*
|
|
ISR(TIMER1_COMPA_vect) {
|
|
status_flags.frame_timeout = true;
|
|
}
|
|
*/
|
|
|
|
ISR(INT6_vect) {
|
|
}
|
|
|
|
/*
|
|
void setup_fps_timer(void) {
|
|
// WGM = 0100: CTC with OCR1nA top
|
|
// CS = 101: clock / 1024
|
|
TCCR1A = 0;
|
|
TCCR1B = _BV(3) | _BV(2) | _BV(0);
|
|
|
|
// Set the top for the counter
|
|
OCR1AH = FPS_TIMEOUT >> 8;
|
|
OCR1AL = FPS_TIMEOUT & 0xff;
|
|
|
|
// Enable compare on A match interrupt
|
|
TIMSK1 |= _BV(1);
|
|
}
|
|
*/
|
|
|
|
void startup_animation(sk9822_t *lights, rgb_t *colors) {
|
|
for (int i = 0; i < LIGHT_COUNT; i++) {
|
|
colors[i].brightness = 1;
|
|
}
|
|
|
|
for (int i = 0; i < LIGHT_COUNT; i++) {
|
|
colors[i].b = 0;
|
|
colors[i].r = 128;
|
|
sk9822_send(lights, colors, LIGHT_COUNT);
|
|
_delay_ms(50);
|
|
}
|
|
|
|
for (int i = 0; i < LIGHT_COUNT; i++) {
|
|
colors[i].r = 0;
|
|
colors[i].g = 128;
|
|
sk9822_send(lights, colors, LIGHT_COUNT);
|
|
_delay_ms(100);
|
|
}
|
|
|
|
for (int i = 0; i < LIGHT_COUNT; i++) {
|
|
colors[i].g = 0;
|
|
colors[i].b = 128;
|
|
sk9822_send(lights, colors, LIGHT_COUNT);
|
|
_delay_ms(100);
|
|
}
|
|
}
|
|
|
|
int main(void) {
|
|
EIMSK = 1 << INT6 | 1 << INT2;
|
|
EICRA |= 1 << ISC21 | 1 << ISC20;
|
|
EICRB |= 1 << ISC61 | 1 << ISC60;
|
|
|
|
status_flags.frame_timeout = false;
|
|
|
|
// Disable unneeded modules
|
|
PRR0 = _BV(7) | _BV(5) | _BV(2);
|
|
PRR1 = _BV(7) | _BV(4) | _BV(3) | _BV(0);
|
|
|
|
rfm_t radio = (rfm_t){
|
|
.spi = {
|
|
.clock = { .ddr = &DDRB, .port = &PORTB, .pin = &PINB, .addr = 1 },
|
|
.data_out = { .ddr = &DDRB, .port = &PORTB, .pin = &PINB, .addr = 2 },
|
|
.data_in = { .ddr = &DDRB, .port = &PORTB, .pin = &PINB, .addr = 3 },
|
|
.chip_select = { .ddr = &DDRB, .port = &PORTB, .pin = &PINB, .addr = 4 },
|
|
},
|
|
.irq = { .ddr = &DDRE, .port = &PORTE, .pin = &PINE, .addr = 6 },
|
|
.reset = { .ddr = &DDRD, .port = &PORTD, .pin = &PIND, .addr = 4 },
|
|
};
|
|
|
|
rfm_error_e error;
|
|
rfm_init(&radio, (uint8_t [4]){ 0xde, 0xca, 0xfb, 0xad }, 4, &error);
|
|
rfm_listen(&radio);
|
|
|
|
sk9822_t lights = {
|
|
.clock_pin = { .ddr = &DDRC, .port = &PORTC, .pin = &PINC, .addr = 7 },
|
|
.data_pin = { .ddr = &DDRD, .port = &PORTD, .pin = &PIND, .addr = 6 }
|
|
};
|
|
|
|
rgb_t error_indicator[LIGHT_COUNT] = {
|
|
{ .brightness = 1, .r = 255, .g = 0, .b = 0 },
|
|
{ .brightness = 0, .r = 0, .g = 0, .b = 0 },
|
|
{ .brightness = 0, .r = 0, .g = 0, .b = 0 },
|
|
{ .brightness = 1, .r = 255, .g = 0, .b = 0 },
|
|
{ .brightness = 0, .r = 0, .g = 0, .b = 0 },
|
|
{ .brightness = 0, .r = 0, .g = 0, .b = 0 },
|
|
{ .brightness = 1, .r = 255, .g = 0, .b = 0 },
|
|
{ .brightness = 0, .r = 0, .g = 0, .b = 0 },
|
|
{ .brightness = 0, .r = 0, .g = 0, .b = 0 },
|
|
};
|
|
|
|
lantern_t lantern = lantern_new(lights);
|
|
lantern_set_mode(&lantern, cse_off);
|
|
|
|
// setup_fps_timer();
|
|
|
|
// set_sleep_mode(SLEEP_MODE_IDLE);
|
|
|
|
rgb_t colors[LIGHT_COUNT] = {
|
|
{ .brightness = 0, .r = 0, .g = 0, .b = 0 },
|
|
{ .brightness = 0, .r = 0, .g = 0, .b = 0 },
|
|
{ .brightness = 0, .r = 0, .g = 0, .b = 0 },
|
|
{ .brightness = 0, .r = 0, .g = 0, .b = 0 },
|
|
{ .brightness = 0, .r = 0, .g = 0, .b = 0 },
|
|
{ .brightness = 0, .r = 0, .g = 0, .b = 0 },
|
|
{ .brightness = 0, .r = 0, .g = 0, .b = 0 },
|
|
{ .brightness = 0, .r = 0, .g = 0, .b = 0 },
|
|
{ .brightness = 0, .r = 0, .g = 0, .b = 0 },
|
|
};
|
|
_delay_ms(1000);
|
|
startup_animation(&lights, (rgb_t *)&colors);
|
|
|
|
while(1) {
|
|
/*
|
|
if (status_flags.frame_timeout) {
|
|
status_flags.frame_timeout = false;
|
|
lantern_step(&lantern, colors);
|
|
sk9822_send(&lights, colors, LIGHT_COUNT);
|
|
}
|
|
*/
|
|
|
|
lantern_step(&lantern, colors);
|
|
sk9822_send(&lights, colors, LIGHT_COUNT);
|
|
|
|
interrupt_flags_t flags = rfm_interrupts(&radio);
|
|
while (flags.fifo_not_empty) {
|
|
uint8_t msg[60];
|
|
uint8_t length;
|
|
|
|
rfm_receive(&radio, (uint8_t *)msg, &length);
|
|
if (length == 1) {
|
|
switch (msg[0]) {
|
|
case lme_off:
|
|
lantern_set_mode(&lantern, cse_off);
|
|
break;
|
|
case lme_normal:
|
|
lantern_set_mode(&lantern, cse_normal);
|
|
break;
|
|
case lme_spooky:
|
|
lantern_set_mode(&lantern, cse_spooky);
|
|
break;
|
|
case lme_eerie:
|
|
lantern_set_mode(&lantern, cse_eerie);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
flags = rfm_interrupts(&radio);
|
|
}
|
|
|
|
_delay_ms(FPS_MS);
|
|
/*
|
|
sei();
|
|
sleep_mode();
|
|
cli();
|
|
*/
|
|
}
|
|
|
|
sk9822_send(&lights, error_indicator, LIGHT_COUNT);
|
|
_delay_ms(1000);
|
|
|
|
return 0;
|
|
}
|