Subversion Repositories FlightCtrl

Rev

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

// ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
// + Copyright (c) Holger Buss, Ingo Busker
// + Nur für den privaten Gebrauch
// + www.MikroKopter.com
// + porting the sources to other systems or using the software on other systems (except hardware from www.mikrokopter.de) is not allowed
// ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
// + Es gilt für das gesamte Projekt (Hardware, Software, Binärfiles, Sourcecode und Dokumentation),
// + dass eine Nutzung (auch auszugsweise) nur für den privaten (nicht-kommerziellen) Gebrauch zulässig ist.
// + Sollten direkte oder indirekte kommerzielle Absichten verfolgt werden, ist mit uns (info@mikrokopter.de) Kontakt
// + bzgl. der Nutzungsbedingungen aufzunehmen.
// + Eine kommerzielle Nutzung ist z.B.Verkauf von MikroKoptern, Bestückung und Verkauf von Platinen oder Bausätzen,
// + Verkauf von Luftbildaufnahmen, usw.
// ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
// + Werden Teile des Quellcodes (mit oder ohne Modifikation) weiterverwendet oder veröffentlicht,
// + unterliegen sie auch diesen Nutzungsbedingungen und diese Nutzungsbedingungen incl. Copyright müssen dann beiliegen
// ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
// + Sollte die Software (auch auszugesweise) oder sonstige Informationen des MikroKopter-Projekts
// + auf anderen Webseiten oder sonstigen Medien veröffentlicht werden, muss unsere Webseite "http://www.mikrokopter.de"
// + eindeutig als Ursprung verlinkt werden
// ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
// + Keine Gewähr auf Fehlerfreiheit, Vollständigkeit oder Funktion
// + Benutzung auf eigene Gefahr
// + Wir übernehmen keinerlei Haftung für direkte oder indirekte Personen- oder Sachschäden
// ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
// + Die Portierung der Software (oder Teile davon) auf andere Systeme (ausser der Hardware von www.mikrokopter.de) ist nur
// + mit unserer Zustimmung zulässig
// ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
// + Die Funktion printf_P() unterliegt ihrer eigenen Lizenz und ist hiervon nicht betroffen
// ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
// + Redistributions of source code (with or without modifications) must retain the above copyright notice,
// + this list of conditions and the following disclaimer.
// +   * Neither the name of the copyright holders nor the names of contributors may be used to endorse or promote products derived
// +     from this software without specific prior written permission.
// +   * The use of this project (hardware, software, binary files, sources and documentation) is only permittet
// +     for non-commercial use (directly or indirectly)
// +     Commercial use (for excample: selling of MikroKopters, selling of PCBs, assembly, ...) is only permitted
// +     with our written permission
// +   * If sources or documentations are redistributet on other webpages, out webpage (http://www.MikroKopter.de) must be
// +     clearly linked as origin
// +   * porting to systems other than hardware from www.mikrokopter.de is not allowed
// +  THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
// +  AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
// +  IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
// +  ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
// +  LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
// +  CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
// +  SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
// +  INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN// +  CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
// +  ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
// +  POSSIBILITY OF SUCH DAMAGE.
// ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
#include <stdlib.h>
#include <avr/io.h>
#include <avr/interrupt.h>

#include "analog.h"
#include "main.h"
#include "timer0.h"
#include "fc.h"
#include "printf_P.h"
#include "eeprom.h"
#include "twimaster.h"
#include "uart0.h"

volatile uint16_t Test = 0;

volatile int16_t UBat = 100;
volatile int16_t AdValueGyroNick = 0, AdValueGyroRoll = 0,  AdValueGyroYaw = 0;
volatile int16_t FilterHiResGyroNick = 0, FilterHiResGyroRoll = 0;
volatile int16_t HiResGyroNick = 2500, HiResGyroRoll = 2500;
volatile int16_t AdValueAccRoll = 0,  AdValueAccNick = 0, AdValueAccTop = 0, AdValueAccZ = 0;
volatile int32_t AirPressure = 32000;
volatile int32_t StartAirPressure;
volatile int16_t AdAirPressure = 1023;
volatile int32_t ReadingHeight = 0;
volatile int16_t ReadingVario = 0;
volatile int32_t SumHeight = 0;
volatile uint16_t MeasurementCounter = 0;
volatile uint8_t ADReady = 1;

uint8_t DacOffsetGyroNick = 115, DacOffsetGyroRoll = 115, DacOffsetGyroYaw = 115;
uint8_t GyroDefectNick = 0, GyroDefectRoll = 0, GyroDefectYaw = 0;
int8_t ExpandBaro = 0;
uint8_t PressureSensorOffset;

/*****************************************************/
/*     Initialize Analog Digital Converter           */
/*****************************************************/
void ADC_Init(void)
{
        uint8_t sreg = SREG;
        // disable all interrupts before reconfiguration
        cli();
        //ADC0 ... ADC7 is connected to PortA pin 0 ... 7
        DDRA = 0x00;
        PORTA = 0x00;
        // Digital Input Disable Register 0
        // Disable digital input buffer for analog adc_channel pins
        DIDR0 = 0xFF;
        // external reference, adjust data to the right
    ADMUX &= ~((1 << REFS1)|(1 << REFS0)|(1 << ADLAR));
    // set muxer to ADC adc_channel 0 (0 to 7 is a valid choice)
    ADMUX = (ADMUX & 0xE0) | 0x00;
    //Set ADC Control and Status Register A
    //Auto Trigger Enable, Prescaler Select Bits to Division Factor 128, i.e. ADC clock = SYSCKL/128 = 156.25 kHz
        ADCSRA = (0<<ADEN)|(0<<ADSC)|(0<<ADATE)|(1<<ADPS2)|(1<<ADPS1)|(1<<ADPS0)|(0<<ADIE);
        //Set ADC Control and Status Register B
        //Trigger Source to Free Running Mode
        ADCSRB &= ~((1 << ADTS2)|(1 << ADTS1)|(1 << ADTS0));
        // Start AD conversion
        ADC_Enable();
    // restore global interrupt flags
    SREG = sreg;
}

void SearchAirPressureOffset(void)
{
        uint8_t off;
        off = GetParamByte(PID_PRESSURE_OFFSET);
        if(off > 20) off -= 10;
        OCR0A = off;
        ExpandBaro = 0;
        Delay_ms_Mess(100);
        if(AdAirPressure < AIR_PRESSURE_SEARCH) off = 0;
        for(; off < 250;off++)
        {
                OCR0A = off;
                Delay_ms_Mess(50);
                printf(".");
                if(AdAirPressure < AIR_PRESSURE_SEARCH) break;
        }
        SetParamByte(PID_PRESSURE_OFFSET, off);
        PressureSensorOffset = off;
        AirPressure = AIR_PRESSURE_SCALE * (int32_t)AdAirPressure; // init IIR Filter
        Delay_ms_Mess(300);
}


void SearchDacGyroOffset(void)
{
        uint8_t i, ready = 0;
        uint16_t timeout ;

        GyroDefectNick = 0; GyroDefectRoll = 0; GyroDefectYaw = 0;

        timeout = SetDelay(2000);
        if(BoardRelease == 13) // the auto offset calibration is available only at board release 1.3
        {
                for(i = 140; i != 0; i--)
                {
                        if(ready == 3 && i > 10) i = 9;
                        ready = 0;
                        if(AdValueGyroNick < 1020) DacOffsetGyroNick--; else if(AdValueGyroNick > 1030) DacOffsetGyroNick++; else ready++;
                        if(AdValueGyroRoll < 1020) DacOffsetGyroRoll--; else if(AdValueGyroRoll > 1030) DacOffsetGyroRoll++; else ready++;
                        if(AdValueGyroYaw  < 1020) DacOffsetGyroYaw-- ; else if(AdValueGyroYaw  > 1030) DacOffsetGyroYaw++ ; else ready++;
                        I2C_Start(TWI_STATE_GYRO_OFFSET_TX);   // initiate data transmission
                        if(DacOffsetGyroNick < 10)  { GyroDefectNick = 1; DacOffsetGyroNick = 10;}; if(DacOffsetGyroNick > 245) { GyroDefectNick = 1; DacOffsetGyroNick = 245;};
                        if(DacOffsetGyroRoll < 10)  { GyroDefectRoll = 1; DacOffsetGyroRoll = 10;}; if(DacOffsetGyroRoll > 245) { GyroDefectRoll = 1; DacOffsetGyroRoll = 245;};
                        if(DacOffsetGyroYaw  < 10)  { GyroDefectYaw  = 1; DacOffsetGyroYaw  = 10;}; if(DacOffsetGyroYaw  > 245) { GyroDefectYaw  = 1; DacOffsetGyroYaw  = 245;};
                        while(twi_state)
                        {
                                if(CheckDelay(timeout))
                                {
                                        printf("\r\n DAC or I2C Error1 check I2C, 3Vref, DAC, and BL-Ctrl");
                                        break;
                                }
                        } // wait for end of data transmission
                        ADReady = 0;
                        ADC_Enable();
                        while(!ADReady);
                        if(i < 10) Delay_ms_Mess(10);
                }
                Delay_ms_Mess(70);
        }
}




/*****************************************************/
/*     Interrupt Service Routine for ADC             */
/*****************************************************/
// runs at 312.5 kHz or 3.2 µs
// if after (60.8µs) all 19 states are processed the interrupt is disabled
// and the update of further ads is stopped

/*
0  nickgyro
1  rollgyro
2  yawgyro
3  accroll
4  accnick
5  nickgyro
6  rollgyro
7  ubat
8  acctop
9  air pressure
10 nickgyro
11 rollgyro
12 yawgyro
13 accroll
14 accnick
15 gyronick
16 gyroroll
17 air pressure
*/



#define AD_GYRO_YAW             0
#define AD_GYRO_ROLL    1
#define AD_GYRO_NICK    2
#define AD_AIRPRESS             3
#define AD_UBAT                 4
#define AD_ACC_TOP              5
#define AD_ACC_ROLL             6
#define AD_ACC_NICK             7

ISR(ADC_vect)
{
    static uint8_t ad_channel = AD_GYRO_NICK, state = 0;
    static uint16_t gyroyaw, gyroroll, gyronick, accroll, accnick;
    static int32_t filtergyronick, filtergyroroll;
    static int32_t tmpAirPressure = 0;
    static uint8_t AirPressCount = 0;

    // state machine
        switch(state++)
        {
                case 0:
                        gyronick = ADC; // get nick gyro voltage 1st sample
                        ad_channel = AD_GYRO_ROLL;
                        break;
                case 1:
                        gyroroll = ADC; // get roll gyro voltage 1st sample
                        ad_channel = AD_GYRO_YAW;
                        break;
                case 2:
                        gyroyaw = ADC; // get yaw gyro voltage 1st sample
                        ad_channel = AD_ACC_ROLL;
                        break;
                case 3:
                        accroll = ADC; // get roll acc voltage 1st sample
                        ad_channel = AD_ACC_NICK;
            break;
                case 4:
                        accnick = ADC; // get nick acc voltage 1st sample
                        ad_channel = AD_GYRO_NICK;
                        break;
                case 5:
                        gyronick += ADC; // get nick gyro voltage 2nd sample
                        ad_channel = AD_GYRO_ROLL;
                        break;
                case 6:
                        gyroroll += ADC; // get roll gyro voltage 2nd sample
                        ad_channel = AD_UBAT;
                        break;
                case 7:
                        // get actual UBat (Volts*10) is ADC*30V/1024*10 = ADC/3
                        UBat = (3 * UBat + ADC / 3) / 4; // low pass filter updates UBat only to 1 quater with actual ADC value
                        ad_channel = AD_ACC_TOP;
            break;
                case 8:
                        AdValueAccZ = ADC; // get plain acceleration in Z direction
                        AdValueAccTop =  (int16_t)ADC - AdBiasAccTop; // get acceleration in Z direction
                        if(AdValueAccTop > 1)
                        {
                                if(AdBiasAccTop < 750)
                                {
                                        AdBiasAccTop += 0.02;
                                        if(ModelIsFlying < 500) AdBiasAccTop += 0.1;
                                }
                        }
                        else if(AdValueAccTop < -1)
                        {
                                if(AdBiasAccTop > 550)
                                {
                                        AdBiasAccTop -= 0.02;
                                        if(ModelIsFlying < 500) AdBiasAccTop -= 0.1;
                                }
                        }
                        // averaging acc
                        ReadingIntegralTop -= ReadingIntegralTop / 1024; // discharge
                        ReadingIntegralTop += AdValueAccTop; // load
                        ad_channel = AD_AIRPRESS;
                        break;
                // case 9 is moved to the end
                case 10:
                        gyronick += ADC; // get nick gyro voltage 3rd sample
                        ad_channel = AD_GYRO_ROLL;
                        break;
                case 11:
                        gyroroll += ADC; // get roll gyro voltage 3rd sample
                        ad_channel = AD_GYRO_YAW;
                        break;
                case 12:
                        gyroyaw += ADC; // get yaw gyro voltage 2nd sample
                        if(BoardRelease == 10) AdValueGyroYaw = (gyroyaw + 1) / 2; // analog gain on board 1.0 is 2 times higher
                        else
                        if(BoardRelease == 20) AdValueGyroYaw = 2047 - gyroyaw; // 2 times higher than a single sample
                        else                   AdValueGyroYaw = gyroyaw;        // 2 times higher than a single sample
                        ad_channel = AD_ACC_ROLL;
                        break;
                case 13:
                        accroll += ADC; // get roll acc voltage 2nd sample
                        AdValueAccRoll = AdBiasAccRoll - accroll; // subtract bias
                        ad_channel = AD_ACC_NICK;
                        break;
                case 14:
                        accnick += ADC; // get nick acc voltage 2nd sample
                        AdValueAccNick = accnick  - AdBiasAccNick; // subtract bias
                        ad_channel = AD_GYRO_NICK;
            break;
        case 15:
                gyronick += ADC; // get nick gyro voltage 4th sample
                if(BoardRelease == 10) gyronick *= 2; // 8 times higer than a single sample, HW gain x2
                else gyronick *= 4; // 16 times higer than a single sample
                        AdValueGyroNick = gyronick / 8; // 2 times higher than a single sample
                        filtergyronick = (filtergyronick + gyronick) / 2; //(16 samples)/2 results in a factor of 8 higher than a single sample) see HIRES_GYRO_AMPLIFY
                        HiResGyroNick = filtergyronick - BiasHiResGyroNick;
                        FilterHiResGyroNick = (FilterHiResGyroNick + HiResGyroNick) / 2;
                        ad_channel = AD_GYRO_ROLL;
                        break;
        case 16:
                gyroroll += ADC; // get roll gyro voltage 4th sample
                if(BoardRelease == 10) gyroroll *= 2; // 8 times higer than a single sample, HW gain x2
                else gyroroll *= 4; // 16 times higher than a single sample
                        AdValueGyroRoll = gyroroll / 8; // 2 times higher than a single sample
                        filtergyroroll = (filtergyroroll + gyroroll) / 2; //(16 samples)/2 results in a factor of 8 higher than a single sample) see HIRES_GYRO_AMPLIFY
                        HiResGyroRoll = filtergyroroll - BiasHiResGyroRoll;
                        FilterHiResGyroRoll = (FilterHiResGyroRoll + HiResGyroRoll) / 2;
                        ad_channel = AD_AIRPRESS;
                        break;
                case 17:
                        state = 0; // restart sequence from beginning
            ADReady = 1; // mark
            MeasurementCounter++; // increment total measurement counter
                        // "break;" is missing to enable fall thru case 9 at the end of the sequence
                case 9:
                        AdAirPressure = ADC; // update meassured air pressure, changes -523 counts per OPA_OFFSET_STEP
                        tmpAirPressure += AdAirPressure;
                        if(++AirPressCount >= AIR_PRESSURE_SCALE)
                        {
                                #define AIRPRESSURE_FILTER_TIME 8
                                AirPressure = (AirPressure * (AIRPRESSURE_FILTER_TIME - 1) + tmpAirPressure + (AIR_PRESSURE_SCALE * EXPANDBARO_ADC_SHIFT) * (int32_t)ExpandBaro + AIRPRESSURE_FILTER_TIME/2)/AIRPRESSURE_FILTER_TIME;
                                ReadingHeight = (StartAirPressure - AirPressure); // change of air pressure to ground
                                SumHeight -= SumHeight/SM_FILTER;
                                SumHeight += ReadingHeight;
                                ReadingVario = (15 * ReadingVario + 8 * (int16_t)(ReadingHeight - SumHeight/SM_FILTER))/16;
                                tmpAirPressure /= 2;
                                AirPressCount = AIR_PRESSURE_SCALE/2;
                        }
                        ad_channel = AD_GYRO_NICK;
                        break;
                default:
                        ad_channel = AD_GYRO_NICK;
                        state = 0;
                        break;
        }
    // set adc muxer to next ad_channel
    ADMUX = (ADMUX & 0xE0) | ad_channel;
    // after full cycle stop further interrupts
    if(state != 0) ADC_Enable();
}