mirror of
https://github.com/ArduPilot/ardupilot
synced 2025-01-06 16:08:28 -04:00
13f8f8ef11
git-svn-id: https://arducopter.googlecode.com/svn/trunk@231 f9c3cf11-9bcb-44bc-f272-b75c42450872
416 lines
11 KiB
C
416 lines
11 KiB
C
/*
|
|
ArduCopter 1.3 - Aug 2010
|
|
www.ArduCopter.com
|
|
Copyright (c) 2010. All rights reserved.
|
|
An Open Source Arduino based multicopter.
|
|
|
|
This program is free software: you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
(at your option) any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include "WProgram.h"
|
|
#include "UserConfig.h"
|
|
|
|
/*******************************************************************/
|
|
// ArduPilot Mega specific hardware and software settings
|
|
//
|
|
// DO NOT EDIT unless you are absolytely sure of your doings.
|
|
// User configurable settings are on UserConfig.h
|
|
/*******************************************************************/
|
|
|
|
|
|
/* APM Hardware definitions */
|
|
#define LED_Yellow 36
|
|
#define LED_Red 35
|
|
#define LED_Green 37
|
|
#define RELE_pin 47
|
|
#define SW1_pin 41
|
|
#define SW2_pin 40
|
|
|
|
//#define VDIV1 AN1
|
|
//#define VDIV2 AN2
|
|
//#define VDIV3 AN3
|
|
//#define VDIV4 AN4
|
|
|
|
//#define AN5
|
|
//#define AN6
|
|
|
|
// Sensor: GYROX, GYROY, GYROZ, ACCELX, ACCELY, ACCELZ
|
|
uint8_t sensors[6] = {1, 2, 0, 4, 5, 6}; // For ArduPilot Mega Sensor Shield Hardware
|
|
|
|
// Sensor: GYROX, GYROY, GYROZ, ACCELX, ACCELY, ACCELZ, MAGX, MAGY, MAGZ
|
|
int SENSOR_SIGN[]={
|
|
1, -1, -1, // GYROX, GYROY, GYROZ
|
|
-1, 1, 1, // ACCELX, ACCELY, ACCELZ
|
|
-1, -1, -1}; // MAGNETOX, MAGNETOY, MAGNETOZ
|
|
//{-1,1,-1,1,-1,1,-1,-1,-1};
|
|
|
|
|
|
/* APM Hardware definitions, END */
|
|
|
|
/* General definitions */
|
|
|
|
#define TRUE 1
|
|
#define FALSE 0
|
|
#define ON 1
|
|
#define OFF 0
|
|
|
|
|
|
// ADC : Voltage reference 3.3v / 12bits(4096 steps) => 0.8mV/ADC step
|
|
// ADXL335 Sensitivity(from datasheet) => 330mV/g, 0.8mV/ADC step => 330/0.8 = 412
|
|
// Tested value : 408
|
|
#define GRAVITY 408 //this equivalent to 1G in the raw data coming from the accelerometer
|
|
#define Accel_Scale(x) x*(GRAVITY/9.81)//Scaling the raw data of the accel to actual acceleration in meters for seconds square
|
|
|
|
#define ToRad(x) (x*0.01745329252) // *pi/180
|
|
#define ToDeg(x) (x*57.2957795131) // *180/pi
|
|
|
|
// IDG500 Sensitivity (from datasheet) => 2.0mV/º/s, 0.8mV/ADC step => 0.8/3.33 = 0.4
|
|
// Tested values :
|
|
#define Gyro_Gain_X 0.4 //X axis Gyro gain
|
|
#define Gyro_Gain_Y 0.41 //Y axis Gyro gain
|
|
#define Gyro_Gain_Z 0.41 //Z axis Gyro gain
|
|
#define Gyro_Scaled_X(x) x*ToRad(Gyro_Gain_X) //Return the scaled ADC raw data of the gyro in radians for second
|
|
#define Gyro_Scaled_Y(x) x*ToRad(Gyro_Gain_Y) //Return the scaled ADC raw data of the gyro in radians for second
|
|
#define Gyro_Scaled_Z(x) x*ToRad(Gyro_Gain_Z) //Return the scaled ADC raw data of the gyro in radians for second
|
|
|
|
/*For debugging purposes*/
|
|
#define OUTPUTMODE 1 //If value = 1 will print the corrected data, 0 will print uncorrected data of the gyros (with drift), 2 Accel only data
|
|
|
|
int AN[6]; //array that store the 6 ADC channels
|
|
int AN_OFFSET[6]; //Array that store the Offset of the gyros and accelerometers
|
|
int gyro_temp;
|
|
|
|
|
|
float G_Dt=0.02; // Integration time for the gyros (DCM algorithm)
|
|
float Accel_Vector[3]= {0, 0, 0}; //Store the acceleration in a vector
|
|
float Accel_Vector_unfiltered[3]= {0, 0, 0}; //Store the acceleration in a vector
|
|
float Gyro_Vector[3]= {0, 0, 0};//Store the gyros rutn rate in a vector
|
|
float Omega_Vector[3]= {0, 0, 0}; //Corrected Gyro_Vector data
|
|
float Omega_P[3]= {0, 0, 0};//Omega Proportional correction
|
|
float Omega_I[3]= {0, 0, 0};//Omega Integrator
|
|
float Omega[3]= {0, 0, 0};
|
|
//float Accel_magnitude;
|
|
//float Accel_weight;
|
|
|
|
float errorRollPitch[3]= {0, 0, 0};
|
|
float errorYaw[3]= {0, 0, 0};
|
|
float errorCourse=0;
|
|
float COGX=0; //Course overground X axis
|
|
float COGY=1; //Course overground Y axis
|
|
|
|
float roll=0;
|
|
float pitch=0;
|
|
float yaw=0;
|
|
|
|
unsigned int counter=0;
|
|
|
|
float DCM_Matrix[3][3]= {
|
|
{
|
|
1,0,0 }
|
|
,{
|
|
0,1,0 }
|
|
,{
|
|
0,0,1 }
|
|
};
|
|
float Update_Matrix[3][3]={
|
|
{
|
|
0,1,2 }
|
|
,{
|
|
3,4,5 }
|
|
,{
|
|
6,7,8 }
|
|
}; //Gyros here
|
|
|
|
float Temporary_Matrix[3][3]={
|
|
{
|
|
0,0,0 }
|
|
,{
|
|
0,0,0 }
|
|
,{
|
|
0,0,0 }
|
|
};
|
|
|
|
// GPS variables
|
|
float speed_3d=0;
|
|
int GPS_ground_speed=0;
|
|
|
|
// Main timers
|
|
long timer=0;
|
|
long timer_old;
|
|
long GPS_timer;
|
|
long GPS_timer_old;
|
|
float GPS_Dt=0.2; // GPS Dt
|
|
|
|
// Attitude control variables
|
|
float command_rx_roll=0; // User commands
|
|
float command_rx_roll_old;
|
|
float command_rx_roll_diff;
|
|
float command_rx_pitch=0;
|
|
float command_rx_pitch_old;
|
|
float command_rx_pitch_diff;
|
|
float command_rx_yaw=0;
|
|
float command_rx_yaw_diff;
|
|
int control_roll; // PID control results
|
|
int control_pitch;
|
|
int control_yaw;
|
|
float K_aux;
|
|
|
|
// Attitude PID controls
|
|
float roll_I=0;
|
|
float roll_D;
|
|
float err_roll;
|
|
float pitch_I=0;
|
|
float pitch_D;
|
|
float err_pitch;
|
|
float yaw_I=0;
|
|
float yaw_D;
|
|
float err_yaw;
|
|
|
|
//Position control
|
|
long target_longitude;
|
|
long target_lattitude;
|
|
byte target_position;
|
|
float gps_err_roll;
|
|
float gps_err_roll_old;
|
|
float gps_roll_D;
|
|
float gps_roll_I=0;
|
|
float gps_err_pitch;
|
|
float gps_err_pitch_old;
|
|
float gps_pitch_D;
|
|
float gps_pitch_I=0;
|
|
float command_gps_roll;
|
|
float command_gps_pitch;
|
|
|
|
//Altitude control
|
|
int Initial_Throttle;
|
|
int target_sonar_altitude;
|
|
int err_altitude;
|
|
int err_altitude_old;
|
|
float command_altitude;
|
|
float altitude_I;
|
|
float altitude_D;
|
|
|
|
// Sonar variables
|
|
int Sonar_value=0;
|
|
#define SonarToCm(x) (x*1.26) // Sonar raw value to centimeters
|
|
int Sonar_Counter=0;
|
|
|
|
// AP_mode : 1=> Position hold 2=>Stabilization assist mode (normal mode)
|
|
byte AP_mode = 2;
|
|
|
|
// Mode LED timers and variables, used to blink LED_Green
|
|
byte gled_status = HIGH;
|
|
long gled_timer;
|
|
int gled_speed;
|
|
|
|
long t0;
|
|
int num_iter;
|
|
float aux_debug;
|
|
|
|
// Radio definitions
|
|
int roll_mid;
|
|
int pitch_mid;
|
|
int yaw_mid;
|
|
|
|
int Neutro_yaw;
|
|
int ch_roll;
|
|
int ch_pitch;
|
|
int ch_throttle;
|
|
int ch_yaw;
|
|
int ch_aux;
|
|
int ch_aux2;
|
|
|
|
int frontMotor;
|
|
int backMotor;
|
|
int leftMotor;
|
|
int rightMotor;
|
|
byte motorArmed = 0;
|
|
int minThrottle = 0;
|
|
|
|
// Serial communication
|
|
char queryType;
|
|
long tlmTimer = 0;
|
|
|
|
// Arming/Disarming
|
|
uint8_t Arming_counter=0;
|
|
uint8_t Disarming_counter=0;
|
|
|
|
|
|
|
|
/*****************************************************/
|
|
// APM Specific Memory variables
|
|
|
|
// Following variables stored in EEPROM
|
|
float KP_QUAD_ROLL;
|
|
float KD_QUAD_ROLL;
|
|
float KI_QUAD_ROLL;
|
|
float KP_QUAD_PITCH;
|
|
float KD_QUAD_PITCH;
|
|
float KI_QUAD_PITCH;
|
|
float KP_QUAD_YAW;
|
|
float KD_QUAD_YAW;
|
|
float KI_QUAD_YAW;
|
|
float STABLE_MODE_KP_RATE;
|
|
float KP_GPS_ROLL;
|
|
float KD_GPS_ROLL;
|
|
float KI_GPS_ROLL;
|
|
float KP_GPS_PITCH;
|
|
float KD_GPS_PITCH;
|
|
float KI_GPS_PITCH;
|
|
float GPS_MAX_ANGLE;
|
|
float KP_ALTITUDE;
|
|
float KD_ALTITUDE;
|
|
float KI_ALTITUDE;
|
|
int acc_offset_x;
|
|
int acc_offset_y;
|
|
int acc_offset_z;
|
|
int gyro_offset_roll;
|
|
int gyro_offset_pitch;
|
|
int gyro_offset_yaw;
|
|
float Kp_ROLLPITCH;
|
|
float Ki_ROLLPITCH;
|
|
float Kp_YAW;
|
|
float Ki_YAW;
|
|
float GEOG_CORRECTION_FACTOR;
|
|
int MAGNETOMETER;
|
|
float Kp_RateRoll;
|
|
float Ki_RateRoll;
|
|
float Kd_RateRoll;
|
|
float Kp_RatePitch;
|
|
float Ki_RatePitch;
|
|
float Kd_RatePitch;
|
|
float Kp_RateYaw;
|
|
float Ki_RateYaw;
|
|
float Kd_RateYaw;
|
|
float xmitFactor;
|
|
|
|
// EEPROM storage addresses
|
|
#define KP_QUAD_ROLL_ADR 0
|
|
#define KD_QUAD_ROLL_ADR 4
|
|
#define KI_QUAD_ROLL_ADR 8
|
|
#define KP_QUAD_PITCH_ADR 12
|
|
#define KD_QUAD_PITCH_ADR 16
|
|
#define KI_QUAD_PITCH_ADR 20
|
|
#define KP_QUAD_YAW_ADR 24
|
|
#define KD_QUAD_YAW_ADR 28
|
|
#define KI_QUAD_YAW_ADR 32
|
|
#define STABLE_MODE_KP_RATE_ADR 36
|
|
#define KP_GPS_ROLL_ADR 40
|
|
#define KD_GPS_ROLL_ADR 44
|
|
#define KI_GPS_ROLL_ADR 48
|
|
#define KP_GPS_PITCH_ADR 52
|
|
#define KD_GPS_PITCH_ADR 56
|
|
#define KI_GPS_PITCH_ADR 60
|
|
#define GPS_MAX_ANGLE_ADR 64
|
|
#define KP_ALTITUDE_ADR 68
|
|
#define KD_ALTITUDE_ADR 72
|
|
#define KI_ALTITUDE_ADR 76
|
|
#define acc_offset_x_ADR 80
|
|
#define acc_offset_y_ADR 84
|
|
#define acc_offset_z_ADR 88
|
|
#define gyro_offset_roll_ADR 92
|
|
#define gyro_offset_pitch_ADR 96
|
|
#define gyro_offset_yaw_ADR 100
|
|
#define Kp_ROLLPITCH_ADR 104
|
|
#define Ki_ROLLPITCH_ADR 108
|
|
#define Kp_YAW_ADR 112
|
|
#define Ki_YAW_ADR 116
|
|
#define GEOG_CORRECTION_FACTOR_ADR 120
|
|
#define MAGNETOMETER_ADR 124
|
|
#define XMITFACTOR_ADR 128
|
|
#define KP_RATEROLL_ADR 132
|
|
#define KI_RATEROLL_ADR 136
|
|
#define KD_RATEROLL_ADR 140
|
|
#define KP_RATEPITCH_ADR 144
|
|
#define KI_RATEPITCH_ADR 148
|
|
#define KD_RATEPITCH_ADR 152
|
|
#define KP_RATEYAW_ADR 156
|
|
#define KI_RATEYAW_ADR 160
|
|
#define KD_RATEYAW_ADR 164
|
|
#define CHROLL_MID 168
|
|
#define CHPITCH_MID 172
|
|
#define CHYAW_MID 176
|
|
|
|
// Utilities for writing and reading from the EEPROM
|
|
float readEEPROM(int address) {
|
|
union floatStore {
|
|
byte floatByte[4];
|
|
float floatVal;
|
|
} floatOut;
|
|
|
|
for (int i = 0; i < 4; i++)
|
|
floatOut.floatByte[i] = EEPROM.read(address + i);
|
|
return floatOut.floatVal;
|
|
}
|
|
|
|
void writeEEPROM(float value, int address) {
|
|
union floatStore {
|
|
byte floatByte[4];
|
|
float floatVal;
|
|
} floatIn;
|
|
|
|
floatIn.floatVal = value;
|
|
for (int i = 0; i < 4; i++)
|
|
EEPROM.write(address + i, floatIn.floatByte[i]);
|
|
}
|
|
|
|
void readUserConfig() {
|
|
KP_QUAD_ROLL = readEEPROM(KP_QUAD_ROLL_ADR);
|
|
KD_QUAD_ROLL = readEEPROM(KD_QUAD_ROLL_ADR);
|
|
KI_QUAD_ROLL = readEEPROM(KI_QUAD_ROLL_ADR);
|
|
KP_QUAD_PITCH = readEEPROM(KP_QUAD_PITCH_ADR);
|
|
KD_QUAD_PITCH = readEEPROM(KD_QUAD_PITCH_ADR);
|
|
KI_QUAD_PITCH = readEEPROM(KI_QUAD_PITCH_ADR);
|
|
KP_QUAD_YAW = readEEPROM(KP_QUAD_YAW_ADR);
|
|
KD_QUAD_YAW = readEEPROM(KD_QUAD_YAW_ADR);
|
|
KI_QUAD_YAW = readEEPROM(KI_QUAD_YAW_ADR);
|
|
STABLE_MODE_KP_RATE = readEEPROM(STABLE_MODE_KP_RATE_ADR);
|
|
KP_GPS_ROLL = readEEPROM(KP_GPS_ROLL_ADR);
|
|
KD_GPS_ROLL = readEEPROM(KD_GPS_ROLL_ADR);
|
|
KI_GPS_ROLL = readEEPROM(KI_GPS_ROLL_ADR);
|
|
KP_GPS_PITCH = readEEPROM(KP_GPS_PITCH_ADR);
|
|
KD_GPS_PITCH = readEEPROM(KD_GPS_PITCH_ADR);
|
|
KI_GPS_PITCH = readEEPROM(KI_GPS_PITCH_ADR);
|
|
GPS_MAX_ANGLE = readEEPROM(GPS_MAX_ANGLE_ADR);
|
|
KP_ALTITUDE = readEEPROM(KP_ALTITUDE_ADR);
|
|
KD_ALTITUDE = readEEPROM(KD_ALTITUDE_ADR);
|
|
KI_ALTITUDE = readEEPROM(KI_ALTITUDE_ADR);
|
|
acc_offset_x = readEEPROM(acc_offset_x_ADR);
|
|
acc_offset_y = readEEPROM(acc_offset_y_ADR);
|
|
acc_offset_z = readEEPROM(acc_offset_z_ADR);
|
|
gyro_offset_roll = readEEPROM(gyro_offset_roll_ADR);
|
|
gyro_offset_pitch = readEEPROM(gyro_offset_pitch_ADR);
|
|
gyro_offset_yaw = readEEPROM(gyro_offset_yaw_ADR);
|
|
Kp_ROLLPITCH = readEEPROM(Kp_ROLLPITCH_ADR);
|
|
Ki_ROLLPITCH = readEEPROM(Ki_ROLLPITCH_ADR);
|
|
Kp_YAW = readEEPROM(Kp_YAW_ADR);
|
|
Ki_YAW = readEEPROM(Ki_YAW_ADR);
|
|
GEOG_CORRECTION_FACTOR = readEEPROM(GEOG_CORRECTION_FACTOR_ADR);
|
|
MAGNETOMETER = readEEPROM(MAGNETOMETER_ADR);
|
|
Kp_RateRoll = readEEPROM(KP_RATEROLL_ADR);
|
|
Ki_RateRoll = readEEPROM(KI_RATEROLL_ADR);
|
|
Kd_RateRoll = readEEPROM(KD_RATEROLL_ADR);
|
|
Kp_RatePitch = readEEPROM(KP_RATEPITCH_ADR);
|
|
Ki_RatePitch = readEEPROM(KI_RATEPITCH_ADR);
|
|
Kd_RatePitch = readEEPROM(KD_RATEPITCH_ADR);
|
|
Kp_RateYaw = readEEPROM(KP_RATEYAW_ADR);
|
|
Ki_RateYaw = readEEPROM(KI_RATEYAW_ADR);
|
|
Kd_RateYaw = readEEPROM(KD_RATEYAW_ADR);
|
|
xmitFactor = readEEPROM(XMITFACTOR_ADR);
|
|
roll_mid = readEEPROM(CHROLL_MID);
|
|
pitch_mid = readEEPROM(CHPITCH_MID);
|
|
yaw_mid = readEEPROM(CHYAW_MID);
|
|
|
|
}
|