code:parafoil_main
Differences
This shows you the differences between two versions of the page.
Both sides previous revisionPrevious revisionNext revision | Previous revision | ||
code:parafoil_main [2008/04/29 04:57] – laurenceb | code:parafoil_main [2008/08/16 21:11] (current) – laurenceb | ||
---|---|---|---|
Line 5: | Line 5: | ||
{{code: | {{code: | ||
+ | ===== TSIP version for lassen iq ===== | ||
+ | |||
+ | [[code: | ||
+ | (nmea version has been " | ||
===== Main.h ===== | ===== Main.h ===== | ||
Line 23: | Line 27: | ||
#include < | #include < | ||
- | #define minutes (float)1.0/ | + | #define minutes (float)1.0/ |
#define baudrate (int)4800 | #define baudrate (int)4800 | ||
- | #define delta_time (float)0.02 //20ms | + | #define delta_time (float)0.02 //20ms |
#define sqrt_delta_time (float)0.141 // | #define sqrt_delta_time (float)0.141 // | ||
#define control_gain (float)90.0 | #define control_gain (float)90.0 | ||
#define damping_constant (float)0.2 | #define damping_constant (float)0.2 | ||
+ | |||
+ | #define top_count (u16)((float)F_CPU*delta_time/ | ||
+ | #define pwm_count_time ((float)F_CPU*0.0005/ | ||
+ | #define safe_time_start (u08)((u16)(5.0*pwm_count_time)>> | ||
+ | #define safe_time_end (u08)((top_count-1000)>> | ||
- | + | #define I_C 0.0001 //control loop | |
- | #define I_C 0.0001 //control loop | + | |
#define P_C 0.02 | #define P_C 0.02 | ||
#define D_C 0.04 | #define D_C 0.04 | ||
Line 52: | Line 60: | ||
#define Deg2rad PI/180.0 | #define Deg2rad PI/180.0 | ||
- | #define battery_factor (float)8.0/ | + | #define battery_factor (float)8.0/ |
- | #define battery_chan 0x05 //ADC6 | + | #define battery_chan 0x05 //ADC6 |
#define toggle_pin PIND=0x20 // | #define toggle_pin PIND=0x20 // | ||
- | #define bauddiv | + | #define bauddiv |
- | #define radio_cts bit_is_set(PIND, | + | #define radio_cts bit_is_set(PIND, |
#define led_left PORTC = ((PORTC & ~0x18) | 0x08) | #define led_left PORTC = ((PORTC & ~0x18) | 0x08) | ||
Line 71: | Line 79: | ||
#define gyro_off PORTB|=1<< | #define gyro_off PORTB|=1<< | ||
#define gyro_on PORTB& | #define gyro_on PORTB& | ||
- | + | ||
#ifdef ADCSRA | #ifdef ADCSRA | ||
#ifndef ADCSR | #ifndef ADCSR | ||
Line 83: | Line 91: | ||
typedef struct | typedef struct | ||
{ | { | ||
- | volatile | + | float longitude; |
- | volatile | + | float latitude; |
- | volatile | + | float altitude; |
- | volatile | + | float speed; |
- | volatile | + | float heading; |
- | volatile | + | u08 packetflag; |
- | volatile | + | u08 status; |
} gps_type; | } gps_type; | ||
- | int put_char(char c, FILE * stream); //talk to the UART | + | int put_char(char c, FILE * stream); //talk to the UART |
int get_char( FILE * stream); | int get_char( FILE * stream); | ||
</ | </ | ||
- | |||
- | |||
- | |||
- | |||
- | |||
- | |||
- | |||
- | |||
- | |||
- | |||
- | |||
- | |||
- | |||
- | |||
===== Main.c ===== | ===== Main.c ===== | ||
Line 123: | Line 117: | ||
volatile kalman_state our_state; | volatile kalman_state our_state; | ||
volatile kalman_model our_model; | volatile kalman_model our_model; | ||
+ | volatile u08 Kalman_flag; | ||
volatile float Heading; | volatile float Heading; | ||
Line 130: | Line 125: | ||
volatile u08 gyro_error=0x00; | volatile u08 gyro_error=0x00; | ||
- | gps_type | + | volatile |
//ground control globals | //ground control globals | ||
#ifdef GROUND | #ifdef GROUND | ||
Line 141: | Line 136: | ||
uint8_t mcusr_mirror __attribute__ ((section (" | uint8_t mcusr_mirror __attribute__ ((section (" | ||
- | | + | void get_mcusr(void) __attribute__((naked)) __attribute__((section(" |
- | | + | void get_mcusr(void) |
- | | + | { |
- | void get_mcusr(void) | + | mcusr_mirror = MCUSR; |
- | { | + | MCUSR = 0; |
- | mcusr_mirror = MCUSR; | + | wdt_disable(); |
- | MCUSR = 0; | + | } //avr-libc wdt code |
- | wdt_disable(); | + | |
- | } //avr-libc wdt code | + | |
void setup() | void setup() | ||
Line 166: | Line 159: | ||
//Timer 1 setup | //Timer 1 setup | ||
- | TCCR1B = ((TCCR1B & ~0x07) | 0x02); //clock at F_CPU/8 | + | TCCR1B = ((TCCR1B & ~0x07) | 0x02); |
TCNT1 = 0; //reset timer1 | TCNT1 = 0; //reset timer1 | ||
Line 177: | Line 170: | ||
// set top count value to give correct period PWM | // set top count value to give correct period PWM | ||
- | ICR1 = (u16)F_CPU*delta_time/ | + | ICR1 = top_count; |
// clear output compare value A | // clear output compare value A | ||
Line 204: | Line 197: | ||
//IO setup | //IO setup | ||
gyro_off; | gyro_off; | ||
- | PORTD|=( (1<< | + | PORTD|=( (1<< |
DDRB=0b00101111; | DDRB=0b00101111; | ||
DDRC=0b00011000; | DDRC=0b00011000; | ||
Line 211: | Line 204: | ||
//Kalman setup | //Kalman setup | ||
our_model.F.tl=1.0; | our_model.F.tl=1.0; | ||
- | our_model.F.tr=delta_time; | + | our_model.F.tr=delta_time; |
our_model.F.bl=0.0; | our_model.F.bl=0.0; | ||
our_model.F.br=1.0-(damping_constant*delta_time); | our_model.F.br=1.0-(damping_constant*delta_time); | ||
- | our_model.B.tl=0.0; | + | our_model.B.tl=0.0; |
our_model.B.tr=(delta_time*delta_time*control_gain)/ | our_model.B.tr=(delta_time*delta_time*control_gain)/ | ||
our_model.B.bl=0.0; | our_model.B.bl=0.0; | ||
Line 224: | Line 217: | ||
our_model.H.bl=0.0; | our_model.H.bl=0.0; | ||
our_model.H.br=1.0; | our_model.H.br=1.0; | ||
- | + | | |
- | our_model.Q.tl=0.1/ | + | our_model.Q.tl=5.0*delta_time; //estimated - corresponds to heading change |
our_model.Q.tr=0.0; | our_model.Q.tr=0.0; | ||
our_model.Q.bl=0.0; | our_model.Q.bl=0.0; | ||
- | our_model.Q.br=1.0/ | + | our_model.Q.br=10.0*delta_time; //estimated - corresponds to rate change |
- | our_model.R.tl=20.0; //or around 20 in the case of a GPD update | + | our_model.R.tl=100.0; //Data covariance |
our_model.R.tr=0.0; | our_model.R.tr=0.0; | ||
our_model.R.bl=0.0; | our_model.R.bl=0.0; | ||
- | our_model.R.br=1.0; // | + | our_model.R.br=2.0; // |
our_state.state.t=0.0; | our_state.state.t=0.0; | ||
our_state.state.b=0.0; | our_state.state.b=0.0; | ||
- | our_state.P.tl=150.0; | + | our_state.P.tl=8000.0; |
our_state.P.tr=0.0; | our_state.P.tr=0.0; | ||
our_state.P.bl=0.0; | our_state.P.bl=0.0; | ||
- | our_state.P.br=20.0; | + | our_state.P.br=400.0; |
} | } | ||
Line 247: | Line 240: | ||
float read_gyro() | float read_gyro() | ||
{ | { | ||
- | static float turn_rate; | ||
u16 a; | u16 a; | ||
- | gyro_on; //SS = 0 | + | gyro_on; //SS = 0 |
SPDR=read_rate; | SPDR=read_rate; | ||
while(!(inb(SPSR) & (1<< | while(!(inb(SPSR) & (1<< | ||
Line 269: | Line 261: | ||
a|=(u16)SPDR; | a|=(u16)SPDR; | ||
gyro_off; | gyro_off; | ||
- | a=(a>> | + | a=(a>> |
return (float)(125.0/ | return (float)(125.0/ | ||
} | } | ||
- | ISR(TIMER1_COMPA_vect) // | + | void set_servo(float setting) |
{ | { | ||
- | sei(); //we need to be able to nest UART interrupts in here | + | OCR1A=(u16)(pwm_count_time*(3.0+setting)); |
+ | } | ||
+ | |||
+ | ISR(TIMER1_COMPA_vect) // | ||
+ | { | ||
+ | sei(); //we need to be able to nest UART interrupts in here | ||
vector measurement_vector; | vector measurement_vector; | ||
static u08 timer; | static u08 timer; | ||
static vector control_vector; | static vector control_vector; | ||
+ | static kalman_state our_state_local; | ||
static float integral; | static float integral; | ||
+ | static float target; | ||
float a; | float a; | ||
control_vector.t=0.0; | control_vector.t=0.0; | ||
+ | if(!timer) | ||
+ | { | ||
+ | our_state_local=our_state; | ||
+ | } | ||
if(Heading_flag) | if(Heading_flag) | ||
{ | { | ||
our_model.H.tl=1.0; | our_model.H.tl=1.0; | ||
- | measurement_vector.t=Heading; | + | measurement_vector.t=Heading; |
+ | target=Target; | ||
Heading_flag=FALSE; | Heading_flag=FALSE; | ||
} | } | ||
Line 297: | Line 301: | ||
{ | { | ||
our_model.H.br=0.0; | our_model.H.br=0.0; | ||
- | measurement_vector.b=0.0; | + | measurement_vector.b=0.0; |
} | } | ||
else | else | ||
Line 303: | Line 307: | ||
our_model.H.br=1.0; | our_model.H.br=1.0; | ||
} | } | ||
- | our_state=predict_and_update(& | + | our_state_local=predict_and_update(& |
- | if(our_state.state.t> | + | if(our_state_local.state.t> |
{ | { | ||
- | our_state.state.t-=360; | + | our_state_local.state.t-=360; |
} | } | ||
- | if(our_state.state.t< | + | if(our_state_local.state.t< |
{ | { | ||
- | our_state.state.t+=360; | + | our_state_local.state.t+=360; |
} | } | ||
- | a=our_state.state.t-Target; | + | a=our_state_local.state.t-target; |
if(a> | if(a> | ||
{ | { | ||
Line 321: | Line 325: | ||
a+=360; | a+=360; | ||
} | } | ||
- | integral+=a; | + | integral+=a; |
if(integral> | if(integral> | ||
{ | { | ||
Line 332: | Line 336: | ||
if(timer> | if(timer> | ||
{ | { | ||
- | control_vector.b=P_C*a+I_C*integral+D_C*our_state.state.b; //PID control | + | control_vector.b=P_C*a+I_C*integral+D_C*our_state_local.state.b; |
- | if(control_vector.b> | + | if(control_vector.b> |
{ | { | ||
control_vector.b=1.0; | control_vector.b=1.0; | ||
Line 341: | Line 345: | ||
control_vector.b=-1.0; | control_vector.b=-1.0; | ||
} | } | ||
- | set_PWM1A( (u16)(((float)F_CPU/ | + | set_servo(control_vector.b); |
} | } | ||
else | else | ||
Line 350: | Line 354: | ||
if(pwm_counter==0x0A) // | if(pwm_counter==0x0A) // | ||
{ | { | ||
- | control_vector.b=(float)(pwm_period-3*pwm_record_factor); | + | control_vector.b=((float)pwm_period*(1.0/pwm_record_factor)-3.0);// |
} | } | ||
#endif | #endif | ||
- | } | + | if(!Kalman_flag) //our global kalman state is unlocked |
- | + | { | |
- | void set_PWM1A(u16 setting) | + | our_state=our_state_local; |
- | { | + | } |
- | u08 sreg = SREG; | + | |
- | | + | |
- | OCR1A=setting; | + | |
- | SREG = sreg; | + | |
} | } | ||
- | u08 cutdownrulecheck(int time) | + | u08 cutdownrulecheck(int time, float * altitude) |
{ | { | ||
- | if(gps.altitude> | + | if(*altitude> |
{ | { | ||
printf(" | printf(" | ||
Line 380: | Line 380: | ||
u08 checksum(char *c_buf) | u08 checksum(char *c_buf) | ||
{ | { | ||
- | u08 i; | + | u08 sum=0; |
- | | + | while(*c_buf) |
- | u08 size=strlen(c_buf); | + | { |
- | | + | sum += *c_buf++; |
- | | + | } |
- | sum += *c_buf; | + | return sum; |
- | } | + | |
- | return sum; | + | |
} | } | ||
Line 403: | Line 401: | ||
void wiggleservo() | void wiggleservo() | ||
{ | { | ||
- | OCR1A =(u16)(((float)F_CPU/ | + | set_servo(1); |
_delay_ms(150); | _delay_ms(150); | ||
- | OCR1A =(u16)(((float)F_CPU/ | + | set_servo(-1); |
_delay_ms(150); | _delay_ms(150); | ||
- | OCR1A =(u16)(((float)F_CPU/ | + | set_servo(0); |
} | } | ||
Line 422: | Line 420: | ||
return battery_factor*((float)(inb(ADCL) | (inb(ADCH)<< | return battery_factor*((float)(inb(ADCL) | (inb(ADCH)<< | ||
} | } | ||
- | + | ||
- | void dataprint() | + | void dataprint(gps_type * gps) |
{ | { | ||
char outputbuffer[80]; | char outputbuffer[80]; | ||
char longitude; | char longitude; | ||
char latitude; | char latitude; | ||
- | if (gps.longitude> | + | if (gps->longitude> |
{ | { | ||
longitude=' | longitude=' | ||
Line 436: | Line 434: | ||
longitude=' | longitude=' | ||
} | } | ||
- | if (gps.latitude> | + | if (gps->latitude> |
{ | { | ||
latitude=' | latitude=' | ||
Line 444: | Line 442: | ||
latitude=' | latitude=' | ||
} | } | ||
- | sprintf(outputbuffer," | + | Kalman_flag=TRUE; |
- | (double)gps.longitude, | + | while((TCNT1>> |
+ | sprintf(outputbuffer," | ||
+ | (double)gps->longitude, | ||
(double)our_state.state.b, | (double)our_state.state.b, | ||
+ | Kalman_flag=FALSE; | ||
if(eeprom_read_byte(& | if(eeprom_read_byte(& | ||
{ | { | ||
Line 461: | Line 462: | ||
int put_char(char c, FILE * stream) | int put_char(char c, FILE * stream) | ||
{ | { | ||
- | loop_until_bit_is_set(UCSR0A, | + | loop_until_bit_is_set(UCSR0A, |
UDR0 = c; | UDR0 = c; | ||
| | ||
Line 478: | Line 479: | ||
int Heightupto=0; | int Heightupto=0; | ||
float K; | float K; | ||
- | float | + | float |
- | float | + | float |
- | float Wind_x; | + | |
- | float Wind_y; | + | |
FILE mystdio = FDEV_SETUP_STREAM(put_char, | FILE mystdio = FDEV_SETUP_STREAM(put_char, | ||
setup(); | setup(); | ||
Line 488: | Line 487: | ||
if(undead_man) // | if(undead_man) // | ||
{ | { | ||
- | while(!gps.status) //wait for a gps lock | + | while(!Gps.status) //wait for a gps lock |
{ | { | ||
- | printf(" | + | printf(" |
wdt_reset(); | wdt_reset(); | ||
- | while(!gps.packetflag) // | + | Gps.packetflag=FALSE; |
- | gps.packetflag=FALSE; | + | while(!Gps.packetflag); //wait for some new gps |
} | } | ||
for (L=0; | for (L=0; | ||
Line 501: | Line 500: | ||
wdt_reset(); | wdt_reset(); | ||
} | } | ||
- | if(eeprom_read_byte(& | + | if(eeprom_read_byte(& |
{ | { | ||
eeprom_write_byte(& | eeprom_write_byte(& | ||
Line 515: | Line 514: | ||
disable_ground_control(); | disable_ground_control(); | ||
#endif | #endif | ||
- | L=0; //need it =0 for next stage | + | L=0; //need it =0 for next stage |
printf(" | printf(" | ||
} | } | ||
- | if(!eeprom_read_byte(& | + | if(!eeprom_read_byte(& |
{ | { | ||
- | while(!cutdownrulecheck(time)) | + | while(!cutdownrulecheck(time,& |
{ | { | ||
| | ||
- | Windx_total | + | Wind_x |
- | Windy_total | + | Wind_y |
| | ||
//find the altitude in 50m incraments | //find the altitude in 50m incraments | ||
- | if ((int)(gps.altitude/ | + | if ((int)(Gps.altitude/ |
{ | { | ||
Heightupto++; | Heightupto++; | ||
- | eeprom_write_byte(& | + | eeprom_write_byte(& |
- | eeprom_write_byte(& | + | eeprom_write_byte(& |
L = 0; | L = 0; | ||
} | } | ||
wiggleservo(); | wiggleservo(); | ||
- | if (radio_cts) | + | if (radio_cts) |
{ | { | ||
- | dataprint(); | + | dataprint(&Gps); |
} | } | ||
wdt_reset(); | wdt_reset(); | ||
- | while(!gps.packetflag) // | + | Gps.packetflag=FALSE; |
- | gps.packetflag=FALSE; | + | while(!Gps.packetflag); //wait for some new gps |
} | } | ||
cutdown(12); | cutdown(12); | ||
- | eeprom_write_byte(& | + | eeprom_write_byte(& |
} | } | ||
sbi(TIMSK1, | sbi(TIMSK1, | ||
for(;;) | for(;;) | ||
{ | { | ||
- | L=(int)gps.altitude/ | + | L=(int)Gps.altitude/ |
Wind_x=(float)(eeprom_read_byte(& | Wind_x=(float)(eeprom_read_byte(& | ||
Wind_y=(float)(eeprom_read_byte(& | Wind_y=(float)(eeprom_read_byte(& | ||
// | // | ||
- | Target = Rad2deg*atan2((targeteast - gps.longitude)*cos(Deg2rad*gps.latitude), | + | Target = Rad2deg*atan2((targeteast - Gps.longitude)*cos(Deg2rad*Gps.latitude), |
//wind compensation | //wind compensation | ||
- | Heading =Rad2deg*atan2(gps.speed*sin(gps.heading)-Wind_x, | + | Heading =Rad2deg*atan2(Gps.speed*sin(Gps.heading)-Wind_x, |
Heading_flag=TRUE; | Heading_flag=TRUE; | ||
- | K=Heading-Target; | + | K=Heading-Target; |
if (K >180) | if (K >180) | ||
{ | { | ||
K -= 360; | K -= 360; | ||
- | } | + | } //all to keep us in +-180 degree range |
if(K < -180) | if(K < -180) | ||
{ | { | ||
Line 574: | Line 573: | ||
led_right; | led_right; | ||
} | } | ||
- | if (radio_cts) | + | if (radio_cts) |
{ | { | ||
- | dataprint(); | + | dataprint(&Gps); |
} | } | ||
#ifdef GROUND | #ifdef GROUND | ||
- | if(gps.altitude< | + | if(Gps.altitude< |
{ | { | ||
enable_ground_control(); | enable_ground_control(); | ||
Line 585: | Line 584: | ||
#endif | #endif | ||
wdt_reset(); | wdt_reset(); | ||
- | while(!gps.packetflag) // | + | |
- | gps.packetflag=FALSE; | + | while(!Gps.packetflag); //wait for some new gps |
} | } | ||
} | } | ||
+ | |||
/* | /* | ||
char *datastring() // | char *datastring() // |
code/parafoil_main.1209445048.txt.gz · Last modified: 2008/07/19 23:31 (external edit)