Subversion Repositories Projects

Rev

Rev 780 | Go to most recent revision | Blame | Compare with Previous | Last modification | View Log | RSS feed

// ***************************************************************
// ** Spektrum Diversity v1.3 - use up to 4 satellite receivers **
// ***************************************************************
// ** Target: An Atmel ATtiny2313 (RC-Oscillator @ 8 MHz)       **
// **         controls a 74HC151 Multiplexer                    **
// ***************************************************************
// ** It monitors the data from 4 satellite receivers and       **
// ** connects a valid one to the output via the multiplexer    **
// ***************************************************************
// ** LED-Modes during startup                                  **
// ** ************************                                  **
// ** LED fast flash: Waiting for first datapulse               **
// **                                                           **
// ** LED-Modes during operation                                **
// ** **************************                                **
// ** LED OFF: Everything is fine                               **
// ** LED ON:  FAILURE - The first selected sat had lost sync   **
// **                                                           **
// ** LED-Modes after Self-Test and pressed button              **
// ** ********************************************              **
// ** LED flashes at 1 Hz: Everything is fine                   **
// ** LED flashes some times: Times flashed -> damaged Channel# **
// ** LED off: check voltage(regulator), button or firmware     **
// ***************************************************************
// ** (c) 2010-0xFFFF Stefan Pendsa                             **
// ** License: don't know - use it and have fun                 **
// ***************************************************************

#include <avr/io.h>
#include <avr/eeprom.h>
#include <avr/interrupt.h>
#include <util/delay.h>


#define LED_OFF PORTD |= 1 << PD0
#define LED_ON  PORTD &= ~(1 << PD0)

volatile unsigned int TimerEvent;

unsigned char eedummy EEMEM;                                                            // Dummy-Variable for Address 0 in EEPROM
unsigned char eecheck EEMEM;                                                            // Check-Variable
unsigned char bind;                                                                                     // Bind-Pulse-Counter (in RAM)
unsigned char eebind EEMEM;                                                                     // Bind-Pulse-Counter (in EEPROM)



ISR(TIMER1_OVF_vect)                                                                            // Triggered every 8,192 msec
{
        TimerEvent++;
}



void SelectSat(unsigned char sat)
{
        PORTD = (PORTD & 0b1100111) | (sat << 3);                               // Select the input for 74HC151
        _delay_us(10);                                                                                  // Wait for stable state
}



void ResetTimer(void)
{
        TCNT1H = 0;
        TCNT1L = 0;
        TimerEvent = 0;
}



void Binding(void)                                                                                      // Binding sequence
{
        unsigned char i = 0;

        DDRB = 0b11110000;                                                                              // Sat-Pins to output
        _delay_ms(80);                                                                                  // Let them time to boot up
       
        for (i=0;i<bind;i++)                                                                    // Send negative 100µs pulses to all sat's
        {
                PORTB = 0b00000000;
                _delay_us(100);
                PORTB = 0b11110000;
                _delay_us(100);
        }

        for (i=0;i<bind;i++)                                                                    // Flash the number of used pulses to the LED
        {
                LED_ON;
                _delay_ms(100);
                LED_OFF;
                _delay_ms(250);
        }

        DDRB = 0b00000000;                                                                              // Sat-Pins to input again
       
        bind++;                                                                                                 // Save increased bind-pulse-counter for next binding.
        if (bind > 6) bind = 3;                                                                 // 3 pulses are for DX7+AR6200, 4-6 pulses have been seen...
        eeprom_write_byte(&eebind, bind);                                               // around the world for other combinations. Hopefully it works for them.
        _delay_ms(500);
}



void Testing(void)                                                                                      // Self Test
{
        unsigned char error = 0;
        unsigned char i = 0;

        DDRB = 0b11110000;                                                                              // Port B Output for satellites, Input for feedback

        PORTB = 0b01010000;                                                                             // Test Pattern
        SelectSat(0);
        if (!(PINB & (1<<PB3))) error = 1;
        SelectSat(1);
        if (PINB & (1<<PB3)) error = 2;
        SelectSat(2);
        if (!(PINB & (1<<PB3))) error = 3;
        SelectSat(3);
        if (PINB & (1<<PB3)) error = 4;

        PORTB = 0b10100000;                                                                             // Another (inverted) Test Pattern
        SelectSat(0);
        if (PINB & (1<<PB3)) error = 1;
        SelectSat(1);
        if (!(PINB & (1<<PB3))) error = 2;
        SelectSat(2);
        if (PINB & (1<<PB3)) error = 3;
        SelectSat(3);
        if (!(PINB & (1<<PB3))) error = 4;

        DDRB = 0b00000000;                                                                              // Port B Input again

        while (PIND & (1<<PD6));                                                                // Wait for Bind-Switch

        while(1)                                                                                                // Never return
        {
                if (error == 0)                                                                         // When no error occured -> LED flashes at 1 Hz
                {
                        LED_ON;
                        _delay_ms(100);
                        LED_OFF;
                        _delay_ms(900);
                }
                else
                {
                        for (i=0;i<error;i++)                                                   // When error occured -> Flash-Out the Errorcode
                        {
                                LED_ON;
                                _delay_ms(100);
                                LED_OFF;
                                _delay_ms(400);
                        }
                        _delay_ms(1000);
                }

        }

}



int main(void)
{
        unsigned char i = 0;
        unsigned char active[4];
        unsigned char sat = 99;

        DDRB = 0b00000000;                                                                              // Port B Input for satellites and feedback
        DDRD = 0b0011001;                                                                               // Port D Output for MUX and LED, Input for Switch & Test
        PORTB = 0b11110000;                                                                             // Port B Pullup's for (unused) satellites
        PORTD = 0b1100001;                                                                              // Port D Pullup's for Switch & Test, LED off

        for (i=0;i<4;i++) active[i] = 0;                                                // Reset active-array


    if (eeprom_read_byte(&eecheck) != 0x42)                                     // Invalid Data in EEPROM -> initialize
        {
                eeprom_write_byte(&eecheck, 0x42);
                bind = 3;
                eeprom_write_byte(&eebind, bind);
        }
        else bind = eeprom_read_byte(&eebind);                                  // Valid Data in EEPROM -> read bind-pulse-counter


        if (!(PIND & (1<<PD5))) Testing();                                              // Initiate Self-Test when Test-Pad is low
        if (!(PIND & (1<<PD6))) Binding();                                              // Initiate Binding when Bind-Button is pressed

        _delay_ms(100);

        TCCR1B = ( 1 << CS10 );                                                                 // Timer1 Prescaler = 1 -> 8,192 msec
        TIMSK = ( 1 << TOIE1 );                                                                 // Timer1 Overflow Interrupt Enable
        sei();                                                                                                  // Global Interrupts enable

        ResetTimer();
        while(sat == 99)                                                                                // Wait for first signal
        {
                if (TimerEvent == 10) LED_ON;                                           // Blink while waiting...
                if (TimerEvent == 20)
                {
                        LED_OFF;
                        TimerEvent = 0;
                }

                for (i=0;i<4;i++)                                                                       // Select first active Satellite
                {
                        if (!(PINB & (1<<(i+4))))
                        {
                                active[i] = 1;
                                sat = i;
                                SelectSat(sat);
                                break;
                        }
                }
        }

        LED_OFF;

        while(1)                                                                                                // Main-Loop
        {
                for (i=0;i<4;i++) active[i] = 0;                                        // Reset active-array

                ResetTimer();
                while((PINB & 0b11110000) == 0b11110000)                        // Wait for first signal (SYNC to frame)
                {
                        if (TimerEvent > 3) break;                                              // (max. 3*8=24ms)
                }
               
                ResetTimer();
                while(TimerEvent < 1)                                                           // Check active satellites (for 1*8=8ms)
                {
                        for (i=0;i<4;i++)
                        {
                                if (!(PINB & (1<<(i+4))))
                                {
                                        active[i] = 1;
                                }
                        }
                }

                if (active[sat] == 0)                                                           // Detect fail on active satellite
                {
                        for (i=0;i<4;i++)                                                               // Select lowest active satellite
                        {
                                if (active[i] == 1)
                                {
                                        sat = i;
                                        SelectSat(sat);
                                        break;
                                }
                        }
                        LED_ON;                                                                                 // Failure-LED ON
                }

                if (!(PIND & (1<<PD6))) LED_OFF;                                        // Reset Failure-LED when Bind-Button is pressed
        }

}