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/05/01 00:06] – laurenceb | code:parafoil_main [2008/08/16 21:11] (current) – laurenceb | ||
---|---|---|---|
Line 4: | Line 4: | ||
{{code: | {{code: | ||
+ | |||
+ | ===== TSIP version for lassen iq ===== | ||
+ | |||
+ | [[code: | ||
+ | (nmea version has been " | ||
===== Main.h ===== | ===== Main.h ===== | ||
Line 98: | Line 103: | ||
int get_char( FILE * stream); | int get_char( FILE * stream); | ||
</ | </ | ||
- | |||
- | |||
- | |||
- | |||
- | |||
- | |||
- | |||
- | |||
- | |||
- | |||
- | |||
- | |||
- | |||
- | |||
- | |||
- | |||
- | |||
- | |||
- | |||
===== Main.c ===== | ===== Main.c ===== | ||
Line 131: | 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 138: | 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 183: | 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 210: | Line 197: | ||
//IO setup | //IO setup | ||
gyro_off; | gyro_off; | ||
- | PORTD|=( (1<< | + | PORTD|=( (1<< |
DDRB=0b00101111; | DDRB=0b00101111; | ||
DDRC=0b00011000; | DDRC=0b00011000; | ||
Line 217: | 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 230: | 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 253: | 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 275: | Line 261: | ||
a|=(u16)SPDR; | a|=(u16)SPDR; | ||
gyro_off; | gyro_off; | ||
- | a=(a>> | + | a=(a>> |
return (float)(125.0/ | return (float)(125.0/ | ||
} | } | ||
- | + | ||
- | void set_PWM1A(u16 setting) | + | void set_servo(float setting) |
{ | { | ||
- | u08 sreg = SREG; | + | OCR1A=(u16)(pwm_count_time*(3.0+setting)); // |
- | cli(); | + | |
- | OCR1A=setting; | + | |
- | SREG = sreg; | + | |
} | } | ||
ISR(TIMER1_COMPA_vect) // | ISR(TIMER1_COMPA_vect) // | ||
{ | { | ||
- | sei(); //we need to be able to nest UART interrupts in here | + | 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; | 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; | target=Target; | ||
Heading_flag=FALSE; | Heading_flag=FALSE; | ||
Line 313: | 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 319: | 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 337: | Line 325: | ||
a+=360; | a+=360; | ||
} | } | ||
- | integral+=a; | + | integral+=a; |
if(integral> | if(integral> | ||
{ | { | ||
Line 348: | 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 357: | 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 366: | 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_state=our_state_local; | ||
+ | } | ||
} | } | ||
- | + | ||
- | u08 cutdownrulecheck(int time) | + | u08 cutdownrulecheck(int time, float * altitude) |
{ | { | ||
- | if(gps.altitude> | + | if(*altitude> |
{ | { | ||
printf(" | printf(" | ||
Line 409: | 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 428: | 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 442: | Line 434: | ||
longitude=' | longitude=' | ||
} | } | ||
- | if (gps.latitude> | + | if (gps->latitude> |
{ | { | ||
latitude=' | latitude=' | ||
Line 450: | 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 467: | 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 484: | 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 494: | 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 507: | Line 500: | ||
wdt_reset(); | wdt_reset(); | ||
} | } | ||
- | if(eeprom_read_byte(& | + | if(eeprom_read_byte(& |
{ | { | ||
eeprom_write_byte(& | eeprom_write_byte(& | ||
Line 521: | 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 580: | 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 591: | 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.1209600360.txt.gz · Last modified: 2008/07/19 23:31 (external edit)