Subversion Repositories svnkaklik

Rev

Rev 512 | Blame | Compare with Previous | Last modification | View Log | Download

//----- Include Files ---------------------------------------------------------
#include <avr/io.h>             // include I/O definitions (port names, pin names, etc)
#include <avr/interrupt.h>      // include interrupt support
#include <math.h>

#include "global.h"             // include our global settings
#include "uart2.h"              // include uart function library
#include "rprintf.h"    // include printf function library
#include "timer.h"              // include timer function library (timing, PWM, etc)
#include "a2d.h"                // include A/D converter function library

//----- Begin Code ------------------------------------------------------------
#define BUFLEN 64

void radtodeg(double fi, u16 *deg, u08 *min, u08 *sec)  //convert radians to degrees mins and seconds
{
  double pom;

        fi=fi*180/PI;
        *deg=(u16)trunc(fi);
        pom=(fi-(*deg))*60;
        *min=(u08)trunc(pom);
        *sec=(u08)round((pom-(*min))*60);
}

inline double quadraticerror(double average, double buf[], u16 size) // compute average quadratic error
{
  u16 i;
  double err=0;

        for(i=0; i<size; i++) err += square(buf[i]-average);            // sum quadratic errors 
        err = sqrt((1/(double)size)*err);
        return err;
}

int main(void)
{
        u16 i,x,y;
        double fi, err, fibuf[BUFLEN];          // buffer for recorded and computed values
        u08 fi_min, fi_sec, err_min, err_sec;   // computed angles
        u16 fi_deg, err_deg;            // computed angles in whole degrees

        // initialize some libraries
        // initialize the UART (serial port)
        uartInit();
        uartSetBaudRate(0,9600);
        // make all rprintf statements use uart for output
        rprintfInit(uart0SendByte);
        // initialize the timer system
        timerInit();
        // turn on and initialize A/D converter
        a2dInit();      
        // configure a2d port (PORTA) as input
        // so we can receive analog signals
        DDRF = 0x00;
        // make sure pull-up resistors are turned off
        PORTF = 0x00;

        // set the a2d prescaler (clock division ratio)
        // - a lower prescale setting will make the a2d converter go faster
        // - a higher setting will make it go slower but the measurements
        //   will be more accurate
        // - other allowed prescale values can be found in a2d.h
        a2dSetPrescaler(ADC_PRESCALE_DIV128);

        // set the a2d reference
        // - the reference is the voltage against which a2d measurements are made
        // - other allowed reference values can be found in a2d.h
        a2dSetReference(ADC_REFERENCE_AREF);

        // use a2dConvert8bit(channel#) to get an 8bit a2d reading
        // use a2dConvert10bit(channel#) to get a 10bit a2d reading

        rprintf("inklinometr 2009\r\n");

        while(1)
        {
                fi=0;
                err=0;
                for(i=0; i<BUFLEN; i++)
                { 
                        x = a2dConvert10bit(ADC_CH_ADC0);
                        y = a2dConvert10bit(ADC_CH_ADC1);
                        fibuf[i] = atan2((double)x-511,(double)y-511);          // record computed angles to buffer for post processing
                }
                for(i=0; i<BUFLEN; i++) fi += fibuf[i];         // sum recorded angles
                fi = (fi/BUFLEN)+PI;            // average recorded angles and expand product to whole circle
                
                err=quadraticerror((fi-PI),fibuf,BUFLEN);
                radtodeg(fi,&fi_deg,&fi_min,&fi_sec);           //translate radians to degrees
                radtodeg(err,&err_deg,&err_min,&err_sec);
                
                rprintf("fi:%d.%d.%d  +- %d.%d.%d \r\n", fi_deg, fi_min, fi_sec, err_deg, err_min, err_sec);
                delay_ms(20);
        }
        return 0;
}