mirror of
https://github.com/ArduPilot/ardupilot
synced 2025-01-12 02:48:28 -04:00
Merge branch 'master' of https://code.google.com/p/ardupilot-mega
This commit is contained in:
commit
eba281e3de
@ -215,10 +215,12 @@ static const char* flight_mode_strings[] = {
|
||||
2 Elevator
|
||||
3 Throttle
|
||||
4 Rudder (if we have ailerons)
|
||||
5 Mode
|
||||
6 TBD
|
||||
7 TBD
|
||||
8 TBD
|
||||
5 Aux5
|
||||
6 Aux6
|
||||
7 Aux7
|
||||
8 Aux8/Mode
|
||||
Each Aux channel can be configured to have any of the available auxiliary functions assigned to it.
|
||||
See libraries/RC_Channel/RC_Channel_aux.h for more information
|
||||
*/
|
||||
|
||||
// Failsafe
|
||||
|
@ -37,7 +37,7 @@
|
||||
// PREPROCESSOR DIRECTIVES
|
||||
// ------------------------------------------------------------------------------------------------------------------------------------------------------------
|
||||
|
||||
#include "ppm_encoder.h"
|
||||
#include "..\Libraries\ppm_encoder.h"
|
||||
#include <util/delay.h>
|
||||
|
||||
|
||||
|
@ -1,2 +1,2 @@
|
||||
|
||||
libraries used by all code bases
|
||||
libraries used by all ArduPPM code bases
|
166
Tools/ArduPPM/WorkBasket/Jeti Duplex/Jetibox/JetiBox.cpp
Normal file
166
Tools/ArduPPM/WorkBasket/Jeti Duplex/Jetibox/JetiBox.cpp
Normal file
@ -0,0 +1,166 @@
|
||||
/*
|
||||
JetiBox.cpp, Version 1.0 beta
|
||||
July 2010, by Uwe Gartmann
|
||||
|
||||
Library acts as a Sensor when connected to a Jeti Receiver
|
||||
written for Arduino Mega / ArduPilot Mega
|
||||
|
||||
This library is free software; you can redistribute it and/or
|
||||
modify it under the terms of the GNU Lesser General Public
|
||||
License as published by the Free Software Foundation; either
|
||||
version 2.1 of the License, or (at your option) any later version.
|
||||
|
||||
This library 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
|
||||
Lesser General Public License for more details.
|
||||
|
||||
You should have received a copy of the GNU Lesser General Public
|
||||
License along with this library; if not, write to the Free Software
|
||||
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
||||
*/
|
||||
|
||||
#include <avr/io.h>
|
||||
#include <avr/interrupt.h>
|
||||
#include "wiring.h"
|
||||
#include "mySerial0.h"
|
||||
#include "JetiBox.h"
|
||||
|
||||
#define LCDLine1 1
|
||||
#define LCDLine2 17
|
||||
#define LCDMaxChar 32
|
||||
|
||||
//define Jeti-Box Display
|
||||
struct jeti_box {
|
||||
unsigned char lcd[32];
|
||||
volatile unsigned char lcdpos;
|
||||
volatile unsigned char keys;
|
||||
volatile unsigned char sendpos;
|
||||
};
|
||||
|
||||
// create jbox
|
||||
jeti_box jb = {{0},0,0,0};
|
||||
unsigned char dummy;
|
||||
unsigned char lastkey = 0;
|
||||
|
||||
ISR(USART3_RX_vect) //serial data available
|
||||
{
|
||||
// save response data to keys
|
||||
dummy = UDR3;
|
||||
if (dummy != 0xFF)
|
||||
{
|
||||
jb.keys = dummy;
|
||||
// disable this interrupt
|
||||
UCSR3B &= ~(1<<RXEN3);
|
||||
}
|
||||
}
|
||||
|
||||
ISR(USART3_UDRE_vect) //transmit buffer empty
|
||||
/*
|
||||
* jbox.sendpos = 0 -> interrupt is enabled (function start() ), send startbyte with 9.bit=0
|
||||
* jbox.sendpos = 1-32 -> send display data with 9.bit=1
|
||||
* jbox.sendpos = 33 -> send endbyte with 9.bit=0
|
||||
* jbox.sendpos = 34 -> reset sendpos=0 -> disable this interrupt
|
||||
*/
|
||||
{
|
||||
switch (jb.sendpos)
|
||||
{
|
||||
case 0:
|
||||
// send start byte with 9.bit=0
|
||||
UCSR3B &= ~(1<<TXB83); // clear bit 9
|
||||
UDR3 = 0xFE;
|
||||
jb.sendpos++;
|
||||
break;
|
||||
|
||||
case 33:
|
||||
// send end byte with 9.bit=0
|
||||
UCSR3B &= ~(1<<TXB83);
|
||||
UDR3 = 0xFF;
|
||||
jb.sendpos++;
|
||||
break;
|
||||
|
||||
case 34:
|
||||
// disable interrupt transmit buffer empty
|
||||
UCSR3B &= ~(1<<UDRIE3);
|
||||
// set sendpos -> 0
|
||||
jb.sendpos = 0;
|
||||
// enable receiver interrupt for reading key byte
|
||||
UCSR3B |= (1<<RXEN3);
|
||||
break;
|
||||
|
||||
default:
|
||||
// set 9.bit=1
|
||||
UCSR3B |= (1<<TXB83);
|
||||
// send byte from LCD buffer
|
||||
UDR3 = jb.lcd[jb.sendpos];
|
||||
// increment to next byte
|
||||
jb.sendpos++;
|
||||
}
|
||||
}
|
||||
|
||||
JetiBox::JetiBox()
|
||||
{
|
||||
// Class constructor
|
||||
}
|
||||
|
||||
// Public Methods --------------------------------------------------------------------
|
||||
void JetiBox::begin()
|
||||
{
|
||||
#ifndef F_CPU
|
||||
#define F_CPU 16000000
|
||||
#endif
|
||||
#define _UBRR3 (F_CPU/8/9600-1) //when U2X0 is not set use divider 16 instead of 8
|
||||
|
||||
// Set baudrate
|
||||
UCSR3A = (1<<U2X3); //U2X0 enabled!
|
||||
UBRR3H=(_UBRR3>>8); //high byte
|
||||
UBRR3L=_UBRR3; //low byte
|
||||
|
||||
// Set frame format: 9data, odd parity, 2stop bit
|
||||
UCSR3C = (0<<UMSEL30)|(3<<UPM30)|(1<<USBS3)|(3<<UCSZ30);
|
||||
|
||||
// Enable receiver and transmitter, set frame size
|
||||
UCSR3B = (1<<TXEN3)|(1<<RXCIE3)|(1<<UCSZ32);
|
||||
}
|
||||
|
||||
void JetiBox::refresh()
|
||||
{
|
||||
UCSR3B |= (1<<UDRIE3); // Enable Interrupt
|
||||
}
|
||||
|
||||
uint8_t JetiBox::keys(void)
|
||||
{
|
||||
unsigned char c = (jb.keys>>4) xor 0x0F;
|
||||
if (lastkey==c)
|
||||
{
|
||||
return 0;
|
||||
}else{
|
||||
return lastkey = c;
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
void JetiBox::write(uint8_t c)
|
||||
{
|
||||
jb.lcd[jb.lcdpos] = c;
|
||||
if (jb.lcdpos < LCDMaxChar)
|
||||
{
|
||||
jb.lcdpos++;
|
||||
}
|
||||
}
|
||||
|
||||
void JetiBox::line1()
|
||||
{
|
||||
jb.lcdpos = LCDLine1;
|
||||
}
|
||||
|
||||
void JetiBox::line2()
|
||||
{
|
||||
jb.lcdpos = LCDLine2;
|
||||
}
|
||||
|
||||
// Preinstantiate Objects //////////////////////////////////////////////////////
|
||||
JetiBox JBox;
|
||||
|
||||
|
||||
|
51
Tools/ArduPPM/WorkBasket/Jeti Duplex/Jetibox/JetiBox.h
Normal file
51
Tools/ArduPPM/WorkBasket/Jeti Duplex/Jetibox/JetiBox.h
Normal file
@ -0,0 +1,51 @@
|
||||
/*
|
||||
JetiBox.h, Version 1.0 beta
|
||||
July 2010, by Uwe Gartmann
|
||||
|
||||
Library acts as a Sensor when connected to a Jeti Receiver
|
||||
written for Arduino Mega / ArduPilot Mega
|
||||
|
||||
This library is free software; you can redistribute it and/or
|
||||
modify it under the terms of the GNU Lesser General Public
|
||||
License as published by the Free Software Foundation; either
|
||||
version 2.1 of the License, or (at your option) any later version.
|
||||
|
||||
This library 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
|
||||
Lesser General Public License for more details.
|
||||
|
||||
You should have received a copy of the GNU Lesser General Public
|
||||
License along with this library; if not, write to the Free Software
|
||||
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
||||
*/
|
||||
|
||||
#ifndef JetiBox_h
|
||||
#define JetiBox_h
|
||||
|
||||
#include <inttypes.h>
|
||||
#include "Print.h"
|
||||
|
||||
#define jbox_key_up 0b0010
|
||||
#define jbox_key_right 0b0001
|
||||
#define jbox_key_down 0b0100
|
||||
#define jbox_key_left 0b1000
|
||||
|
||||
struct jeti_box;
|
||||
|
||||
class JetiBox : public Print
|
||||
{
|
||||
public:
|
||||
JetiBox();
|
||||
void begin();
|
||||
void refresh();
|
||||
uint8_t keys(void);
|
||||
virtual void write(uint8_t);
|
||||
using Print::write; // pull in write(str) and write(buf, size) from Print
|
||||
void line1();
|
||||
void line2();
|
||||
};
|
||||
|
||||
extern JetiBox JBox;
|
||||
|
||||
#endif
|
@ -0,0 +1,925 @@
|
||||
//
|
||||
// Example and reference ArduPilot Mega configuration file
|
||||
// =======================================================
|
||||
//
|
||||
// This file contains documentation and examples for configuration options
|
||||
// supported by the ArduPilot Mega software.
|
||||
//
|
||||
// Most of these options are just that - optional. You should create
|
||||
// the APM_Config.h file and use this file as a reference for options
|
||||
// that you want to change. Don't copy this file directly; the options
|
||||
// described here and their default values may change over time.
|
||||
//
|
||||
// Each item is marked with a keyword describing when you should set it:
|
||||
//
|
||||
// REQUIRED
|
||||
// You must configure this in your APM_Config.h file. The
|
||||
// software will not compile if the option is not set.
|
||||
//
|
||||
// OPTIONAL
|
||||
// The option has a sensible default (which will be described
|
||||
// here), but you may wish to override it.
|
||||
//
|
||||
// EXPERIMENTAL
|
||||
// You should configure this option unless you are prepared
|
||||
// to deal with potential problems. It may relate to a feature
|
||||
// still in development, or which is not yet adequately tested.
|
||||
//
|
||||
// DEBUG
|
||||
// The option should only be set if you are debugging the
|
||||
// software, or if you are gathering information for a bug
|
||||
// report.
|
||||
//
|
||||
// NOTE:
|
||||
// Many of these settings are considered 'factory defaults', and the
|
||||
// live value is stored and managed in the ArduPilot Mega EEPROM.
|
||||
// Use the setup 'factoryreset' command after changing options in
|
||||
// your APM_Config.h file.
|
||||
//
|
||||
// Units
|
||||
// -----
|
||||
//
|
||||
// Unless indicated otherwise, numeric quantities use the following units:
|
||||
//
|
||||
// Measurement | Unit
|
||||
// ------------+-------------------------------------
|
||||
// angle | degrees
|
||||
// distance | metres
|
||||
// speed | metres per second
|
||||
// servo angle | microseconds
|
||||
// voltage | volts
|
||||
// times | seconds
|
||||
// throttle | percent
|
||||
//
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// HARDWARE CONFIGURATION AND CONNECTIONS
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// GPS_PROTOCOL REQUIRED
|
||||
//
|
||||
// GPS configuration, must be one of:
|
||||
//
|
||||
// GPS_PROTOCOL_NONE No GPS attached
|
||||
// GPS_PROTOCOL_IMU X-Plane interface or ArduPilot IMU.
|
||||
// GPS_PROTOCOL_MTK MediaTek-based GPS.
|
||||
// GPS_PROTOCOL_UBLOX UBLOX GPS
|
||||
// GPS_PROTOCOL_SIRF SiRF-based GPS in Binary mode. NOT TESTED
|
||||
// GPS_PROTOCOL_NMEA Standard NMEA GPS. NOT SUPPORTED (yet?)
|
||||
//
|
||||
#define GPS_PROTOCOL GPS_PROTOCOL_UBLOX
|
||||
//
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// AIRSPEED_SENSOR OPTIONAL
|
||||
// AIRSPEED_RATIO OPTIONAL
|
||||
//
|
||||
// Set AIRSPEED_SENSOR to ENABLED if you have an airspeed sensor attached.
|
||||
// Adjust AIRSPEED_RATIO in small increments to calibrate the airspeed
|
||||
// sensor relative to your GPS. The calculation and default value are optimized for speeds around 12 m/s
|
||||
//
|
||||
// The default assumes that an airspeed sensor is connected.
|
||||
//
|
||||
#define AIRSPEED_SENSOR ENABLED
|
||||
#define AIRSPEED_RATIO 1.9936
|
||||
//
|
||||
#define MAGNETOMETER 1
|
||||
#define USE_MAGNETOMETER ENABLED
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// GCS_PROTOCOL OPTIONAL
|
||||
// GCS_PORT OPTIONAL
|
||||
//
|
||||
// The GCS_PROTOCOL option determines which (if any) ground control station
|
||||
// protocol will be used. Must be one of:
|
||||
//
|
||||
// GCS_PROTOCOL_NONE No GCS output
|
||||
// GCS_PROTOCOL_STANDARD standard APM protocol
|
||||
// GCS_PROTOCOL_SPECIAL special test protocol (?)
|
||||
// GCS_PROTOCOL_LEGACY legacy ArduPilot protocol
|
||||
// GCS_PROTOCOL_XPLANE HIL simulation ground station
|
||||
// GCS_PROTOCOL_IMU ArdiPilot IMU output
|
||||
// GCS_PROTOCOL_JASON Jason's special secret GCS protocol
|
||||
// GCS_PROTOCOL_DEBUGTERMINAL In-flight debug console (experimental)
|
||||
//
|
||||
// The GCS_PORT option determines which serial port will be used by the
|
||||
// GCS protocol. The usual values are 0 for the console/USB port,
|
||||
// or 3 for the telemetry port on the oilpan. Note that some protocols
|
||||
// will ignore this value and always use the console port.
|
||||
//
|
||||
// The default GCS protocol is the standard ArduPilot Mega protocol.
|
||||
//
|
||||
// The default serial port is the telemetry port for GCS_PROTOCOL_STANDARD
|
||||
// and GCS_PROTOCOL_LEGACY. For all other protocols, the default serial
|
||||
// port is the FTDI/console port. GCS_PORT normally should not be set
|
||||
// in your configuration.
|
||||
//
|
||||
#define GCS_PROTOCOL GCS_PROTOCOL_XDIY
|
||||
#define GCS_PORT 0
|
||||
//
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// Serial port speeds.
|
||||
//
|
||||
// SERIAL0_BAUD OPTIONAL
|
||||
//
|
||||
// Baudrate for the console port. Default is 38400bps.
|
||||
//
|
||||
// SERIAL3_BAUD OPTIONAL
|
||||
//
|
||||
// Baudrate for the telemetry port. Default is 115200bps.
|
||||
//
|
||||
//#define SERIAL0_BAUD 38400
|
||||
//#define SERIAL3_BAUD 115200
|
||||
//
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// Battery monitoring OPTIONAL
|
||||
//
|
||||
// See the manual for details on selecting divider resistors for battery
|
||||
// monitoring via the oilpan.
|
||||
//
|
||||
// BATTERY_EVENT OPTIONAL
|
||||
//
|
||||
// Set BATTERY_EVENT to ENABLED to enable battery monitoring. The default is
|
||||
// DISABLED.
|
||||
//
|
||||
// BATTERY_TYPE OPTIONAL if BATTERY_EVENT is set
|
||||
//
|
||||
// Set to 0 for 3s LiPo, 1 for 4s LiPo. The default is 0, selecting a 3s
|
||||
// battery.
|
||||
//
|
||||
// LOW_VOLTAGE OPTIONAL if BATTERY_EVENT is set.
|
||||
//
|
||||
// Normally derived from BATTERY_TYPE, the automatic value can be overridden
|
||||
// here. Value in volts at which ArduPilot Mega should consider the
|
||||
// battery to be "low".
|
||||
//
|
||||
// VOLT_DIV_RATIO OPTIONAL
|
||||
//
|
||||
// See the manual for details. The default value corresponds to the resistors
|
||||
// recommended by the manual.
|
||||
//
|
||||
//#define BATTERY_EVENT DISABLED
|
||||
//#define BATTERY_TYPE 0
|
||||
//#define LOW_VOLTAGE 11.4
|
||||
//#define VOLT_DIV_RATIO 3.0
|
||||
//
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// INPUT_VOLTAGE OPTIONAL
|
||||
//
|
||||
// In order to have accurate pressure and battery voltage readings, this
|
||||
// value should be set to the voltage measured on the 5V rail on the oilpan.
|
||||
//
|
||||
// See the manual for more details. The default value should be close.
|
||||
//
|
||||
#define INPUT_VOLTAGE 5.0
|
||||
//
|
||||
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// RADIO CONFIGURATION
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// FLIGHT_MODE OPTIONAL
|
||||
// FLIGHT_MODE_CHANNEL OPTIONAL
|
||||
//
|
||||
// Flight modes assigned to the control channel, and the input channel that
|
||||
// is read for the control mode.
|
||||
//
|
||||
// Use a servo tester, or the ArduPilotMega_demo test program to check your
|
||||
// switch settings.
|
||||
//
|
||||
// ATTENTION: Some ArduPilot Mega boards have radio channels marked 0-7, and
|
||||
// others have them marked the standard 1-8. The FLIGHT_MODE_CHANNEL option
|
||||
// uses channel numbers 1-8 (and defaults to 8).
|
||||
//
|
||||
// If you only have a three-position switch or just want three modes, set your
|
||||
// switch to produce 1165, 1425, and 1815 microseconds and configure
|
||||
// FLIGHT_MODE 1 & 2, 3 & 4 and 5 & 6 to be the same. This is the default.
|
||||
//
|
||||
// If you have FLIGHT_MODE_CHANNEL set to 8 (the default) and your control
|
||||
// channel connected to input channel 8, the hardware failsafe mode will
|
||||
// activate for any control input over 1750ms.
|
||||
//
|
||||
// For more modes (up to six), set your switch(es) to produce any of 1165,
|
||||
// 1295, 1425, 1555, 1685, and 1815 microseconds.
|
||||
//
|
||||
// Flight mode | Switch Setting (ms)
|
||||
// ------------+---------------------
|
||||
// 1 | 1165
|
||||
// 2 | 1295
|
||||
// 3 | 1425
|
||||
// 4 | 1555
|
||||
// 5 | 1685
|
||||
// 6 | 1815 (FAILSAFE if using channel 8)
|
||||
//
|
||||
// The following standard flight modes are available:
|
||||
//
|
||||
// Name | Description
|
||||
// -----------------+--------------------------------------------
|
||||
// |
|
||||
// MANUAL | Full manual control via the hardware multiplexer.
|
||||
// |
|
||||
// STABILIZE | Tries to maintain level flight, but can be overridden with radio control inputs.
|
||||
// |
|
||||
// FLY_BY_WIRE_A | Autopilot style control via user input, with manual throttle.
|
||||
// |
|
||||
// FLY_BY_WIRE_B | Autopilot style control via user input, aispeed controlled with throttle.
|
||||
// |
|
||||
// RTL | Returns to the Home location and then LOITERs at a safe altitude.
|
||||
// |
|
||||
// AUTO | Autonomous flight based on programmed waypoints. Use the WaypointWriter
|
||||
// | application or your Ground Control System to edit and upload
|
||||
// | waypoints and other commands.
|
||||
// |
|
||||
//
|
||||
//
|
||||
// The following non-standard modes are EXPERIMENTAL:
|
||||
//
|
||||
// Name | Description
|
||||
// -----------------+--------------------------------------------
|
||||
// |
|
||||
// LOITER | Flies in a circle around the current location.
|
||||
// |
|
||||
// CIRCLE | Flies in a stabilized 'dumb' circle.
|
||||
// |
|
||||
//
|
||||
//
|
||||
// If you are using channel 8 for mode switching then FLIGHT_MODE_5 and
|
||||
// FLIGHT_MODE_6 should be MANUAL.
|
||||
//
|
||||
//
|
||||
#define FLIGHT_MODE_CHANNEL 8
|
||||
//
|
||||
//#define FLIGHT_MODE_1 RTL
|
||||
//#define FLIGHT_MODE_2 RTL
|
||||
//#define FLIGHT_MODE_3 FLY_BY_WIRE_A
|
||||
//#define FLIGHT_MODE_4 FLY_BY_WIRE_A
|
||||
//#define FLIGHT_MODE_5 MANUAL
|
||||
//#define FLIGHT_MODE_6 MANUAL
|
||||
|
||||
// Define the pulse width when to switch to next higher FLIGHT_MODE
|
||||
#define FLIGHT_MODE_1_BOUNDARY 1125
|
||||
#define FLIGHT_MODE_2_BOUNDARY 1335
|
||||
#define FLIGHT_MODE_3_BOUNDARY 1550
|
||||
#define FLIGHT_MODE_4_BOUNDARY 1690
|
||||
#define FLIGHT_MODE_5_BOUNDARY 1880
|
||||
|
||||
|
||||
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// THROTTLE_FAILSAFE OPTIONAL
|
||||
// THROTTLE_FS_VALUE OPTIONAL
|
||||
// THROTTLE_FAILSAFE_ACTION OPTIONAL
|
||||
//
|
||||
// The throttle failsafe allows you to configure a software failsafe activated
|
||||
// by a setting on the throttle input channel (channel 3).
|
||||
//
|
||||
// This can be used to achieve a failsafe override on loss of radio control
|
||||
// without having to sacrifice one of your FLIGHT_MODE settings, as the
|
||||
// throttle failsafe overrides the switch-selected mode.
|
||||
//
|
||||
// Throttle failsafe is enabled by setting THROTTLE_FAILSAFE to 1. The default
|
||||
// is for it to be disabled.
|
||||
//
|
||||
// If the throttle failsafe is enabled, THROTTLE_FS_VALUE sets the channel value
|
||||
// below which the failsafe engages. The default is 975ms, which is a very low
|
||||
// throttle setting. Most transmitters will let you trim the manual throttle
|
||||
// position up so that you cannot engage the failsafe with a regular stick movement.
|
||||
//
|
||||
// Configure your receiver's failsafe setting for the throttle channel to the
|
||||
// absolute minimum, and use the ArduPilotMega_demo program to check that
|
||||
// you cannot reach that value with the throttle control. Leave a margin of
|
||||
// at least 50 microseconds between the lowest throttle setting and
|
||||
// THROTTLE_FS_VALUE.
|
||||
//
|
||||
// The FAILSAFE_ACTION setting determines what APM will do when throttle failsafe
|
||||
// mode is entered while flying in AUTO mode. This is important in order to avoid
|
||||
// accidental failsafe behaviour when flying waypoints that take the aircraft
|
||||
// temporarily out of radio range.
|
||||
//
|
||||
// If FAILSAFE_ACTION is 1, when failsafe is entered in AUTO or LOITER modes,
|
||||
// the aircraft will head for home in RTL mode. If the throttle channel moves
|
||||
// back up, it will return to AUTO or LOITER mode.
|
||||
//
|
||||
// The default behaviour is to ignore throttle failsafe in AUTO and LOITER modes.
|
||||
//
|
||||
//#define THROTTLE_FAILSAFE DISABLED
|
||||
//#define THROTTLE_FS_VALUE 975
|
||||
//#define THROTTLE_FAILSAFE_ACTION 2
|
||||
//
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// AUTO_TRIM OPTIONAL
|
||||
//
|
||||
// ArduPilot Mega can update its trim settings by looking at the
|
||||
// radio inputs when switching out of MANUAL mode. This allows you to
|
||||
// manually trim your aircraft before switching to an assisted mode, but it
|
||||
// also means that you should avoid switching out of MANUAL while you have
|
||||
// any control stick deflection.
|
||||
//
|
||||
// The default is to enable AUTO_TRIM.
|
||||
//
|
||||
#define AUTO_TRIM ENABLED
|
||||
//
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// THROTTLE_REVERSE OPTIONAL
|
||||
//
|
||||
// A few speed controls require the throttle servo signal be reversed. Setting
|
||||
// this to ENABLED will reverse the throttle output signal. Ensure that your
|
||||
// throttle needs to be reversed by using the hardware failsafe and the
|
||||
// ArduPilotMega_demo program before setting this option.
|
||||
//
|
||||
// The default is to not reverse the signal.
|
||||
//
|
||||
//#define THROTTLE_REVERSE DISABLED
|
||||
//
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// ENABLE_STICK_MIXING OPTIONAL
|
||||
//
|
||||
// If this option is set to ENABLED, manual control inputs are are respected
|
||||
// while in the autopilot modes (AUTO, RTL, LOITER, CIRCLE etc.)
|
||||
//
|
||||
// The default is to enable stick mixing, allowing the pilot to take
|
||||
// emergency action without switching modes.
|
||||
//
|
||||
#define ENABLE_STICK_MIXING ENABLED
|
||||
//
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// THROTTLE_OUT DEBUG
|
||||
//
|
||||
// When debugging, it can be useful to disable the throttle output. Set
|
||||
// this option to DISABLED to disable throttle output signals.
|
||||
//
|
||||
// The default is to not disable throttle output.
|
||||
//
|
||||
#define THROTTLE_OUT DISABLED
|
||||
//
|
||||
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// STARTUP BEHAVIOUR
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// GROUND_START_DELAY OPTIONAL
|
||||
//
|
||||
// If configured, inserts a delay between power-up and the beginning of IMU
|
||||
// calibration during a ground start.
|
||||
//
|
||||
// Use this setting to give you time to position the aircraft horizontally
|
||||
// for the IMU calibration.
|
||||
//
|
||||
// The default is to begin IMU calibration immediately at startup.
|
||||
//
|
||||
//#define GROUND_START_DELAY 0
|
||||
//
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// ENABLE_AIR_START OPTIONAL
|
||||
//
|
||||
// If air start is disabled then you will get a ground start (including IMU
|
||||
// calibration) every time the AP is powered up. This means that if you get
|
||||
// a power glitch or reboot for some reason in the air, you will probably
|
||||
// crash, but it prevents a lot of problems on the ground like unintentional
|
||||
// motor start-ups, etc.
|
||||
//
|
||||
// If air start is enabled then you will get an air start at power up and a
|
||||
// ground start will be performed if the speed is near zero when we get gps
|
||||
// lock.
|
||||
//
|
||||
// The default is to disable air start.
|
||||
//
|
||||
#define ENABLE_AIR_START 0
|
||||
//
|
||||
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// FLIGHT AND NAVIGATION CONTROL
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// Altitude measurement and control.
|
||||
//
|
||||
// AOA OPTIONAL
|
||||
//
|
||||
// The angle in 100ths of a degree that the nose of the aircraft should be
|
||||
// raised from horizontal in level flight. The default is 1 degree.
|
||||
//
|
||||
//#define AOA 100 // note, 100ths of a degree
|
||||
//
|
||||
// ALT_EST_GAIN OPTIONAL
|
||||
//
|
||||
// The gain of the altitude estimation function; a lower number results
|
||||
// in slower error correction and smoother output. The default is a
|
||||
// reasonable starting point.
|
||||
//
|
||||
//#define ALT_EST_GAIN 0.01
|
||||
//
|
||||
// ALTITUDE_MIX OPTIONAL
|
||||
//
|
||||
// Configures the blend between GPS and pressure altitude.
|
||||
// 0 = GPS altitude, 1 = Press alt, 0.5 = half and half, etc.
|
||||
//
|
||||
// The default is to use only GPS altitude.
|
||||
//
|
||||
//#define ALTITUDE_MIX 0
|
||||
//
|
||||
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// ENABLE_HIL OPTIONAL
|
||||
//
|
||||
// This will output a binary control string to for use in HIL sims
|
||||
// such as Xplane 9 or FlightGear.
|
||||
//
|
||||
//#define ENABLE_HIL ENABLED
|
||||
//
|
||||
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// AIRSPEED_CRUISE OPTIONAL
|
||||
//
|
||||
// The speed in metres per second to maintain during cruise. The default
|
||||
// is 10m/s, which is a conservative value suitable for relatively small,
|
||||
// light aircraft.
|
||||
//
|
||||
#define AIRSPEED_CRUISE 15
|
||||
//
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// FLY_BY_WIRE_B airspeed control
|
||||
//
|
||||
// AIRSPEED_FBW_MIN OPTIONAL
|
||||
// AIRSPEED_FBW_MAX OPTIONAL
|
||||
//
|
||||
// Airspeed corresponding to minimum and maximum throttle in Fly By Wire B mode.
|
||||
// The defaults are 6 and 30 metres per second.
|
||||
//
|
||||
// THROTTLE_ALT_P OPTIONAL
|
||||
// THROTTLE_ALT_I OPTIONAL
|
||||
// THROTTLE_ALT_D OPTIONAL
|
||||
//
|
||||
// P, I and D terms for the throttle control loop. Defaults are 0.5, 0, 0.
|
||||
//
|
||||
// THROTTLE_ALT_INT_MAX OPTIONAL
|
||||
//
|
||||
// Maximum throttle input due to the integral. Limits the throttle input
|
||||
// due to persistent inability to maintain the commanded speed. Helps
|
||||
// prevent the throttle from staying wide open when the control is reduced
|
||||
// after a period at maxium speed.
|
||||
// Default is 20 (20%).
|
||||
//
|
||||
//#define AIRSPEED_FBW_MIN 6
|
||||
//#define AIRSPEED_FBW_MAX 30
|
||||
//#define THROTTLE_ALT_P 0.32
|
||||
//#define THROTTLE_ALT_I 0.04
|
||||
//#define THROTTLE_ALT_D 0.0
|
||||
//#define THROTTLE_ALT_INT_MAX 20
|
||||
//
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// Throttle control
|
||||
//
|
||||
// THROTTLE_MIN OPTIONAL
|
||||
//
|
||||
// The minimum throttle setting to which the autopilot will reduce the
|
||||
// throttle while descending. The default is zero, which is
|
||||
// suitable for aircraft with a steady power-off glide. Increase this
|
||||
// value if your aircraft needs throttle to maintain a stable descent in
|
||||
// level flight.
|
||||
//
|
||||
// THROTTLE_CRUISE OPTIONAL
|
||||
//
|
||||
// The approximate throttle setting to achieve AIRSPEED_CRUISE in level flight.
|
||||
// The default is 45%, which is reasonable for a modestly powered aircraft.
|
||||
//
|
||||
// THROTTLE_MAX OPTIONAL
|
||||
//
|
||||
// The maximum throttle setting the autopilot will apply. The default is 75%.
|
||||
// Reduce this value if your aicraft is overpowered, or has complex flight
|
||||
// characteristics at high throttle settings.
|
||||
//
|
||||
//#define THROTTLE_MIN 0
|
||||
//#define THROTTLE_CRUISE 45
|
||||
//#define THROTTLE_MAX 75
|
||||
//
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// Autopilot control limits
|
||||
//
|
||||
// HEAD_MAX OPTIONAL
|
||||
//
|
||||
// The maximum commanded bank angle in either direction.
|
||||
// The default is 45 degrees. Decrease this value if your aircraft is not
|
||||
// stable or has difficulty maintaining altitude in a steep bank.
|
||||
//
|
||||
// PITCH_MAX OPTIONAL
|
||||
//
|
||||
// The maximum commanded pitch up angle.
|
||||
// The default is 15 degrees. Care should be taken not to set this value too
|
||||
// large, as the aircraft may stall.
|
||||
//
|
||||
// PITCH_MIN
|
||||
//
|
||||
// The maximum commanded pitch down angle. Note that this value must be
|
||||
// negative. The default is -25 degrees. Care should be taken not to set
|
||||
// this value too large as it may result in overspeeding the aircraft.
|
||||
//
|
||||
//#define HEAD_MAX 45
|
||||
//#define PITCH_MAX 15
|
||||
//#define PITCH_MIN -25
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// Attitude control gains
|
||||
//
|
||||
// Tuning values for the attitude control PID loops.
|
||||
//
|
||||
// The P term is the primary tuning value. This determines how the control
|
||||
// deflection varies in proportion to the required correction.
|
||||
//
|
||||
// The I term is used to help control surfaces settle. This value should
|
||||
// normally be kept low.
|
||||
//
|
||||
// The D term is used to control overshoot. Avoid using or adjusting this
|
||||
// term if you are not familiar with tuning PID loops. It should normally
|
||||
// be zero for most aircraft.
|
||||
//
|
||||
// Note: When tuning these values, start with changes of no more than 25% at
|
||||
// a time.
|
||||
//
|
||||
// SERVO_ROLL_P OPTIONAL
|
||||
// SERVO_ROLL_I OPTIONAL
|
||||
// SERVO_ROLL_D OPTIONAL
|
||||
//
|
||||
// P, I and D terms for roll control. Defaults are 0.4, 0, 0.
|
||||
//
|
||||
// SERVO_ROLL_INT_MAX OPTIONAL
|
||||
//
|
||||
// Maximum control offset due to the integral. This prevents the control
|
||||
// output from being overdriven due to a persistent offset (e.g. crosstracking).
|
||||
// Default is 5 degrees.
|
||||
//
|
||||
// ROLL_SLEW_LIMIT EXPERIMENTAL
|
||||
//
|
||||
// Limits the slew rate of the roll control in degrees per second. If zero,
|
||||
// slew rate is not limited. Default is to not limit the roll control slew rate.
|
||||
// (This feature is currently not implemented.)
|
||||
//
|
||||
// SERVO_PITCH_P OPTIONAL
|
||||
// SERVO_PITCH_I OPTIONAL
|
||||
// SERVO_PITCH_D OPTIONAL
|
||||
//
|
||||
// P, I and D terms for the pitch control. Defaults are 0.6, 0, 0.
|
||||
//
|
||||
// SERVO_PITCH_INT_MAX OPTIONAL
|
||||
//
|
||||
// Maximum control offset due to the integral. This prevents the control
|
||||
// output from being overdriven due to a persistent offset (e.g. native flight
|
||||
// AoA). If you find this value is insufficient, consider adjusting the AOA
|
||||
// parameter.
|
||||
// Default is 5 degrees.
|
||||
//
|
||||
// PITCH_COMP OPTIONAL
|
||||
//
|
||||
// Adds pitch input to compensate for the loss of lift due to roll control.
|
||||
// Default is 0.20 (20% of roll control also applied to pitch control).
|
||||
//
|
||||
// SERVO_YAW_P OPTIONAL
|
||||
// SERVO_YAW_I OPTIONAL
|
||||
// SERVO_YAW_D OPTIONAL
|
||||
//
|
||||
// P, I and D terms for the YAW control. Defaults are 0.5, 0, 0.
|
||||
//
|
||||
// SERVO_YAW_INT_MAX OPTIONAL
|
||||
//
|
||||
// Maximum control offset due to the integral. This prevents the control
|
||||
// output from being overdriven due to a persistent offset (e.g. crosstracking).
|
||||
// Default is 5 degrees.
|
||||
//
|
||||
// RUDDER_MIX OPTIONAL
|
||||
//
|
||||
// Roll to yaw mixing. This allows for co-ordinated turns.
|
||||
// Default is 0.50 (50% of roll control also applied to yaw control.)
|
||||
//
|
||||
//#define SERVO_ROLL_P 0.4
|
||||
//#define SERVO_ROLL_I 0.0
|
||||
//#define SERVO_ROLL_D 0.0
|
||||
//#define SERVO_ROLL_INT_MAX 5
|
||||
//#define ROLL_SLEW_LIMIT 0
|
||||
//#define SERVO_PITCH_P 0.6
|
||||
//#define SERVO_PITCH_I 0.0
|
||||
//#define SERVO_PITCH_D 0.0
|
||||
//#define SERVO_PITCH_INT_MAX 5
|
||||
//#define PITCH_COMP 0.2
|
||||
//#define SERVO_YAW_P 0.5
|
||||
//#define SERVO_YAW_I 0.0
|
||||
//#define SERVO_YAW_D 0.0
|
||||
//#define SERVO_YAW_INT_MAX 5
|
||||
//#define RUDDER_MIX 0.5
|
||||
//
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// Navigation control gains
|
||||
//
|
||||
// Tuning values for the navigation control PID loops.
|
||||
//
|
||||
// The P term is the primary tuning value. This determines how the control
|
||||
// deflection varies in proportion to the required correction.
|
||||
//
|
||||
// The I term is used to control drift.
|
||||
//
|
||||
// The D term is used to control overshoot. Avoid adjusting this term if
|
||||
// you are not familiar with tuning PID loops.
|
||||
//
|
||||
// Note: When tuning these values, start with changes of no more than 25% at
|
||||
// a time.
|
||||
//
|
||||
// NAV_ROLL_P OPTIONAL
|
||||
// NAV_ROLL_I OPTIONAL
|
||||
// NAV_ROLL_D OPTIONAL
|
||||
//
|
||||
// P, I and D terms for navigation control over roll, normally used for
|
||||
// controlling the aircraft's course. The P term controls how aggressively
|
||||
// the aircraft will bank to change or hold course.
|
||||
// Defaults are 0.7, 0.01, 0.02.
|
||||
//
|
||||
// NAV_ROLL_INT_MAX OPTIONAL
|
||||
//
|
||||
// Maximum control offset due to the integral. This prevents the control
|
||||
// output from being overdriven due to a persistent offset (e.g. crosstracking).
|
||||
// Default is 5 degrees.
|
||||
//
|
||||
// NAV_PITCH_ASP_P OPTIONAL
|
||||
// NAV_PITCH_ASP_I OPTIONAL
|
||||
// NAV_PITCH_ASP_D OPTIONAL
|
||||
//
|
||||
// P, I and D terms for pitch adjustments made to maintain airspeed.
|
||||
// Defaults are 0.65, 0, 0.
|
||||
//
|
||||
// NAV_PITCH_ASP_INT_MAX OPTIONAL
|
||||
//
|
||||
// Maximum pitch offset due to the integral. This limits the control
|
||||
// output from being overdriven due to a persistent offset (eg. inability
|
||||
// to maintain the programmed airspeed).
|
||||
// Default is 5 degrees.
|
||||
//
|
||||
// NAV_PITCH_ALT_P OPTIONAL
|
||||
// NAV_PITCH_ALT_I OPTIONAL
|
||||
// NAV_PITCH_ALT_D OPTIONAL
|
||||
//
|
||||
// P, I and D terms for pitch adjustments made to maintain altitude.
|
||||
// Defaults are 0.65, 0, 0.
|
||||
//
|
||||
// NAV_PITCH_ALT_INT_MAX OPTIONAL
|
||||
//
|
||||
// Maximum pitch offset due to the integral. This limits the control
|
||||
// output from being overdriven due to a persistent offset (eg. inability
|
||||
// to maintain the programmed altitude).
|
||||
// Default is 5 degrees.
|
||||
//
|
||||
//#define NAV_ROLL_P 0.7
|
||||
//#define NAV_ROLL_I 0.01
|
||||
//#define NAV_ROLL_D 0.02
|
||||
//#define NAV_ROLL_INT_MAX 5
|
||||
//#define NAV_PITCH_ASP_P 0.65
|
||||
//#define NAV_PITCH_ASP_I 0.0
|
||||
//#define NAV_PITCH_ASP_D 0.0
|
||||
//#define NAV_PITCH_ASP_INT_MAX 5
|
||||
//#define NAV_PITCH_ALT_P 0.65
|
||||
//#define NAV_PITCH_ALT_I 0.0
|
||||
//#define NAV_PITCH_ALT_D 0.0
|
||||
//#define NAV_PITCH_ALT_INT_MAX 5
|
||||
//
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// Energy/Altitude control gains
|
||||
//
|
||||
// The Energy/altitude control system uses throttle input to control aircraft
|
||||
// altitude.
|
||||
//
|
||||
// The P term is the primary tuning value. This determines how the throttle
|
||||
// setting varies in proportion to the required correction.
|
||||
//
|
||||
// The I term is used to compensate for small offsets.
|
||||
//
|
||||
// The D term is used to control overshoot. Avoid adjusting this term if
|
||||
// you are not familiar with tuning PID loops.
|
||||
//
|
||||
// THROTTLE_TE_P OPTIONAL
|
||||
// THROTTLE_TE_I OPTIONAL
|
||||
// THROTTLE_TE_D OPTIONAL
|
||||
//
|
||||
// P, I and D terms for throttle adjustments made to control altitude.
|
||||
// Defaults are 0.5, 0, 0.
|
||||
//
|
||||
// THROTTLE_TE_INT_MAX OPTIONAL
|
||||
//
|
||||
// Maximum throttle input due to the integral term. This limits the
|
||||
// throttle from being overdriven due to a persistent offset (e.g.
|
||||
// inability to maintain the programmed altitude).
|
||||
// Default is 20%.
|
||||
//
|
||||
// THROTTLE_SLEW_LIMIT OPTIONAL
|
||||
//
|
||||
// Limits the slew rate of the throttle, in percent per second. Helps
|
||||
// avoid sudden throttle changes, which can destabilise the aircraft.
|
||||
// A setting of zero disables the feature.
|
||||
// Default is zero (disabled).
|
||||
//
|
||||
// P_TO_T OPTIONAL
|
||||
//
|
||||
// Pitch to throttle feed-forward gain. Used when AIRSPEED_SENSOR
|
||||
// is DISABLED. Default is 2.5.
|
||||
//
|
||||
//#define THROTTLE_TE_P 0.50
|
||||
//#define THROTTLE_TE_I 0.0
|
||||
//#define THROTTLE_TE_D 0.0
|
||||
//#define THROTTLE_TE_INT_MAX 20
|
||||
//#define THROTTLE_SLEW_LIMIT 0
|
||||
//#define P_TO_T 2.5
|
||||
//
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// Crosstrack compensation
|
||||
//
|
||||
// XTRACK_GAIN OPTIONAL
|
||||
//
|
||||
// Crosstrack compensation in degrees per metre off track.
|
||||
// Default value is 0.01 degrees per metre. Values lower than 0.001 will
|
||||
// disable crosstrack compensation.
|
||||
//
|
||||
// XTRACK_ENTRY_ANGLE OPTIONAL
|
||||
//
|
||||
// Maximum angle used to correct for track following.
|
||||
// Default value is 30 degrees.
|
||||
//
|
||||
//#define XTRACK_GAIN 0.01
|
||||
//#define XTRACK_ENTRY_ANGLE 30
|
||||
//
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// DEBUGGING
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// Subsystem test and debug.
|
||||
//
|
||||
// DEBUG_SUBSYSTEM DEBUG
|
||||
//
|
||||
// Selects a subsystem debug mode. Default is 0.
|
||||
//
|
||||
// 0 = no debug
|
||||
// 1 = Debug the Radio input
|
||||
// 2 = Radio Setup / Servo output
|
||||
// 4 = Debug the GPS input
|
||||
// 5 = Debug the GPS input - RAW OUTPUT
|
||||
// 6 = Debug the IMU
|
||||
// 7 = Debug the Control Switch
|
||||
// 8 = Debug the Servo DIP switches
|
||||
// 9 = Debug the Relay out
|
||||
// 10 = Debug the Magnetometer
|
||||
// 11 = Debug the ABS pressure sensor
|
||||
// 12 = Debug the stored waypoints
|
||||
// 13 = Debug the Throttle
|
||||
// 14 = Debug the Radio Min Max
|
||||
// 15 = Debug the EEPROM - Hex Dump
|
||||
// 16 = XBee X-CTU Range and RSSI Test
|
||||
// 17 = Debug IMU - raw gyro and accel outputs
|
||||
// 20 = Debug Analog Sensors
|
||||
//
|
||||
//
|
||||
//#define DEBUG_SUBSYSTEM 0
|
||||
//
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// DEBUG_LEVEL DEBUG
|
||||
//
|
||||
// Selects the lowest level of debug messages passed to the telemetry system.
|
||||
// Default is SEVERITY_LOW. May be one of:
|
||||
//
|
||||
// SEVERITY_LOW
|
||||
// SEVERITY_MEDIUM
|
||||
// SEVERITY_HIGH
|
||||
// SEVERITY_CRITICAL
|
||||
//
|
||||
//#define DEBUG_LEVEL SEVERITY_LOW
|
||||
//
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// Dataflash logging control
|
||||
//
|
||||
// Each of these logging options may be set to ENABLED to enable, or DISABLED
|
||||
// to disable the logging of the respective data.
|
||||
//
|
||||
// LOG_ATTITUDE_FAST DEBUG
|
||||
//
|
||||
// Logs basic attitude info to the dataflash at 50Hz (uses more space).
|
||||
// Defaults to DISABLED.
|
||||
//
|
||||
// LOG_ATTITUDE_MED OPTIONAL
|
||||
//
|
||||
// Logs basic attitude info to the dataflash at 10Hz (uses less space than
|
||||
// LOG_ATTITUDE_FAST). Defaults to ENABLED.
|
||||
//
|
||||
// LOG_GPS OPTIONAL
|
||||
//
|
||||
// Logs GPS info to the dataflash at 10Hz. Defaults to ENABLED.
|
||||
//
|
||||
// LOG_PM OPTIONAL
|
||||
//
|
||||
// Logs IMU performance monitoring info every 20 seconds.
|
||||
// Defaults to DISABLED.
|
||||
//
|
||||
// LOG_CTUN OPTIONAL
|
||||
//
|
||||
// Logs control loop tuning info at 10 Hz. This information is useful for tuning
|
||||
// servo control loop gain values. Defaults to DISABLED.
|
||||
//
|
||||
// LOG_NTUN OPTIONAL
|
||||
//
|
||||
// Logs navigation tuning info at 10 Hz. This information is useful for tuning
|
||||
// navigation control loop gain values. Defaults to DISABLED.
|
||||
//
|
||||
// LOG_ MODE OPTIONAL
|
||||
//
|
||||
// Logs changes to the flight mode upon occurrence. Defaults to ENABLED.
|
||||
//
|
||||
// LOG_RAW DEBUG
|
||||
//
|
||||
// Logs raw accelerometer and gyro data at 50 Hz (uses more space).
|
||||
// Defaults to DISABLED.
|
||||
//
|
||||
// LOG_CMD OPTIONAL
|
||||
//
|
||||
// Logs new commands when they process.
|
||||
// Defaults to ENABLED.
|
||||
//
|
||||
//#define LOG_ATTITUDE_FAST DISABLED
|
||||
//#define LOG_ATTITUDE_MED ENABLED
|
||||
//#define LOG_GPS ENABLED
|
||||
//#define LOG_PM ENABLED
|
||||
//#define LOG_CTUN DISABLED
|
||||
//#define LOG_NTUN DISABLED
|
||||
//#define LOG_MODE ENABLED
|
||||
//#define LOG_RAW DISABLED
|
||||
//#define LOG_CMD ENABLED
|
||||
//
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// Navigation defaults
|
||||
//
|
||||
// WP_RADIUS_DEFAULT OPTIONAL
|
||||
//
|
||||
// When the user performs a factory reset on the APM, set the waypoint radius
|
||||
// (the radius from a target waypoint within which the APM will consider
|
||||
// itself to have arrived at the waypoint) to this value in meters. This is
|
||||
// mainly intended to allow users to start using the APM without running the
|
||||
// WaypointWriter first.
|
||||
//
|
||||
// LOITER_RADIUS_DEFAULT OPTIONAL
|
||||
//
|
||||
// When the user performs a factory reset on the APM, set the loiter radius
|
||||
// (the distance the APM will attempt to maintain from a waypoint while
|
||||
// loitering) to this value in meters. This is mainly intended to allow
|
||||
// users to start using the APM without running the WaypointWriter first.
|
||||
//
|
||||
//#define WP_RADIUS_DEFAULT 20
|
||||
//#define LOITER_RADIUS_DEFAULT 30
|
||||
//
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// Debugging interface
|
||||
//
|
||||
// DEBUG_PORT OPTIONAL
|
||||
//
|
||||
// The APM will periodically send messages reporting what it is doing; this
|
||||
// variable determines to which serial port they will be sent. Port 0 is the
|
||||
// USB serial port on the shield, port 3 is the telemetry port.
|
||||
//
|
||||
//#define DEBUG_PORT 0
|
||||
//
|
||||
|
||||
//
|
||||
// Do not remove - this is to discourage users from copying this file
|
||||
// and using it as-is rather than editing their own.
|
||||
//
|
||||
|
@ -0,0 +1,914 @@
|
||||
//
|
||||
// Example and reference ArduPilot Mega configuration file
|
||||
// =======================================================
|
||||
//
|
||||
// This file contains documentation and examples for configuration options
|
||||
// supported by the ArduPilot Mega software.
|
||||
//
|
||||
// Most of these options are just that - optional. You should create
|
||||
// the APM_Config.h file and use this file as a reference for options
|
||||
// that you want to change. Don't copy this file directly; the options
|
||||
// described here and their default values may change over time.
|
||||
//
|
||||
// Each item is marked with a keyword describing when you should set it:
|
||||
//
|
||||
// REQUIRED
|
||||
// You must configure this in your APM_Config.h file. The
|
||||
// software will not compile if the option is not set.
|
||||
//
|
||||
// OPTIONAL
|
||||
// The option has a sensible default (which will be described
|
||||
// here), but you may wish to override it.
|
||||
//
|
||||
// EXPERIMENTAL
|
||||
// You should configure this option unless you are prepared
|
||||
// to deal with potential problems. It may relate to a feature
|
||||
// still in development, or which is not yet adequately tested.
|
||||
//
|
||||
// DEBUG
|
||||
// The option should only be set if you are debugging the
|
||||
// software, or if you are gathering information for a bug
|
||||
// report.
|
||||
//
|
||||
// NOTE:
|
||||
// Many of these settings are considered 'factory defaults', and the
|
||||
// live value is stored and managed in the ArduPilot Mega EEPROM.
|
||||
// Use the setup 'factoryreset' command after changing options in
|
||||
// your APM_Config.h file.
|
||||
//
|
||||
// Units
|
||||
// -----
|
||||
//
|
||||
// Unless indicated otherwise, numeric quantities use the following units:
|
||||
//
|
||||
// Measurement | Unit
|
||||
// ------------+-------------------------------------
|
||||
// angle | degrees
|
||||
// distance | metres
|
||||
// speed | metres per second
|
||||
// servo angle | microseconds
|
||||
// voltage | volts
|
||||
// times | seconds
|
||||
// throttle | percent
|
||||
//
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// HARDWARE CONFIGURATION AND CONNECTIONS
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// GPS_PROTOCOL REQUIRED
|
||||
//
|
||||
// GPS configuration, must be one of:
|
||||
//
|
||||
// GPS_PROTOCOL_NONE No GPS attached
|
||||
// GPS_PROTOCOL_IMU X-Plane interface or ArduPilot IMU.
|
||||
// GPS_PROTOCOL_MTK MediaTek-based GPS.
|
||||
// GPS_PROTOCOL_UBLOX UBLOX GPS
|
||||
// GPS_PROTOCOL_SIRF SiRF-based GPS in Binary mode. NOT TESTED
|
||||
// GPS_PROTOCOL_NMEA Standard NMEA GPS. NOT SUPPORTED (yet?)
|
||||
//
|
||||
//#define GPS_PROTOCOL GPS_PROTOCOL_UBLOX
|
||||
//
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// AIRSPEED_SENSOR OPTIONAL
|
||||
// AIRSPEED_RATIO OPTIONAL
|
||||
//
|
||||
// Set AIRSPEED_SENSOR to ENABLED if you have an airspeed sensor attached.
|
||||
// Adjust AIRSPEED_RATIO in small increments to calibrate the airspeed
|
||||
// sensor relative to your GPS. The calculation and default value are optimized for speeds around 12 m/s
|
||||
//
|
||||
// The default assumes that an airspeed sensor is connected.
|
||||
//
|
||||
//#define AIRSPEED_SENSOR ENABLED
|
||||
//#define AIRSPEED_RATIO 1.9936
|
||||
//
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// GCS_PROTOCOL OPTIONAL
|
||||
// GCS_PORT OPTIONAL
|
||||
//
|
||||
// The GCS_PROTOCOL option determines which (if any) ground control station
|
||||
// protocol will be used. Must be one of:
|
||||
//
|
||||
// GCS_PROTOCOL_NONE No GCS output
|
||||
// GCS_PROTOCOL_STANDARD standard APM protocol
|
||||
// GCS_PROTOCOL_SPECIAL special test protocol (?)
|
||||
// GCS_PROTOCOL_LEGACY legacy ArduPilot protocol
|
||||
// GCS_PROTOCOL_XPLANE HIL simulation ground station
|
||||
// GCS_PROTOCOL_IMU ArdiPilot IMU output
|
||||
// GCS_PROTOCOL_JASON Jason's special secret GCS protocol
|
||||
// GCS_PROTOCOL_DEBUGTERMINAL In-flight debug console (experimental)
|
||||
//
|
||||
// The GCS_PORT option determines which serial port will be used by the
|
||||
// GCS protocol. The usual values are 0 for the console/USB port,
|
||||
// or 3 for the telemetry port on the oilpan. Note that some protocols
|
||||
// will ignore this value and always use the console port.
|
||||
//
|
||||
// The default GCS protocol is the standard ArduPilot Mega protocol.
|
||||
//
|
||||
// The default serial port is the telemetry port for GCS_PROTOCOL_STANDARD
|
||||
// and GCS_PROTOCOL_LEGACY. For all other protocols, the default serial
|
||||
// port is the FTDI/console port. GCS_PORT normally should not be set
|
||||
// in your configuration.
|
||||
//
|
||||
//#define GCS_PROTOCOL GCS_PROTOCOL_STANDARD
|
||||
//#define GCS_PORT 3
|
||||
//
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// Serial port speeds.
|
||||
//
|
||||
// SERIAL0_BAUD OPTIONAL
|
||||
//
|
||||
// Baudrate for the console port. Default is 38400bps.
|
||||
//
|
||||
// SERIAL3_BAUD OPTIONAL
|
||||
//
|
||||
// Baudrate for the telemetry port. Default is 115200bps.
|
||||
//
|
||||
//#define SERIAL0_BAUD 38400
|
||||
//#define SERIAL3_BAUD 115200
|
||||
//
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// Battery monitoring OPTIONAL
|
||||
//
|
||||
// See the manual for details on selecting divider resistors for battery
|
||||
// monitoring via the oilpan.
|
||||
//
|
||||
// BATTERY_EVENT OPTIONAL
|
||||
//
|
||||
// Set BATTERY_EVENT to ENABLED to enable battery monitoring. The default is
|
||||
// DISABLED.
|
||||
//
|
||||
// BATTERY_TYPE OPTIONAL if BATTERY_EVENT is set
|
||||
//
|
||||
// Set to 0 for 3s LiPo, 1 for 4s LiPo. The default is 0, selecting a 3s
|
||||
// battery.
|
||||
//
|
||||
// LOW_VOLTAGE OPTIONAL if BATTERY_EVENT is set.
|
||||
//
|
||||
// Normally derived from BATTERY_TYPE, the automatic value can be overridden
|
||||
// here. Value in volts at which ArduPilot Mega should consider the
|
||||
// battery to be "low".
|
||||
//
|
||||
// VOLT_DIV_RATIO OPTIONAL
|
||||
//
|
||||
// See the manual for details. The default value corresponds to the resistors
|
||||
// recommended by the manual.
|
||||
//
|
||||
//#define BATTERY_EVENT DISABLED
|
||||
//#define BATTERY_TYPE 0
|
||||
//#define LOW_VOLTAGE 11.4
|
||||
//#define VOLT_DIV_RATIO 3.0
|
||||
//
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// INPUT_VOLTAGE OPTIONAL
|
||||
//
|
||||
// In order to have accurate pressure and battery voltage readings, this
|
||||
// value should be set to the voltage measured on the 5V rail on the oilpan.
|
||||
//
|
||||
// See the manual for more details. The default value should be close.
|
||||
//
|
||||
//#define INPUT_VOLTAGE 5.0
|
||||
//
|
||||
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// RADIO CONFIGURATION
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// FLIGHT_MODE OPTIONAL
|
||||
// FLIGHT_MODE_CHANNEL OPTIONAL
|
||||
//
|
||||
// Flight modes assigned to the control channel, and the input channel that
|
||||
// is read for the control mode.
|
||||
//
|
||||
// Use a servo tester, or the ArduPilotMega_demo test program to check your
|
||||
// switch settings.
|
||||
//
|
||||
// ATTENTION: Some ArduPilot Mega boards have radio channels marked 0-7, and
|
||||
// others have them marked the standard 1-8. The FLIGHT_MODE_CHANNEL option
|
||||
// uses channel numbers 1-8 (and defaults to 8).
|
||||
//
|
||||
// If you only have a three-position switch or just want three modes, set your
|
||||
// switch to produce 1165, 1425, and 1815 microseconds and configure
|
||||
// FLIGHT_MODE 1 & 2, 3 & 4 and 5 & 6 to be the same. This is the default.
|
||||
//
|
||||
// If you have FLIGHT_MODE_CHANNEL set to 8 (the default) and your control
|
||||
// channel connected to input channel 8, the hardware failsafe mode will
|
||||
// activate for any control input over 1750ms.
|
||||
//
|
||||
// For more modes (up to six), set your switch(es) to produce any of 1165,
|
||||
// 1295, 1425, 1555, 1685, and 1815 microseconds.
|
||||
//
|
||||
// Flight mode | Switch Setting (ms)
|
||||
// ------------+---------------------
|
||||
// 1 | 1165
|
||||
// 2 | 1295
|
||||
// 3 | 1425
|
||||
// 4 | 1555
|
||||
// 5 | 1685
|
||||
// 6 | 1815 (FAILSAFE if using channel 8)
|
||||
//
|
||||
// The following standard flight modes are available:
|
||||
//
|
||||
// Name | Description
|
||||
// -----------------+--------------------------------------------
|
||||
// |
|
||||
// MANUAL | Full manual control via the hardware multiplexer.
|
||||
// |
|
||||
// STABILIZE | Tries to maintain level flight, but can be overridden with radio control inputs.
|
||||
// |
|
||||
// FLY_BY_WIRE_A | Autopilot style control via user input, with manual throttle.
|
||||
// |
|
||||
// FLY_BY_WIRE_B | Autopilot style control via user input, aispeed controlled with throttle.
|
||||
// |
|
||||
// RTL | Returns to the Home location and then LOITERs at a safe altitude.
|
||||
// |
|
||||
// AUTO | Autonomous flight based on programmed waypoints. Use the WaypointWriter
|
||||
// | application or your Ground Control System to edit and upload
|
||||
// | waypoints and other commands.
|
||||
// |
|
||||
//
|
||||
//
|
||||
// The following non-standard modes are EXPERIMENTAL:
|
||||
//
|
||||
// Name | Description
|
||||
// -----------------+--------------------------------------------
|
||||
// |
|
||||
// LOITER | Flies in a circle around the current location.
|
||||
// |
|
||||
// CIRCLE | Flies in a stabilized 'dumb' circle.
|
||||
// |
|
||||
//
|
||||
//
|
||||
// If you are using channel 8 for mode switching then FLIGHT_MODE_5 and
|
||||
// FLIGHT_MODE_6 should be MANUAL.
|
||||
//
|
||||
//
|
||||
//#define FLIGHT_MODE_CHANNEL 8
|
||||
//
|
||||
//#define FLIGHT_MODE_1 RTL
|
||||
//#define FLIGHT_MODE_2 RTL
|
||||
//#define FLIGHT_MODE_3 FLY_BY_WIRE_A
|
||||
//#define FLIGHT_MODE_4 FLY_BY_WIRE_A
|
||||
//#define FLIGHT_MODE_5 MANUAL
|
||||
//#define FLIGHT_MODE_6 MANUAL
|
||||
//
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// THROTTLE_FAILSAFE OPTIONAL
|
||||
// THROTTLE_FS_VALUE OPTIONAL
|
||||
// THROTTLE_FAILSAFE_ACTION OPTIONAL
|
||||
//
|
||||
// The throttle failsafe allows you to configure a software failsafe activated
|
||||
// by a setting on the throttle input channel (channel 3).
|
||||
//
|
||||
// This can be used to achieve a failsafe override on loss of radio control
|
||||
// without having to sacrifice one of your FLIGHT_MODE settings, as the
|
||||
// throttle failsafe overrides the switch-selected mode.
|
||||
//
|
||||
// Throttle failsafe is enabled by setting THROTTLE_FAILSAFE to 1. The default
|
||||
// is for it to be disabled.
|
||||
//
|
||||
// If the throttle failsafe is enabled, THROTTLE_FS_VALUE sets the channel value
|
||||
// below which the failsafe engages. The default is 975ms, which is a very low
|
||||
// throttle setting. Most transmitters will let you trim the manual throttle
|
||||
// position up so that you cannot engage the failsafe with a regular stick movement.
|
||||
//
|
||||
// Configure your receiver's failsafe setting for the throttle channel to the
|
||||
// absolute minimum, and use the ArduPilotMega_demo program to check that
|
||||
// you cannot reach that value with the throttle control. Leave a margin of
|
||||
// at least 50 microseconds between the lowest throttle setting and
|
||||
// THROTTLE_FS_VALUE.
|
||||
//
|
||||
// The FAILSAFE_ACTION setting determines what APM will do when throttle failsafe
|
||||
// mode is entered while flying in AUTO mode. This is important in order to avoid
|
||||
// accidental failsafe behaviour when flying waypoints that take the aircraft
|
||||
// temporarily out of radio range.
|
||||
//
|
||||
// If FAILSAFE_ACTION is 1, when failsafe is entered in AUTO or LOITER modes,
|
||||
// the aircraft will head for home in RTL mode. If the throttle channel moves
|
||||
// back up, it will return to AUTO or LOITER mode.
|
||||
//
|
||||
// The default behaviour is to ignore throttle failsafe in AUTO and LOITER modes.
|
||||
//
|
||||
//#define THROTTLE_FAILSAFE DISABLED
|
||||
//#define THROTTLE_FS_VALUE 975
|
||||
//#define THROTTLE_FAILSAFE_ACTION 2
|
||||
//
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// AUTO_TRIM OPTIONAL
|
||||
//
|
||||
// ArduPilot Mega can update its trim settings by looking at the
|
||||
// radio inputs when switching out of MANUAL mode. This allows you to
|
||||
// manually trim your aircraft before switching to an assisted mode, but it
|
||||
// also means that you should avoid switching out of MANUAL while you have
|
||||
// any control stick deflection.
|
||||
//
|
||||
// The default is to enable AUTO_TRIM.
|
||||
//
|
||||
//#define AUTO_TRIM ENABLED
|
||||
//
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// THROTTLE_REVERSE OPTIONAL
|
||||
//
|
||||
// A few speed controls require the throttle servo signal be reversed. Setting
|
||||
// this to ENABLED will reverse the throttle output signal. Ensure that your
|
||||
// throttle needs to be reversed by using the hardware failsafe and the
|
||||
// ArduPilotMega_demo program before setting this option.
|
||||
//
|
||||
// The default is to not reverse the signal.
|
||||
//
|
||||
//#define THROTTLE_REVERSE DISABLED
|
||||
//
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// ENABLE_STICK_MIXING OPTIONAL
|
||||
//
|
||||
// If this option is set to ENABLED, manual control inputs are are respected
|
||||
// while in the autopilot modes (AUTO, RTL, LOITER, CIRCLE etc.)
|
||||
//
|
||||
// The default is to enable stick mixing, allowing the pilot to take
|
||||
// emergency action without switching modes.
|
||||
//
|
||||
//#define ENABLE_STICK_MIXING ENABLED
|
||||
//
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// THROTTLE_OUT DEBUG
|
||||
//
|
||||
// When debugging, it can be useful to disable the throttle output. Set
|
||||
// this option to DISABLED to disable throttle output signals.
|
||||
//
|
||||
// The default is to not disable throttle output.
|
||||
//
|
||||
//#define THROTTLE_OUT ENABLED
|
||||
//
|
||||
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// STARTUP BEHAVIOUR
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// GROUND_START_DELAY OPTIONAL
|
||||
//
|
||||
// If configured, inserts a delay between power-up and the beginning of IMU
|
||||
// calibration during a ground start.
|
||||
//
|
||||
// Use this setting to give you time to position the aircraft horizontally
|
||||
// for the IMU calibration.
|
||||
//
|
||||
// The default is to begin IMU calibration immediately at startup.
|
||||
//
|
||||
//#define GROUND_START_DELAY 0
|
||||
//
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// ENABLE_AIR_START OPTIONAL
|
||||
//
|
||||
// If air start is disabled then you will get a ground start (including IMU
|
||||
// calibration) every time the AP is powered up. This means that if you get
|
||||
// a power glitch or reboot for some reason in the air, you will probably
|
||||
// crash, but it prevents a lot of problems on the ground like unintentional
|
||||
// motor start-ups, etc.
|
||||
//
|
||||
// If air start is enabled then you will get an air start at power up and a
|
||||
// ground start will be performed if the speed is near zero when we get gps
|
||||
// lock.
|
||||
//
|
||||
// The default is to disable air start.
|
||||
//
|
||||
//#define ENABLE_AIR_START 0
|
||||
//
|
||||
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// FLIGHT AND NAVIGATION CONTROL
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// Altitude measurement and control.
|
||||
//
|
||||
// AOA OPTIONAL
|
||||
//
|
||||
// The angle in 100ths of a degree that the nose of the aircraft should be
|
||||
// raised from horizontal in level flight. The default is 1 degree.
|
||||
//
|
||||
//#define AOA 100 // note, 100ths of a degree
|
||||
//
|
||||
// ALT_EST_GAIN OPTIONAL
|
||||
//
|
||||
// The gain of the altitude estimation function; a lower number results
|
||||
// in slower error correction and smoother output. The default is a
|
||||
// reasonable starting point.
|
||||
//
|
||||
//#define ALT_EST_GAIN 0.01
|
||||
//
|
||||
// ALTITUDE_MIX OPTIONAL
|
||||
//
|
||||
// Configures the blend between GPS and pressure altitude.
|
||||
// 0 = GPS altitude, 1 = Press alt, 0.5 = half and half, etc.
|
||||
//
|
||||
// The default is to use only GPS altitude.
|
||||
//
|
||||
//#define ALTITUDE_MIX 0
|
||||
//
|
||||
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// ENABLE_HIL OPTIONAL
|
||||
//
|
||||
// This will output a binary control string to for use in HIL sims
|
||||
// such as Xplane 9 or FlightGear.
|
||||
//
|
||||
//#define ENABLE_HIL ENABLED
|
||||
//
|
||||
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// AIRSPEED_CRUISE OPTIONAL
|
||||
//
|
||||
// The speed in metres per second to maintain during cruise. The default
|
||||
// is 10m/s, which is a conservative value suitable for relatively small,
|
||||
// light aircraft.
|
||||
//
|
||||
//#define AIRSPEED_CRUISE 10
|
||||
//
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// FLY_BY_WIRE_B airspeed control
|
||||
//
|
||||
// AIRSPEED_FBW_MIN OPTIONAL
|
||||
// AIRSPEED_FBW_MAX OPTIONAL
|
||||
//
|
||||
// Airspeed corresponding to minimum and maximum throttle in Fly By Wire B mode.
|
||||
// The defaults are 6 and 30 metres per second.
|
||||
//
|
||||
// THROTTLE_ALT_P OPTIONAL
|
||||
// THROTTLE_ALT_I OPTIONAL
|
||||
// THROTTLE_ALT_D OPTIONAL
|
||||
//
|
||||
// P, I and D terms for the throttle control loop. Defaults are 0.5, 0, 0.
|
||||
//
|
||||
// THROTTLE_ALT_INT_MAX OPTIONAL
|
||||
//
|
||||
// Maximum throttle input due to the integral. Limits the throttle input
|
||||
// due to persistent inability to maintain the commanded speed. Helps
|
||||
// prevent the throttle from staying wide open when the control is reduced
|
||||
// after a period at maxium speed.
|
||||
// Default is 20 (20%).
|
||||
//
|
||||
//#define AIRSPEED_FBW_MIN 6
|
||||
//#define AIRSPEED_FBW_MAX 30
|
||||
//#define THROTTLE_ALT_P 0.32
|
||||
//#define THROTTLE_ALT_I 0.04
|
||||
//#define THROTTLE_ALT_D 0.0
|
||||
//#define THROTTLE_ALT_INT_MAX 20
|
||||
//
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// Throttle control
|
||||
//
|
||||
// THROTTLE_MIN OPTIONAL
|
||||
//
|
||||
// The minimum throttle setting to which the autopilot will reduce the
|
||||
// throttle while descending. The default is zero, which is
|
||||
// suitable for aircraft with a steady power-off glide. Increase this
|
||||
// value if your aircraft needs throttle to maintain a stable descent in
|
||||
// level flight.
|
||||
//
|
||||
// THROTTLE_CRUISE OPTIONAL
|
||||
//
|
||||
// The approximate throttle setting to achieve AIRSPEED_CRUISE in level flight.
|
||||
// The default is 45%, which is reasonable for a modestly powered aircraft.
|
||||
//
|
||||
// THROTTLE_MAX OPTIONAL
|
||||
//
|
||||
// The maximum throttle setting the autopilot will apply. The default is 75%.
|
||||
// Reduce this value if your aicraft is overpowered, or has complex flight
|
||||
// characteristics at high throttle settings.
|
||||
//
|
||||
//#define THROTTLE_MIN 0
|
||||
//#define THROTTLE_CRUISE 45
|
||||
//#define THROTTLE_MAX 75
|
||||
//
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// Autopilot control limits
|
||||
//
|
||||
// HEAD_MAX OPTIONAL
|
||||
//
|
||||
// The maximum commanded bank angle in either direction.
|
||||
// The default is 45 degrees. Decrease this value if your aircraft is not
|
||||
// stable or has difficulty maintaining altitude in a steep bank.
|
||||
//
|
||||
// PITCH_MAX OPTIONAL
|
||||
//
|
||||
// The maximum commanded pitch up angle.
|
||||
// The default is 15 degrees. Care should be taken not to set this value too
|
||||
// large, as the aircraft may stall.
|
||||
//
|
||||
// PITCH_MIN
|
||||
//
|
||||
// The maximum commanded pitch down angle. Note that this value must be
|
||||
// negative. The default is -25 degrees. Care should be taken not to set
|
||||
// this value too large as it may result in overspeeding the aircraft.
|
||||
//
|
||||
//#define HEAD_MAX 45
|
||||
//#define PITCH_MAX 15
|
||||
//#define PITCH_MIN -25
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// Attitude control gains
|
||||
//
|
||||
// Tuning values for the attitude control PID loops.
|
||||
//
|
||||
// The P term is the primary tuning value. This determines how the control
|
||||
// deflection varies in proportion to the required correction.
|
||||
//
|
||||
// The I term is used to help control surfaces settle. This value should
|
||||
// normally be kept low.
|
||||
//
|
||||
// The D term is used to control overshoot. Avoid using or adjusting this
|
||||
// term if you are not familiar with tuning PID loops. It should normally
|
||||
// be zero for most aircraft.
|
||||
//
|
||||
// Note: When tuning these values, start with changes of no more than 25% at
|
||||
// a time.
|
||||
//
|
||||
// SERVO_ROLL_P OPTIONAL
|
||||
// SERVO_ROLL_I OPTIONAL
|
||||
// SERVO_ROLL_D OPTIONAL
|
||||
//
|
||||
// P, I and D terms for roll control. Defaults are 0.4, 0, 0.
|
||||
//
|
||||
// SERVO_ROLL_INT_MAX OPTIONAL
|
||||
//
|
||||
// Maximum control offset due to the integral. This prevents the control
|
||||
// output from being overdriven due to a persistent offset (e.g. crosstracking).
|
||||
// Default is 5 degrees.
|
||||
//
|
||||
// ROLL_SLEW_LIMIT EXPERIMENTAL
|
||||
//
|
||||
// Limits the slew rate of the roll control in degrees per second. If zero,
|
||||
// slew rate is not limited. Default is to not limit the roll control slew rate.
|
||||
// (This feature is currently not implemented.)
|
||||
//
|
||||
// SERVO_PITCH_P OPTIONAL
|
||||
// SERVO_PITCH_I OPTIONAL
|
||||
// SERVO_PITCH_D OPTIONAL
|
||||
//
|
||||
// P, I and D terms for the pitch control. Defaults are 0.6, 0, 0.
|
||||
//
|
||||
// SERVO_PITCH_INT_MAX OPTIONAL
|
||||
//
|
||||
// Maximum control offset due to the integral. This prevents the control
|
||||
// output from being overdriven due to a persistent offset (e.g. native flight
|
||||
// AoA). If you find this value is insufficient, consider adjusting the AOA
|
||||
// parameter.
|
||||
// Default is 5 degrees.
|
||||
//
|
||||
// PITCH_COMP OPTIONAL
|
||||
//
|
||||
// Adds pitch input to compensate for the loss of lift due to roll control.
|
||||
// Default is 0.20 (20% of roll control also applied to pitch control).
|
||||
//
|
||||
// SERVO_YAW_P OPTIONAL
|
||||
// SERVO_YAW_I OPTIONAL
|
||||
// SERVO_YAW_D OPTIONAL
|
||||
//
|
||||
// P, I and D terms for the YAW control. Defaults are 0.5, 0, 0.
|
||||
//
|
||||
// SERVO_YAW_INT_MAX OPTIONAL
|
||||
//
|
||||
// Maximum control offset due to the integral. This prevents the control
|
||||
// output from being overdriven due to a persistent offset (e.g. crosstracking).
|
||||
// Default is 5 degrees.
|
||||
//
|
||||
// RUDDER_MIX OPTIONAL
|
||||
//
|
||||
// Roll to yaw mixing. This allows for co-ordinated turns.
|
||||
// Default is 0.50 (50% of roll control also applied to yaw control.)
|
||||
//
|
||||
//#define SERVO_ROLL_P 0.4
|
||||
//#define SERVO_ROLL_I 0.0
|
||||
//#define SERVO_ROLL_D 0.0
|
||||
//#define SERVO_ROLL_INT_MAX 5
|
||||
//#define ROLL_SLEW_LIMIT 0
|
||||
//#define SERVO_PITCH_P 0.6
|
||||
//#define SERVO_PITCH_I 0.0
|
||||
//#define SERVO_PITCH_D 0.0
|
||||
//#define SERVO_PITCH_INT_MAX 5
|
||||
//#define PITCH_COMP 0.2
|
||||
//#define SERVO_YAW_P 0.5
|
||||
//#define SERVO_YAW_I 0.0
|
||||
//#define SERVO_YAW_D 0.0
|
||||
//#define SERVO_YAW_INT_MAX 5
|
||||
//#define RUDDER_MIX 0.5
|
||||
//
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// Navigation control gains
|
||||
//
|
||||
// Tuning values for the navigation control PID loops.
|
||||
//
|
||||
// The P term is the primary tuning value. This determines how the control
|
||||
// deflection varies in proportion to the required correction.
|
||||
//
|
||||
// The I term is used to control drift.
|
||||
//
|
||||
// The D term is used to control overshoot. Avoid adjusting this term if
|
||||
// you are not familiar with tuning PID loops.
|
||||
//
|
||||
// Note: When tuning these values, start with changes of no more than 25% at
|
||||
// a time.
|
||||
//
|
||||
// NAV_ROLL_P OPTIONAL
|
||||
// NAV_ROLL_I OPTIONAL
|
||||
// NAV_ROLL_D OPTIONAL
|
||||
//
|
||||
// P, I and D terms for navigation control over roll, normally used for
|
||||
// controlling the aircraft's course. The P term controls how aggressively
|
||||
// the aircraft will bank to change or hold course.
|
||||
// Defaults are 0.7, 0.01, 0.02.
|
||||
//
|
||||
// NAV_ROLL_INT_MAX OPTIONAL
|
||||
//
|
||||
// Maximum control offset due to the integral. This prevents the control
|
||||
// output from being overdriven due to a persistent offset (e.g. crosstracking).
|
||||
// Default is 5 degrees.
|
||||
//
|
||||
// NAV_PITCH_ASP_P OPTIONAL
|
||||
// NAV_PITCH_ASP_I OPTIONAL
|
||||
// NAV_PITCH_ASP_D OPTIONAL
|
||||
//
|
||||
// P, I and D terms for pitch adjustments made to maintain airspeed.
|
||||
// Defaults are 0.65, 0, 0.
|
||||
//
|
||||
// NAV_PITCH_ASP_INT_MAX OPTIONAL
|
||||
//
|
||||
// Maximum pitch offset due to the integral. This limits the control
|
||||
// output from being overdriven due to a persistent offset (eg. inability
|
||||
// to maintain the programmed airspeed).
|
||||
// Default is 5 degrees.
|
||||
//
|
||||
// NAV_PITCH_ALT_P OPTIONAL
|
||||
// NAV_PITCH_ALT_I OPTIONAL
|
||||
// NAV_PITCH_ALT_D OPTIONAL
|
||||
//
|
||||
// P, I and D terms for pitch adjustments made to maintain altitude.
|
||||
// Defaults are 0.65, 0, 0.
|
||||
//
|
||||
// NAV_PITCH_ALT_INT_MAX OPTIONAL
|
||||
//
|
||||
// Maximum pitch offset due to the integral. This limits the control
|
||||
// output from being overdriven due to a persistent offset (eg. inability
|
||||
// to maintain the programmed altitude).
|
||||
// Default is 5 degrees.
|
||||
//
|
||||
//#define NAV_ROLL_P 0.7
|
||||
//#define NAV_ROLL_I 0.01
|
||||
//#define NAV_ROLL_D 0.02
|
||||
//#define NAV_ROLL_INT_MAX 5
|
||||
//#define NAV_PITCH_ASP_P 0.65
|
||||
//#define NAV_PITCH_ASP_I 0.0
|
||||
//#define NAV_PITCH_ASP_D 0.0
|
||||
//#define NAV_PITCH_ASP_INT_MAX 5
|
||||
//#define NAV_PITCH_ALT_P 0.65
|
||||
//#define NAV_PITCH_ALT_I 0.0
|
||||
//#define NAV_PITCH_ALT_D 0.0
|
||||
//#define NAV_PITCH_ALT_INT_MAX 5
|
||||
//
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// Energy/Altitude control gains
|
||||
//
|
||||
// The Energy/altitude control system uses throttle input to control aircraft
|
||||
// altitude.
|
||||
//
|
||||
// The P term is the primary tuning value. This determines how the throttle
|
||||
// setting varies in proportion to the required correction.
|
||||
//
|
||||
// The I term is used to compensate for small offsets.
|
||||
//
|
||||
// The D term is used to control overshoot. Avoid adjusting this term if
|
||||
// you are not familiar with tuning PID loops.
|
||||
//
|
||||
// THROTTLE_TE_P OPTIONAL
|
||||
// THROTTLE_TE_I OPTIONAL
|
||||
// THROTTLE_TE_D OPTIONAL
|
||||
//
|
||||
// P, I and D terms for throttle adjustments made to control altitude.
|
||||
// Defaults are 0.5, 0, 0.
|
||||
//
|
||||
// THROTTLE_TE_INT_MAX OPTIONAL
|
||||
//
|
||||
// Maximum throttle input due to the integral term. This limits the
|
||||
// throttle from being overdriven due to a persistent offset (e.g.
|
||||
// inability to maintain the programmed altitude).
|
||||
// Default is 20%.
|
||||
//
|
||||
// THROTTLE_SLEW_LIMIT OPTIONAL
|
||||
//
|
||||
// Limits the slew rate of the throttle, in percent per second. Helps
|
||||
// avoid sudden throttle changes, which can destabilise the aircraft.
|
||||
// A setting of zero disables the feature.
|
||||
// Default is zero (disabled).
|
||||
//
|
||||
// P_TO_T OPTIONAL
|
||||
//
|
||||
// Pitch to throttle feed-forward gain. Used when AIRSPEED_SENSOR
|
||||
// is DISABLED. Default is 2.5.
|
||||
//
|
||||
//#define THROTTLE_TE_P 0.50
|
||||
//#define THROTTLE_TE_I 0.0
|
||||
//#define THROTTLE_TE_D 0.0
|
||||
//#define THROTTLE_TE_INT_MAX 20
|
||||
//#define THROTTLE_SLEW_LIMIT 0
|
||||
//#define P_TO_T 2.5
|
||||
//
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// Crosstrack compensation
|
||||
//
|
||||
// XTRACK_GAIN OPTIONAL
|
||||
//
|
||||
// Crosstrack compensation in degrees per metre off track.
|
||||
// Default value is 0.01 degrees per metre. Values lower than 0.001 will
|
||||
// disable crosstrack compensation.
|
||||
//
|
||||
// XTRACK_ENTRY_ANGLE OPTIONAL
|
||||
//
|
||||
// Maximum angle used to correct for track following.
|
||||
// Default value is 30 degrees.
|
||||
//
|
||||
//#define XTRACK_GAIN 0.01
|
||||
//#define XTRACK_ENTRY_ANGLE 30
|
||||
//
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// DEBUGGING
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// Subsystem test and debug.
|
||||
//
|
||||
// DEBUG_SUBSYSTEM DEBUG
|
||||
//
|
||||
// Selects a subsystem debug mode. Default is 0.
|
||||
//
|
||||
// 0 = no debug
|
||||
// 1 = Debug the Radio input
|
||||
// 2 = Radio Setup / Servo output
|
||||
// 4 = Debug the GPS input
|
||||
// 5 = Debug the GPS input - RAW OUTPUT
|
||||
// 6 = Debug the IMU
|
||||
// 7 = Debug the Control Switch
|
||||
// 8 = Debug the Servo DIP switches
|
||||
// 9 = Debug the Relay out
|
||||
// 10 = Debug the Magnetometer
|
||||
// 11 = Debug the ABS pressure sensor
|
||||
// 12 = Debug the stored waypoints
|
||||
// 13 = Debug the Throttle
|
||||
// 14 = Debug the Radio Min Max
|
||||
// 15 = Debug the EEPROM - Hex Dump
|
||||
// 16 = XBee X-CTU Range and RSSI Test
|
||||
// 17 = Debug IMU - raw gyro and accel outputs
|
||||
// 20 = Debug Analog Sensors
|
||||
//
|
||||
//
|
||||
//#define DEBUG_SUBSYSTEM 0
|
||||
//
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// DEBUG_LEVEL DEBUG
|
||||
//
|
||||
// Selects the lowest level of debug messages passed to the telemetry system.
|
||||
// Default is SEVERITY_LOW. May be one of:
|
||||
//
|
||||
// SEVERITY_LOW
|
||||
// SEVERITY_MEDIUM
|
||||
// SEVERITY_HIGH
|
||||
// SEVERITY_CRITICAL
|
||||
//
|
||||
//#define DEBUG_LEVEL SEVERITY_LOW
|
||||
//
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// Dataflash logging control
|
||||
//
|
||||
// Each of these logging options may be set to ENABLED to enable, or DISABLED
|
||||
// to disable the logging of the respective data.
|
||||
//
|
||||
// LOG_ATTITUDE_FAST DEBUG
|
||||
//
|
||||
// Logs basic attitude info to the dataflash at 50Hz (uses more space).
|
||||
// Defaults to DISABLED.
|
||||
//
|
||||
// LOG_ATTITUDE_MED OPTIONAL
|
||||
//
|
||||
// Logs basic attitude info to the dataflash at 10Hz (uses less space than
|
||||
// LOG_ATTITUDE_FAST). Defaults to ENABLED.
|
||||
//
|
||||
// LOG_GPS OPTIONAL
|
||||
//
|
||||
// Logs GPS info to the dataflash at 10Hz. Defaults to ENABLED.
|
||||
//
|
||||
// LOG_PM OPTIONAL
|
||||
//
|
||||
// Logs IMU performance monitoring info every 20 seconds.
|
||||
// Defaults to DISABLED.
|
||||
//
|
||||
// LOG_CTUN OPTIONAL
|
||||
//
|
||||
// Logs control loop tuning info at 10 Hz. This information is useful for tuning
|
||||
// servo control loop gain values. Defaults to DISABLED.
|
||||
//
|
||||
// LOG_NTUN OPTIONAL
|
||||
//
|
||||
// Logs navigation tuning info at 10 Hz. This information is useful for tuning
|
||||
// navigation control loop gain values. Defaults to DISABLED.
|
||||
//
|
||||
// LOG_ MODE OPTIONAL
|
||||
//
|
||||
// Logs changes to the flight mode upon occurrence. Defaults to ENABLED.
|
||||
//
|
||||
// LOG_RAW DEBUG
|
||||
//
|
||||
// Logs raw accelerometer and gyro data at 50 Hz (uses more space).
|
||||
// Defaults to DISABLED.
|
||||
//
|
||||
// LOG_CMD OPTIONAL
|
||||
//
|
||||
// Logs new commands when they process.
|
||||
// Defaults to ENABLED.
|
||||
//
|
||||
//#define LOG_ATTITUDE_FAST DISABLED
|
||||
//#define LOG_ATTITUDE_MED ENABLED
|
||||
//#define LOG_GPS ENABLED
|
||||
//#define LOG_PM ENABLED
|
||||
//#define LOG_CTUN DISABLED
|
||||
//#define LOG_NTUN DISABLED
|
||||
//#define LOG_MODE ENABLED
|
||||
//#define LOG_RAW DISABLED
|
||||
//#define LOG_CMD ENABLED
|
||||
//
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// Navigation defaults
|
||||
//
|
||||
// WP_RADIUS_DEFAULT OPTIONAL
|
||||
//
|
||||
// When the user performs a factory reset on the APM, set the waypoint radius
|
||||
// (the radius from a target waypoint within which the APM will consider
|
||||
// itself to have arrived at the waypoint) to this value in meters. This is
|
||||
// mainly intended to allow users to start using the APM without running the
|
||||
// WaypointWriter first.
|
||||
//
|
||||
// LOITER_RADIUS_DEFAULT OPTIONAL
|
||||
//
|
||||
// When the user performs a factory reset on the APM, set the loiter radius
|
||||
// (the distance the APM will attempt to maintain from a waypoint while
|
||||
// loitering) to this value in meters. This is mainly intended to allow
|
||||
// users to start using the APM without running the WaypointWriter first.
|
||||
//
|
||||
//#define WP_RADIUS_DEFAULT 20
|
||||
//#define LOITER_RADIUS_DEFAULT 30
|
||||
//
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// Debugging interface
|
||||
//
|
||||
// DEBUG_PORT OPTIONAL
|
||||
//
|
||||
// The APM will periodically send messages reporting what it is doing; this
|
||||
// variable determines to which serial port they will be sent. Port 0 is the
|
||||
// USB serial port on the shield, port 3 is the telemetry port.
|
||||
//
|
||||
//#define DEBUG_PORT 0
|
||||
//
|
||||
|
||||
//
|
||||
// Do not remove - this is to discourage users from copying this file
|
||||
// and using it as-is rather than editing their own.
|
||||
//
|
||||
#error You should not copy APM_Config.h.reference - make your own APM_Config.h file with just the options you need.
|
@ -0,0 +1,19 @@
|
||||
#define DEBUG_SUBSYSTEM 0
|
||||
|
||||
#define FLIGHT_MODE_CHANNEL 8
|
||||
#define FLIGHT_MODE_1 AUTO
|
||||
#define FLIGHT_MODE_2 RTL
|
||||
#define FLIGHT_MODE_3 FLY_BY_WIRE_A
|
||||
#define FLIGHT_MODE_4 FLY_BY_WIRE_B
|
||||
#define FLIGHT_MODE_5 STABILIZE
|
||||
#define FLIGHT_MODE_6 MANUAL
|
||||
|
||||
#define GCS_PROTOCOL GCS_PROTOCOL_DEBUGTERMINAL
|
||||
#define ENABLE_HIL ENABLED
|
||||
#define GCS_PORT 3
|
||||
#define GPS_PROTOCOL GPS_PROTOCOL_IMU
|
||||
#define AIRSPEED_CRUISE 25
|
||||
|
||||
#define THROTTLE_FAILSAFE ENABLED
|
||||
#define AIRSPEED_SENSOR ENABLED
|
||||
|
@ -0,0 +1,845 @@
|
||||
// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: t -*-
|
||||
|
||||
/*
|
||||
ArduPilotMega Version 1.0.3 Public Alpha
|
||||
Authors: Doug Weibel, Jose Julio, Jordi Munoz, Jason Short
|
||||
Thanks to: Chris Anderson, HappyKillMore, Bill Premerlani, James Cohen, JB from rotorFX, Automatik, Fefenin, Peter Meister, Remzibi
|
||||
Please contribute your ideas!
|
||||
|
||||
|
||||
This firmware is free software; you can redistribute it and/or
|
||||
modify it under the terms of the GNU Lesser General Public
|
||||
License as published by the Free Software Foundation; either
|
||||
version 2.1 of the License, or (at your option) any later version.
|
||||
*/
|
||||
|
||||
// Libraries
|
||||
#include <FastSerial.h>
|
||||
#include <JETI_Box.h>
|
||||
#include <AP_Common.h>
|
||||
#include <APM_RC.h> // ArduPilot Mega RC Library
|
||||
#include <APM_ADC.h> // ArduPilot Mega Analog to Digital Converter Library
|
||||
#include <AP_GPS.h> // ArduPilot GPS library
|
||||
#include <Wire.h>
|
||||
#include <APM_BMP085.h> // ArduPilot Mega BMP085 Library
|
||||
#include <DataFlash.h> // ArduPilot Mega Flash Memory Library
|
||||
#include <APM_Compass.h> // ArduPilot Mega Magnetometer Library
|
||||
|
||||
// AVR runtime
|
||||
#include <avr/io.h>
|
||||
#include <avr/eeprom.h>
|
||||
#include <avr/pgmspace.h>
|
||||
#include <math.h>
|
||||
|
||||
// Configuration
|
||||
#include "config.h"
|
||||
|
||||
// Local modules
|
||||
#include "defines.h"
|
||||
|
||||
// Serial ports
|
||||
//
|
||||
// Note that FastSerial port buffers are allocated at ::begin time,
|
||||
// so there is not much of a penalty to defining ports that we don't
|
||||
// use.
|
||||
//
|
||||
FastSerialPort0(Serial); // FTDI/console
|
||||
FastSerialPort1(Serial1); // GPS port (except for GPS_PROTOCOL_IMU)
|
||||
//FastSerialPort3(Serial3); // Telemetry port (optional, Standard and ArduPilot protocols only)
|
||||
|
||||
// GPS selection
|
||||
#if GPS_PROTOCOL == GPS_PROTOCOL_NMEA
|
||||
AP_GPS_NMEA GPS(&Serial1);
|
||||
#elif GPS_PROTOCOL == GPS_PROTOCOL_SIRF
|
||||
AP_GPS_SIRF GPS(&Serial1);
|
||||
#elif GPS_PROTOCOL == GPS_PROTOCOL_UBLOX
|
||||
AP_GPS_UBLOX GPS(&Serial1);
|
||||
#elif GPS_PROTOCOL == GPS_PROTOCOL_IMU
|
||||
AP_GPS_IMU GPS(&Serial); // note, console port
|
||||
#elif GPS_PROTOCOL == GPS_PROTOCOL_MTK
|
||||
AP_GPS_MTK GPS(&Serial1);
|
||||
#elif GPS_PROTOCOL == GPS_PROTOCOL_NONE
|
||||
AP_GPS_NONE GPS(NULL);
|
||||
#else
|
||||
# error Must define GPS_PROTOCOL in your configuration file.
|
||||
#endif
|
||||
|
||||
// The X-Plane GCS requires the IMU GPS configuration
|
||||
#if (ENABLE_HIL == ENABLED) && (GPS_PROTOCOL != GPS_PROTOCOL_IMU)
|
||||
# error Must select GPS_PROTOCOL_IMU when configuring for X-Plane
|
||||
#endif
|
||||
|
||||
// GENERAL VARIABLE DECLARATIONS
|
||||
// --------------------------------------------
|
||||
byte control_mode = MANUAL;
|
||||
boolean failsafe = false; // did our throttle dip below the failsafe value?
|
||||
boolean ch3_failsafe = false;
|
||||
byte crash_timer;
|
||||
byte oldSwitchPosition; // for remembering the control mode switch
|
||||
boolean reverse_switch = 1; // do we read the reversing switches after startup?
|
||||
|
||||
byte ground_start_count = 6; // have we achieved first lock and set Home?
|
||||
int ground_start_avg; // 5 samples to avg speed for ground start
|
||||
boolean ground_start = false; // have we started on the ground?
|
||||
char *comma = ",";
|
||||
|
||||
char* flight_mode_strings[] = {
|
||||
"Manual",
|
||||
"Circle",
|
||||
"Stabilize",
|
||||
"",
|
||||
"",
|
||||
"FBW_A",
|
||||
"FBW_B",
|
||||
"",
|
||||
"",
|
||||
"",
|
||||
"Auto",
|
||||
"RTL",
|
||||
"Loiter",
|
||||
"Takeoff",
|
||||
"Land"};
|
||||
|
||||
|
||||
/* Radio values
|
||||
Channel assignments
|
||||
1 Ailerons (rudder if no ailerons)
|
||||
2 Elevator
|
||||
3 Throttle
|
||||
4 Rudder (if we have ailerons)
|
||||
5 Mode
|
||||
6 TBD
|
||||
7 TBD
|
||||
8 TBD
|
||||
*/
|
||||
|
||||
int radio_min[8]; // may be reset by init sequence
|
||||
int radio_trim[8]; // may be reset by init sequence
|
||||
int radio_max[8]; // may be reset by init sequence
|
||||
int radio_in[8]; // current values from the transmitter - microseconds
|
||||
int radio_out[8]; // Send to the PWM library
|
||||
int servo_out[8]; // current values to the servos - degrees * 100 (approx assuming servo is -45 to 45 degrees except [3] is 0 to 100
|
||||
|
||||
int elevon1_trim = 1500;
|
||||
int elevon2_trim = 1500;
|
||||
int ch1_temp = 1500; // Used for elevon mixing
|
||||
int ch2_temp = 1500;
|
||||
|
||||
int reverse_roll = 1;
|
||||
int reverse_pitch = 1;
|
||||
int reverse_rudder = 1;
|
||||
byte mix_mode = 0; // 0 = normal , 1 = elevons
|
||||
int reverse_elevons = 1;
|
||||
int reverse_ch1_elevon = 1;
|
||||
int reverse_ch2_elevon = 1;
|
||||
byte yaw_mode;
|
||||
|
||||
byte flight_mode_channel;
|
||||
byte flight_modes[6];
|
||||
byte auto_trim;
|
||||
|
||||
// for elevons radio_in[CH_ROLL] and radio_in[CH_PITCH] are equivalent aileron and elevator, not left and right elevon
|
||||
|
||||
|
||||
/* PID Control variables
|
||||
Cases
|
||||
1 Aileron servo control (rudder if no ailerons)
|
||||
2 Elevator servo control
|
||||
3 Rudder servo control (if we have ailerons)
|
||||
4 Roll set-point control
|
||||
5 Pitch set-point based on airspeed error control
|
||||
6 Pitch set-point based on altitude error control (if we do not have airspeed sensor)
|
||||
7 Throttle based on Energy Height (Total Energy) error control
|
||||
8 Throttle based on altitude error control
|
||||
*/
|
||||
|
||||
float kp[8];
|
||||
float ki[8];
|
||||
float kd[8];
|
||||
uint16_t integrator_max[8]; // PID Integrator Max deg * 100
|
||||
float integrator[8]; // PID Integrators deg * 100
|
||||
long last_error[8]; // PID last error for derivative
|
||||
float nav_gain_scaler = 1; // Gain scaling for headwind/tailwind
|
||||
|
||||
|
||||
/* Feed Forward gains
|
||||
Cases
|
||||
1 Bank angle to desired pitch (Pitch Comp)
|
||||
2 Aileron Servo to Rudder Servo
|
||||
3 Pitch to Throttle
|
||||
*/
|
||||
|
||||
float kff[3];
|
||||
|
||||
|
||||
// GPS variables
|
||||
// -------------
|
||||
const float t7 = 10000000.0; // used to scale GPS values for EEPROM storage
|
||||
float scaleLongUp; // used to reverse longtitude scaling
|
||||
float scaleLongDown; // used to reverse longtitude scaling
|
||||
boolean GPS_light = false; // status of the GPS light
|
||||
|
||||
// Location & Navigation
|
||||
// ---------------------
|
||||
byte wp_radius = 20; // meters
|
||||
long hold_course = -1; // deg * 100 dir of plane
|
||||
long nav_bearing; // deg * 100 : 0 to 360 current desired bearing to navigate
|
||||
long target_bearing; // deg * 100 : 0 to 360 location of the plane to the target
|
||||
long crosstrack_bearing; // deg * 100 : 0 to 360 desired angle of plane to target
|
||||
int climb_rate; // m/s * 100 - For future implementation of controlled ascent/descent by rate
|
||||
byte loiter_radius; // meters
|
||||
float x_track_gain;
|
||||
int x_track_angle;
|
||||
|
||||
long head_max;
|
||||
long pitch_max;
|
||||
long pitch_min;
|
||||
float altitude_mix;
|
||||
|
||||
byte command_must_index; // current command memory location
|
||||
byte command_may_index; // current command memory location
|
||||
byte command_must_ID; // current command ID
|
||||
byte command_may_ID; // current command ID
|
||||
//byte EEPROM_command // 1 = from the list, 0 = generated
|
||||
|
||||
|
||||
// Airspeed
|
||||
// --------
|
||||
int airspeed; // m/s * 100
|
||||
int airspeed_cruise; // m/s * 100 : target airspeed sensor value
|
||||
float airspeed_ratio; // used to scale airspeed
|
||||
byte airspeed_fbw_min; // m/s
|
||||
byte airspeed_fbw_max; // m/s
|
||||
|
||||
// Throttle Failsafe
|
||||
// ------------------
|
||||
byte throttle_failsafe_enabled;
|
||||
int throttle_failsafe_value;
|
||||
byte throttle_failsafe_action;
|
||||
uint16_t log_bitmask;
|
||||
|
||||
// Location Errors
|
||||
// ---------------
|
||||
long bearing_error; // deg * 100 : 0 to 36000
|
||||
long altitude_error; // meters * 100 we are off in altitude
|
||||
float airspeed_error; // m/s * 100
|
||||
float crosstrack_error; // meters we are off trackline
|
||||
long energy_error; // energy state error (kinetic + potential) for altitude hold
|
||||
long airspeed_energy_error; // kinetic portion of energy error
|
||||
|
||||
// Sensors
|
||||
// --------
|
||||
float airpressure_raw; // Airspeed Sensor - is a float to better handle filtering
|
||||
int airpressure_offset; // analog air pressure sensor while still
|
||||
int airpressure; // airspeed as a pressure value
|
||||
float battery_voltage = LOW_VOLTAGE * 1.05; // Battery Voltage of total battery, initialized above threshold for filter
|
||||
float battery_voltage1 = LOW_VOLTAGE * 1.05; // Battery Voltage of cell 1, initialized above threshold for filter
|
||||
float battery_voltage2 = LOW_VOLTAGE * 1.05; // Battery Voltage of cells 1+2, initialized above threshold for filter
|
||||
float battery_voltage3 = LOW_VOLTAGE * 1.05; // Battery Voltage of cells 1+2+3, initialized above threshold for filter
|
||||
float battery_voltage4 = LOW_VOLTAGE * 1.05; // Battery Voltage of cells 1+2+3+4, initialized above threshold for filter
|
||||
|
||||
// From IMU
|
||||
long roll_sensor; // degrees * 100
|
||||
long pitch_sensor; // degrees * 100
|
||||
long yaw_sensor; // degrees * 100
|
||||
|
||||
float roll; // radians
|
||||
float pitch; // radians
|
||||
float yaw; // radians
|
||||
|
||||
|
||||
// Magnetometer variables
|
||||
int magnetom_x; // Doug to do
|
||||
int magnetom_y;
|
||||
int magnetom_z;
|
||||
float MAG_Heading;
|
||||
|
||||
// Pressure Sensor variables
|
||||
unsigned long abs_press;
|
||||
unsigned long abs_press_filt;
|
||||
unsigned long abs_press_gnd;
|
||||
int ground_temperature;
|
||||
int temp_unfilt;
|
||||
long ground_alt; // Ground altitude from gps at startup in centimeters
|
||||
long press_alt; // Pressure altitude
|
||||
|
||||
// flight mode specific
|
||||
// --------------------
|
||||
//boolean takeoff_complete = false; // Flag for using take-off controls
|
||||
boolean land_complete = false;
|
||||
int landing_pitch; // pitch for landing set by commands
|
||||
int takeoff_pitch;
|
||||
int takeoff_altitude;
|
||||
int landing_distance; // meters;
|
||||
|
||||
// Loiter management
|
||||
// -----------------
|
||||
long old_target_bearing; // deg * 100
|
||||
int loiter_total; // deg : how many times to loiter * 360
|
||||
int loiter_delta; // deg : how far we just turned
|
||||
int loiter_sum; // deg : how far we have turned around a waypoint
|
||||
long loiter_time; // millis : when we started LOITER mode
|
||||
int loiter_time_max; // millis : how long to stay in LOITER mode
|
||||
|
||||
// these are the values for navigation control functions
|
||||
// ----------------------------------------------------
|
||||
long nav_roll; // deg * 100 : target roll angle
|
||||
long nav_pitch; // deg * 100 : target pitch angle
|
||||
long altitude_estimate; // for smoothing GPS output
|
||||
|
||||
int throttle_min; // 0-100 : target throttle output for average speed
|
||||
int throttle_cruise; // 0-100 : target throttle output for average speed
|
||||
int throttle_max; // 0-100 : target throttle output for average speed
|
||||
|
||||
// Waypoints
|
||||
// ---------
|
||||
long wp_distance; // meters - distance between plane and next waypoint
|
||||
long wp_totalDistance; // meters - distance between old and next waypoint
|
||||
byte wp_total; // # of Commands total including way
|
||||
byte wp_index; // Current active command index
|
||||
byte next_wp_index; // Current active command index
|
||||
|
||||
// repeating event control
|
||||
// -----------------------
|
||||
byte event_id; // what to do - see defines
|
||||
long event_timer; // when the event was asked for in ms
|
||||
int event_delay; // how long to delay the next firing of event in millis
|
||||
int event_repeat; // how many times to fire : 0 = forever, 1 = do once, 2 = do twice
|
||||
int event_value; // per command value, such as PWM for servos
|
||||
int event_undo_value; // the value used to undo commands
|
||||
byte repeat_forever;
|
||||
byte undo_event; // counter for timing the undo
|
||||
|
||||
// delay command
|
||||
// --------------
|
||||
int delay_timeout = 0; // used to delay commands
|
||||
long delay_start = 0; // used to delay commands
|
||||
|
||||
// 3D Location vectors
|
||||
// -------------------
|
||||
struct Location home; // home location
|
||||
struct Location prev_WP; // last waypoint
|
||||
struct Location current_loc; // current location
|
||||
struct Location next_WP; // next waypoint
|
||||
struct Location tell_command; // command for telemetry
|
||||
struct Location next_command; // command preloaded
|
||||
long target_altitude; // used for
|
||||
long offset_altitude; // used for
|
||||
int hold_alt_above_home;
|
||||
boolean home_is_set = false; // Flag for if we have gps lock and have set the home location
|
||||
|
||||
|
||||
// IMU variables
|
||||
// -------------
|
||||
|
||||
//Sensor: GYROX, GYROY, GYROZ, ACCELX, ACCELY, ACCELZ
|
||||
uint8_t sensors[6] = {1,2,0,4,5,6}; // For ArduPilot Mega Sensor Shield Hardware
|
||||
int SENSOR_SIGN[] = { 1, -1, -1,
|
||||
-1, 1, 1,
|
||||
-1, -1, -1};
|
||||
|
||||
// Temp compensation curve constants
|
||||
// These must be produced by measuring data and curve fitting
|
||||
// [X / Y/Z gyro][A / B/C or 0 order / 1st order / 2nd order constants]
|
||||
// values may migrate to a Config file
|
||||
float GTC[3][3]={{1665, 0, 0},
|
||||
{1665, 0, 0},
|
||||
{1665, 0, 0}};
|
||||
|
||||
float AN[6]; // array that store the 6 ADC channels used by IMU
|
||||
float AN_OFFSET[6]; // Array that store the Offset of the gyros and accelerometers
|
||||
float G_Dt = 0.02; // Integration time for the gyros (DCM algorithm)
|
||||
float Accel_Vector[3]; // Store the acceleration in a vector
|
||||
float Accel_Vector_unfiltered[3]; // Store the acceleration in a vector
|
||||
float Gyro_Vector[3]; // Store the gyros turn rate in a vector
|
||||
float Omega_Vector[3]; // Corrected Gyro_Vector data
|
||||
float Omega_P[3]; // Omega Proportional correction
|
||||
float Omega_I[3]; // Omega Integrator
|
||||
float Omega[3];
|
||||
float errorRollPitch[3];
|
||||
float errorYaw[3];
|
||||
float errorCourse;
|
||||
float COGX; // Course overground X axis
|
||||
float COGY = 1; // Course overground Y axis
|
||||
|
||||
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}};
|
||||
|
||||
float Temporary_Matrix[3][3];
|
||||
|
||||
// Performance monitoring
|
||||
// ----------------------
|
||||
long perf_mon_timer;
|
||||
float imu_health; // Metric based on accel gain deweighting
|
||||
int G_Dt_max; // Max main loop cycle time in milliseconds
|
||||
byte gyro_sat_count;
|
||||
byte adc_constraints;
|
||||
byte renorm_sqrt_count;
|
||||
byte renorm_blowup_count;
|
||||
int gps_fix_count;
|
||||
byte gcs_messages_sent;
|
||||
|
||||
|
||||
// GCS
|
||||
// ---
|
||||
char GCS_buffer[53];
|
||||
|
||||
// System Timers
|
||||
// --------------
|
||||
unsigned long fast_loopTimer; // Time in miliseconds of main control loop
|
||||
unsigned long medium_loopTimer; // Time in miliseconds of navigation control loop
|
||||
byte medium_loopCounter = 0; // Counters for branching from main control loop to slower loops
|
||||
byte slow_loopCounter = 0;
|
||||
byte superslow_loopCounter = 0;
|
||||
unsigned long deltaMiliSeconds; // Delta Time in miliseconds
|
||||
unsigned long dTnav; // Delta Time in milliseconds for navigation computations
|
||||
int mainLoop_count;
|
||||
unsigned long elapsedTime; // for doing custom events
|
||||
unsigned int GPS_timer;
|
||||
|
||||
//********************************************************************************************************************************
|
||||
//********************************************************************************************************************************
|
||||
//********************************************************************************************************************************
|
||||
//********************************************************************************************************************************
|
||||
//********************************************************************************************************************************
|
||||
|
||||
// Basic Initialization
|
||||
//---------------------
|
||||
void setup() {
|
||||
jeti_init();
|
||||
jeti_status(" ** X-DIY **");
|
||||
jeti_update;
|
||||
init_ardupilot();
|
||||
}
|
||||
|
||||
void loop()
|
||||
{
|
||||
// We want this to execute at 50Hz if possible
|
||||
// -------------------------------------------
|
||||
if (millis()-fast_loopTimer > 19) {
|
||||
deltaMiliSeconds = millis() - fast_loopTimer;
|
||||
G_Dt = (float)deltaMiliSeconds / 1000.f;
|
||||
fast_loopTimer = millis();
|
||||
mainLoop_count++;
|
||||
|
||||
// Execute the fast loop
|
||||
// ---------------------
|
||||
fast_loop();
|
||||
|
||||
// Execute the medium loop
|
||||
// -----------------------
|
||||
medium_loop();
|
||||
|
||||
if (millis()- perf_mon_timer > 20000) {
|
||||
if (mainLoop_count != 0) {
|
||||
send_message(MSG_PERF_REPORT);
|
||||
if (log_bitmask & MASK_LOG_PM)
|
||||
Log_Write_Performance();
|
||||
|
||||
resetPerfData();
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void fast_loop()
|
||||
{
|
||||
// This is the fast loop - we want it to execute at 50Hz if possible
|
||||
// -----------------------------------------------------------------
|
||||
if (deltaMiliSeconds > G_Dt_max)
|
||||
G_Dt_max = deltaMiliSeconds;
|
||||
|
||||
// Read radio
|
||||
// ----------
|
||||
read_radio();
|
||||
|
||||
// check for throttle failsafe condition
|
||||
// ------------------------------------
|
||||
#if THROTTLE_FAILSAFE == 1
|
||||
set_failsafe(ch3_failsafe);
|
||||
#endif
|
||||
|
||||
|
||||
// read in the plane's attitude
|
||||
// ----------------------------
|
||||
#if GPS_PROTOCOL == GPS_PROTOCOL_IMU
|
||||
GPS.update();
|
||||
airspeed = (float)GPS.airspeed; //airspeed is * 100 coming in from Xplane for accuracy
|
||||
calc_airspeed_errors();
|
||||
|
||||
if(digitalRead(SLIDE_SWITCH_PIN) == 0) {
|
||||
read_AHRS();
|
||||
roll_sensor = -roll_sensor;
|
||||
pitch_sensor = -pitch_sensor;
|
||||
//yaw_sensor = -yaw_sensor;
|
||||
}else{
|
||||
roll_sensor = GPS.roll_sensor;
|
||||
pitch_sensor = GPS.pitch_sensor;
|
||||
yaw_sensor = GPS.ground_course;
|
||||
}
|
||||
|
||||
#else
|
||||
// Read Airspeed
|
||||
// -------------
|
||||
#if AIRSPEED_SENSOR == 1
|
||||
read_airspeed();
|
||||
#endif
|
||||
|
||||
read_AHRS();
|
||||
|
||||
if (log_bitmask & MASK_LOG_ATTITUDE_FAST)
|
||||
Log_Write_Attitude((int)roll_sensor, (int)pitch_sensor, (uint16_t)yaw_sensor);
|
||||
|
||||
if (log_bitmask & MASK_LOG_RAW)
|
||||
Log_Write_Raw();
|
||||
#endif
|
||||
|
||||
// altitude smoothing
|
||||
// ------------------
|
||||
if (control_mode != FLY_BY_WIRE_B)
|
||||
calc_altitude_error();
|
||||
|
||||
// custom code/exceptions for flight modes
|
||||
// ---------------------------------------
|
||||
update_current_flight_mode();
|
||||
|
||||
// apply desired roll, pitch and yaw to the plane
|
||||
// ----------------------------------------------
|
||||
if (control_mode > MANUAL)
|
||||
stabilize();
|
||||
|
||||
// write out the servo PWM values
|
||||
// ------------------------------
|
||||
set_servos_4();
|
||||
|
||||
#if ENABLE_HIL
|
||||
output_HIL();
|
||||
#endif
|
||||
|
||||
#if GCS_PROTOCOL == GCS_PROTOCOL_DEBUGTERMINAL
|
||||
readgcsinput();
|
||||
#endif
|
||||
}
|
||||
|
||||
void medium_loop()
|
||||
{
|
||||
// This is the start of the medium (10 Hz) loop pieces
|
||||
// -----------------------------------------
|
||||
switch(medium_loopCounter) {
|
||||
|
||||
// This case deals with the GPS
|
||||
//-------------------------------
|
||||
case 0:
|
||||
medium_loopCounter++;
|
||||
update_GPS();
|
||||
|
||||
#if MAGNETOMETER == 1
|
||||
APM_Compass.Read(); // Read magnetometer
|
||||
APM_Compass.Calculate(roll,pitch); // Calculate heading
|
||||
#endif
|
||||
|
||||
break;
|
||||
|
||||
// This case performs some navigation computations
|
||||
//------------------------------------------------
|
||||
case 1:
|
||||
medium_loopCounter++;
|
||||
|
||||
if(GPS.new_data){
|
||||
dTnav = millis() - medium_loopTimer;
|
||||
medium_loopTimer = millis();
|
||||
}
|
||||
|
||||
// calculate the plane's desired bearing
|
||||
// -------------------------------------
|
||||
navigate();
|
||||
break;
|
||||
|
||||
// unused? no, jeti gets updated :-)
|
||||
//------------------------------
|
||||
case 2:
|
||||
medium_loopCounter++;
|
||||
|
||||
// perform next command
|
||||
// --------------------
|
||||
update_commands();
|
||||
jeti_update();
|
||||
break;
|
||||
|
||||
// This case deals with sending high rate telemetry
|
||||
//-------------------------------------------------
|
||||
case 3:
|
||||
medium_loopCounter++;
|
||||
|
||||
if ((log_bitmask & MASK_LOG_ATTITUDE_MED) && !(log_bitmask & MASK_LOG_ATTITUDE_FAST))
|
||||
Log_Write_Attitude((int)roll_sensor, (int)pitch_sensor, (uint16_t)yaw_sensor);
|
||||
|
||||
if (log_bitmask & MASK_LOG_CTUN)
|
||||
Log_Write_Control_Tuning();
|
||||
|
||||
if (log_bitmask & MASK_LOG_NTUN)
|
||||
Log_Write_Nav_Tuning();
|
||||
|
||||
if (log_bitmask & MASK_LOG_GPS)
|
||||
Log_Write_GPS(GPS.time, current_loc.lat, current_loc.lng, GPS.altitude, current_loc.alt, (long) GPS.ground_speed, GPS.ground_course, GPS.fix, GPS.num_sats);
|
||||
|
||||
send_message(MSG_ATTITUDE); // Sends attitude data
|
||||
break;
|
||||
|
||||
// This case controls the slow loop
|
||||
//---------------------------------
|
||||
case 4:
|
||||
medium_loopCounter=0;
|
||||
slow_loop();
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
void slow_loop()
|
||||
{
|
||||
// This is the slow (3 1/3 Hz) loop pieces
|
||||
//----------------------------------------
|
||||
switch (slow_loopCounter){
|
||||
case 0:
|
||||
slow_loopCounter++;
|
||||
superslow_loopCounter++;
|
||||
if(superslow_loopCounter >=15) {
|
||||
// keep track of what page is in use in the log
|
||||
// *** We need to come up with a better scheme to handle this...
|
||||
eeprom_write_word((uint16_t *) EE_LAST_LOG_PAGE, DataFlash.GetWritePage());
|
||||
superslow_loopCounter = 0;
|
||||
}
|
||||
break;
|
||||
|
||||
case 1:
|
||||
slow_loopCounter++;
|
||||
|
||||
// Read 3-position switch on radio
|
||||
// -------------------------------
|
||||
read_control_switch();
|
||||
|
||||
// Read Control Surfaces/Mix switches
|
||||
// ----------------------------------
|
||||
if(reverse_switch){
|
||||
update_servo_switches();
|
||||
}
|
||||
|
||||
// Read main battery voltage if hooked up - does not read the 5v from radio
|
||||
// ------------------------------------------------------------------------
|
||||
#if BATTERY_EVENT == 1
|
||||
read_battery();
|
||||
#endif
|
||||
|
||||
// Read Baro pressure
|
||||
// ------------------
|
||||
read_airpressure();
|
||||
break;
|
||||
|
||||
case 2:
|
||||
slow_loopCounter = 0;
|
||||
update_events();
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
void update_GPS(void)
|
||||
{
|
||||
GPS.update();
|
||||
update_GPS_light();
|
||||
|
||||
if (GPS.new_data && GPS.fix) {
|
||||
GPS_timer = millis();
|
||||
send_message(MSG_LOCATION);
|
||||
|
||||
// for performance
|
||||
// ---------------
|
||||
gps_fix_count++;
|
||||
|
||||
if(ground_start_count > 1){
|
||||
ground_start_count--;
|
||||
ground_start_avg += GPS.ground_speed;
|
||||
|
||||
} else if (ground_start_count == 1) {
|
||||
// We countdown N number of good GPS fixes
|
||||
// so that the altitude is more accurate
|
||||
// -------------------------------------
|
||||
if (current_loc.lat == 0) {
|
||||
SendDebugln("!! bad loc");
|
||||
ground_start_count = 5;
|
||||
|
||||
} else {
|
||||
if(ENABLE_AIR_START == 1 && (ground_start_avg / 5) < SPEEDFILT) {
|
||||
startup_ground();
|
||||
|
||||
if (log_bitmask & MASK_LOG_CMD)
|
||||
Log_Write_Startup(TYPE_GROUNDSTART_MSG);
|
||||
|
||||
init_home();
|
||||
} else if (ENABLE_AIR_START == 0) {
|
||||
init_home();
|
||||
}
|
||||
|
||||
ground_start_count = 0;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
current_loc.lng = GPS.longitude; // Lon * 10**7
|
||||
current_loc.lat = GPS.latitude; // Lat * 10**7
|
||||
|
||||
#if GPS_PROTOCOL == GPS_PROTOCOL_IMU
|
||||
current_loc.alt = GPS.altitude;
|
||||
#else
|
||||
current_loc.alt = ((1 - altitude_mix) * GPS.altitude) + (altitude_mix * press_alt); // alt_MSL centimeters (meters * 100)
|
||||
#endif
|
||||
|
||||
COGX = cos(ToRad(GPS.ground_course/100.0));
|
||||
COGY = sin(ToRad(GPS.ground_course/100.0));
|
||||
}
|
||||
}
|
||||
|
||||
void update_current_flight_mode(void)
|
||||
{
|
||||
if(control_mode == AUTO){
|
||||
crash_checker();
|
||||
|
||||
switch(command_must_ID){
|
||||
case CMD_TAKEOFF:
|
||||
#if USE_MAGNETOMETER == ENABLED
|
||||
calc_nav_roll();
|
||||
#else
|
||||
nav_roll = 0;
|
||||
#endif
|
||||
float scaler;
|
||||
#if AIRSPEED_SENSOR == 1
|
||||
scaler = (float)airspeed / (float)airspeed_cruise;
|
||||
nav_pitch = scaler * takeoff_pitch * 50;
|
||||
#else
|
||||
scaler = (float)GPS.ground_speed / (float)airspeed_cruise;
|
||||
nav_pitch = scaler * takeoff_pitch * 50;
|
||||
#endif
|
||||
|
||||
nav_pitch = constrain(nav_pitch, 0l, (long)takeoff_pitch);
|
||||
|
||||
servo_out[CH_THROTTLE] = throttle_max; //TODO: Replace with THROTTLE_TAKEOFF or other method of controlling throttle
|
||||
// ******************************
|
||||
|
||||
// override pitch_sensor to limit porpoising
|
||||
// pitch_sensor = constrain(pitch_sensor, -6000, takeoff_pitch * 100);
|
||||
// throttle = passthrough
|
||||
break;
|
||||
|
||||
case CMD_LAND:
|
||||
calc_nav_roll();
|
||||
|
||||
#if AIRSPEED_SENSOR == 1
|
||||
calc_nav_pitch();
|
||||
calc_throttle();
|
||||
#else
|
||||
calc_nav_pitch(); // calculate nav_pitch just to use for calc_throttle
|
||||
calc_throttle(); // throttle basedtest landing_pitch; // pitch held constant
|
||||
#endif
|
||||
|
||||
if (land_complete) {
|
||||
servo_out[CH_THROTTLE] = 0;
|
||||
}
|
||||
break;
|
||||
|
||||
default:
|
||||
hold_course = -1;
|
||||
calc_nav_roll();
|
||||
calc_nav_pitch();
|
||||
calc_throttle();
|
||||
break;
|
||||
}
|
||||
}else{
|
||||
switch(control_mode){
|
||||
case RTL:
|
||||
case LOITER:
|
||||
hold_course = -1;
|
||||
crash_checker();
|
||||
calc_nav_roll();
|
||||
calc_nav_pitch();
|
||||
calc_throttle();
|
||||
break;
|
||||
|
||||
case FLY_BY_WIRE_A:
|
||||
// fake Navigation output using sticks
|
||||
nav_roll = ((radio_in[CH_ROLL] - radio_trim[CH_ROLL]) * head_max * reverse_roll) / 350;
|
||||
nav_pitch = ((radio_in[CH_PITCH] - radio_trim[CH_PITCH]) * 3500l * reverse_pitch) / 350;
|
||||
|
||||
nav_roll = constrain(nav_roll, -head_max, head_max);
|
||||
nav_pitch = constrain(nav_pitch, -3000, 3000); // trying to give more pitch authority
|
||||
break;
|
||||
|
||||
case FLY_BY_WIRE_B:
|
||||
// fake Navigation output using sticks
|
||||
// We use pitch_min because its magnitude is normally greater than pitch_max
|
||||
nav_roll = ((radio_in[CH_ROLL] - radio_trim[CH_ROLL]) * head_max * reverse_roll) / 350;
|
||||
altitude_error = ((radio_in[CH_PITCH] - radio_trim[CH_PITCH]) * pitch_min * -reverse_pitch) / 350;
|
||||
nav_roll = constrain(nav_roll, -head_max, head_max);
|
||||
|
||||
#if AIRSPEED_SENSOR == 1
|
||||
//airspeed_error = ((AIRSPEED_FBW_MAX - AIRSPEED_FBW_MIN) * servo_out[CH_THROTTLE] / 100) + AIRSPEED_FBW_MIN;
|
||||
airspeed_error = ((int)(airspeed_fbw_max - airspeed_fbw_min) * servo_out[CH_THROTTLE]) + ((int)airspeed_fbw_min * 100);
|
||||
// Intermediate calculation - airspeed_error is just desired airspeed at this point
|
||||
airspeed_energy_error = (long)(((long)airspeed_error * (long)airspeed_error) - ((long)airspeed * (long)airspeed))/20000; //Changed 0.00005f * to / 20000 to avoid floating point calculation
|
||||
airspeed_error = (airspeed_error - airspeed);
|
||||
|
||||
#endif
|
||||
|
||||
calc_throttle();
|
||||
calc_nav_pitch();
|
||||
break;
|
||||
|
||||
case STABILIZE:
|
||||
nav_roll = 0;
|
||||
nav_pitch = 0;
|
||||
// throttle is passthrough
|
||||
break;
|
||||
|
||||
case CIRCLE:
|
||||
// we have no GPS installed and have lost radio contact
|
||||
// or we just want to fly around in a gentle circle w/o GPS
|
||||
// ----------------------------------------------------
|
||||
nav_roll = head_max / 3;
|
||||
nav_pitch = 0;
|
||||
|
||||
if (failsafe == true){
|
||||
servo_out[CH_THROTTLE] = throttle_cruise;
|
||||
}
|
||||
break;
|
||||
|
||||
case MANUAL:
|
||||
// servo_out is for Sim control only
|
||||
// ---------------------------------
|
||||
servo_out[CH_ROLL] = reverse_roll * (radio_in[CH_ROLL] - radio_trim[CH_ROLL]) * 9;
|
||||
servo_out[CH_PITCH] = reverse_pitch * (radio_in[CH_PITCH] - radio_trim[CH_PITCH]) * 9;
|
||||
servo_out[CH_RUDDER] = reverse_rudder * (radio_in[CH_RUDDER] - radio_trim[CH_RUDDER]) * 9;
|
||||
break;
|
||||
//roll: -13788.000, pitch: -13698.000, thr: 0.000, rud: -13742.000
|
||||
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void read_AHRS(void)
|
||||
{
|
||||
// Get gyro and accel data and perform IMU calculations
|
||||
//-----------------------------------------------------
|
||||
Read_adc_raw(); // Get current values for IMU sensors
|
||||
Matrix_update(); // Integrate the DCM matrix
|
||||
Normalize(); // Normalize the DCM matrix
|
||||
Drift_correction(); // Perform drift correction
|
||||
Euler_angles(); // Calculate pitch, roll, yaw for stabilization and navigation
|
||||
}
|
||||
|
||||
|
||||
|
||||
|
@ -0,0 +1,343 @@
|
||||
//****************************************************************
|
||||
// Function that controls aileron/rudder, elevator, rudder (if 4 channel control) and throttle to produce desired attitude and airspeed.
|
||||
//****************************************************************
|
||||
|
||||
void stabilize()
|
||||
{
|
||||
float ch1_inf = 1.0;
|
||||
float ch2_inf = 1.0;
|
||||
float ch4_inf = 1.0;
|
||||
|
||||
if(crash_timer > 0){
|
||||
nav_roll = 0;
|
||||
}
|
||||
|
||||
// For Testing Only
|
||||
// roll_sensor = (radio_in[CH_RUDDER] - radio_trim[CH_RUDDER]) * 10;
|
||||
// Serial.print(" roll_sensor ");
|
||||
// Serial.print(roll_sensor,DEC);
|
||||
|
||||
// Calculate dersired servo output for the roll
|
||||
// ---------------------------------------------
|
||||
servo_out[CH_ROLL] = PID((nav_roll - roll_sensor), deltaMiliSeconds, CASE_SERVO_ROLL, 1.0);
|
||||
servo_out[CH_PITCH] = PID((nav_pitch + abs(roll_sensor * kff[CASE_PITCH_COMP]) - pitch_sensor), deltaMiliSeconds, CASE_SERVO_PITCH, 1.0);
|
||||
//Serial.print(" servo_out[CH_ROLL] ");
|
||||
//Serial.print(servo_out[CH_ROLL],DEC);
|
||||
|
||||
// Mix Stick input to allow users to override control surfaces
|
||||
// -----------------------------------------------------------
|
||||
if ((control_mode < FLY_BY_WIRE_A) || (ENABLE_STICK_MIXING == 1 && control_mode > FLY_BY_WIRE_B)) {
|
||||
|
||||
ch1_inf = (float)radio_in[CH_ROLL] - (float)radio_trim[CH_ROLL];
|
||||
ch1_inf = abs(ch1_inf);
|
||||
ch1_inf = min(ch1_inf, 400.0);
|
||||
ch1_inf = ((400.0 - ch1_inf) /400.0);
|
||||
|
||||
ch2_inf = (float)radio_in[CH_PITCH] - radio_trim[CH_PITCH];
|
||||
ch2_inf = abs(ch2_inf);
|
||||
ch2_inf = min(ch2_inf, 400.0);
|
||||
ch2_inf = ((400.0 - ch2_inf) /400.0);
|
||||
|
||||
// scale the sensor input based on the stick input
|
||||
// -----------------------------------------------
|
||||
servo_out[CH_ROLL] *= ch1_inf;
|
||||
servo_out[CH_PITCH] *= ch2_inf;
|
||||
|
||||
// Mix in stick inputs
|
||||
// -------------------
|
||||
servo_out[CH_ROLL] += reverse_roll * (radio_in[CH_ROLL] - radio_trim[CH_ROLL]) * 9;
|
||||
servo_out[CH_PITCH] += reverse_pitch * (radio_in[CH_PITCH] - radio_trim[CH_PITCH]) * 9;
|
||||
|
||||
//Serial.print(" servo_out[CH_ROLL] ");
|
||||
//Serial.println(servo_out[CH_ROLL],DEC);
|
||||
}
|
||||
|
||||
// stick mixing performed for rudder for all cases including FBW unless disabled for higher modes
|
||||
// important for steering on the ground during landing
|
||||
// -----------------------------------------------
|
||||
if (control_mode <= FLY_BY_WIRE_B || ENABLE_STICK_MIXING == 1) {
|
||||
ch4_inf = (float)radio_in[CH_RUDDER] - (float)radio_trim[CH_RUDDER];
|
||||
ch4_inf = abs(ch4_inf);
|
||||
ch4_inf = min(ch4_inf, 400.0);
|
||||
ch4_inf = ((400.0 - ch4_inf) /400.0);
|
||||
}
|
||||
|
||||
// Apply output to Rudder
|
||||
// ----------------------
|
||||
calc_nav_yaw();
|
||||
servo_out[CH_RUDDER] *= ch4_inf;
|
||||
servo_out[CH_RUDDER] += reverse_rudder * (radio_in[CH_RUDDER] - radio_trim[CH_RUDDER]) * 9;
|
||||
|
||||
// Call slew rate limiter if used
|
||||
// ------------------------------
|
||||
//#if(ROLL_SLEW_LIMIT != 0)
|
||||
// servo_out[CH_ROLL] = roll_slew_limit(servo_out[CH_ROLL]);
|
||||
//#endif
|
||||
}
|
||||
|
||||
void crash_checker()
|
||||
{
|
||||
if(pitch_sensor < -4500){
|
||||
crash_timer = 255;
|
||||
}
|
||||
if(crash_timer > 0)
|
||||
crash_timer--;
|
||||
}
|
||||
|
||||
|
||||
#if AIRSPEED_SENSOR == 0
|
||||
void calc_throttle()
|
||||
{
|
||||
// no airspeed sensor, we use nav pitch to determin the proper throttle output
|
||||
// AUTO, RTL, etc
|
||||
// ---------------------------------------------------------------------------
|
||||
if (nav_pitch >= 0) {
|
||||
servo_out[CH_THROTTLE] = throttle_cruise + (THROTTLE_MAX - throttle_cruise) * nav_pitch / pitch_max;
|
||||
} else {
|
||||
servo_out[CH_THROTTLE] = throttle_cruise - (throttle_cruise - THROTTLE_MIN) * nav_pitch / pitch_min;
|
||||
}
|
||||
servo_out[CH_THROTTLE] = max(servo_out[CH_THROTTLE], 0);
|
||||
|
||||
// are we going too slow? up the throttle to get some more groundspeed
|
||||
// move into PID loop in the future
|
||||
// lower the contstant value to limit the effect : 50 = default
|
||||
int gs_boost = 30 * (1.0 - ((float)GPS.ground_speed / (float)airspeed_cruise));
|
||||
gs_boost = max(0, gs_boost);
|
||||
servo_out[CH_THROTTLE] += gs_boost;
|
||||
servo_out[CH_THROTTLE] = constrain(servo_out[CH_THROTTLE], THROTTLE_MIN, THROTTLE_MAX);
|
||||
}
|
||||
#endif
|
||||
|
||||
#if AIRSPEED_SENSOR == 1
|
||||
void calc_throttle()
|
||||
{
|
||||
// throttle control with airspeed compensation
|
||||
// -------------------------------------------
|
||||
energy_error = airspeed_energy_error + (float)altitude_error * 0.098f;
|
||||
|
||||
// positive energy errors make the throttle go higher
|
||||
servo_out[CH_THROTTLE] = throttle_cruise + PID(energy_error, dTnav, CASE_TE_THROTTLE, 1.0);
|
||||
servo_out[CH_THROTTLE] = max(servo_out[CH_THROTTLE], 0);
|
||||
|
||||
// are we going too slow? up the throttle to get some more groundspeed
|
||||
// move into PID loop in the future
|
||||
// lower the contstant value to limit the effect : 50 = default
|
||||
|
||||
//JASON - We really should change this to act on airspeed in this case. Let's visit about it....
|
||||
/*int gs_boost = 30 * (1.0 - ((float)GPS.ground_speed / (float)airspeed_cruise));
|
||||
gs_boost = max(0,gs_boost);
|
||||
servo_out[CH_THROTTLE] += gs_boost;*/
|
||||
servo_out[CH_THROTTLE] = constrain(servo_out[CH_THROTTLE], THROTTLE_MIN, THROTTLE_MAX);
|
||||
}
|
||||
#endif
|
||||
|
||||
|
||||
|
||||
/*****************************************
|
||||
* Calculate desired roll angle (in medium freq loop)
|
||||
*****************************************/
|
||||
// Yaw is separated into a function for future implementation of heading hold on take-off
|
||||
// ----------------------------------------------------------------------------------------
|
||||
void calc_nav_yaw()
|
||||
{
|
||||
// read_adc(4) is y axis accel;
|
||||
// Control is a feedforward from the aileron control + a PID to coordinate the turn (drive y axis accel to zero)
|
||||
servo_out[CH_RUDDER] = kff[CASE_RUDDER_MIX] * servo_out[CH_ROLL] + PID((long)read_adc(4), deltaMiliSeconds, CASE_SERVO_RUDDER, 1.0);
|
||||
|
||||
}
|
||||
|
||||
|
||||
#if AIRSPEED_SENSOR == 1
|
||||
void calc_nav_pitch()
|
||||
{
|
||||
// Calculate the Pitch of the plane
|
||||
// --------------------------------
|
||||
nav_pitch = -PID(airspeed_error, dTnav, CASE_NAV_PITCH_ASP, 1.0);
|
||||
nav_pitch = constrain(nav_pitch, pitch_min, pitch_max);
|
||||
}
|
||||
#endif
|
||||
|
||||
#if AIRSPEED_SENSOR == 0
|
||||
void calc_nav_pitch()
|
||||
{
|
||||
// Calculate the Pitch of the plane
|
||||
// --------------------------------
|
||||
nav_pitch = PID(altitude_error, dTnav, CASE_NAV_PITCH_ALT, 1.0);
|
||||
nav_pitch = constrain(nav_pitch, pitch_min, pitch_max);
|
||||
}
|
||||
#endif
|
||||
|
||||
|
||||
void calc_nav_roll()
|
||||
{
|
||||
|
||||
// Adjust gain based on ground speed - We need lower nav gain going in to a headwind, etc.
|
||||
// This does not make provisions for wind speed in excess of airframe speed
|
||||
nav_gain_scaler = (float)GPS.ground_speed / (float)(AIRSPEED_CRUISE * 100);
|
||||
nav_gain_scaler = constrain(nav_gain_scaler, 0.2, 1.4);
|
||||
|
||||
// Jason -> I will implement the servo gain scaler some time, but it is independant of this scaling.
|
||||
// Doug to implement the high speed servo gain scale
|
||||
// use head max to limit turns, make a var
|
||||
|
||||
// negative error = left turn
|
||||
// positive error = right turn
|
||||
// Calculate the required roll of the plane
|
||||
// ----------------------------------------
|
||||
nav_roll = PID(bearing_error, dTnav, CASE_NAV_ROLL, nav_gain_scaler); //returns desired bank angle in degrees*100
|
||||
nav_roll = constrain(nav_roll,-head_max, head_max);
|
||||
|
||||
}
|
||||
|
||||
|
||||
/*****************************************
|
||||
* Roll servo slew limit
|
||||
*****************************************/
|
||||
/*
|
||||
float roll_slew_limit(float servo)
|
||||
{
|
||||
static float last;
|
||||
float temp = constrain(servo, last-ROLL_SLEW_LIMIT * deltaMiliSeconds/1000.f, last + ROLL_SLEW_LIMIT * deltaMiliSeconds/1000.f);
|
||||
last = servo;
|
||||
return temp;
|
||||
}*/
|
||||
|
||||
/*****************************************
|
||||
* Throttle slew limit
|
||||
*****************************************/
|
||||
/*float throttle_slew_limit(float throttle)
|
||||
{
|
||||
static float last;
|
||||
float temp = constrain(throttle, last-THROTTLE_SLEW_LIMIT * deltaMiliSeconds/1000.f, last + THROTTLE_SLEW_LIMIT * deltaMiliSeconds/1000.f);
|
||||
last = throttle;
|
||||
return temp;
|
||||
}
|
||||
*/
|
||||
|
||||
/*****************************************
|
||||
* Proportional Integrator Derivative Control
|
||||
*****************************************/
|
||||
|
||||
float PID(long PID_error, long dt, int PID_case, float scaler)
|
||||
{
|
||||
// PID_case is used to keep track of which PID controller is being used - e.g. PID_servo_out[CH_ROLL]
|
||||
float output, derivative;
|
||||
|
||||
derivative = 1000.0f * (float)(PID_error - last_error[PID_case]) / (float)dt;
|
||||
last_error[PID_case] = PID_error;
|
||||
output = (kp[PID_case] * scaler * (float)PID_error); // Compute proportional component
|
||||
//Positive error produces positive output
|
||||
|
||||
integrator[PID_case] += (float)PID_error * ki[PID_case] * scaler * (float)dt / 1000.0f;
|
||||
integrator[PID_case] = constrain(integrator[PID_case], -1.0*(float)integrator_max[PID_case], (float)integrator_max[PID_case]);
|
||||
|
||||
output += integrator[PID_case]; // Add the integral component
|
||||
output += kd[PID_case] * scaler * derivative; // Add the derivative component
|
||||
return output;
|
||||
}
|
||||
|
||||
// Zeros out navigation Integrators if we are changing mode, have passed a waypoint, etc.
|
||||
// Keeps outdated data out of our calculations
|
||||
void reset_I(void)
|
||||
{
|
||||
integrator[CASE_NAV_ROLL] = 0;
|
||||
integrator[CASE_NAV_PITCH_ASP] = 0;
|
||||
integrator[CASE_NAV_PITCH_ALT] = 0;
|
||||
integrator[CASE_TE_THROTTLE] = 0;
|
||||
integrator[CASE_ALT_THROTTLE] = 0;
|
||||
|
||||
last_error[CASE_NAV_ROLL] = 0;
|
||||
last_error[CASE_NAV_PITCH_ASP] = 0;
|
||||
last_error[CASE_NAV_PITCH_ALT] = 0;
|
||||
last_error[CASE_TE_THROTTLE] = 0;
|
||||
last_error[CASE_ALT_THROTTLE] = 0;
|
||||
}
|
||||
|
||||
/*****************************************
|
||||
* Set the flight control servos based on the current calculated values
|
||||
*****************************************/
|
||||
void set_servos_4(void)
|
||||
{
|
||||
if(control_mode == MANUAL){
|
||||
// do a direct pass through of radio values
|
||||
for(int y=0; y<4; y++) {
|
||||
radio_out[y] = radio_in[y];
|
||||
}
|
||||
|
||||
} else {
|
||||
if (mix_mode == 0){
|
||||
radio_out[CH_ROLL] = radio_trim[CH_ROLL] + (reverse_roll * ((float)servo_out[CH_ROLL] * 0.11111));
|
||||
radio_out[CH_PITCH] = radio_trim[CH_PITCH] + (reverse_pitch * ((float)servo_out[CH_PITCH] * 0.11111));
|
||||
radio_out[CH_RUDDER] = radio_trim[CH_RUDDER] + (reverse_rudder * ((float)servo_out[CH_RUDDER] * 0.11111));
|
||||
}else{
|
||||
/*Elevon mode*/
|
||||
float ch1;
|
||||
float ch2;
|
||||
ch1 = reverse_elevons * (servo_out[CH_PITCH] - servo_out[CH_ROLL]);
|
||||
ch2 = servo_out[CH_PITCH] + servo_out[CH_ROLL];
|
||||
radio_out[CH_ROLL] = elevon1_trim + (reverse_ch1_elevon * (ch1 * 0.11111));
|
||||
radio_out[CH_PITCH] = elevon2_trim + (reverse_ch2_elevon * (ch2 * 0.11111));
|
||||
}
|
||||
|
||||
#if THROTTLE_OUT == 0
|
||||
radio_out[CH_THROTTLE] = 0;
|
||||
#endif
|
||||
|
||||
|
||||
// convert 0 to 100% into PWM
|
||||
servo_out[CH_THROTTLE] = constrain(servo_out[CH_THROTTLE], 0, 100);
|
||||
radio_out[CH_THROTTLE] = servo_out[CH_THROTTLE] * ((radio_max[CH_THROTTLE] - radio_min[CH_THROTTLE]) / 100);
|
||||
radio_out[CH_THROTTLE] += radio_min[CH_THROTTLE];
|
||||
|
||||
//Radio_in: 1763 PWM output: 2000 Throttle: 78.7695999145 PWM Min: 1110 PWM Max: 1938
|
||||
|
||||
#if THROTTLE_REVERSE == 1
|
||||
radio_out[CH_THROTTLE] = radio_max[CH_THROTTLE] + radio_min[CH_THROTTLE] - radio_out[CH_THROTTLE];
|
||||
#endif
|
||||
}
|
||||
|
||||
// send values to the PWM timers for output
|
||||
// ----------------------------------------
|
||||
for(int y=0; y<4; y++) {
|
||||
//radio_out[y] = constrain(radio_out[y], radio_min[y], radio_max[y]);
|
||||
APM_RC.OutputCh(y, radio_out[y]); // send to Servos
|
||||
//Serial.print(radio_out[y],DEC);
|
||||
//Serial.print(", ");
|
||||
}
|
||||
//Serial.println(" ");
|
||||
}
|
||||
|
||||
void demo_servos(byte i) {
|
||||
|
||||
while(i > 0){
|
||||
send_message(SEVERITY_LOW,"Demo Servos!");
|
||||
APM_RC.OutputCh(1, 1400);
|
||||
delay(400);
|
||||
APM_RC.OutputCh(1, 1600);
|
||||
delay(200);
|
||||
APM_RC.OutputCh(1, 1500);
|
||||
delay(400);
|
||||
i--;
|
||||
}
|
||||
}
|
||||
|
||||
int readOutputCh(unsigned char ch)
|
||||
{
|
||||
int pwm;
|
||||
switch(ch)
|
||||
{
|
||||
case 0: pwm = OCR5B; break; // ch0
|
||||
case 1: pwm = OCR5C; break; // ch1
|
||||
case 2: pwm = OCR1B; break; // ch2
|
||||
case 3: pwm = OCR1C; break; // ch3
|
||||
case 4: pwm = OCR4C; break; // ch4
|
||||
case 5: pwm = OCR4B; break; // ch5
|
||||
case 6: pwm = OCR3C; break; // ch6
|
||||
case 7: pwm = OCR3B; break; // ch7
|
||||
case 8: pwm = OCR5A; break; // ch8, PL3
|
||||
case 9: pwm = OCR1A; break; // ch9, PB5
|
||||
case 10: pwm = OCR3A; break; // ch10, PE3
|
||||
}
|
||||
pwm >>= 1; // pwm / 2;
|
||||
return pwm;
|
||||
}
|
387
Tools/ArduPPM/WorkBasket/Jeti Duplex/X-DIY_Jeti_V2/X-DIY/DCM.pde
Normal file
387
Tools/ArduPPM/WorkBasket/Jeti Duplex/X-DIY_Jeti_V2/X-DIY/DCM.pde
Normal file
@ -0,0 +1,387 @@
|
||||
|
||||
// Read the 6 ADC channels needed for the IMU
|
||||
// -----------------
|
||||
void Read_adc_raw(void)
|
||||
{
|
||||
int tc_temp = APM_ADC.Ch(GYRO_TEMP_CH);
|
||||
for (int i = 0; i < 6; i++) {
|
||||
AN[i] = APM_ADC.Ch(sensors[i]);
|
||||
if (i < 3) {
|
||||
AN[i] -= gyro_temp_comp(i, tc_temp); // Subtract temp compensated typical gyro bias
|
||||
} else {
|
||||
AN[i] -= 2025; // Subtract typical accel bias
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// Returns the temperature compensated raw gyro value
|
||||
//---------------------------------------------------
|
||||
float gyro_temp_comp(int i, int temp)
|
||||
{
|
||||
// We use a 2nd order curve of the form Gtc = A + B * Graw + C * (Graw)**2
|
||||
//------------------------------------------------------------------------
|
||||
return GTC[i][0] + GTC[i][1] * temp + GTC[i][2] * temp * temp;
|
||||
}
|
||||
|
||||
// Returns an analog value with the offset removed
|
||||
// -----------------
|
||||
float read_adc(int select)
|
||||
{
|
||||
float temp;
|
||||
if (SENSOR_SIGN[select] < 0)
|
||||
temp = (AN_OFFSET[select]-AN[select]);
|
||||
else
|
||||
temp = (AN[select]-AN_OFFSET[select]);
|
||||
|
||||
if (abs(temp) > ADC_CONSTRAINT)
|
||||
adc_constraints++; // We keep track of the number of times we constrain the ADC output for performance reporting
|
||||
|
||||
/*
|
||||
// For checking the pitch/roll drift correction gain time constants
|
||||
switch (select) {
|
||||
case 3:
|
||||
return 0;
|
||||
break;
|
||||
case 4:
|
||||
return 0;
|
||||
break;
|
||||
case 5:
|
||||
return 400;
|
||||
break;
|
||||
}
|
||||
*/
|
||||
|
||||
|
||||
//End of drift correction gain test code
|
||||
|
||||
return constrain(temp, -ADC_CONSTRAINT, ADC_CONSTRAINT); // Throw out nonsensical values
|
||||
}
|
||||
|
||||
/**************************************************/
|
||||
void Normalize(void)
|
||||
{
|
||||
float error = 0;
|
||||
float temporary[3][3];
|
||||
float renorm = 0;
|
||||
boolean problem = FALSE;
|
||||
|
||||
error= -Vector_Dot_Product(&DCM_Matrix[0][0], &DCM_Matrix[1][0]) * .5; // eq.19
|
||||
|
||||
Vector_Scale(&temporary[0][0], &DCM_Matrix[1][0], error); // eq.19
|
||||
Vector_Scale(&temporary[1][0], &DCM_Matrix[0][0], error); // eq.19
|
||||
|
||||
Vector_Add(&temporary[0][0], &temporary[0][0], &DCM_Matrix[0][0]); //eq.19
|
||||
Vector_Add(&temporary[1][0], &temporary[1][0], &DCM_Matrix[1][0]); //eq.19
|
||||
|
||||
Vector_Cross_Product(&temporary[2][0], &temporary[0][0], &temporary[1][0]); // c= a x b // eq.20
|
||||
|
||||
renorm = Vector_Dot_Product(&temporary[0][0], &temporary[0][0]);
|
||||
if (renorm < 1.5625f && renorm > 0.64f) { // Check if we are OK with Taylor expansion
|
||||
renorm = .5 * (3 - renorm); // eq.21
|
||||
} else if (renorm < 100.0f && renorm > 0.01f) {
|
||||
renorm = 1. / sqrt(renorm);
|
||||
renorm_sqrt_count++;
|
||||
} else {
|
||||
problem = TRUE;
|
||||
renorm_blowup_count++;
|
||||
}
|
||||
Vector_Scale(&DCM_Matrix[0][0], &temporary[0][0], renorm);
|
||||
|
||||
renorm = Vector_Dot_Product(&temporary[1][0], &temporary[1][0]);
|
||||
if (renorm < 1.5625f && renorm > 0.64f) { // Check if we are OK with Taylor expansion
|
||||
renorm = .5 * (3 - renorm); // eq.21
|
||||
} else if (renorm < 100.0f && renorm > 0.01f) {
|
||||
renorm = 1. / sqrt(renorm);
|
||||
renorm_sqrt_count++;
|
||||
} else {
|
||||
problem = TRUE;
|
||||
renorm_blowup_count++;
|
||||
}
|
||||
Vector_Scale(&DCM_Matrix[1][0], &temporary[1][0], renorm);
|
||||
|
||||
renorm = Vector_Dot_Product(&temporary[2][0], &temporary[2][0]);
|
||||
if (renorm < 1.5625f && renorm > 0.64f) { // Check if we are OK with Taylor expansion
|
||||
renorm = .5 * (3 - renorm); // eq.21
|
||||
} else if (renorm < 100.0f && renorm > 0.01f) {
|
||||
renorm = 1. / sqrt(renorm);
|
||||
renorm_sqrt_count++;
|
||||
} else {
|
||||
problem = TRUE;
|
||||
renorm_blowup_count++;
|
||||
}
|
||||
Vector_Scale(&DCM_Matrix[2][0], &temporary[2][0], renorm);
|
||||
|
||||
if (problem) { // Our solution is blowing up and we will force back to initial condition. Hope we are not upside down!
|
||||
DCM_Matrix[0][0] = 1.0f;
|
||||
DCM_Matrix[0][1] = 0.0f;
|
||||
DCM_Matrix[0][2] = 0.0f;
|
||||
DCM_Matrix[1][0] = 0.0f;
|
||||
DCM_Matrix[1][1] = 1.0f;
|
||||
DCM_Matrix[1][2] = 0.0f;
|
||||
DCM_Matrix[2][0] = 0.0f;
|
||||
DCM_Matrix[2][1] = 0.0f;
|
||||
DCM_Matrix[2][2] = 1.0f;
|
||||
problem = FALSE;
|
||||
}
|
||||
}
|
||||
|
||||
/**************************************************/
|
||||
void Drift_correction(void)
|
||||
{
|
||||
//Compensation the Roll, Pitch and Yaw drift.
|
||||
float mag_heading_x;
|
||||
float mag_heading_y;
|
||||
float errorCourse = 0;
|
||||
static float Scaled_Omega_P[3];
|
||||
static float Scaled_Omega_I[3];
|
||||
float Accel_magnitude;
|
||||
float Accel_weight;
|
||||
float Integrator_magnitude;
|
||||
|
||||
//*****Roll and Pitch***************
|
||||
|
||||
// Calculate the magnitude of the accelerometer vector
|
||||
Accel_magnitude = sqrt(Accel_Vector[0] * Accel_Vector[0] + Accel_Vector[1] * Accel_Vector[1] + Accel_Vector[2] * Accel_Vector[2]);
|
||||
Accel_magnitude = Accel_magnitude / GRAVITY; // Scale to gravity.
|
||||
|
||||
// Dynamic weighting of accelerometer info (reliability filter)
|
||||
// Weight for accelerometer info (<0.5G = 0.0, 1G = 1.0 , >1.5G = 0.0)
|
||||
Accel_weight = constrain(1 - 2 * abs(1 - Accel_magnitude), 0, 1); //
|
||||
|
||||
// We monitor the amount that the accelerometer based drift correction is deweighted for performanc reporting
|
||||
imu_health = imu_health + 0.02 * (Accel_weight-.5);
|
||||
imu_health = constrain(imu_health, 0, 1);
|
||||
|
||||
Vector_Cross_Product(&errorRollPitch[0], &Accel_Vector[0], &DCM_Matrix[2][0]); // adjust the ground of reference
|
||||
// errorRollPitch are in Accel ADC units
|
||||
// Limit max errorRollPitch to limit max Omega_P and Omega_I
|
||||
errorRollPitch[0] = constrain(errorRollPitch[0], -50, 50);
|
||||
errorRollPitch[1] = constrain(errorRollPitch[1], -50, 50);
|
||||
errorRollPitch[2] = constrain(errorRollPitch[2], -50, 50);
|
||||
|
||||
Vector_Scale(&Omega_P[0], &errorRollPitch[0], Kp_ROLLPITCH * Accel_weight);
|
||||
|
||||
Vector_Scale(&Scaled_Omega_I[0], &errorRollPitch[0], Ki_ROLLPITCH * Accel_weight);
|
||||
Vector_Add(Omega_I, Omega_I, Scaled_Omega_I);
|
||||
|
||||
//*****YAW***************
|
||||
|
||||
#if MAGNETOMETER == ENABLED
|
||||
// We make the gyro YAW drift correction based on compass magnetic heading
|
||||
errorCourse= (DCM_Matrix[0][0] * APM_Compass.Heading_Y) - (DCM_Matrix[1][0] * APM_Compass.Heading_X); // Calculating YAW error
|
||||
#else // Use GPS Ground course to correct yaw gyro drift=
|
||||
if(GPS.ground_speed >= SPEEDFILT){
|
||||
// Optimization: We have precalculated COGX and COGY (Course over Ground X and Y) from GPS info
|
||||
errorCourse = (DCM_Matrix[0][0] * COGY) - (DCM_Matrix[1][0] * COGX); // Calculating YAW error
|
||||
}
|
||||
#endif
|
||||
|
||||
Vector_Scale(errorYaw, &DCM_Matrix[2][0], errorCourse); // Applys the yaw correction to the XYZ rotation of the aircraft, depeding the position.
|
||||
|
||||
Vector_Scale(&Scaled_Omega_P[0], &errorYaw[0], Kp_YAW);
|
||||
Vector_Add(Omega_P, Omega_P, Scaled_Omega_P); //Adding Proportional.
|
||||
|
||||
Vector_Scale(&Scaled_Omega_I[0], &errorYaw[0], Ki_YAW);
|
||||
Vector_Add(Omega_I, Omega_I, Scaled_Omega_I); //adding integrator to the Omega_I
|
||||
|
||||
|
||||
// Here we will place a limit on the integrator so that the integrator cannot ever exceed half the saturation limit of the gyros
|
||||
Integrator_magnitude = sqrt(Vector_Dot_Product(Omega_I, Omega_I));
|
||||
if (Integrator_magnitude > ToRad(300)) {
|
||||
Vector_Scale(Omega_I, Omega_I, 0.5f * ToRad(300) / Integrator_magnitude);
|
||||
}
|
||||
}
|
||||
|
||||
/**************************************************/
|
||||
void Accel_adjust(void)
|
||||
{
|
||||
Accel_Vector[1] += Accel_Scale((GPS.ground_speed / 100) * Omega[2]); // Centrifugal force on Acc_y = GPS_speed * GyroZ
|
||||
Accel_Vector[2] -= Accel_Scale((GPS.ground_speed / 100) * Omega[1]); // Centrifugal force on Acc_z = GPS_speed * GyroY
|
||||
}
|
||||
|
||||
|
||||
/**************************************************/
|
||||
void Matrix_update(void)
|
||||
{
|
||||
Gyro_Vector[0] = Gyro_Scaled_X(read_adc(0)); // gyro x roll
|
||||
Gyro_Vector[1] = Gyro_Scaled_Y(read_adc(1)); // gyro y pitch
|
||||
Gyro_Vector[2] = Gyro_Scaled_Z(read_adc(2)); // gyro Z yaw
|
||||
|
||||
//Record when you saturate any of the gyros.
|
||||
if((abs(Gyro_Vector[0]) >= ToRad(300)) || (abs(Gyro_Vector[1]) >= ToRad(300)) || (abs(Gyro_Vector[2]) >= ToRad(300)))
|
||||
gyro_sat_count++;
|
||||
|
||||
/*
|
||||
Serial.print (AN[0]);
|
||||
Serial.print (" ");
|
||||
Serial.print (AN_OFFSET[0]);
|
||||
Serial.print (" ");
|
||||
Serial.print (Gyro_Vector[0]);
|
||||
Serial.print (" ");
|
||||
Serial.print (AN[1]);
|
||||
Serial.print (" ");
|
||||
Serial.print (AN_OFFSET[1]);
|
||||
Serial.print (" ");
|
||||
Serial.print (Gyro_Vector[1]);
|
||||
Serial.print (" ");
|
||||
Serial.print (AN[2]);
|
||||
Serial.print (" ");
|
||||
Serial.print (AN_OFFSET[2]);
|
||||
Serial.print (" ");
|
||||
Serial.println (Gyro_Vector[2]);
|
||||
*/
|
||||
|
||||
// Accel_Vector[0]=read_adc(3); // acc x
|
||||
// Accel_Vector[1]=read_adc(4); // acc y
|
||||
// Accel_Vector[2]=read_adc(5); // acc z
|
||||
// Low pass filter on accelerometer data (to filter vibrations)
|
||||
Accel_Vector[0] = Accel_Vector[0] * 0.6 + (float)read_adc(3) * 0.4; // acc x
|
||||
Accel_Vector[1] = Accel_Vector[1] * 0.6 + (float)read_adc(4) * 0.4; // acc y
|
||||
Accel_Vector[2] = Accel_Vector[2] * 0.6 + (float)read_adc(5) * 0.4; // acc z
|
||||
|
||||
|
||||
Vector_Add(&Omega[0], &Gyro_Vector[0], &Omega_I[0]); // adding proportional term
|
||||
Vector_Add(&Omega_Vector[0], &Omega[0], &Omega_P[0]); // adding Integrator term
|
||||
|
||||
Accel_adjust(); // Remove centrifugal acceleration.
|
||||
|
||||
#if OUTPUTMODE == 1
|
||||
Update_Matrix[0][0] = 0;
|
||||
Update_Matrix[0][1] = -G_Dt * Omega_Vector[2]; // -z
|
||||
Update_Matrix[0][2] = G_Dt * Omega_Vector[1]; // y
|
||||
Update_Matrix[1][0] = G_Dt * Omega_Vector[2]; // z
|
||||
Update_Matrix[1][1] = 0;
|
||||
Update_Matrix[1][2] = -G_Dt * Omega_Vector[0]; // -x
|
||||
Update_Matrix[2][0] = -G_Dt * Omega_Vector[1]; // -y
|
||||
Update_Matrix[2][1] = G_Dt * Omega_Vector[0]; // x
|
||||
Update_Matrix[2][2] = 0;
|
||||
#else // Uncorrected data (no drift correction)
|
||||
Update_Matrix[0][0] = 0;
|
||||
Update_Matrix[0][1] = -G_Dt * Gyro_Vector[2]; // -z
|
||||
Update_Matrix[0][2] = G_Dt * Gyro_Vector[1]; // y
|
||||
Update_Matrix[1][0] = G_Dt * Gyro_Vector[2]; // z
|
||||
Update_Matrix[1][1] = 0;
|
||||
Update_Matrix[1][2] = -G_Dt * Gyro_Vector[0];
|
||||
Update_Matrix[2][0] = -G_Dt * Gyro_Vector[1];
|
||||
Update_Matrix[2][1] = G_Dt * Gyro_Vector[0];
|
||||
Update_Matrix[2][2] = 0;
|
||||
#endif
|
||||
|
||||
Matrix_Multiply(DCM_Matrix, Update_Matrix, Temporary_Matrix); // a * b = c
|
||||
|
||||
for(int x = 0; x < 3; x++){ // Matrix Addition (update)
|
||||
for(int y = 0; y < 3; y++){
|
||||
DCM_Matrix[x][y] += Temporary_Matrix[x][y];
|
||||
}
|
||||
}
|
||||
|
||||
/*
|
||||
Serial.print (G_Dt * 1000);
|
||||
Serial.print (" ");
|
||||
Serial.print (DCM_Matrix[0][0]);
|
||||
Serial.print (" ");
|
||||
Serial.print (DCM_Matrix[0][1]);
|
||||
Serial.print (" ");
|
||||
Serial.print (DCM_Matrix[0][2]);
|
||||
Serial.print (" ");
|
||||
Serial.print (DCM_Matrix[1][0]);
|
||||
Serial.print (" ");
|
||||
Serial.print (DCM_Matrix[1][1]);
|
||||
Serial.print (" ");
|
||||
Serial.print (DCM_Matrix[1][2]);
|
||||
Serial.print (" ");
|
||||
Serial.print (DCM_Matrix[2][0]);
|
||||
Serial.print (" ");
|
||||
Serial.print (DCM_Matrix[2][1]);
|
||||
Serial.print (" ");
|
||||
Serial.println (DCM_Matrix[2][2]);
|
||||
*/
|
||||
}
|
||||
|
||||
/**************************************************/
|
||||
void Euler_angles(void)
|
||||
{
|
||||
#if (OUTPUTMODE == 2) // Only accelerometer info (debugging purposes)
|
||||
roll = atan2(Accel_Vector[1], Accel_Vector[2]); // atan2(acc_y, acc_z)
|
||||
pitch = -asin((Accel_Vector[0]) / (double)GRAVITY); // asin(acc_x)
|
||||
yaw = 0;
|
||||
roll_sensor = ToDeg(roll) * 100;
|
||||
pitch_sensor = ToDeg(pitch) * 100;
|
||||
#else
|
||||
pitch = -asin(DCM_Matrix[2][0]);
|
||||
roll = atan2(DCM_Matrix[2][1], DCM_Matrix[2][2]);
|
||||
yaw = atan2(DCM_Matrix[1][0], DCM_Matrix[0][0]);
|
||||
pitch_sensor = ToDeg(pitch) * 100;
|
||||
roll_sensor = ToDeg(roll) * 100;
|
||||
yaw_sensor = ToDeg(yaw) * 100;
|
||||
if (yaw_sensor < 0) yaw_sensor += 36000;
|
||||
#endif
|
||||
|
||||
/*
|
||||
Serial.print ("Roll ");
|
||||
Serial.print (roll_sensor / 100);
|
||||
Serial.print (", Pitch ");
|
||||
Serial.print (pitch_sensor / 100);
|
||||
Serial.print (", Yaw ");
|
||||
Serial.println (yaw_sensor / 100);
|
||||
*/
|
||||
}
|
||||
|
||||
/**************************************************/
|
||||
//Computes the dot product of two vectors
|
||||
float Vector_Dot_Product(float vector1[3], float vector2[3])
|
||||
{
|
||||
float op = 0;
|
||||
|
||||
for(int c = 0; c < 3; c++)
|
||||
{
|
||||
op += vector1[c] * vector2[c];
|
||||
}
|
||||
|
||||
return op;
|
||||
}
|
||||
|
||||
/**************************************************/
|
||||
//Computes the cross product of two vectors
|
||||
void Vector_Cross_Product(float vectorOut[3], float v1[3], float v2[3])
|
||||
{
|
||||
vectorOut[0]= (v1[1] * v2[2]) - (v1[2] * v2[1]);
|
||||
vectorOut[1]= (v1[2] * v2[0]) - (v1[0] * v2[2]);
|
||||
vectorOut[2]= (v1[0] * v2[1]) - (v1[1] * v2[0]);
|
||||
}
|
||||
|
||||
/**************************************************/
|
||||
//Multiply the vector by a scalar.
|
||||
void Vector_Scale(float vectorOut[3], float vectorIn[3], float scale2)
|
||||
{
|
||||
for(int c = 0; c < 3; c++)
|
||||
{
|
||||
vectorOut[c] = vectorIn[c] * scale2;
|
||||
}
|
||||
}
|
||||
|
||||
/**************************************************/
|
||||
void Vector_Add(float vectorOut[3], float vectorIn1[3], float vectorIn2[3])
|
||||
{
|
||||
for(int c = 0; c < 3; c++)
|
||||
{
|
||||
vectorOut[c] = vectorIn1[c]+vectorIn2[c];
|
||||
}
|
||||
}
|
||||
|
||||
/********* MATRIX FUNCTIONS *****************************************/
|
||||
//Multiply two 3x3 matrixs. This function developed by Jordi can be easily adapted to multiple n*n matrix's. (Pero me da flojera!).
|
||||
void Matrix_Multiply(float a[3][3], float b[3][3], float mat[3][3])
|
||||
{
|
||||
float op[3];
|
||||
for(int x = 0; x < 3; x++){
|
||||
for(int y = 0; y < 3; y++){
|
||||
for(int w = 0; w < 3; w++){
|
||||
op[w] = a[x][w] * b[w][y];
|
||||
}
|
||||
mat[x][y] = 0;
|
||||
mat[x][y] = op[0] + op[1] + op[2];
|
||||
float test = mat[x][y];
|
||||
}
|
||||
}
|
||||
}
|
@ -0,0 +1,389 @@
|
||||
0 0000 16 int radio_trim 1
|
||||
1 0001 ..
|
||||
2 0002 16 int radio_trim 2
|
||||
3 0003 ..
|
||||
4 0004 16 int radio_trim 3
|
||||
5 0005 ..
|
||||
6 0006 16 int radio_trim 4
|
||||
7 0007 ..
|
||||
8 0008 16 int radio_trim 5
|
||||
9 0009 ..
|
||||
10 000A 16 int radio_trim 6
|
||||
11 000B ..
|
||||
12 000C 16 int radio_trim 7
|
||||
13 000D ..
|
||||
14 000E 16 int radio_trim 8
|
||||
15 000F ..
|
||||
16 0010 16 int radio_min 1
|
||||
17 0011 ..
|
||||
18 0012 16 int radio_min 2
|
||||
19 0013 ..
|
||||
20 0014 16 int radio_min 3
|
||||
21 0015 ..
|
||||
22 0016 16 int radio_min 4
|
||||
23 0017 ..
|
||||
24 0018 16 int radio_min 5
|
||||
25 0019 ..
|
||||
26 001A 16 int radio_min 6
|
||||
27 001B ..
|
||||
28 001C 16 int radio_min 7
|
||||
29 001D ..
|
||||
30 001E 16 int radio_min 8
|
||||
31 001F ..
|
||||
32 0020 16 int radio_max 1
|
||||
33 0021 ..
|
||||
34 0022 16 int radio_max 2
|
||||
35 0023 ..
|
||||
36 0024 16 int radio_max 3
|
||||
37 0025 ..
|
||||
38 0026 16 int radio_max 4
|
||||
39 0027 ..
|
||||
40 0028 16 int radio_max 5
|
||||
41 0029 ..
|
||||
42 002A 16 int radio_max 6
|
||||
43 002B ..
|
||||
44 002C 16 int radio_max 7
|
||||
45 002D ..
|
||||
46 002E 16 int radio_max 8
|
||||
47 002F ..
|
||||
48 0030 16 int elevon_trim 1
|
||||
49 0031 ..
|
||||
50 0032 16 int elevon_trim 2
|
||||
51 0033 ..
|
||||
52 0034 16 int XTRACK_GAIN
|
||||
53 0035 ..
|
||||
54 0036 16 int XTRACK_ENTRY_ANGLE
|
||||
55 0037 ..
|
||||
56 0038 8 int HEAD_MAX
|
||||
57 0039 8 int PITCH_MAX
|
||||
58 003A 8 int Pitch_min
|
||||
59 003B 32 float EE_ALT_MIX
|
||||
60 003C ..
|
||||
61 003D ..
|
||||
62 003E ..
|
||||
63 003F
|
||||
64 0040 32 float Kp 0
|
||||
65 0041 ..
|
||||
66 0042 ..
|
||||
67 0043 ..
|
||||
68 0044 32 float Kp 1
|
||||
69 0045 ..
|
||||
70 0046 ..
|
||||
71 0047 ..
|
||||
72 0048 32 float Kp 2
|
||||
73 0049 ..
|
||||
74 004A ..
|
||||
75 004B ..
|
||||
76 004C 32 float Kp 3
|
||||
77 004D ..
|
||||
78 004E ..
|
||||
79 004F ..
|
||||
80 0050 32 float Kp 4
|
||||
81 0051 ..
|
||||
82 0052 ..
|
||||
83 0053 ..
|
||||
84 0054 32 float Kp 5
|
||||
85 0055 ..
|
||||
86 0056 ..
|
||||
87 0057 ..
|
||||
88 0058 32 float Kp 6
|
||||
89 0059 ..
|
||||
90 005A ..
|
||||
91 005B ..
|
||||
92 005C 32 float Kp 7
|
||||
93 005D ..
|
||||
94 005E ..
|
||||
95 005F ..
|
||||
96 0060 32 float Ki 0
|
||||
97 0061 ..
|
||||
98 0062 ..
|
||||
99 0063 ..
|
||||
100 0064 32 float Ki 1
|
||||
101 0065 ..
|
||||
102 0066 ..
|
||||
103 0067 ..
|
||||
104 0068 32 float Ki 2
|
||||
105 0069 ..
|
||||
106 006A ..
|
||||
107 006B ..
|
||||
108 006C 32 float Ki 3
|
||||
109 006D ..
|
||||
110 006E ..
|
||||
111 006F ..
|
||||
112 0070 32 float Ki 4
|
||||
113 0071 ..
|
||||
114 0072 ..
|
||||
115 0073 ..
|
||||
116 0074 32 float Ki 5
|
||||
117 0075 ..
|
||||
118 0076 ..
|
||||
119 0077 ..
|
||||
120 0078 32 float Ki 6
|
||||
121 0079 ..
|
||||
122 007A ..
|
||||
123 007B ..
|
||||
124 007C 32 float Ki 7
|
||||
125 007D ..
|
||||
126 007E ..
|
||||
127 007F ..
|
||||
128 0080 32 float kd 0
|
||||
129 0081 ..
|
||||
130 0082 ..
|
||||
131 0083 ..
|
||||
132 0084 32 float kd 1
|
||||
133 0085 ..
|
||||
134 0086 ..
|
||||
135 0087 ..
|
||||
136 0088 32 float kd 2
|
||||
137 0089 ..
|
||||
138 008A ..
|
||||
139 008B ..
|
||||
140 008C 32 float kd 3
|
||||
141 008D ..
|
||||
142 008E ..
|
||||
143 008F ..
|
||||
144 0090 32 float kd 4
|
||||
145 0091 ..
|
||||
146 0092 ..
|
||||
147 0093 ..
|
||||
148 0094 32 float kd 5
|
||||
149 0095 ..
|
||||
150 0096 ..
|
||||
151 0097 ..
|
||||
152 0098 32 float kd 6
|
||||
153 0099 ..
|
||||
154 009A ..
|
||||
155 009B ..
|
||||
156 009C 32 float kd 7
|
||||
157 009D ..
|
||||
158 009E ..
|
||||
159 009F ..
|
||||
160 00A0 32 float integrator_max 0
|
||||
161 00A1 ..
|
||||
162 00A2 ..
|
||||
163 00A3 ..
|
||||
164 00A4 32 float integrator_max 1
|
||||
165 00A5 ..
|
||||
166 00A6 ..
|
||||
167 00A7 ..
|
||||
168 00A8 32 float integrator_max 2
|
||||
169 00A9 ..
|
||||
170 00AA ..
|
||||
171 00AB ..
|
||||
172 00AC 32 float integrator_max 3
|
||||
173 00AD ..
|
||||
174 00AE ..
|
||||
175 00AF ..
|
||||
176 00B0 32 float integrator_max 4
|
||||
177 00B1 ..
|
||||
178 00B2 ..
|
||||
179 00B3 ..
|
||||
180 00B4 32 float integrator_max 5
|
||||
181 00B5 ..
|
||||
182 00B6 ..
|
||||
183 00B7 ..
|
||||
184 00B8 32 float integrator_max 6
|
||||
185 00B9 ..
|
||||
186 00BA ..
|
||||
187 00BB ..
|
||||
188 00BC 32 float integrator_max 7
|
||||
189 00BD ..
|
||||
190 00BE ..
|
||||
191 00BF ..
|
||||
192 00C0 32 float Kff 0
|
||||
193 00C1 ..
|
||||
194 00C2 ..
|
||||
195 00C3 ..
|
||||
196 00C4 32 float Kff 1
|
||||
197 00C5 ..
|
||||
198 00C6 ..
|
||||
199 00C7 ..
|
||||
200 00C8 32 float Kff 2
|
||||
201 00C9 ..
|
||||
202 00CA ..
|
||||
203 00CB ..
|
||||
204 00CC 32 float Kff 3
|
||||
205 00CD ..
|
||||
206 00CE ..
|
||||
207 00CF ..
|
||||
208 00D0 32 float Kff 4
|
||||
209 00D1 ..
|
||||
210 00D2 ..
|
||||
211 00D3 ..
|
||||
212 00D4 32 float Kff 5
|
||||
213 00D5 ..
|
||||
214 00D6 ..
|
||||
215 00D7 ..
|
||||
216 00D8 32 float Kff 6
|
||||
217 00D9 ..
|
||||
218 00DA ..
|
||||
219 00DB ..
|
||||
220 00DC 32 float Kff 7
|
||||
221 00DD ..
|
||||
222 00DE ..
|
||||
223 00DF ..
|
||||
224 00E0 32 int EE_ACCEL_OFFSET 0
|
||||
225 00E1 ..
|
||||
226 00E2 ..
|
||||
227 00E3 ..
|
||||
228 00E4 32 int EE_ACCEL_OFFSET 1
|
||||
229 00E5 ..
|
||||
230 00E6 ..
|
||||
231 00E7 ..
|
||||
232 00E8 32 int EE_ACCEL_OFFSET 2
|
||||
233 00E9 ..
|
||||
234 00EA ..
|
||||
235 00EB ..
|
||||
236 00EC 32 int EE_ACCEL_OFFSET 3
|
||||
237 00ED ..
|
||||
238 00EE ..
|
||||
239 00EF ..
|
||||
240 00F0 32 int EE_ACCEL_OFFSET 4
|
||||
241 00F1 ..
|
||||
242 00F2 ..
|
||||
243 00F3 ..
|
||||
244 00F4 32 int EE_ACCEL_OFFSET 5
|
||||
245 00F5 ..
|
||||
246 00F6 ..
|
||||
247 00F7 ..
|
||||
248 00F8 8 EE_CONFIG
|
||||
249 00F9 8 EE_WP_MODE
|
||||
250 00FA 8 EE_YAW_MODE
|
||||
251 00FB 8 EE_WP_TOTAL 0x106
|
||||
252 00FC 8 EE_WP_INDEX 0x107
|
||||
253 00FD 8 EE_WP_RADIUS 0x108
|
||||
254 00FE 8 EE_LOITER_RADIUS 0x109
|
||||
255 00FF 32 EE_ALT_HOLD_HOME 0x10A
|
||||
256 0100 ..
|
||||
257 0101 ..
|
||||
258 0102 ..
|
||||
259 0103 8 AIRSPEED_CRUISE
|
||||
260 0104 32 AIRSPEED_RATIO
|
||||
261 0105 ..
|
||||
262 0106 ..
|
||||
263 0107 ..
|
||||
264 0108 AIRSPEED_FBW_MIN
|
||||
265 0109 AIRSPEED_FBW_MAX
|
||||
266 010A 8 THROTTLE_MIN
|
||||
267 010B 8 THROTTLE_CRUISE
|
||||
268 010C 8 THROTTLE_MAX
|
||||
269 010D 8 THROTTLE_FAILSAFE
|
||||
270 010E 16 THROTTLE_FS_VALUE
|
||||
271 010F ..
|
||||
272 0110 8 THROTTLE_FAILSAFE_ACTION
|
||||
273 0111
|
||||
274 0112 8 FLIGHT_MODE_CHANNEL
|
||||
275 0113 8 AUTO_TRIM
|
||||
276 0114 uint16 LOGging Bitmask
|
||||
277 0115 ..
|
||||
278 0116 uint32 EE_ABS_PRESS_GND
|
||||
279 0117 ..
|
||||
280 0118 ..
|
||||
281 0119 ..
|
||||
282 011A 16 int EE_GND_TEMP
|
||||
283 011B ..
|
||||
284 011C 16 int EE_GND_ALT
|
||||
285 011D ..
|
||||
286 011E 16 int EE_AP_OFFSET
|
||||
287 011F ..
|
||||
288 0120 8 byte EE_SWITCH_ENABLE
|
||||
289 0121 8 ARRAY FLIGHT_MODES_1
|
||||
290 0122 8 ARRAY FLIGHT_MODES_2
|
||||
291 0123 8 ARRAY FLIGHT_MODES_3
|
||||
292 0124 8 ARRAY FLIGHT_MODES_4
|
||||
293 0125 8 ARRAY FLIGHT_MODES_5
|
||||
294 0126 8 ARRAY FLIGHT_MODES_6
|
||||
295 0127
|
||||
296 0128
|
||||
297 0129
|
||||
298 012A
|
||||
299 012B
|
||||
300 012C
|
||||
301 012D
|
||||
302 012E
|
||||
303 012F (0x130 is the Start Byte for commands)
|
||||
304 0130 byte Command id (Home) 0
|
||||
305 0131 byte Parameter 1
|
||||
306 0132 long Altitude, Parameter 2
|
||||
307 0133 ..
|
||||
308 0134 ..
|
||||
309 0135 ..
|
||||
310 0136 long Latitude, Parameter 3
|
||||
311 0137 ..
|
||||
312 0138 ..
|
||||
313 0139 ..
|
||||
314 013A long Longitude, Parameter 3
|
||||
315 013B ..
|
||||
316 013C ..
|
||||
317 013D byte Command id (WP 1) 1
|
||||
318 013E byte Parameter 1
|
||||
319 013F long Altitude, Parameter 2
|
||||
320 0140 ..
|
||||
321 0141 ..
|
||||
322 0142 ..
|
||||
323 0143 long Latitude, Parameter 3
|
||||
324 0144 ..
|
||||
325 0145 ..
|
||||
326 0146 ..
|
||||
327 0147 long Longitude, Parameter 3
|
||||
328 0148 ..
|
||||
329 0149 ..
|
||||
330 014A ..
|
||||
331 014B
|
||||
332 014C
|
||||
333 014D
|
||||
334 014E
|
||||
335 014F
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
**********************************************************************************************
|
||||
SECTION 6 - LOGS
|
||||
|
||||
3584 0xE00 int Last log page used
|
||||
3585 0xE01 ..
|
||||
3587 0xE02 byte last log number
|
||||
3588 0xE03 unused
|
||||
3589 0xE04 int Log 1 start page
|
||||
3590 0xE05 ..
|
||||
3591 0xE06 int Log 2 start page
|
||||
3592 0xE07 ..
|
||||
3593 0xE08 int Log 3 start page
|
||||
3594 0xE09 ..
|
||||
3595 0xE0A int Log 4 start page
|
||||
3596 0xE0B ..
|
||||
3597 0xE0C int Log 5 start page
|
||||
3598 0xE0D ..
|
||||
3599 0xE0E int Log 6 start page
|
||||
3600 0xE0F ..
|
||||
3601 0xE10 int Log 7 start page
|
||||
3602 0xE11 ..
|
||||
3603 0xE12 int Log 8 start page
|
||||
3604 0xE13 ..
|
||||
3605 0xE14 int Log 9 start page
|
||||
3606 0xE15 ..
|
||||
3607 0xE16 int Log 10 start page
|
||||
3608 0xE17 ..
|
||||
3609 0xE18 int Log 11 start page
|
||||
3610 0xE19 ..
|
||||
3611 0xE1A int Log 12 start page
|
||||
3612 0xE1B ..
|
||||
3613 0xE1C int Log 13 start page
|
||||
3614 0xE1D ..
|
||||
3615 0xE1E int Log 14 start page
|
||||
3616 0xE1F ..
|
||||
3617 0xE20 int Log 15 start page
|
||||
3618 0xE21 ..
|
||||
3619 0xE22 int Log 16 start page
|
||||
3620 0xE23 ..
|
||||
3621 0xE24 int Log 17 start page
|
||||
3622 0xE25 ..
|
||||
3623 0xE26 int Log 18 start page
|
||||
3624 0xE27 ..
|
||||
3625 0xE28 int Log 19 start page
|
||||
3626 0xE29 ..
|
||||
|
||||
|
||||
|
@ -0,0 +1,286 @@
|
||||
// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: t -*-
|
||||
// ************************************************************************************
|
||||
// This function gets critical data from EEPROM to get us underway if restarting in air
|
||||
// ************************************************************************************
|
||||
void read_EEPROM_startup(void)
|
||||
{
|
||||
read_EEPROM_gains();
|
||||
read_EEPROM_radio_minmax(); // read Radio limits
|
||||
read_EEPROM_trims(); // read Radio trims
|
||||
read_user_configs();
|
||||
read_EEPROM_waypoint_info();
|
||||
}
|
||||
|
||||
void read_EEPROM_airstart_critical(void)
|
||||
{
|
||||
int16_t temp = 0;
|
||||
read_EEPROM_IMU_offsets();
|
||||
|
||||
// For debugging only
|
||||
/*
|
||||
Serial.print ("Offsets "); Serial.print (AN_OFFSET[0]);
|
||||
Serial.print (" "); Serial.print (AN_OFFSET[1]);
|
||||
Serial.print (" "); Serial.print (AN_OFFSET[2]);
|
||||
Serial.print (" "); Serial.print (AN_OFFSET[3]);
|
||||
Serial.print (" "); Serial.print (AN_OFFSET[4]);
|
||||
Serial.print (" "); Serial.println (AN_OFFSET[5]);
|
||||
*/
|
||||
|
||||
// Read the home location
|
||||
//-----------------------
|
||||
home = get_wp_with_index(0);
|
||||
|
||||
// Read pressure sensor values
|
||||
//----------------------------
|
||||
read_pressure_data();
|
||||
}
|
||||
|
||||
void save_EEPROM_groundstart(void)
|
||||
{
|
||||
save_EEPROM_trims();
|
||||
save_EEPROM_IMU_offsets();
|
||||
// pressure sensor data saved by init_home
|
||||
}
|
||||
|
||||
|
||||
/********************************************************************************/
|
||||
long read_alt_to_hold()
|
||||
{
|
||||
byte options = eeprom_read_byte((byte *) EE_CONFIG);
|
||||
|
||||
// save the alitude above home option
|
||||
if(options & HOLD_ALT_ABOVE_HOME){
|
||||
int32_t temp = eeprom_read_dword((const uint32_t *) EE_ALT_HOLD_HOME);
|
||||
return temp + home.alt;
|
||||
}else{
|
||||
return current_loc.alt;
|
||||
}
|
||||
}
|
||||
|
||||
long save_alt_to_hold(int32_t alt_to_hold)
|
||||
{
|
||||
byte options = eeprom_read_byte((byte *) EE_CONFIG);
|
||||
|
||||
// save the alitude above home option
|
||||
if(options & HOLD_ALT_ABOVE_HOME)
|
||||
eeprom_write_block((const void *)&alt_to_hold, (void *)EE_ALT_HOLD_HOME, sizeof (alt_to_hold));
|
||||
}
|
||||
|
||||
|
||||
/********************************************************************************/
|
||||
void read_EEPROM_waypoint_info(void)
|
||||
{
|
||||
wp_total = eeprom_read_byte((uint8_t *) EE_WP_TOTAL);
|
||||
wp_radius = eeprom_read_byte((uint8_t *) EE_WP_RADIUS);
|
||||
loiter_radius = eeprom_read_byte((uint8_t *) EE_LOITER_RADIUS);
|
||||
}
|
||||
|
||||
void save_EEPROM_waypoint_info(void)
|
||||
{
|
||||
eeprom_write_byte((uint8_t *) EE_WP_TOTAL, wp_total);
|
||||
eeprom_write_byte((uint8_t *) EE_WP_RADIUS, wp_radius);
|
||||
eeprom_write_byte((uint8_t *) EE_LOITER_RADIUS, loiter_radius);
|
||||
}
|
||||
|
||||
/********************************************************************************/
|
||||
void read_EEPROM_gains(void)
|
||||
{
|
||||
eeprom_read_block((void*)&kp, (const void*)EE_KP, sizeof(kp));
|
||||
eeprom_read_block((void*)&ki, (const void*)EE_KI, sizeof(ki));
|
||||
eeprom_read_block((void*)&kd, (const void*)EE_KD, sizeof(kd));
|
||||
eeprom_read_block((void*)&integrator_max, (const void*)EE_IMAX, sizeof(integrator_max));
|
||||
eeprom_read_block((void*)&kff, (const void*)EE_KFF, sizeof(kff));
|
||||
|
||||
// stored as degree * 100
|
||||
x_track_gain = eeprom_read_word((uint16_t *) EE_XTRACK_GAIN);
|
||||
|
||||
// stored as degrees
|
||||
x_track_angle = eeprom_read_word((uint16_t *) EE_XTRACK_ANGLE) * 100;
|
||||
|
||||
// stored as degrees
|
||||
head_max = eeprom_read_byte((uint8_t *) EE_HEAD_MAX) * 100; // scale to degress * 100
|
||||
pitch_max = eeprom_read_byte((uint8_t *) EE_PITCH_MAX) * 100; // scale to degress * 100
|
||||
pitch_min = -eeprom_read_byte((uint8_t *) EE_PITCH_MIN) * 100; // scale to degress * 100
|
||||
|
||||
// stored as a float
|
||||
eeprom_read_block((void*)&altitude_mix, (const void*)EE_ALT_MIX, sizeof(altitude_mix));
|
||||
}
|
||||
|
||||
void save_EEPROM_gains(void)
|
||||
{
|
||||
eeprom_write_block((const void *)&kp, (void *)EE_KP, sizeof (kp));
|
||||
eeprom_write_block((const void *)&ki, (void *)EE_KI, sizeof (ki));
|
||||
eeprom_write_block((const void *)&kd, (void *)EE_KD, sizeof (kd));
|
||||
eeprom_write_block((const void *)&integrator_max, (void *)EE_IMAX, sizeof (integrator_max));
|
||||
eeprom_write_block((const void *)&kff, (void *)EE_KFF, sizeof (kff));
|
||||
|
||||
// stored as degree * 100
|
||||
eeprom_write_word((uint16_t *) EE_XTRACK_GAIN, x_track_gain);
|
||||
|
||||
// stored as degrees
|
||||
eeprom_write_word((uint16_t *) EE_XTRACK_ANGLE, x_track_angle / 100);
|
||||
|
||||
// stored as degrees
|
||||
eeprom_write_byte((uint8_t *) EE_HEAD_MAX, head_max / 100);
|
||||
eeprom_write_byte((uint8_t *) EE_PITCH_MAX, pitch_max / 100);
|
||||
eeprom_write_byte((uint8_t *) EE_PITCH_MIN, -pitch_min / 100);
|
||||
|
||||
// stored as a float
|
||||
eeprom_write_block((const void*)&altitude_mix, (void*)EE_ALT_MIX, sizeof(altitude_mix));
|
||||
}
|
||||
|
||||
|
||||
/********************************************************************************/
|
||||
|
||||
void read_EEPROM_trims(void)
|
||||
{
|
||||
// Read Radio trim settings
|
||||
eeprom_read_block((void*)&radio_trim, (const void*)EE_TRIM, sizeof(radio_trim));
|
||||
elevon1_trim = eeprom_read_word((uint16_t *) EE_ELEVON1_TRIM);
|
||||
elevon2_trim = eeprom_read_word((uint16_t *) EE_ELEVON2_TRIM);
|
||||
}
|
||||
|
||||
void save_EEPROM_trims(void)
|
||||
{
|
||||
// Save Radio trim settings
|
||||
eeprom_write_block((const void *)&radio_trim, (void *)EE_TRIM, sizeof(radio_trim));
|
||||
eeprom_write_word((uint16_t *) EE_ELEVON1_TRIM, elevon1_trim);
|
||||
eeprom_write_word((uint16_t *) EE_ELEVON2_TRIM, elevon2_trim);
|
||||
}
|
||||
|
||||
/********************************************************************************/
|
||||
|
||||
void save_EEPROM_IMU_offsets(void)
|
||||
{
|
||||
eeprom_write_block((const void *)&AN_OFFSET, (void *)EE_AN_OFFSET, sizeof (AN_OFFSET));
|
||||
}
|
||||
|
||||
void read_EEPROM_IMU_offsets(void)
|
||||
{
|
||||
eeprom_read_block((void*)&AN_OFFSET, (const void*)EE_AN_OFFSET, sizeof(AN_OFFSET));
|
||||
}
|
||||
|
||||
/********************************************************************************/
|
||||
|
||||
void save_command_index(void)
|
||||
{
|
||||
eeprom_write_byte((uint8_t *) EE_WP_INDEX, command_must_index);
|
||||
}
|
||||
|
||||
void read_command_index(void)
|
||||
{
|
||||
wp_index = command_must_index = eeprom_read_byte((uint8_t *) EE_WP_INDEX);
|
||||
}
|
||||
|
||||
/********************************************************************************/
|
||||
|
||||
void save_pressure_data(void)
|
||||
{
|
||||
eeprom_write_dword((uint32_t *)EE_ABS_PRESS_GND, abs_press_gnd);
|
||||
eeprom_write_word((uint16_t *)EE_GND_TEMP, ground_temperature);
|
||||
eeprom_write_word((uint16_t *)EE_GND_ALT, (ground_alt / 100));
|
||||
|
||||
#if AIRSPEED_SENSOR == 1
|
||||
eeprom_write_word((uint16_t *)EE_AP_OFFSET, airpressure_offset);
|
||||
#endif
|
||||
}
|
||||
|
||||
void read_pressure_data(void)
|
||||
{
|
||||
abs_press_gnd = eeprom_read_dword((uint32_t *) EE_ABS_PRESS_GND);
|
||||
abs_press_filt = abs_press_gnd; // Better than zero for an air start value
|
||||
ground_temperature = eeprom_read_word((uint16_t *) EE_GND_TEMP);
|
||||
ground_alt = eeprom_read_word((uint16_t *) EE_GND_ALT) * 100;
|
||||
|
||||
#if AIRSPEED_SENSOR == 1
|
||||
airpressure_offset = eeprom_read_word((uint16_t *) EE_AP_OFFSET);
|
||||
#endif
|
||||
}
|
||||
|
||||
|
||||
|
||||
/********************************************************************************/
|
||||
|
||||
void read_EEPROM_radio_minmax(void)
|
||||
{
|
||||
// Read Radio min/max settings
|
||||
eeprom_read_block((void*)&radio_max, (const void*)EE_MAX, sizeof(radio_max));
|
||||
eeprom_read_block((void*)&radio_min, (const void*)EE_MIN, sizeof(radio_min));
|
||||
}
|
||||
|
||||
|
||||
void save_EEPROM_radio_minmax(void)
|
||||
{
|
||||
// Save Radio min/max settings
|
||||
eeprom_write_block((const void *)&radio_max, (void *)EE_MAX, sizeof(radio_max));
|
||||
eeprom_write_block((const void *)&radio_min, (void *)EE_MIN, sizeof(radio_min));
|
||||
}
|
||||
|
||||
|
||||
|
||||
/********************************************************************************/
|
||||
|
||||
void read_user_configs(void)
|
||||
{
|
||||
// Read Radio min/max settings
|
||||
airspeed_cruise = eeprom_read_byte((byte *) EE_AIRSPEED_CRUISE)*100;
|
||||
eeprom_read_block((void*)&airspeed_ratio, (const void*)EE_AIRSPEED_RATIO, sizeof(airspeed_ratio));
|
||||
|
||||
throttle_min = eeprom_read_byte((byte *) EE_THROTTLE_MIN);
|
||||
throttle_max = eeprom_read_byte((byte *) EE_THROTTLE_MAX);
|
||||
throttle_cruise = eeprom_read_byte((byte *) EE_THROTTLE_CRUISE);
|
||||
|
||||
airspeed_fbw_min = eeprom_read_byte((byte *) EE_AIRSPEED_FBW_MIN);
|
||||
airspeed_fbw_max = eeprom_read_byte((byte *) EE_AIRSPEED_FBW_MAX);
|
||||
|
||||
throttle_failsafe_enabled = eeprom_read_byte((byte *) EE_THROTTLE_FAILSAFE);
|
||||
throttle_failsafe_action = eeprom_read_byte((byte *) EE_THROTTLE_FAILSAFE_ACTION);
|
||||
throttle_failsafe_value = eeprom_read_word((uint16_t *) EE_THROTTLE_FS_VALUE);
|
||||
|
||||
//flight_mode_channel = eeprom_read_byte((byte *) EE_FLIGHT_MODE_CHANNEL);
|
||||
auto_trim = eeprom_read_byte((byte *) EE_AUTO_TRIM);
|
||||
log_bitmask = eeprom_read_word((uint16_t *) EE_LOG_BITMASK);
|
||||
|
||||
read_EEPROM_flight_modes();
|
||||
reverse_switch = eeprom_read_byte((byte *) EE_REVERSE_SWITCH);
|
||||
}
|
||||
|
||||
|
||||
void save_user_configs(void)
|
||||
{
|
||||
eeprom_write_byte((byte *) EE_AIRSPEED_CRUISE, airspeed_cruise/100);
|
||||
eeprom_write_block((const void *)&airspeed_ratio, (void *)EE_AIRSPEED_RATIO, sizeof(airspeed_ratio));
|
||||
|
||||
eeprom_write_byte((byte *) EE_THROTTLE_MIN, throttle_min);
|
||||
eeprom_write_byte((byte *) EE_THROTTLE_MAX, throttle_max);
|
||||
eeprom_write_byte((byte *) EE_THROTTLE_CRUISE, throttle_cruise);
|
||||
|
||||
eeprom_write_byte((byte *) EE_AIRSPEED_FBW_MIN, airspeed_fbw_min);
|
||||
eeprom_write_byte((byte *) EE_AIRSPEED_FBW_MAX, airspeed_fbw_max);
|
||||
|
||||
eeprom_write_byte((byte *) EE_THROTTLE_FAILSAFE, throttle_failsafe_enabled);
|
||||
eeprom_write_byte((byte *) EE_THROTTLE_FAILSAFE_ACTION,throttle_failsafe_action);
|
||||
eeprom_write_word((uint16_t *) EE_THROTTLE_FS_VALUE, throttle_failsafe_value);
|
||||
|
||||
//eeprom_write_byte((byte *) EE_FLIGHT_MODE_CHANNEL, flight_mode_channel);
|
||||
eeprom_write_byte((byte *) EE_AUTO_TRIM, auto_trim);
|
||||
eeprom_write_word((uint16_t *) EE_LOG_BITMASK, log_bitmask);
|
||||
|
||||
//save_EEPROM_flight_modes();
|
||||
eeprom_write_byte((byte *) EE_REVERSE_SWITCH, reverse_switch);
|
||||
}
|
||||
|
||||
/********************************************************************************/
|
||||
void read_EEPROM_flight_modes(void)
|
||||
{
|
||||
// Read Radio min/max settings
|
||||
eeprom_read_block((void*)&flight_modes, (const void*)EE_FLIGHT_MODES, sizeof(flight_modes));
|
||||
}
|
||||
|
||||
|
||||
void save_EEPROM_flight_modes(void)
|
||||
{
|
||||
// Save Radio min/max settings
|
||||
eeprom_write_block((const void *)&flight_modes, (void *)EE_FLIGHT_MODES, sizeof(flight_modes));
|
||||
}
|
||||
|
@ -0,0 +1,176 @@
|
||||
|
||||
// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: t -*-
|
||||
|
||||
#if GCS_PROTOCOL == GCS_PROTOCOL_LEGACY
|
||||
|
||||
#if GCS_PORT == 3
|
||||
# define SendSer Serial3.print
|
||||
# define SendSerln Serial3.println
|
||||
#else
|
||||
# define SendSer Serial.print
|
||||
# define SendSerln Serial.println
|
||||
#endif
|
||||
|
||||
// This is the standard GCS output file for ArduPilot
|
||||
|
||||
/*
|
||||
Message Prefixes
|
||||
!!! Position Low rate telemetry
|
||||
+++ Attitude High rate telemetry
|
||||
### Mode Change in control mode
|
||||
%%% Waypoint Current and previous waypoints
|
||||
XXX Alert Text alert - NOTE: Alert message generation is not localized to a function
|
||||
PPP IMU Performance Sent every 20 seconds for performance monitoring
|
||||
|
||||
Message Suffix
|
||||
*** All messages use this suffix
|
||||
*/
|
||||
|
||||
/*
|
||||
void acknowledge(byte id, byte check1, byte check2) {}
|
||||
void send_message(byte id) {}
|
||||
void send_message(byte id, long param) {}
|
||||
void send_message(byte severity, const char *str) {}
|
||||
*/
|
||||
|
||||
void acknowledge(byte id, byte check1, byte check2)
|
||||
{
|
||||
}
|
||||
|
||||
void send_message(byte severity, const char *str) // This is the instance of send_message for message 0x05
|
||||
{
|
||||
if(severity >= DEBUG_LEVEL){
|
||||
SendSer("MSG: ");
|
||||
SendSerln(str);
|
||||
}
|
||||
}
|
||||
|
||||
void send_message(byte id)
|
||||
{
|
||||
send_message(id,0l);
|
||||
}
|
||||
|
||||
void send_message(byte id, long param)
|
||||
{
|
||||
switch(id) {
|
||||
case MSG_ATTITUDE: // ** Attitude message
|
||||
print_attitude();
|
||||
break;
|
||||
case MSG_LOCATION: // ** Location/GPS message
|
||||
print_position();
|
||||
break;
|
||||
case MSG_HEARTBEAT: // ** Location/GPS message
|
||||
print_control_mode();
|
||||
break;
|
||||
//case MSG_PERF_REPORT:
|
||||
// printPerfData();
|
||||
}
|
||||
}
|
||||
|
||||
void print_current_waypoints()
|
||||
{
|
||||
}
|
||||
|
||||
void print_attitude(void)
|
||||
{
|
||||
SendSer("+++");
|
||||
SendSer("ASP:");
|
||||
SendSer((int)airspeed / 100, DEC);
|
||||
SendSer(",THH:");
|
||||
SendSer(servo_out[CH_THROTTLE], DEC);
|
||||
SendSer (",RLL:");
|
||||
SendSer(roll_sensor / 100, DEC);
|
||||
SendSer (",PCH:");
|
||||
SendSer(pitch_sensor / 100, DEC);
|
||||
SendSerln(",***");
|
||||
}
|
||||
|
||||
void print_control_mode(void)
|
||||
{
|
||||
switch (control_mode){
|
||||
case MANUAL:
|
||||
SendSerln("###MANUAL***");
|
||||
break;
|
||||
case STABILIZE:
|
||||
SendSerln("###STABILIZE***");
|
||||
break;
|
||||
case CIRCLE:
|
||||
SendSerln("###CIRCLE***");
|
||||
break;
|
||||
case FLY_BY_WIRE_A:
|
||||
SendSerln("###FBW A***");
|
||||
break;
|
||||
case FLY_BY_WIRE_B:
|
||||
SendSerln("###FBW B***");
|
||||
break;
|
||||
case AUTO:
|
||||
SendSerln("###AUTO***");
|
||||
break;
|
||||
case RTL:
|
||||
SendSerln("###RTL***");
|
||||
break;
|
||||
case LOITER:
|
||||
SendSerln("###LOITER***");
|
||||
break;
|
||||
case TAKEOFF:
|
||||
SendSerln("##TAKEOFF***");
|
||||
break;
|
||||
case LAND:
|
||||
SendSerln("##LAND***");
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
void print_position(void)
|
||||
{
|
||||
SendSer("!!!");
|
||||
SendSer("LAT:");
|
||||
SendSer(current_loc.lat/10,DEC);
|
||||
SendSer(",LON:");
|
||||
SendSer(current_loc.lng/10,DEC); //wp_current_lat
|
||||
SendSer(",SPD:");
|
||||
SendSer(GPS.ground_speed/100,DEC);
|
||||
SendSer(",CRT:");
|
||||
SendSer(climb_rate,DEC);
|
||||
SendSer(",ALT:");
|
||||
SendSer(current_loc.alt/100,DEC);
|
||||
SendSer(",ALH:");
|
||||
SendSer(next_WP.alt/100,DEC);
|
||||
SendSer(",CRS:");
|
||||
SendSer(yaw_sensor/100,DEC);
|
||||
SendSer(",BER:");
|
||||
SendSer(target_bearing/100,DEC);
|
||||
SendSer(",WPN:");
|
||||
SendSer(wp_index,DEC);//Actually is the waypoint.
|
||||
SendSer(",DST:");
|
||||
SendSer(wp_distance,DEC);
|
||||
SendSer(",BTV:");
|
||||
SendSer(battery_voltage,DEC);
|
||||
SendSer(",RSP:");
|
||||
SendSer(servo_out[CH_ROLL]/100,DEC);
|
||||
SendSer(",TOW:");
|
||||
SendSer(GPS.time);
|
||||
SendSerln(",***");
|
||||
}
|
||||
|
||||
void print_waypoint(struct Location *cmd,byte index)
|
||||
{
|
||||
SendSer("command #: ");
|
||||
SendSer(index, DEC);
|
||||
SendSer(" id: ");
|
||||
SendSer(cmd->id,DEC);
|
||||
SendSer(" p1: ");
|
||||
SendSer(cmd->p1,DEC);
|
||||
SendSer(" p2: ");
|
||||
SendSer(cmd->alt,DEC);
|
||||
SendSer(" p3: ");
|
||||
SendSer(cmd->lat,DEC);
|
||||
SendSer(" p4: ");
|
||||
SendSerln(cmd->lng,DEC);
|
||||
}
|
||||
|
||||
void print_waypoints()
|
||||
{
|
||||
}
|
||||
|
||||
#endif
|
File diff suppressed because it is too large
Load Diff
@ -0,0 +1,192 @@
|
||||
// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: t -*-
|
||||
|
||||
#if GCS_PROTOCOL == GCS_PROTOCOL_IMU
|
||||
|
||||
// This is the standard GCS output file for ArduPilot
|
||||
|
||||
/*
|
||||
Message Prefixes
|
||||
!!! Position Low rate telemetry
|
||||
+++ Attitude High rate telemetry
|
||||
### Mode Change in control mode
|
||||
%%% Waypoint Current and previous waypoints
|
||||
XXX Alert Text alert - NOTE: Alert message generation is not localized to a function
|
||||
PPP IMU Performance Sent every 20 seconds for performance monitoring
|
||||
|
||||
Message Suffix
|
||||
*** All messages use this suffix
|
||||
*/
|
||||
|
||||
/*
|
||||
void acknowledge(byte id, byte check1, byte check2) {}
|
||||
void send_message(byte id) {}
|
||||
void send_message(byte id, long param) {}
|
||||
void send_message(byte severity, const char *str) {}
|
||||
*/
|
||||
|
||||
void acknowledge(byte id, byte check1, byte check2)
|
||||
{
|
||||
}
|
||||
|
||||
void send_message(byte id)
|
||||
{
|
||||
send_message(id,0l);
|
||||
}
|
||||
|
||||
void send_message(byte id, long param)
|
||||
{
|
||||
switch(id) {
|
||||
case MSG_ATTITUDE:
|
||||
print_attitude();
|
||||
break;
|
||||
|
||||
case MSG_LOCATION: // ** Location/GPS message
|
||||
print_location();
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
void send_message(byte severity, const char *str)
|
||||
{
|
||||
if(severity >= DEBUG_LEVEL){
|
||||
Serial.print("MSG: ");
|
||||
Serial.println(str);
|
||||
}
|
||||
}
|
||||
|
||||
void print_current_waypoints()
|
||||
{
|
||||
}
|
||||
|
||||
void print_control_mode(void)
|
||||
{
|
||||
}
|
||||
|
||||
void print_attitude(void)
|
||||
{
|
||||
//Serial.print("!!!VER:");
|
||||
//Serial.print(SOFTWARE_VER); //output the software version
|
||||
//Serial.print(",");
|
||||
|
||||
// Analogs
|
||||
Serial.print("AN0:");
|
||||
Serial.print(read_adc(0)); //Reversing the sign.
|
||||
Serial.print(",AN1:");
|
||||
Serial.print(read_adc(1));
|
||||
Serial.print(",AN2:");
|
||||
Serial.print(read_adc(2));
|
||||
Serial.print(",AN3:");
|
||||
Serial.print(read_adc(3));
|
||||
Serial.print (",AN4:");
|
||||
Serial.print(read_adc(4));
|
||||
Serial.print (",AN5:");
|
||||
Serial.print(read_adc(5));
|
||||
Serial.print (",");
|
||||
|
||||
// DCM
|
||||
Serial.print ("EX0:");
|
||||
Serial.print(convert_to_dec(DCM_Matrix[0][0]));
|
||||
Serial.print (",EX1:");
|
||||
Serial.print(convert_to_dec(DCM_Matrix[0][1]));
|
||||
Serial.print (",EX2:");
|
||||
Serial.print(convert_to_dec(DCM_Matrix[0][2]));
|
||||
Serial.print (",EX3:");
|
||||
Serial.print(convert_to_dec(DCM_Matrix[1][0]));
|
||||
Serial.print (",EX4:");
|
||||
Serial.print(convert_to_dec(DCM_Matrix[1][1]));
|
||||
Serial.print (",EX5:");
|
||||
Serial.print(convert_to_dec(DCM_Matrix[1][2]));
|
||||
Serial.print (",EX6:");
|
||||
Serial.print(convert_to_dec(DCM_Matrix[2][0]));
|
||||
Serial.print (",EX7:");
|
||||
Serial.print(convert_to_dec(DCM_Matrix[2][1]));
|
||||
Serial.print (",EX8:");
|
||||
Serial.print(convert_to_dec(DCM_Matrix[2][2]));
|
||||
Serial.print (",");
|
||||
|
||||
// Euler
|
||||
Serial.print("RLL:");
|
||||
Serial.print(ToDeg(roll));
|
||||
Serial.print(",PCH:");
|
||||
Serial.print(ToDeg(pitch));
|
||||
Serial.print(",YAW:");
|
||||
Serial.print(ToDeg(yaw));
|
||||
Serial.print(",IMUH:");
|
||||
Serial.print(((int)imu_health>>8)&0xff);
|
||||
Serial.print (",");
|
||||
|
||||
|
||||
/*
|
||||
#if MAGNETOMETER == ENABLED
|
||||
Serial.print("MGX:");
|
||||
Serial.print(magnetom_x);
|
||||
Serial.print (",MGY:");
|
||||
Serial.print(magnetom_y);
|
||||
Serial.print (",MGZ:");
|
||||
Serial.print(magnetom_z);
|
||||
Serial.print (",MGH:");
|
||||
Serial.print(MAG_Heading);
|
||||
Serial.print (",");
|
||||
#endif
|
||||
*/
|
||||
|
||||
//Serial.print("Temp:");
|
||||
//Serial.print(temp_unfilt/20.0); // Convert into degrees C
|
||||
//alti();
|
||||
//Serial.print(",Pressure: ");
|
||||
//Serial.print(press);
|
||||
//Serial.print(",Alt: ");
|
||||
//Serial.print(press_alt/1000); // Original floating point full solution in meters
|
||||
//Serial.print (",");
|
||||
Serial.println("***");
|
||||
}
|
||||
|
||||
void print_location(void)
|
||||
{
|
||||
Serial.print("LAT:");
|
||||
Serial.print(current_loc.lat);
|
||||
Serial.print(",LON:");
|
||||
Serial.print(current_loc.lng);
|
||||
Serial.print(",ALT:");
|
||||
Serial.print(current_loc.alt/100); // meters
|
||||
Serial.print(",COG:");
|
||||
Serial.print(GPS.ground_course);
|
||||
Serial.print(",SOG:");
|
||||
Serial.print(GPS.ground_speed);
|
||||
Serial.print(",FIX:");
|
||||
Serial.print((int)GPS.fix);
|
||||
Serial.print(",SAT:");
|
||||
Serial.print((int)GPS.num_sats);
|
||||
Serial.print (",");
|
||||
Serial.print("TOW:");
|
||||
Serial.print(GPS.time);
|
||||
Serial.println("***");
|
||||
}
|
||||
|
||||
void print_waypoints()
|
||||
{
|
||||
}
|
||||
|
||||
void print_waypoint(struct Location *cmd,byte index)
|
||||
{
|
||||
Serial.print("command #: ");
|
||||
Serial.print(index, DEC);
|
||||
Serial.print(" id: ");
|
||||
Serial.print(cmd->id,DEC);
|
||||
Serial.print(" p1: ");
|
||||
Serial.print(cmd->p1,DEC);
|
||||
Serial.print(" p2: ");
|
||||
Serial.print(cmd->alt,DEC);
|
||||
Serial.print(" p3: ");
|
||||
Serial.print(cmd->lat,DEC);
|
||||
Serial.print(" p4: ");
|
||||
Serial.println(cmd->lng,DEC);
|
||||
|
||||
}
|
||||
|
||||
long convert_to_dec(float x)
|
||||
{
|
||||
return x*10000000;
|
||||
}
|
||||
|
||||
#endif
|
@ -0,0 +1,238 @@
|
||||
// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: t -*-
|
||||
|
||||
|
||||
#if GCS_PROTOCOL == GCS_PROTOCOL_JASON
|
||||
|
||||
// this is my personal GCS - Jason
|
||||
|
||||
|
||||
void send_message(byte severity, const char *str) // This is the instance of send_message for message 0x05
|
||||
{
|
||||
if(severity >= DEBUG_LEVEL){
|
||||
Serial.print("MSG: ");
|
||||
Serial.println(str);
|
||||
}
|
||||
}
|
||||
|
||||
void send_message(byte id)
|
||||
{
|
||||
send_message(id,(long)0);
|
||||
}
|
||||
|
||||
void send_message(byte id, long param)
|
||||
{
|
||||
|
||||
switch(id) {
|
||||
case MSG_HEARTBEAT:
|
||||
print_control_mode();
|
||||
break;
|
||||
|
||||
case MSG_ATTITUDE:
|
||||
print_attitude();
|
||||
break;
|
||||
|
||||
case MSG_LOCATION:
|
||||
print_position();
|
||||
break;
|
||||
|
||||
case MSG_COMMAND:
|
||||
struct Location cmd = get_wp_with_index(param);
|
||||
print_waypoint(&cmd, param);
|
||||
break;
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
void pipe()
|
||||
{
|
||||
Serial.print("|");
|
||||
}
|
||||
|
||||
void print_current_waypoints()
|
||||
{
|
||||
Serial.print("MSG: ");
|
||||
|
||||
Serial.print("CUR:");
|
||||
Serial.print("\t");
|
||||
Serial.print(current_loc.lat,DEC);
|
||||
Serial.print(",\t");
|
||||
Serial.print(current_loc.lng,DEC);
|
||||
Serial.print(",\t");
|
||||
Serial.println(current_loc.alt,DEC);
|
||||
|
||||
Serial.print("NWP:");
|
||||
Serial.print(wp_index,DEC);
|
||||
Serial.print(",\t");
|
||||
Serial.print(next_WP.lat,DEC);
|
||||
Serial.print(",\t");
|
||||
Serial.print(next_WP.lng,DEC);
|
||||
Serial.print(",\t");
|
||||
Serial.println(next_WP.alt,DEC);
|
||||
}
|
||||
|
||||
void print_position(void)
|
||||
{
|
||||
Serial.print("!!");
|
||||
Serial.print(current_loc.lat,DEC); // 0
|
||||
pipe();
|
||||
Serial.print(current_loc.lng,DEC); // 1
|
||||
pipe();
|
||||
Serial.print(current_loc.alt,DEC); // 2
|
||||
pipe();
|
||||
Serial.print(GPS.ground_speed,DEC); // 3
|
||||
pipe();
|
||||
Serial.print(airspeed,DEC); // 4
|
||||
pipe();
|
||||
Serial.print(get_altitude_above_home(),DEC); // 5
|
||||
pipe();
|
||||
Serial.print(climb_rate,DEC); // 6
|
||||
pipe();
|
||||
Serial.print(wp_distance,DEC); // 7
|
||||
pipe();
|
||||
Serial.print(throttle_cruise,DEC); // 8
|
||||
pipe();
|
||||
Serial.println(altitude_error,DEC); // 9
|
||||
|
||||
}
|
||||
|
||||
void print_attitude(void)
|
||||
{
|
||||
//return;
|
||||
|
||||
Serial.print("++");
|
||||
Serial.print(radio_in[0],DEC); // 0
|
||||
pipe();
|
||||
Serial.print(radio_in[1],DEC); // 1
|
||||
pipe();
|
||||
Serial.print(radio_in[CH_THROTTLE],DEC); // 2
|
||||
pipe();
|
||||
Serial.print(roll_sensor,DEC); // 3
|
||||
pipe();
|
||||
Serial.print(pitch_sensor,DEC); // 4
|
||||
pipe();
|
||||
Serial.print("0"); // 5 ir_max - removed, no longer applicable
|
||||
pipe();
|
||||
Serial.print(yaw_sensor,DEC); // 6
|
||||
pipe();
|
||||
Serial.print(target_bearing,DEC); // 7
|
||||
pipe();
|
||||
Serial.print(nav_roll,DEC); // 8
|
||||
pipe();
|
||||
Serial.println(loiter_sum,DEC); // 8
|
||||
|
||||
}
|
||||
|
||||
// required by Groundstation to plot lateral tracking course
|
||||
void print_new_wp_info()
|
||||
{
|
||||
Serial.print("??");
|
||||
Serial.print(wp_index,DEC); //0
|
||||
pipe();
|
||||
Serial.print(prev_WP.lat,DEC); //1
|
||||
pipe();
|
||||
Serial.print(prev_WP.lng,DEC); //2
|
||||
pipe();
|
||||
Serial.print(prev_WP.alt,DEC); //3
|
||||
pipe();
|
||||
Serial.print(next_WP.lat,DEC); //4
|
||||
pipe();
|
||||
Serial.print(next_WP.lng,DEC); //5
|
||||
pipe();
|
||||
Serial.print(next_WP.alt,DEC); //6
|
||||
pipe();
|
||||
Serial.print(wp_totalDistance,DEC); //7
|
||||
pipe();
|
||||
Serial.print(radio_trim[CH_ROLL],DEC); //8
|
||||
pipe();
|
||||
Serial.println(radio_trim[CH_PITCH],DEC); //9
|
||||
}
|
||||
|
||||
void print_control_mode(void)
|
||||
{
|
||||
switch (control_mode){
|
||||
case MANUAL:
|
||||
Serial.println("##MANUAL");
|
||||
break;
|
||||
case STABILIZE:
|
||||
Serial.println("##STABILIZE");
|
||||
break;
|
||||
case FLY_BY_WIRE_A:
|
||||
Serial.println("##FBW A");
|
||||
break;
|
||||
case FLY_BY_WIRE_B:
|
||||
Serial.println("##FBW B");
|
||||
break;
|
||||
case AUTO:
|
||||
Serial.println("##AUTO");
|
||||
break;
|
||||
case RTL:
|
||||
Serial.println("##RTL");
|
||||
break;
|
||||
case LOITER:
|
||||
Serial.println("##LOITER");
|
||||
break;
|
||||
case 98:
|
||||
Serial.println("##Air Start Complete");
|
||||
break;
|
||||
case 99:
|
||||
Serial.println("##Ground Start Complete");
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
void print_tuning(void) {
|
||||
Serial.print("TUN:");
|
||||
Serial.print(servo_out[CH_ROLL]/100);
|
||||
Serial.print(", ");
|
||||
Serial.print(nav_roll/100,DEC);
|
||||
Serial.print(", ");
|
||||
Serial.print(roll_sensor/100,DEC);
|
||||
Serial.print(", ");
|
||||
Serial.print(servo_out[CH_PITCH]/100);
|
||||
Serial.print(", ");
|
||||
Serial.print(nav_pitch/100,DEC);
|
||||
Serial.print(", ");
|
||||
Serial.println(pitch_sensor/100,DEC);
|
||||
}
|
||||
|
||||
void printPerfData(void)
|
||||
{
|
||||
}
|
||||
|
||||
|
||||
void print_waypoint(struct Location *cmd,byte index)
|
||||
{
|
||||
Serial.print("command #: ");
|
||||
Serial.print(index, DEC);
|
||||
Serial.print(" id: ");
|
||||
Serial.print(cmd->id,DEC);
|
||||
Serial.print(" p1: ");
|
||||
Serial.print(cmd->p1,DEC);
|
||||
Serial.print(" p2: ");
|
||||
Serial.print(cmd->alt,DEC);
|
||||
Serial.print(" p3: ");
|
||||
Serial.print(cmd->lat,DEC);
|
||||
Serial.print(" p4: ");
|
||||
Serial.println(cmd->lng,DEC);
|
||||
}
|
||||
|
||||
void print_waypoints()
|
||||
{
|
||||
Serial.println("Commands in memory");
|
||||
Serial.print("commands total: ");
|
||||
Serial.println(wp_total, DEC);
|
||||
// create a location struct to hold the temp Waypoints for printing
|
||||
//Location tmp;
|
||||
Serial.println("Home: ");
|
||||
struct Location cmd = get_wp_with_index(0);
|
||||
print_waypoint(&cmd, 0);
|
||||
Serial.println("Commands: ");
|
||||
|
||||
for (int i=1; i<wp_total; i++){
|
||||
cmd = get_wp_with_index(i);
|
||||
print_waypoint(&cmd, i);
|
||||
}
|
||||
}
|
||||
|
||||
#endif
|
||||
|
@ -0,0 +1,371 @@
|
||||
// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: t -*-
|
||||
// Doug _ command index is a byte and not an int
|
||||
|
||||
#if GCS_PROTOCOL == GCS_PROTOCOL_STANDARD
|
||||
|
||||
#if GCS_PORT == 3
|
||||
# define SendSer Serial3.print
|
||||
#else
|
||||
# define SendSer Serial.print
|
||||
#endif
|
||||
|
||||
// The functions included in this file are for use with the standard binary communication protocol and standard Ground Control Station
|
||||
|
||||
void acknowledge(byte id, byte check1, byte check2) {
|
||||
byte mess_buffer[6];
|
||||
byte mess_ck_a = 0;
|
||||
byte mess_ck_b = 0;
|
||||
int ck;
|
||||
|
||||
SendSer("4D"); // This is the message preamble
|
||||
mess_buffer[0] = 0x03;
|
||||
ck = 3;
|
||||
mess_buffer[1] = 0x00; // Message ID
|
||||
mess_buffer[2] = 0x01; // Message version
|
||||
|
||||
mess_buffer[3] = id;
|
||||
mess_buffer[4] = check1;
|
||||
mess_buffer[5] = check2;
|
||||
|
||||
|
||||
for (int i = 0; i < ck + 3; i++) SendSer (mess_buffer[i]);
|
||||
|
||||
for (int i = 0; i < ck + 3; i++) {
|
||||
mess_ck_a += mess_buffer[i]; // Calculates checksums
|
||||
mess_ck_b += mess_ck_a;
|
||||
}
|
||||
SendSer(mess_ck_a);
|
||||
SendSer(mess_ck_b);
|
||||
}
|
||||
|
||||
void send_message(byte id) {
|
||||
send_message(id, 0l);
|
||||
}
|
||||
|
||||
void send_message(byte id, long param) {
|
||||
byte mess_buffer[54];
|
||||
byte mess_ck_a = 0;
|
||||
byte mess_ck_b = 0;
|
||||
int tempint;
|
||||
int ck;
|
||||
long templong;
|
||||
|
||||
SendSer("4D"); // This is the message preamble
|
||||
|
||||
switch(id) {
|
||||
case MSG_HEARTBEAT: // ** System Status message
|
||||
mess_buffer[0] = 0x07;
|
||||
ck = 7;
|
||||
mess_buffer[3] = control_mode; // Mode
|
||||
templong = millis() / 1000; // Timestamp - Seconds since power - up
|
||||
mess_buffer[4] = templong & 0xff;
|
||||
mess_buffer[5] = (templong >> 8) & 0xff;
|
||||
tempint = battery_voltage1 * 100; // Battery voltage ( * 100)
|
||||
mess_buffer[6] = tempint & 0xff;
|
||||
mess_buffer[7] = (tempint >> 8) & 0xff;
|
||||
tempint = command_must_index; // Command Index (waypoint level)
|
||||
mess_buffer[8] = tempint & 0xff;
|
||||
mess_buffer[9] = (tempint >> 8) & 0xff;
|
||||
break;
|
||||
|
||||
case MSG_ATTITUDE: // ** Attitude message
|
||||
mess_buffer[0] = 0x06;
|
||||
ck = 6;
|
||||
tempint = roll_sensor; // Roll (degrees * 100)
|
||||
mess_buffer[3] = tempint & 0xff;
|
||||
mess_buffer[4] = (tempint >> 8) & 0xff;
|
||||
tempint = pitch_sensor; // Pitch (degrees * 100)
|
||||
mess_buffer[5] = tempint & 0xff;
|
||||
mess_buffer[6] = (tempint >> 8) & 0xff;
|
||||
tempint = yaw_sensor; // Yaw (degrees * 100)
|
||||
mess_buffer[7] = tempint & 0xff;
|
||||
mess_buffer[8] = (tempint >> 8) & 0xff;
|
||||
break;
|
||||
|
||||
case MSG_LOCATION: // ** Location / GPS message
|
||||
mess_buffer[0] = 0x12;
|
||||
ck = 18;
|
||||
templong = current_loc.lat; // Latitude *10 * *7 in 4 bytes
|
||||
mess_buffer[3] = templong & 0xff;
|
||||
mess_buffer[4] = (templong >> 8) & 0xff;
|
||||
mess_buffer[5] = (templong >> 16) & 0xff;
|
||||
mess_buffer[6] = (templong >> 24) & 0xff;
|
||||
|
||||
templong = current_loc.lng; // Longitude *10 * *7 in 4 bytes
|
||||
mess_buffer[7] = templong & 0xff;
|
||||
mess_buffer[8] = (templong >> 8) & 0xff;
|
||||
mess_buffer[9] = (templong >> 16) & 0xff;
|
||||
mess_buffer[10] = (templong >> 24) & 0xff;
|
||||
|
||||
tempint = GPS.altitude / 100; // Altitude MSL in meters * 10 in 2 bytes
|
||||
mess_buffer[11] = tempint & 0xff;
|
||||
mess_buffer[12] = (tempint >> 8) & 0xff;
|
||||
|
||||
tempint = GPS.ground_speed; // Speed in M / S * 100 in 2 bytes
|
||||
mess_buffer[13] = tempint & 0xff;
|
||||
mess_buffer[14] = (tempint >> 8) & 0xff;
|
||||
|
||||
tempint = yaw_sensor; // Ground Course in degreees * 100 in 2 bytes
|
||||
mess_buffer[15] = tempint & 0xff;
|
||||
mess_buffer[16] = (tempint >> 8) & 0xff;
|
||||
|
||||
templong = GPS.time; // Time of Week (milliseconds) in 4 bytes
|
||||
mess_buffer[17] = templong & 0xff;
|
||||
mess_buffer[18] = (templong >> 8) & 0xff;
|
||||
mess_buffer[19] = (templong >> 16) & 0xff;
|
||||
mess_buffer[20] = (templong >> 24) & 0xff;
|
||||
break;
|
||||
|
||||
case MSG_PRESSURE: // ** Pressure message
|
||||
mess_buffer[0] = 0x04;
|
||||
ck = 4;
|
||||
tempint = current_loc.alt / 10; // Altitude MSL in meters * 10 in 2 bytes
|
||||
mess_buffer[3] = tempint & 0xff;
|
||||
mess_buffer[4] = (tempint >> 8) & 0xff;
|
||||
tempint = (int)airspeed / 100; // Airspeed pressure
|
||||
mess_buffer[5] = tempint & 0xff;
|
||||
mess_buffer[6] = (tempint >> 8) & 0xff;
|
||||
break;
|
||||
|
||||
// case 0xMSG_STATUS_TEXT: // ** Status Text message
|
||||
// mess_buffer[0]=sizeof(status_message[0])+1;
|
||||
// ck=mess_buffer[0];
|
||||
// mess_buffer[2] = param&0xff;
|
||||
// for (int i=3;i<ck+2;i++) mess_buffer[i] = status_message[i-3];
|
||||
// break;
|
||||
|
||||
case MSG_PERF_REPORT: // ** Performance Monitoring message
|
||||
mess_buffer[0] = 0x10;
|
||||
ck = 16;
|
||||
templong = millis() - perf_mon_timer; // Report interval (milliseconds) in 4 bytes
|
||||
mess_buffer[3] = templong & 0xff;
|
||||
mess_buffer[4] = (templong >> 8) & 0xff;
|
||||
mess_buffer[5] = (templong >> 16) & 0xff;
|
||||
mess_buffer[6] = (templong >> 24) & 0xff;
|
||||
tempint = mainLoop_count; // Main Loop cycles
|
||||
mess_buffer[7] = tempint & 0xff;
|
||||
mess_buffer[8] = (tempint >> 8) & 0xff;
|
||||
mess_buffer[9] = G_Dt_max & 0xff;
|
||||
mess_buffer[10] = gyro_sat_count; // Problem counts
|
||||
mess_buffer[11] = adc_constraints;
|
||||
mess_buffer[12] = renorm_sqrt_count;
|
||||
mess_buffer[13] = renorm_blowup_count;
|
||||
mess_buffer[14] = gps_fix_count;
|
||||
tempint = (int)(imu_health * 1000); // IMU health metric
|
||||
mess_buffer[15] = tempint & 0xff;
|
||||
mess_buffer[16] = (tempint >> 8) & 0xff;
|
||||
tempint = gcs_messages_sent; // GCS messages sent
|
||||
mess_buffer[17] = tempint & 0xff;
|
||||
mess_buffer[18] = (tempint >> 8) & 0xff;
|
||||
break;
|
||||
|
||||
case MSG_VALUE: // ** Requested Value message
|
||||
mess_buffer[0] = 0x06;
|
||||
ck = 6;
|
||||
templong = param; // Parameter being sent
|
||||
mess_buffer[3] = templong & 0xff;
|
||||
mess_buffer[4] = (templong >> 8) & 0xff;
|
||||
switch(param) {
|
||||
// case 0x00: templong = roll_mode; break;
|
||||
// case 0x01: templong = pitch_mode; break;
|
||||
case 0x02: templong = yaw_mode; break;
|
||||
// case 0x03: templong = throttle_mode; break;
|
||||
case 0x04: templong = elevon1_trim; break;
|
||||
case 0x05: templong = elevon2_trim; break;
|
||||
case 0x10: templong = integrator[0] * 1000; break;
|
||||
case 0x11: templong = integrator[1] * 1000; break;
|
||||
case 0x12: templong = integrator[2] * 1000; break;
|
||||
case 0x13: templong = integrator[3] * 1000; break;
|
||||
case 0x14: templong = integrator[4] * 1000; break;
|
||||
case 0x15: templong = integrator[5] * 1000; break;
|
||||
case 0x16: templong = integrator[6] * 1000; break;
|
||||
case 0x17: templong = integrator[7] * 1000; break;
|
||||
case 0x1a: templong = kff[0]; break;
|
||||
case 0x1b: templong = kff[1]; break;
|
||||
case 0x1c: templong = kff[2]; break;
|
||||
case 0x20: templong = target_bearing; break;
|
||||
case 0x21: templong = nav_bearing; break;
|
||||
case 0x22: templong = bearing_error; break;
|
||||
case 0x23: templong = crosstrack_bearing; break;
|
||||
case 0x24: templong = crosstrack_error; break;
|
||||
case 0x25: templong = altitude_error; break;
|
||||
case 0x26: templong = wp_radius; break;
|
||||
case 0x27: templong = loiter_radius; break;
|
||||
// case 0x28: templong = wp_mode; break;
|
||||
// case 0x29: templong = loop_commands; break;
|
||||
case 0x2a: templong = nav_gain_scaler; break;
|
||||
}
|
||||
mess_buffer[5] = templong & 0xff;
|
||||
mess_buffer[6] = (templong >> 8) & 0xff;
|
||||
mess_buffer[7] = (templong >> 16) & 0xff;
|
||||
mess_buffer[8] = (templong >> 24) & 0xff;
|
||||
break;
|
||||
|
||||
case MSG_COMMAND: // Command list item message
|
||||
mess_buffer[0] = 0x10;
|
||||
ck = 16;
|
||||
tempint = param; // item number
|
||||
mess_buffer[3] = tempint & 0xff;
|
||||
mess_buffer[4] = (tempint >> 8) & 0xff;
|
||||
tempint = wp_total; // list length (# of commands in mission)
|
||||
mess_buffer[5] = tempint & 0xff;
|
||||
mess_buffer[6] = (tempint >> 8) & 0xff;
|
||||
tell_command = get_wp_with_index((int)param);
|
||||
mess_buffer[7] = tell_command.id; // command id
|
||||
mess_buffer[8] = tell_command.p1; // P1
|
||||
tempint = tell_command.alt; // P2
|
||||
mess_buffer[9] = tempint & 0xff;
|
||||
mess_buffer[10] = (tempint >> 8) & 0xff;
|
||||
templong = tell_command.lat; // P3
|
||||
mess_buffer[11] = templong & 0xff;
|
||||
mess_buffer[12] = (templong >> 8) & 0xff;
|
||||
mess_buffer[13] = (templong >> 16) & 0xff;
|
||||
mess_buffer[14] = (templong >> 24) & 0xff;
|
||||
templong = tell_command.lng; // P4
|
||||
mess_buffer[15] = templong & 0xff;
|
||||
mess_buffer[16] = (templong >> 8) & 0xff;
|
||||
mess_buffer[17] = (templong >> 16) & 0xff;
|
||||
mess_buffer[18] = (templong >> 24) & 0xff;
|
||||
break;
|
||||
|
||||
case MSG_TRIMS: // Radio Trims message
|
||||
mess_buffer[0] = 0x10;
|
||||
ck = 16;
|
||||
for(int i = 0; i < 8; i++) {
|
||||
tempint = radio_trim[i]; // trim values
|
||||
mess_buffer[3 + 2 * i] = tempint & 0xff;
|
||||
mess_buffer[4 + 2 * i] = (tempint >> 8) & 0xff;
|
||||
}
|
||||
break;
|
||||
|
||||
case MSG_MINS: // Radio Mins message
|
||||
mess_buffer[0] = 0x10;
|
||||
ck = 16;
|
||||
for(int i = 0; i < 8; i++) {
|
||||
tempint = radio_min[i]; // min values
|
||||
mess_buffer[3 + 2 * i] = tempint & 0xff;
|
||||
mess_buffer[4 + 2 * i] = (tempint >> 8) & 0xff;
|
||||
}
|
||||
break;
|
||||
|
||||
case MSG_MAXS: // Radio Maxs message
|
||||
mess_buffer[0] = 0x10;
|
||||
ck = 16;
|
||||
for(int i = 0; i < 8; i++) {
|
||||
tempint = radio_max[i]; // max values
|
||||
mess_buffer[3 + 2 * i] = tempint & 0xff;
|
||||
mess_buffer[4 + 2 * i] = (tempint >> 8) & 0xff;
|
||||
}
|
||||
break;
|
||||
|
||||
case MSG_PID: // PID Gains message
|
||||
mess_buffer[0] = 0x0f;
|
||||
ck = 15;
|
||||
mess_buffer[3] = param & 0xff; // PID set #
|
||||
templong = (kp[param] * 1000000); // P gain
|
||||
mess_buffer[4] = templong & 0xff;
|
||||
mess_buffer[5] = (templong >> 8) & 0xff;
|
||||
mess_buffer[6] = (templong >> 16) & 0xff;
|
||||
mess_buffer[7] = (templong >> 24) & 0xff;
|
||||
templong = (ki[param] * 1000000); // I gain
|
||||
mess_buffer[8] = templong & 0xff;
|
||||
mess_buffer[9] = (templong >> 8) & 0xff;
|
||||
mess_buffer[10] = (templong >> 16) & 0xff;
|
||||
mess_buffer[11] = (templong >> 24) & 0xff;
|
||||
templong = (kd[param] * 1000000); // D gain
|
||||
mess_buffer[12] = templong & 0xff;
|
||||
mess_buffer[13] = (templong >> 8) & 0xff;
|
||||
mess_buffer[14] = (templong >> 16) & 0xff;
|
||||
mess_buffer[15] = (templong >> 24) & 0xff;
|
||||
tempint = integrator_max[param]; // Integrator max value
|
||||
mess_buffer[16] = tempint & 0xff;
|
||||
mess_buffer[17] = (tempint >> 8) & 0xff;
|
||||
break;
|
||||
}
|
||||
|
||||
mess_buffer[1] = id; // Message ID
|
||||
mess_buffer[2] = 0x01; // Message version
|
||||
|
||||
for (int i = 0; i < ck + 3; i++) SendSer (mess_buffer[i]);
|
||||
|
||||
for (int i = 0; i < ck + 3; i++) {
|
||||
mess_ck_a += mess_buffer[i]; // Calculates checksums
|
||||
mess_ck_b += mess_ck_a;
|
||||
}
|
||||
|
||||
SendSer(mess_ck_a);
|
||||
SendSer(mess_ck_b);
|
||||
|
||||
gcs_messages_sent++;
|
||||
}
|
||||
|
||||
void send_message(byte severity, const char *str) // This is the instance of send_message for message MSG_STATUS_TEXT
|
||||
{
|
||||
if(severity >= DEBUG_LEVEL){
|
||||
byte length = strlen(str) + 1;
|
||||
|
||||
byte mess_buffer[54];
|
||||
byte mess_ck_a = 0;
|
||||
byte mess_ck_b = 0;
|
||||
int ck;
|
||||
|
||||
SendSer("4D"); // This is the message preamble
|
||||
if(length > 50) length = 50;
|
||||
mess_buffer[0] = length;
|
||||
ck = length;
|
||||
mess_buffer[1] = 0x05; // Message ID
|
||||
mess_buffer[2] = 0x01; // Message version
|
||||
|
||||
mess_buffer[3] = severity;
|
||||
|
||||
for (int i = 3; i < ck + 2; i++)
|
||||
mess_buffer[i] = str[i - 3]; // places the text into mess_buffer at locations 3+
|
||||
|
||||
for (int i = 0; i < ck + 3; i++)
|
||||
SendSer(mess_buffer[i]);
|
||||
|
||||
for (int i = 0; i < ck + 3; i++) {
|
||||
mess_ck_a += mess_buffer[i]; // Calculates checksums
|
||||
mess_ck_b += mess_ck_a;
|
||||
}
|
||||
SendSer(mess_ck_a);
|
||||
SendSer(mess_ck_b);
|
||||
}
|
||||
}
|
||||
|
||||
void print_current_waypoints()
|
||||
{
|
||||
}
|
||||
|
||||
void print_waypoint(struct Location *cmd, byte index)
|
||||
{
|
||||
Serial.print("command #: ");
|
||||
Serial.print(index, DEC);
|
||||
Serial.print(" id: ");
|
||||
Serial.print(cmd->id, DEC);
|
||||
Serial.print(" p1: ");
|
||||
Serial.print(cmd->p1, DEC);
|
||||
Serial.print(" p2: ");
|
||||
Serial.print(cmd->alt, DEC);
|
||||
Serial.print(" p3: ");
|
||||
Serial.print(cmd->lat, DEC);
|
||||
Serial.print(" p4: ");
|
||||
Serial.println(cmd->lng, DEC);
|
||||
}
|
||||
|
||||
void print_waypoints()
|
||||
{
|
||||
}
|
||||
|
||||
|
||||
#endif
|
||||
|
||||
#if GCS_PROTOCOL == GCS_PROTOCOL_NONE
|
||||
void acknowledge(byte id, byte check1, byte check2) {}
|
||||
void send_message(byte id) {}
|
||||
void send_message(byte id, long param) {}
|
||||
void send_message(byte severity, const char *str) {}
|
||||
void print_current_waypoints(){}
|
||||
void print_waypoint(struct Location *cmd, byte index){}
|
||||
void print_waypoints(){}
|
||||
#endif
|
@ -0,0 +1,9 @@
|
||||
#if GCS_PROTOCOL == GCS_PROTOCOL_XDIY
|
||||
void acknowledge(byte id, byte check1, byte check2) {}
|
||||
void send_message(byte id) {}
|
||||
void send_message(byte id, long param) {}
|
||||
void send_message(byte severity, const char *str) {}
|
||||
void print_current_waypoints(){}
|
||||
void print_waypoint(struct Location *cmd, byte index){}
|
||||
void print_waypoints(){}
|
||||
#endif
|
@ -0,0 +1,127 @@
|
||||
// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: t -*-
|
||||
|
||||
#if GCS_PROTOCOL == GCS_PROTOCOL_XPLANE
|
||||
|
||||
|
||||
void acknowledge(byte id, byte check1, byte check2)
|
||||
{
|
||||
}
|
||||
|
||||
void send_message(byte severity, const char *str) // This is the instance of send_message for message 0x05
|
||||
{
|
||||
if(severity >= DEBUG_LEVEL){
|
||||
Serial.print("MSG: ");
|
||||
Serial.println(str);
|
||||
}
|
||||
}
|
||||
|
||||
void send_message(byte id) {
|
||||
send_message(id,0l);
|
||||
}
|
||||
|
||||
void send_message(byte id, long param) {
|
||||
switch(id) {
|
||||
case MSG_HEARTBEAT:
|
||||
print_control_mode();
|
||||
break;
|
||||
|
||||
case MSG_ATTITUDE:
|
||||
//print_attitude();
|
||||
break;
|
||||
|
||||
case MSG_LOCATION:
|
||||
//print_position();
|
||||
break;
|
||||
|
||||
case MSG_COMMAND:
|
||||
struct Location cmd = get_wp_with_index(param);
|
||||
print_waypoint(&cmd, param);
|
||||
break;
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
// required by Groundstation to plot lateral tracking course
|
||||
void print_new_wp_info()
|
||||
{
|
||||
}
|
||||
|
||||
void print_control_mode(void)
|
||||
{
|
||||
Serial.print("MSG: ");
|
||||
Serial.print(flight_mode_strings[control_mode]);
|
||||
}
|
||||
|
||||
|
||||
void print_current_waypoints()
|
||||
{
|
||||
Serial.print("MSG: ");
|
||||
Serial.print("CUR:");
|
||||
Serial.print("\t");
|
||||
Serial.print(current_loc.lat,DEC);
|
||||
Serial.print(",\t");
|
||||
Serial.print(current_loc.lng,DEC);
|
||||
Serial.print(",\t");
|
||||
Serial.println(current_loc.alt,DEC);
|
||||
|
||||
Serial.print("MSG: ");
|
||||
Serial.print("NWP:");
|
||||
Serial.print(wp_index,DEC);
|
||||
Serial.print(",\t");
|
||||
Serial.print(next_WP.lat,DEC);
|
||||
Serial.print(",\t");
|
||||
Serial.print(next_WP.lng,DEC);
|
||||
Serial.print(",\t");
|
||||
Serial.println(next_WP.alt,DEC);
|
||||
}
|
||||
|
||||
void print_position(void)
|
||||
{
|
||||
}
|
||||
|
||||
void printPerfData(void)
|
||||
{
|
||||
}
|
||||
|
||||
void print_attitude(void)
|
||||
{
|
||||
}
|
||||
void print_tuning(void) {
|
||||
}
|
||||
void print_waypoint(struct Location *cmd,byte index)
|
||||
{
|
||||
Serial.print("MSG: command #: ");
|
||||
Serial.print(index, DEC);
|
||||
Serial.print(" id: ");
|
||||
Serial.print(cmd->id,DEC);
|
||||
Serial.print(" p1: ");
|
||||
Serial.print(cmd->p1,DEC);
|
||||
Serial.print(" p2: ");
|
||||
Serial.print(cmd->alt,DEC);
|
||||
Serial.print(" p3: ");
|
||||
Serial.print(cmd->lat,DEC);
|
||||
Serial.print(" p4: ");
|
||||
Serial.println(cmd->lng,DEC);
|
||||
}
|
||||
|
||||
void print_waypoints()
|
||||
{
|
||||
Serial.println("Commands in memory");
|
||||
Serial.print("commands total: ");
|
||||
Serial.println(wp_total, DEC);
|
||||
// create a location struct to hold the temp Waypoints for printing
|
||||
//Location tmp;
|
||||
Serial.println("Home: ");
|
||||
struct Location cmd = get_wp_with_index(0);
|
||||
print_waypoint(&cmd, 0);
|
||||
Serial.println("Commands: ");
|
||||
|
||||
for (int i=1; i<wp_total; i++){
|
||||
cmd = get_wp_with_index(i);
|
||||
print_waypoint(&cmd, i);
|
||||
}
|
||||
}
|
||||
|
||||
#endif
|
||||
|
@ -0,0 +1,78 @@
|
||||
// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: t -*-
|
||||
|
||||
#if ENABLE_HIL
|
||||
// An Xplane/Flightgear output
|
||||
// Use with the GPS_IMU to do Hardware in teh loop simulations
|
||||
|
||||
byte buf_len = 0;
|
||||
byte out_buffer[32];
|
||||
|
||||
void output_HIL(void)
|
||||
{
|
||||
// output real-time sensor data
|
||||
Serial.print("AAA"); // Message preamble
|
||||
output_int((int)(servo_out[CH_ROLL])); // 0 bytes 0, 1
|
||||
output_int((int)(servo_out[CH_PITCH])); // 1 bytes 2, 3
|
||||
output_int((int)(servo_out[CH_THROTTLE])); // 2 bytes 4, 5
|
||||
output_int((int)(servo_out[CH_RUDDER])); // 3 bytes 6, 7
|
||||
output_int((int)wp_distance); // 4 bytes 8,9
|
||||
output_int((int)bearing_error); // 5 bytes 10,11
|
||||
output_int((int)next_WP.alt / 100); // 6 bytes 12, 13
|
||||
output_int((int)energy_error); // 7 bytes 14,15
|
||||
output_byte(wp_index); // 8 bytes 16
|
||||
output_byte(control_mode); // 9 bytes 17
|
||||
|
||||
// print out the buffer and checksum
|
||||
// ---------------------------------
|
||||
print_buffer();
|
||||
}
|
||||
|
||||
// This is for debugging only!,
|
||||
// I just move the underscore to keep the above version pristene.
|
||||
void output_HIL_(void)
|
||||
{
|
||||
// output real-time sensor data
|
||||
Serial.print("AAA"); // Message preamble
|
||||
output_int((int)(servo_out[CH_ROLL])); // 0 bytes 0, 1
|
||||
output_int((int)(servo_out[CH_PITCH])); // 1 bytes 2, 3
|
||||
output_int((int)(servo_out[CH_THROTTLE])); // 2 bytes 4, 5
|
||||
output_int((int)(servo_out[CH_RUDDER])); // 3 bytes 6, 7
|
||||
output_int((int)wp_distance); // 4 bytes 8, 9
|
||||
output_int((int)bearing_error); // 5 bytes 10,11
|
||||
output_int((int)roll_sensor); // 6 bytes 12,13
|
||||
output_int((int)loiter_total); // 7 bytes 14,15
|
||||
output_byte(wp_index); // 8 bytes 16
|
||||
output_byte(control_mode); // 9 bytes 17
|
||||
|
||||
// print out the buffer and checksum
|
||||
// ---------------------------------
|
||||
print_buffer();
|
||||
}
|
||||
|
||||
void output_int(int value)
|
||||
{
|
||||
//buf_len += 2;
|
||||
out_buffer[buf_len++] = value & 0xff;
|
||||
out_buffer[buf_len++] = (value >> 8) & 0xff;
|
||||
}
|
||||
|
||||
void output_byte(byte value)
|
||||
{
|
||||
out_buffer[buf_len++] = value;
|
||||
}
|
||||
|
||||
void print_buffer()
|
||||
{
|
||||
byte ck_a = 0;
|
||||
byte ck_b = 0;
|
||||
for (byte i = 0;i < buf_len; i++){
|
||||
Serial.print (out_buffer[i]);
|
||||
}
|
||||
Serial.print('\r');
|
||||
Serial.print('\n');
|
||||
buf_len = 0;
|
||||
}
|
||||
|
||||
|
||||
#endif
|
||||
|
@ -0,0 +1,264 @@
|
||||
// menu structure
|
||||
//
|
||||
// [status] - [data1] - [data2] - [data3] - [data4] - [dataXY]
|
||||
// |_________| _______|_________|_________|_________|
|
||||
// |
|
||||
// [SETUP1] - [SETUP2] - [SETUPXY]
|
||||
// | | |
|
||||
// [para1.1] [PARAM2.1] [PARAMX.Y]
|
||||
// | | |
|
||||
// [para1.2] [PARAM2.2] [PARAMX.Y]
|
||||
// | | |
|
||||
// [para1.3] [PARAM2.3] [PARAMX.Y]
|
||||
|
||||
// Statusmenu
|
||||
const uint8_t mnuSTATUS = 0;
|
||||
const uint8_t mnuGPS1 = 1;
|
||||
const uint8_t mnuGPS2 = 2;
|
||||
const uint8_t mnuIMU = 3;
|
||||
const uint8_t mnuACC = 4;
|
||||
const uint8_t mnuGYRO = 5;
|
||||
const uint8_t mnuCOMPASS = 6;
|
||||
const uint8_t mnuSERVO1 = 7;
|
||||
const uint8_t mnuSERVO2 = 8;
|
||||
const uint8_t mnuDATA9 = 9;
|
||||
|
||||
// Setup 1
|
||||
const uint8_t mnuSETUP1 = 10;
|
||||
const uint8_t mnuPARAM11 = 11;
|
||||
const uint8_t mnuPARAM12 = 12;
|
||||
const uint8_t mnuPARAM13 = 13;
|
||||
const uint8_t mnuPARAM14 = 14;
|
||||
|
||||
// Setup 2
|
||||
const uint8_t mnuSETUP2 = 20;
|
||||
const uint8_t mnuPARAM21 = 21;
|
||||
const uint8_t mnuPARAM22 = 22;
|
||||
const uint8_t mnuPARAM23 = 23;
|
||||
const uint8_t mnuPARAM24 = 24;
|
||||
|
||||
// Setup 3
|
||||
const uint8_t mnuSETUP3 = 30;
|
||||
const uint8_t mnuPARAM31 = 31;
|
||||
const uint8_t mnuPARAM32 = 32;
|
||||
const uint8_t mnuPARAM33 = 33;
|
||||
const uint8_t mnuPARAM34 = 34;
|
||||
|
||||
// Setup 4
|
||||
const uint8_t mnuSETUP4 = 40;
|
||||
const uint8_t mnuPARAM41 = 41;
|
||||
const uint8_t mnuPARAM42 = 42;
|
||||
const uint8_t mnuPARAM43 = 43;
|
||||
const uint8_t mnuPARAM44 = 44;
|
||||
|
||||
#define LCDMaxPos 32
|
||||
|
||||
static uint8_t MnuPtr = 0;
|
||||
static uint8_t statustxt[LCDMaxPos];
|
||||
|
||||
void jeti_status(const char *str)
|
||||
{
|
||||
byte i;
|
||||
byte length = strlen(str) + 1;
|
||||
if (length > LCDMaxPos) length = LCDMaxPos;
|
||||
for (i = 0; i<LCDMaxPos; i++) {
|
||||
if (i<length) {
|
||||
statustxt[i] = str[i];
|
||||
} else {
|
||||
statustxt[i] = 32; // if text < 32 chars -> fill up with SPACE
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void jeti_menuctrl(uint8_t btns) {
|
||||
// Algorithm for menubuttons
|
||||
// if MnuPtr == 0 then Status
|
||||
// if MnuPtr <10 then Data
|
||||
// if (MnuPtr mod 10) = 0 then Setupp
|
||||
// if (MnuPtr mod 10) <> 0 then Parameter
|
||||
|
||||
if (MnuPtr == mnuSTATUS) { // MnuPtr = Status
|
||||
switch (btns) {
|
||||
case JB_key_right:
|
||||
MnuPtr += 1;
|
||||
break;
|
||||
|
||||
case JB_key_left:
|
||||
MnuPtr = mnuDATA9;
|
||||
break;
|
||||
|
||||
case JB_key_down:
|
||||
MnuPtr = mnuSETUP1;
|
||||
break;
|
||||
}
|
||||
}
|
||||
else if (MnuPtr < 10) { // MnuPtr = Data
|
||||
switch (btns) {
|
||||
case JB_key_right:
|
||||
if (MnuPtr < mnuDATA9) MnuPtr += 1; else MnuPtr = mnuSTATUS;
|
||||
break;
|
||||
|
||||
case JB_key_left:
|
||||
if (MnuPtr > mnuSTATUS) MnuPtr -= 1; else MnuPtr = mnuDATA9;
|
||||
break;
|
||||
}
|
||||
}
|
||||
else if (MnuPtr % 10 == 0) { // MnuPtr = Setup
|
||||
switch (btns) {
|
||||
case JB_key_right:
|
||||
if (MnuPtr < mnuSETUP4) MnuPtr += 10; else MnuPtr = mnuSETUP1;
|
||||
break;
|
||||
|
||||
case JB_key_left:
|
||||
if (MnuPtr > mnuSETUP1) MnuPtr -= 10; else MnuPtr = mnuSETUP4;
|
||||
break;
|
||||
|
||||
case JB_key_up:
|
||||
MnuPtr = mnuSTATUS;
|
||||
break;
|
||||
|
||||
case JB_key_down:
|
||||
MnuPtr += 1;
|
||||
break;
|
||||
}
|
||||
}
|
||||
else { // MnuPtr = Parameter
|
||||
switch (btns) {
|
||||
case JB_key_down:
|
||||
if ((MnuPtr % 10) < 4) MnuPtr += 1;
|
||||
break;
|
||||
|
||||
case JB_key_up:
|
||||
if ((MnuPtr % 10) > 0) MnuPtr -= 1;
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void jeti_init() {
|
||||
// Init Jeti Serial Port
|
||||
JB.begin();
|
||||
}
|
||||
|
||||
void jeti_update() {
|
||||
JB.clear(0);
|
||||
uint8_t Buttons = JB.readbuttons();
|
||||
jeti_menuctrl(Buttons);
|
||||
switch (MnuPtr) {
|
||||
case mnuGPS1:
|
||||
JB.print("GPS> ");
|
||||
if (GPS.fix == 1) {
|
||||
JB.print("FIX");
|
||||
JB.print(" SAT>");
|
||||
JB.print((int)GPS.num_sats);
|
||||
JB.setcursor(LCDLine2);
|
||||
JB.print("Alt: ");
|
||||
JB.print((int)GPS.altitude/100);
|
||||
} else {
|
||||
JB.print("no FIX");
|
||||
}
|
||||
break;
|
||||
|
||||
case mnuGPS2:
|
||||
if (GPS.fix ==1 ) {
|
||||
JB.print("Lat ");
|
||||
JB.print((float)GPS.latitude/10000000, 9);
|
||||
JB.setcursor(LCDLine2);
|
||||
JB.print("Lon ");
|
||||
JB.print((float)GPS.longitude/10000000, 9);
|
||||
} else {
|
||||
JB.print("no Data avail.");
|
||||
}
|
||||
break;
|
||||
|
||||
case mnuIMU:
|
||||
//read_AHRS();
|
||||
JB.print("IMU> R:");
|
||||
JB.print(roll_sensor/100,DEC);
|
||||
JB.print(" P:");
|
||||
JB.print(pitch_sensor/100,DEC);
|
||||
JB.setcursor(LCDLine2);
|
||||
JB.print("Y:");
|
||||
JB.print(yaw_sensor/100,DEC);
|
||||
break;
|
||||
|
||||
case mnuACC:
|
||||
for (int i = 0; i < 6; i++) {
|
||||
AN[i] = APM_ADC.Ch(sensors[i]);
|
||||
}
|
||||
JB.print("ACC> X:");
|
||||
JB.print((int)AN[3]);
|
||||
JB.setcursor(LCDLine2);
|
||||
JB.print(" Y:");
|
||||
JB.print((int)AN[4]);
|
||||
JB.print(" Z:");
|
||||
JB.print((int)AN[5]);
|
||||
break;
|
||||
|
||||
case mnuGYRO:
|
||||
for (int i = 0; i < 6; i++) {
|
||||
AN[i] = APM_ADC.Ch(sensors[i]);
|
||||
}
|
||||
JB.print("GYRO> Y:");
|
||||
JB.print((int)AN[0]);
|
||||
JB.setcursor(LCDLine2);
|
||||
JB.print(" R:");
|
||||
JB.print((int)AN[1]);
|
||||
JB.print(" P:");
|
||||
JB.print((int)AN[2]);
|
||||
break;
|
||||
|
||||
case mnuCOMPASS:
|
||||
JB.print("MAGN> Head:");
|
||||
JB.print((int)ToDeg(APM_Compass.Heading));
|
||||
JB.setcursor(LCDLine2);
|
||||
JB.println("[");
|
||||
JB.print((int)APM_Compass.Mag_X);
|
||||
JB.print(comma);
|
||||
JB.print((int)APM_Compass.Mag_Y);
|
||||
JB.print(comma);
|
||||
JB.print((int)APM_Compass.Mag_Z);
|
||||
JB.println("]");
|
||||
break;
|
||||
|
||||
case mnuSERVO1:
|
||||
JB.print("#1:");
|
||||
JB.print((int)radio_in[CH_1]);
|
||||
JB.print(" #2:");
|
||||
JB.print((int)radio_in[CH_2]);
|
||||
JB.setcursor(LCDLine2);
|
||||
JB.print("#3:");
|
||||
JB.print((int)radio_in[CH_3]);
|
||||
JB.print(" #4:");
|
||||
JB.print((int)radio_in[CH_4]);
|
||||
break;
|
||||
|
||||
case mnuSERVO2:
|
||||
JB.print("#Q:");
|
||||
JB.print((int)servo_out[CH_ROLL]/100);
|
||||
JB.print(" #H:");
|
||||
JB.print((int)servo_out[CH_PITCH]/100);
|
||||
JB.setcursor(LCDLine2);
|
||||
JB.print("#G:");
|
||||
JB.print((int)servo_out[CH_THROTTLE]);
|
||||
JB.print(" #S:");
|
||||
JB.print((int)servo_out[CH_RUDDER]/100);
|
||||
break;
|
||||
|
||||
case mnuSETUP1:
|
||||
JB.print("SETUP 1");
|
||||
break;
|
||||
|
||||
case mnuSTATUS:
|
||||
for (byte i = 0; i<32; i++) {
|
||||
JB.print(statustxt[i]);
|
||||
}
|
||||
//JB.print("STATUS*");
|
||||
break;
|
||||
|
||||
default:
|
||||
JB.print("Menu: n/a #");
|
||||
JB.print((int) MnuPtr);
|
||||
break;
|
||||
}
|
||||
}
|
596
Tools/ArduPPM/WorkBasket/Jeti Duplex/X-DIY_Jeti_V2/X-DIY/Log.pde
Normal file
596
Tools/ArduPPM/WorkBasket/Jeti Duplex/X-DIY_Jeti_V2/X-DIY/Log.pde
Normal file
@ -0,0 +1,596 @@
|
||||
// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: t -*-
|
||||
|
||||
// Code to Write and Read packets from DataFlash log memory
|
||||
// Code to interact with the user to dump or erase logs
|
||||
|
||||
#define HEAD_BYTE1 0xA3 // Decimal 163
|
||||
#define HEAD_BYTE2 0x95 // Decimal 149
|
||||
#define END_BYTE 0xBA // Decimal 186
|
||||
|
||||
|
||||
// These are function definitions so the Menu can be constructed before the functions
|
||||
// are defined below. Order matters to the compiler.
|
||||
static int8_t print_log_menu(uint8_t argc, const Menu::arg *argv);
|
||||
static int8_t dump_log(uint8_t argc, const Menu::arg *argv);
|
||||
static int8_t erase_logs(uint8_t argc, const Menu::arg *argv);
|
||||
static int8_t select_logs(uint8_t argc, const Menu::arg *argv);
|
||||
|
||||
// This is the help function
|
||||
// PSTR is an AVR macro to read strings from flash memory
|
||||
// printf_P is a version of print_f that reads from flash memory
|
||||
static int8_t help_log(uint8_t argc, const Menu::arg *argv)
|
||||
{
|
||||
Serial.printf_P(PSTR("\n"
|
||||
"Commands:\n"
|
||||
" dump <n> dump log <n>\n"
|
||||
" erase erase all logs\n"
|
||||
" enable <name>|all enable logging <name> or everything\n"
|
||||
" disable <name>|all disable logging <name> or everything\n"
|
||||
"\n"));
|
||||
}
|
||||
|
||||
// Creates a constant array of structs representing menu options
|
||||
// and stores them in Flash memory, not RAM.
|
||||
// User enters the string in the console to call the functions on the right.
|
||||
// See class Menu in AP_Coommon for implementation details
|
||||
const struct Menu::command log_menu_commands[] PROGMEM = {
|
||||
{"dump", dump_log},
|
||||
{"erase", erase_logs},
|
||||
{"enable", select_logs},
|
||||
{"disable", select_logs},
|
||||
{"help", help_log}
|
||||
};
|
||||
|
||||
// A Macro to create the Menu
|
||||
MENU2(log_menu, "Log", log_menu_commands, print_log_menu);
|
||||
|
||||
static bool
|
||||
print_log_menu(void)
|
||||
{
|
||||
int log_start;
|
||||
int log_end;
|
||||
byte last_log_num = eeprom_read_byte((uint8_t *) EE_LAST_LOG_NUM);
|
||||
|
||||
Serial.printf_P(PSTR("logs enabled: "));
|
||||
if (0 == log_bitmask) {
|
||||
Serial.printf_P(PSTR("none"));
|
||||
} else {
|
||||
// Macro to make the following code a bit easier on the eye.
|
||||
// Pass it the capitalised name of the log option, as defined
|
||||
// in defines.h but without the LOG_ prefix. It will check for
|
||||
// the bit being set and print the name of the log option to suit.
|
||||
#define PLOG(_s) if (log_bitmask & LOGBIT_ ## _s) Serial.printf_P(PSTR(" %S"), PSTR(#_s))
|
||||
PLOG(ATTITUDE_FAST);
|
||||
PLOG(ATTITUDE_MED);
|
||||
PLOG(GPS);
|
||||
PLOG(PM);
|
||||
PLOG(CTUN);
|
||||
PLOG(NTUN);
|
||||
PLOG(MODE);
|
||||
PLOG(RAW);
|
||||
PLOG(CMD);
|
||||
#undef PLOG
|
||||
}
|
||||
Serial.println();
|
||||
|
||||
if (last_log_num == 0) {
|
||||
Serial.printf_P(PSTR("\nNo logs available for download\n"));
|
||||
} else {
|
||||
|
||||
Serial.printf_P(PSTR("\n%d logs available for download\n"), last_log_num);
|
||||
for(int i=1;i<last_log_num+1;i++) {
|
||||
log_start = eeprom_read_word((uint16_t *) (EE_LOG_1_START+(i-1)*0x02));
|
||||
log_end = eeprom_read_word((uint16_t *) (EE_LOG_1_START+(i)*0x02))-1;
|
||||
if (i == last_log_num) {
|
||||
log_end = eeprom_read_word((uint16_t *) EE_LAST_LOG_PAGE);
|
||||
}
|
||||
Serial.printf_P(PSTR("Log number %d, start page %d, end page %d\n"),
|
||||
i, log_start, log_end);
|
||||
}
|
||||
Serial.println();
|
||||
}
|
||||
return(true);
|
||||
}
|
||||
|
||||
static int8_t
|
||||
dump_log(uint8_t argc, const Menu::arg *argv)
|
||||
{
|
||||
byte dump_log;
|
||||
int dump_log_start;
|
||||
int dump_log_end;
|
||||
byte last_log_num;
|
||||
|
||||
// check that the requested log number can be read
|
||||
dump_log = argv[1].i;
|
||||
last_log_num = eeprom_read_byte((uint8_t *) EE_LAST_LOG_NUM);
|
||||
if ((argc != 2) || (dump_log < 1) || (dump_log > last_log_num)) {
|
||||
Serial.printf_P(PSTR("bad log number\n"));
|
||||
return(-1);
|
||||
}
|
||||
|
||||
dump_log_start = eeprom_read_word((uint16_t *) (EE_LOG_1_START+(dump_log-1)*0x02));
|
||||
dump_log_end = eeprom_read_word((uint16_t *) (EE_LOG_1_START+(dump_log)*0x02))-1;
|
||||
if (dump_log == last_log_num) {
|
||||
dump_log_end = eeprom_read_word((uint16_t *) EE_LAST_LOG_PAGE);
|
||||
}
|
||||
Serial.printf_P(PSTR("Dumping Log number %d, start page %d, end page %d\n"),
|
||||
dump_log, dump_log_start, dump_log_end);
|
||||
Log_Read(dump_log_start, dump_log_end);
|
||||
Serial.printf_P(PSTR("Log read complete\n"));
|
||||
}
|
||||
|
||||
static int8_t
|
||||
erase_logs(uint8_t argc, const Menu::arg *argv)
|
||||
{
|
||||
for(int i = 10 ; i > 0; i--) {
|
||||
Serial.printf_P(PSTR("ATTENTION - Erasing log in %d seconds. Power off now to save log! \n"), i);
|
||||
delay(1000);
|
||||
}
|
||||
Serial.printf_P(PSTR("\nErasing log...\n"));
|
||||
for(int j = 1; j < 4001; j++)
|
||||
DataFlash.PageErase(j);
|
||||
eeprom_write_byte((uint8_t *)EE_LAST_LOG_NUM, 0);
|
||||
eeprom_write_byte((uint8_t *)EE_LAST_LOG_PAGE, 1);
|
||||
Serial.printf_P(PSTR("\nLog erased.\n"));
|
||||
}
|
||||
|
||||
static int8_t
|
||||
select_logs(uint8_t argc, const Menu::arg *argv)
|
||||
{
|
||||
uint16_t bits;
|
||||
|
||||
if (argc != 2) {
|
||||
Serial.printf_P(PSTR("missing log type\n"));
|
||||
return(-1);
|
||||
}
|
||||
|
||||
bits = 0;
|
||||
|
||||
// Macro to make the following code a bit easier on the eye.
|
||||
// Pass it the capitalised name of the log option, as defined
|
||||
// in defines.h but without the LOG_ prefix. It will check for
|
||||
// that name as the argument to the command, and set the bit in
|
||||
// bits accordingly.
|
||||
//
|
||||
if (!strcasecmp_P(argv[1].str, PSTR("all"))) {
|
||||
bits = ~(bits = 0);
|
||||
} else {
|
||||
#define TARG(_s) if (!strcasecmp_P(argv[1].str, PSTR(#_s))) bits |= LOGBIT_ ## _s
|
||||
TARG(ATTITUDE_FAST);
|
||||
TARG(ATTITUDE_MED);
|
||||
TARG(GPS);
|
||||
TARG(PM);
|
||||
TARG(CTUN);
|
||||
TARG(NTUN);
|
||||
TARG(MODE);
|
||||
TARG(RAW);
|
||||
TARG(CMD);
|
||||
#undef TARG
|
||||
}
|
||||
|
||||
if (!strcasecmp_P(argv[0].str, PSTR("enable"))) {
|
||||
log_bitmask |= bits;
|
||||
} else {
|
||||
log_bitmask &= ~bits;
|
||||
}
|
||||
save_user_configs(); // XXX this is a bit heavyweight...
|
||||
|
||||
return(0);
|
||||
}
|
||||
|
||||
int8_t
|
||||
process_logs(uint8_t argc, const Menu::arg *argv)
|
||||
{
|
||||
log_menu.run();
|
||||
}
|
||||
|
||||
// Write an attitude packet. Total length : 10 bytes
|
||||
void Log_Write_Attitude(int log_roll, int log_pitch, uint16_t log_yaw)
|
||||
{
|
||||
DataFlash.WriteByte(HEAD_BYTE1);
|
||||
DataFlash.WriteByte(HEAD_BYTE2);
|
||||
DataFlash.WriteByte(LOG_ATTITUDE_MSG);
|
||||
DataFlash.WriteInt(log_roll);
|
||||
DataFlash.WriteInt(log_pitch);
|
||||
DataFlash.WriteInt(log_yaw);
|
||||
DataFlash.WriteByte(END_BYTE);
|
||||
}
|
||||
|
||||
// Write a performance monitoring packet. Total length : 19 bytes
|
||||
void Log_Write_Performance()
|
||||
{
|
||||
DataFlash.WriteByte(HEAD_BYTE1);
|
||||
DataFlash.WriteByte(HEAD_BYTE2);
|
||||
DataFlash.WriteByte(LOG_PERFORMANCE_MSG);
|
||||
DataFlash.WriteLong(millis()- perf_mon_timer);
|
||||
DataFlash.WriteInt(mainLoop_count);
|
||||
DataFlash.WriteInt(G_Dt_max);
|
||||
DataFlash.WriteByte(gyro_sat_count);
|
||||
DataFlash.WriteByte(adc_constraints);
|
||||
DataFlash.WriteByte(renorm_sqrt_count);
|
||||
DataFlash.WriteByte(renorm_blowup_count);
|
||||
DataFlash.WriteByte(gps_fix_count);
|
||||
DataFlash.WriteInt((int)(imu_health*1000));
|
||||
DataFlash.WriteByte(END_BYTE);
|
||||
}
|
||||
|
||||
// Write a command processing packet. Total length : 19 bytes
|
||||
//void Log_Write_Cmd(byte num, byte id, byte p1, long alt, long lat, long lng)
|
||||
void Log_Write_Cmd(byte num, struct Location *wp)
|
||||
{
|
||||
DataFlash.WriteByte(HEAD_BYTE1);
|
||||
DataFlash.WriteByte(HEAD_BYTE2);
|
||||
DataFlash.WriteByte(LOG_CMD_MSG);
|
||||
DataFlash.WriteByte(num);
|
||||
DataFlash.WriteByte(wp->id);
|
||||
DataFlash.WriteByte(wp->p1);
|
||||
DataFlash.WriteLong(wp->alt);
|
||||
DataFlash.WriteLong(wp->lat);
|
||||
DataFlash.WriteLong(wp->lng);
|
||||
DataFlash.WriteByte(END_BYTE);
|
||||
}
|
||||
|
||||
void Log_Write_Startup(byte type)
|
||||
{
|
||||
DataFlash.WriteByte(HEAD_BYTE1);
|
||||
DataFlash.WriteByte(HEAD_BYTE2);
|
||||
DataFlash.WriteByte(LOG_STARTUP_MSG);
|
||||
DataFlash.WriteByte(type);
|
||||
DataFlash.WriteByte(wp_total);
|
||||
DataFlash.WriteByte(END_BYTE);
|
||||
|
||||
// create a location struct to hold the temp Waypoints for printing
|
||||
struct Location cmd = get_wp_with_index(0);
|
||||
Log_Write_Cmd(0, &cmd);
|
||||
|
||||
for (int i=1; i<wp_total; i++){
|
||||
cmd = get_wp_with_index(i);
|
||||
Log_Write_Cmd(i, &cmd);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
// Write a control tuning packet. Total length : 22 bytes
|
||||
void Log_Write_Control_Tuning()
|
||||
{
|
||||
DataFlash.WriteByte(HEAD_BYTE1);
|
||||
DataFlash.WriteByte(HEAD_BYTE2);
|
||||
DataFlash.WriteByte(LOG_CONTROL_TUNING_MSG);
|
||||
DataFlash.WriteInt((int)(servo_out[CH_ROLL]));
|
||||
DataFlash.WriteInt((int)nav_roll);
|
||||
DataFlash.WriteInt((int)roll_sensor);
|
||||
DataFlash.WriteInt((int)(servo_out[CH_PITCH]));
|
||||
DataFlash.WriteInt((int)nav_pitch);
|
||||
DataFlash.WriteInt((int)pitch_sensor);
|
||||
DataFlash.WriteInt((int)(servo_out[CH_THROTTLE]));
|
||||
DataFlash.WriteInt((int)(servo_out[CH_RUDDER]));
|
||||
DataFlash.WriteInt((int)AN[4]);
|
||||
DataFlash.WriteByte(END_BYTE);
|
||||
}
|
||||
|
||||
// Write a navigation tuning packet. Total length : 18 bytes
|
||||
void Log_Write_Nav_Tuning()
|
||||
{
|
||||
DataFlash.WriteByte(HEAD_BYTE1);
|
||||
DataFlash.WriteByte(HEAD_BYTE2);
|
||||
DataFlash.WriteByte(LOG_NAV_TUNING_MSG);
|
||||
DataFlash.WriteInt((uint16_t)yaw_sensor);
|
||||
DataFlash.WriteInt((int)wp_distance);
|
||||
DataFlash.WriteInt((uint16_t)target_bearing);
|
||||
DataFlash.WriteInt((uint16_t)nav_bearing);
|
||||
DataFlash.WriteInt(altitude_error);
|
||||
DataFlash.WriteInt((int)airspeed);
|
||||
DataFlash.WriteInt((int)(nav_gain_scaler*1000));
|
||||
DataFlash.WriteByte(END_BYTE);
|
||||
}
|
||||
|
||||
// Write a mode packet. Total length : 5 bytes
|
||||
void Log_Write_Mode(byte mode)
|
||||
{
|
||||
DataFlash.WriteByte(HEAD_BYTE1);
|
||||
DataFlash.WriteByte(HEAD_BYTE2);
|
||||
DataFlash.WriteByte(LOG_MODE_MSG);
|
||||
DataFlash.WriteByte(mode);
|
||||
DataFlash.WriteByte(END_BYTE);
|
||||
}
|
||||
|
||||
// Write an GPS packet. Total length : 30 bytes
|
||||
void Log_Write_GPS( long log_Time, long log_Lattitude, long log_Longitude, long log_mix_alt, long log_gps_alt,
|
||||
long log_Ground_Speed, long log_Ground_Course, byte log_Fix, byte log_NumSats)
|
||||
{
|
||||
DataFlash.WriteByte(HEAD_BYTE1);
|
||||
DataFlash.WriteByte(HEAD_BYTE2);
|
||||
DataFlash.WriteByte(LOG_GPS_MSG);
|
||||
DataFlash.WriteLong(log_Time);
|
||||
DataFlash.WriteByte(log_Fix);
|
||||
DataFlash.WriteByte(log_NumSats);
|
||||
DataFlash.WriteLong(log_Lattitude);
|
||||
DataFlash.WriteLong(log_Longitude);
|
||||
DataFlash.WriteLong(log_mix_alt);
|
||||
DataFlash.WriteLong(log_gps_alt);
|
||||
DataFlash.WriteLong(log_Ground_Speed);
|
||||
DataFlash.WriteLong(log_Ground_Course);
|
||||
DataFlash.WriteByte(END_BYTE);
|
||||
DataFlash.WriteByte(END_BYTE);
|
||||
}
|
||||
|
||||
// Write an raw accel/gyro data packet. Total length : 28 bytes
|
||||
void Log_Write_Raw()
|
||||
{
|
||||
DataFlash.WriteByte(HEAD_BYTE1);
|
||||
DataFlash.WriteByte(HEAD_BYTE2);
|
||||
DataFlash.WriteByte(LOG_RAW_MSG);
|
||||
for(int i=0;i<6;i++)
|
||||
DataFlash.WriteLong((long)(AN[i]*1000.0));
|
||||
DataFlash.WriteByte(END_BYTE);
|
||||
}
|
||||
|
||||
// Read an control tuning packet
|
||||
void Log_Read_Control_Tuning()
|
||||
{
|
||||
float logvar;
|
||||
|
||||
Serial.print("CTUN:");
|
||||
for (int y=1;y<10;y++) {
|
||||
logvar = DataFlash.ReadInt();
|
||||
if(y < 9) logvar = logvar/100.f;
|
||||
Serial.print(logvar);
|
||||
Serial.print(comma);
|
||||
}
|
||||
Serial.println(" ");
|
||||
}
|
||||
|
||||
// Read a nav tuning packet
|
||||
void Log_Read_Nav_Tuning()
|
||||
{
|
||||
Serial.print("NTUN:");
|
||||
Serial.print((float)((uint16_t)DataFlash.ReadInt())/100.0); // Yaw from IMU
|
||||
Serial.print(comma);
|
||||
Serial.print(DataFlash.ReadInt()); // wp_distance
|
||||
Serial.print(comma);
|
||||
Serial.print((float)((uint16_t)DataFlash.ReadInt())/100.0); // target_bearing - Bearing to Target
|
||||
Serial.print(comma);
|
||||
Serial.print((float)((uint16_t)DataFlash.ReadInt())/100.0); // nav_bearing - Bearing to steer
|
||||
Serial.print(comma);
|
||||
Serial.print((float)DataFlash.ReadInt()/100.0); // Altitude error
|
||||
Serial.print(comma);
|
||||
Serial.print((float)DataFlash.ReadInt()/100.0); // Airspeed
|
||||
Serial.print(comma);
|
||||
Serial.print((float)DataFlash.ReadInt()/1000.0); // nav_gain_scaler
|
||||
Serial.println(comma);
|
||||
}
|
||||
|
||||
// Read a performance packet
|
||||
void Log_Read_Performance()
|
||||
{
|
||||
long pm_time;
|
||||
int logvar;
|
||||
|
||||
Serial.print("PM:");
|
||||
pm_time = DataFlash.ReadLong();
|
||||
Serial.print(pm_time);
|
||||
Serial.print(comma);
|
||||
for (int y=1;y<9;y++) {
|
||||
if(y<3 || y>7) logvar = DataFlash.ReadInt();
|
||||
else logvar = DataFlash.ReadByte();
|
||||
//if(y > 7) logvar = logvar/1000.f;
|
||||
Serial.print(logvar);
|
||||
Serial.print(comma);
|
||||
}
|
||||
Serial.println(" ");
|
||||
}
|
||||
|
||||
// Read a command processing packet
|
||||
void Log_Read_Cmd()
|
||||
{
|
||||
byte logvarb;
|
||||
long logvarl;
|
||||
|
||||
Serial.print("CMD:");
|
||||
for(int i=1;i<4;i++) {
|
||||
logvarb = DataFlash.ReadByte();
|
||||
Serial.print(logvarb,DEC);
|
||||
Serial.print(comma);
|
||||
}
|
||||
for(int i=1;i<4;i++) {
|
||||
logvarl = DataFlash.ReadLong();
|
||||
Serial.print(logvarl,DEC);
|
||||
Serial.print(comma);
|
||||
}
|
||||
Serial.println(" ");
|
||||
}
|
||||
|
||||
void Log_Read_Startup()
|
||||
{
|
||||
byte logbyte = DataFlash.ReadByte();
|
||||
if (logbyte == TYPE_AIRSTART_MSG)
|
||||
Serial.print("AIR START - ");
|
||||
else if (logbyte == TYPE_GROUNDSTART_MSG)
|
||||
Serial.print("GROUND START - ");
|
||||
else
|
||||
Serial.print("UNKNOWN STARTUP TYPE -");
|
||||
Serial.print(DataFlash.ReadByte(), DEC);
|
||||
Serial.println(" commands in memory");
|
||||
}
|
||||
|
||||
// Read an attitude packet
|
||||
void Log_Read_Attitude()
|
||||
{
|
||||
int log_roll;
|
||||
int log_pitch;
|
||||
uint16_t log_yaw;
|
||||
|
||||
log_roll = DataFlash.ReadInt();
|
||||
log_pitch = DataFlash.ReadInt();
|
||||
log_yaw = (uint16_t)DataFlash.ReadInt();
|
||||
|
||||
Serial.print("ATT:");
|
||||
Serial.print(log_roll);
|
||||
Serial.print(comma);
|
||||
Serial.print(log_pitch);
|
||||
Serial.print(comma);
|
||||
Serial.print(log_yaw);
|
||||
Serial.println();
|
||||
}
|
||||
|
||||
// Read a mode packet
|
||||
void Log_Read_Mode()
|
||||
{
|
||||
byte mode;
|
||||
|
||||
mode = DataFlash.ReadByte();
|
||||
Serial.print("MOD:");
|
||||
switch (mode) {
|
||||
case 0:
|
||||
Serial.println("Manual");
|
||||
break;
|
||||
case 1:
|
||||
Serial.println("Stab");
|
||||
break;
|
||||
case 5:
|
||||
Serial.println("FBW_A");
|
||||
break;
|
||||
case 6:
|
||||
Serial.println("FBW_B");
|
||||
break;
|
||||
case 10:
|
||||
Serial.println("AUTO");
|
||||
break;
|
||||
case 11:
|
||||
Serial.println("RTL");
|
||||
break;
|
||||
case 12:
|
||||
Serial.println("Loiter");
|
||||
break;
|
||||
case 98:
|
||||
Serial.println("AS_COM");
|
||||
break;
|
||||
case 99:
|
||||
Serial.println("GS_COM");
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
// Read a GPS packet
|
||||
void Log_Read_GPS()
|
||||
{
|
||||
|
||||
Serial.print("GPS:");
|
||||
Serial.print(DataFlash.ReadLong()); // Time
|
||||
Serial.print(comma);
|
||||
Serial.print((int)DataFlash.ReadByte()); // Fix
|
||||
Serial.print(comma);
|
||||
Serial.print((int)DataFlash.ReadByte()); // Num of Sats
|
||||
Serial.print(comma);
|
||||
Serial.print((float)DataFlash.ReadLong()/t7, 7); // Lattitude
|
||||
Serial.print(comma);
|
||||
Serial.print((float)DataFlash.ReadLong()/t7, 7); // Longitude
|
||||
Serial.print(comma);
|
||||
Serial.print((float)DataFlash.ReadLong()/100.0); // Baro/gps altitude mix
|
||||
Serial.print(comma);
|
||||
Serial.print((float)DataFlash.ReadLong()/100.0); // GPS altitude
|
||||
Serial.print(comma);
|
||||
Serial.print((float)DataFlash.ReadLong()/100.0); // Ground Speed
|
||||
Serial.print(comma);
|
||||
Serial.println((float)DataFlash.ReadLong()/100.0); // Ground Course
|
||||
|
||||
}
|
||||
|
||||
// Read a raw accel/gyro packet
|
||||
void Log_Read_Raw()
|
||||
{
|
||||
float logvar;
|
||||
Serial.print("RAW:");
|
||||
for (int y=0;y<6;y++) {
|
||||
logvar = DataFlash.ReadLong()/1000.f;
|
||||
Serial.print(logvar);
|
||||
Serial.print(comma);
|
||||
}
|
||||
Serial.println(" ");
|
||||
}
|
||||
|
||||
// Read the DataFlash log memory : Packet Parser
|
||||
void Log_Read(int start_page, int end_page)
|
||||
{
|
||||
byte data;
|
||||
byte log_step=0;
|
||||
int packet_count=0;
|
||||
int page = start_page;
|
||||
|
||||
|
||||
DataFlash.StartRead(start_page);
|
||||
while (page < end_page && page != -1)
|
||||
{
|
||||
data = DataFlash.ReadByte();
|
||||
switch(log_step) //This is a state machine to read the packets
|
||||
{
|
||||
case 0:
|
||||
if(data==HEAD_BYTE1) // Head byte 1
|
||||
log_step++;
|
||||
break;
|
||||
case 1:
|
||||
if(data==HEAD_BYTE2) // Head byte 2
|
||||
log_step++;
|
||||
else
|
||||
log_step = 0;
|
||||
break;
|
||||
case 2:
|
||||
if(data==LOG_ATTITUDE_MSG){
|
||||
Log_Read_Attitude();
|
||||
log_step++;
|
||||
|
||||
}else if(data==LOG_MODE_MSG){
|
||||
Log_Read_Mode();
|
||||
log_step++;
|
||||
|
||||
}else if(data==LOG_CONTROL_TUNING_MSG){
|
||||
Log_Read_Control_Tuning();
|
||||
log_step++;
|
||||
|
||||
}else if(data==LOG_NAV_TUNING_MSG){
|
||||
Log_Read_Nav_Tuning();
|
||||
log_step++;
|
||||
|
||||
}else if(data==LOG_PERFORMANCE_MSG){
|
||||
Log_Read_Performance();
|
||||
log_step++;
|
||||
|
||||
}else if(data==LOG_RAW_MSG){
|
||||
Log_Read_Raw();
|
||||
log_step++;
|
||||
|
||||
}else if(data==LOG_CMD_MSG){
|
||||
Log_Read_Cmd();
|
||||
log_step++;
|
||||
|
||||
}else if(data==LOG_STARTUP_MSG){
|
||||
Log_Read_Startup();
|
||||
log_step++;
|
||||
}else {
|
||||
if(data==LOG_GPS_MSG){
|
||||
Log_Read_GPS();
|
||||
log_step++;
|
||||
} else {
|
||||
Serial.print("Error Reading Packet: ");
|
||||
Serial.print(packet_count);
|
||||
log_step=0; // Restart, we have a problem...
|
||||
}
|
||||
}
|
||||
break;
|
||||
case 3:
|
||||
if(data==END_BYTE){
|
||||
packet_count++;
|
||||
}else{
|
||||
Serial.print("Error Reading END_BYTE ");
|
||||
Serial.println(data,DEC);
|
||||
}
|
||||
log_step=0; // Restart sequence: new packet...
|
||||
break;
|
||||
}
|
||||
page = DataFlash.GetPage();
|
||||
}
|
||||
Serial.print("Number of packets read: ");
|
||||
Serial.println(packet_count);
|
||||
}
|
||||
|
||||
|
@ -0,0 +1,75 @@
|
||||
ArduPilotMega 1.0.0 Commands
|
||||
|
||||
Command Structure in bytes
|
||||
0 0x00 byte Command ID
|
||||
1 0x01 byte Parameter 1
|
||||
2 0x02 int Parameter 2
|
||||
3 0x03 ..
|
||||
4 0x04 long Parameter 3
|
||||
5 0x05 ..
|
||||
6 0x06 ..
|
||||
7 0x07 ..
|
||||
8 0x08 long Parameter 4
|
||||
9 0x09 ..
|
||||
10 0x0A ..
|
||||
11 0x0B ..
|
||||
|
||||
0x00 Reserved
|
||||
....
|
||||
0x0F Reserved
|
||||
|
||||
***********************************
|
||||
Commands 0x10 to 0x2F are commands that have a end criteria, eg "reached waypoint" or "reached altitude"
|
||||
***********************************
|
||||
Command ID Name Parameter 1 Altitude Latitude Longitude
|
||||
0x10 CMD_WAYPOINT - altitude lat lon
|
||||
0x11 CMD_LOITER (indefinitely) - altitude lat lon
|
||||
0x12 CMD_LOITER_N_TURNS turns altitude lat lon
|
||||
0x13 CMD_LOITER_TIME time (seconds*10) altitude lat lon
|
||||
0x14 CMD_RTL - altitude lat lon
|
||||
0x15 CMD_LAND - altitude lat lon
|
||||
0x16 CMD_TAKEOFF angle - - -
|
||||
|
||||
|
||||
***********************************
|
||||
May Commands - these commands are optional to finish
|
||||
Command ID Name Parameter 1 Parameter 2 Parameter 3 Parameter 4
|
||||
0x20 CMD_DELAY - - time (milliseconds) -
|
||||
0x21 CMD_CLIMB rate (cm/sec) alt (finish) - -
|
||||
0x22 CMD_LAND_OPTIONS distance to WP airspeed m/s, throttle %, pitch deg
|
||||
|
||||
|
||||
|
||||
***********************************
|
||||
Unexecuted commands >= 0x20 are dropped when ready for the next command <= 0x1F so plan/queue commands accordingly!
|
||||
For example if you had a string of 0x21 commands following a 0x10 command that had not finished when the waypoint was
|
||||
reached, the unexecuted 0x21 commands would be skipped and the next command <= 0x1F would be loaded
|
||||
***********************************
|
||||
Now Commands - these commands are executed once until no more new now commands are available
|
||||
0x31 CMD_RESET_INDEX
|
||||
0x32 CMD_GOTO_INDEX index repeat count
|
||||
0x33 CMD_GETVAR_INDEX variable ID
|
||||
0x34 CMD_SENDVAR_INDEX off/on = 0/1
|
||||
0x35 CMD_TELEMETRY off/on = 0/1
|
||||
|
||||
0x40 CMD_THROTTLE_CRUISE speed
|
||||
0x41 CMD_AIRSPEED_CRUISE speed
|
||||
0x44 CMD_RESET_HOME altitude lat lon
|
||||
|
||||
0x60 CMD_KP_GAIN array index gain value*100,000
|
||||
0x61 CMD_KI_GAIN array index gain value*100,000
|
||||
0x62 CMD_KD_GAIN array index gain value*100,000
|
||||
0x63 CMD_KI_MAX array index gain value*100,000
|
||||
0x64 CMD_KFF_GAIN array index gain value*100,000
|
||||
|
||||
0x70 CMD_RADIO_TRIM array index value
|
||||
0x71 CMD_RADIO_MAX array index value
|
||||
0x72 CMD_RADIO_MIN array index value
|
||||
0x73 CMD_ELEVON_TRIM array index value (index 0 = elevon 1 trim, 1 = elevon 2 trim)
|
||||
0x75 CMD_INDEX index
|
||||
|
||||
0x80 CMD_REPEAT type value delay in sec repeat count
|
||||
0x81 CMD_RELAY (0,1 to change swicth position)
|
||||
0x82 CMD_SERVO number value
|
||||
|
||||
|
@ -0,0 +1,228 @@
|
||||
// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: t -*-
|
||||
|
||||
void init_commands()
|
||||
{
|
||||
read_EEPROM_waypoint_info();
|
||||
wp_index = 0;
|
||||
command_must_index = 0;
|
||||
command_may_index = 0;
|
||||
next_command.id = CMD_BLANK;
|
||||
}
|
||||
|
||||
void update_auto()
|
||||
{
|
||||
if (wp_index == wp_total){
|
||||
return_to_launch();
|
||||
//wp_index = 0;
|
||||
}
|
||||
}
|
||||
|
||||
void reload_commands()
|
||||
{
|
||||
init_commands();
|
||||
read_command_index(); // Get wp_index = command_must_index from EEPROM
|
||||
if(wp_index > 0){
|
||||
decrement_WP_index;
|
||||
}
|
||||
}
|
||||
|
||||
// Getters
|
||||
// -------
|
||||
struct Location get_wp_with_index(int i)
|
||||
{
|
||||
struct Location temp;
|
||||
long mem;
|
||||
|
||||
|
||||
// Find out proper location in memory by using the start_byte position + the index
|
||||
// --------------------------------------------------------------------------------
|
||||
if (i > wp_total) {
|
||||
temp.id = CMD_BLANK;
|
||||
}else{
|
||||
// read WP position
|
||||
mem = (WP_START_BYTE) + (i * WP_SIZE);
|
||||
temp.id = eeprom_read_byte((uint8_t*)mem);
|
||||
mem++;
|
||||
temp.p1 = eeprom_read_byte((uint8_t*)mem);
|
||||
mem++;
|
||||
temp.alt = (long)eeprom_read_dword((uint32_t*)mem);
|
||||
mem += 4;
|
||||
temp.lat = (long)eeprom_read_dword((uint32_t*)mem);
|
||||
mem += 4;
|
||||
temp.lng = (long)eeprom_read_dword((uint32_t*)mem);
|
||||
}
|
||||
return temp;
|
||||
}
|
||||
|
||||
// Setters
|
||||
// -------
|
||||
void set_wp_with_index(struct Location temp, int i)
|
||||
{
|
||||
|
||||
i = constrain(i,0,wp_total);
|
||||
uint32_t mem = WP_START_BYTE + (i * WP_SIZE);
|
||||
|
||||
eeprom_write_byte((uint8_t *) mem, temp.id);
|
||||
|
||||
mem++;
|
||||
eeprom_write_byte((uint8_t *) mem, temp.p1);
|
||||
|
||||
mem++;
|
||||
eeprom_write_dword((uint32_t *) mem, temp.alt);
|
||||
|
||||
mem += 4;
|
||||
eeprom_write_dword((uint32_t *) mem, temp.lat);
|
||||
|
||||
mem += 4;
|
||||
eeprom_write_dword((uint32_t *) mem, temp.lng);
|
||||
}
|
||||
|
||||
void increment_WP_index()
|
||||
{
|
||||
if(wp_index < wp_total){
|
||||
wp_index++;
|
||||
SendDebug("MSG <increment_WP_index> WP index is incremented to ");
|
||||
SendDebugln(wp_index,DEC);
|
||||
}else{
|
||||
SendDebug("MSG <increment_WP_index> Failed to increment WP index of ");
|
||||
SendDebugln(wp_index,DEC);
|
||||
}
|
||||
}
|
||||
void decrement_WP_index()
|
||||
{
|
||||
if(wp_index > 0){
|
||||
wp_index--;
|
||||
}
|
||||
}
|
||||
|
||||
void loiter_at_location()
|
||||
{
|
||||
next_WP = current_loc;
|
||||
}
|
||||
|
||||
// add a new command at end of command set to RTL.
|
||||
void return_to_launch(void)
|
||||
{
|
||||
// home is WP 0
|
||||
// ------------
|
||||
wp_index = 0;
|
||||
|
||||
// Loads WP from Memory
|
||||
// --------------------
|
||||
set_next_WP(&home);
|
||||
|
||||
// Altitude to hold over home
|
||||
// Set by configuration tool
|
||||
// -------------------------
|
||||
next_WP.alt = read_alt_to_hold();
|
||||
}
|
||||
|
||||
struct Location get_LOITER_home_wp()
|
||||
{
|
||||
// read home position
|
||||
struct Location temp = get_wp_with_index(0);
|
||||
temp.id = CMD_LOITER;
|
||||
|
||||
temp.alt = read_alt_to_hold();
|
||||
return temp;
|
||||
}
|
||||
|
||||
/*
|
||||
This function stores waypoint commands
|
||||
It looks to see what the next command type is and finds the last command.
|
||||
*/
|
||||
void set_next_WP(struct Location *wp)
|
||||
{
|
||||
//send_message(SEVERITY_LOW,"load WP");
|
||||
SendDebug("MSG <set_next_wp> wp_index: ");
|
||||
SendDebugln(wp_index,DEC);
|
||||
send_message(MSG_COMMAND, wp_index);
|
||||
|
||||
// copy the current WP into the OldWP slot
|
||||
// ---------------------------------------
|
||||
prev_WP = current_loc;
|
||||
// Load the next_WP slot
|
||||
// ---------------------
|
||||
next_WP = *wp;
|
||||
// offset the altitude relative to home position
|
||||
// ---------------------------------------------
|
||||
next_WP.alt += home.alt;
|
||||
|
||||
// used to control FBW and limit the rate of climb
|
||||
// -----------------------------------------------
|
||||
target_altitude = current_loc.alt;
|
||||
offset_altitude = next_WP.alt - prev_WP.alt;
|
||||
|
||||
// zero out our loiter vals to watch for missed waypoints
|
||||
loiter_delta = 0;
|
||||
loiter_sum = 0;
|
||||
loiter_total = 0;
|
||||
|
||||
float rads = (abs(next_WP.lat)/t7) * 0.0174532925;
|
||||
//377,173,810 / 10,000,000 = 37.717381 * 0.0174532925 = 0.658292482926943
|
||||
scaleLongDown = cos(rads);
|
||||
scaleLongUp = 1.0f/cos(rads);
|
||||
|
||||
// this is handy for the groundstation
|
||||
wp_totalDistance = getDistance(¤t_loc, &next_WP);
|
||||
wp_distance = wp_totalDistance;
|
||||
|
||||
print_current_waypoints();
|
||||
|
||||
target_bearing = get_bearing(¤t_loc, &next_WP);
|
||||
old_target_bearing = target_bearing;
|
||||
// this is used to offset the shrinking longitude as we go towards the poles
|
||||
|
||||
// set a new crosstrack bearing
|
||||
// ----------------------------
|
||||
reset_crosstrack();
|
||||
}
|
||||
|
||||
|
||||
// run this at setup on the ground
|
||||
// -------------------------------
|
||||
void init_home()
|
||||
{
|
||||
SendDebugln("MSG: <init_home> init home");
|
||||
|
||||
// Extra read just in case
|
||||
// -----------------------
|
||||
//GPS.Read();
|
||||
|
||||
// block until we get a good fix
|
||||
// -----------------------------
|
||||
while (!GPS.new_data || !GPS.fix) {
|
||||
GPS.update();
|
||||
}
|
||||
home.id = CMD_WAYPOINT;
|
||||
home.lng = GPS.longitude; // Lon * 10**7
|
||||
home.lat = GPS.latitude; // Lat * 10**7
|
||||
home.alt = GPS.altitude;
|
||||
home_is_set = TRUE;
|
||||
|
||||
// ground altitude in centimeters for pressure alt calculations
|
||||
// ------------------------------------------------------------
|
||||
ground_alt = GPS.altitude;
|
||||
press_alt = GPS.altitude; // Set initial value for filter
|
||||
save_pressure_data();
|
||||
|
||||
// Save Home to EEPROM
|
||||
// -------------------
|
||||
set_wp_with_index(home, 0);
|
||||
|
||||
// Save prev loc
|
||||
// -------------
|
||||
prev_WP = home;
|
||||
|
||||
// Signal ready to fly
|
||||
// Make the servos wiggle - 3 times
|
||||
// -----------------------
|
||||
demo_servos(3);
|
||||
send_message(SEVERITY_LOW,"\n\n Ready to FLY.");
|
||||
// 12345678901234567890123456789012
|
||||
jeti_status(" X-DIY Ready to FLY");
|
||||
jeti_update();
|
||||
}
|
||||
|
||||
|
||||
|
@ -0,0 +1,480 @@
|
||||
// called by 10 Hz loop
|
||||
// --------------------
|
||||
void update_commands(void)
|
||||
{
|
||||
// This function loads commands into three buffers
|
||||
// when a new command is loaded, it is processed with process_XXX()
|
||||
// ----------------------------------------------------------------
|
||||
if((home_is_set == FALSE) || (control_mode != AUTO)){
|
||||
return; // don't do commands
|
||||
}
|
||||
|
||||
if(control_mode == AUTO){
|
||||
load_next_command();
|
||||
process_next_command();
|
||||
}
|
||||
|
||||
//verify_must();
|
||||
//verify_may();
|
||||
}
|
||||
|
||||
|
||||
void load_next_command()
|
||||
{
|
||||
// fetch next command if it's empty
|
||||
// --------------------------------
|
||||
if(next_command.id == CMD_BLANK){
|
||||
next_command = get_wp_with_index(wp_index+1);
|
||||
if(next_command.id != CMD_BLANK){
|
||||
//SendDebug("MSG <load_next_command> fetch found new cmd from list at index ");
|
||||
//SendDebug((wp_index+1),DEC);
|
||||
//SendDebug(" with cmd id ");
|
||||
//SendDebugln(next_command.id,DEC);
|
||||
}
|
||||
}
|
||||
|
||||
// If the preload failed, return or just Loiter
|
||||
// generate a dynamic command for RTL
|
||||
// --------------------------------------------
|
||||
if(next_command.id == CMD_BLANK){
|
||||
// we are out of commands!
|
||||
//send_message(SEVERITY_LOW,"out of commands!");
|
||||
//SendDebug("MSG <load_next_command> out of commands, wp_index: ");
|
||||
//SendDebugln(wp_index,DEC);
|
||||
|
||||
|
||||
switch (control_mode){
|
||||
case LAND:
|
||||
// don't get a new command
|
||||
break;
|
||||
|
||||
default:
|
||||
next_command = get_LOITER_home_wp();
|
||||
//SendDebug("MSG <load_next_command> Preload RTL cmd id: ");
|
||||
//SendDebugln(next_command.id,DEC);
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void process_next_command()
|
||||
{
|
||||
// these are waypoint/Must commands
|
||||
// ---------------------------------
|
||||
if (command_must_index == 0){ // no current command loaded
|
||||
if (next_command.id >= 0x10 && next_command.id <= 0x1F ){
|
||||
increment_WP_index();
|
||||
save_command_index(); // to Recover from in air Restart
|
||||
command_must_index = wp_index;
|
||||
|
||||
//SendDebug("MSG <process_next_command> new command_must_id ");
|
||||
//SendDebug(next_command.id,DEC);
|
||||
//SendDebug(" index:");
|
||||
//SendDebugln(command_must_index,DEC);
|
||||
if (log_bitmask & MASK_LOG_CMD)
|
||||
Log_Write_Cmd(wp_index, &next_command);
|
||||
process_must();
|
||||
}
|
||||
}
|
||||
|
||||
// these are May commands
|
||||
// ----------------------
|
||||
if (command_may_index == 0){
|
||||
if (next_command.id >= 0x20 && next_command.id <= 0x2F ){
|
||||
increment_WP_index();// this command is from the command list in EEPROM
|
||||
command_may_index = wp_index;
|
||||
//Serial.print("new command_may_index ");
|
||||
//Serial.println(command_may_index,DEC);
|
||||
if (log_bitmask & MASK_LOG_CMD)
|
||||
Log_Write_Cmd(wp_index, &next_command);
|
||||
process_may();
|
||||
}
|
||||
}
|
||||
|
||||
// these are do it now commands
|
||||
// ---------------------------
|
||||
if (next_command.id >= 0x30){
|
||||
increment_WP_index();// this command is from the command list in EEPROM
|
||||
if (log_bitmask & MASK_LOG_CMD)
|
||||
Log_Write_Cmd(wp_index, &next_command);
|
||||
process_now();
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
/*
|
||||
These functions implement the waypoint commands.
|
||||
*/
|
||||
void process_must()
|
||||
{
|
||||
//SendDebug("process must index: ");
|
||||
//SendDebugln(command_must_index,DEC);
|
||||
|
||||
send_message(SEVERITY_LOW,"New cmd: <process_must>");
|
||||
send_message(MSG_COMMAND, wp_index);
|
||||
|
||||
// clear May indexes
|
||||
command_may_index = 0;
|
||||
command_may_ID = 0;
|
||||
|
||||
command_must_ID = next_command.id;
|
||||
|
||||
// invalidate command so a new one is loaded
|
||||
// -----------------------------------------
|
||||
next_command.id = 0;
|
||||
|
||||
// reset navigation integrators
|
||||
// -------------------------
|
||||
reset_I();
|
||||
|
||||
// loads the waypoint into Next_WP struct
|
||||
// --------------------------------------
|
||||
set_next_WP(&next_command);
|
||||
|
||||
switch(command_must_ID){
|
||||
|
||||
case CMD_TAKEOFF: // TAKEOFF!
|
||||
// pitch in deg, airspeed m/s, throttle %, track WP 1 or 0
|
||||
takeoff_pitch = next_command.p1 * 100;
|
||||
takeoff_altitude = next_WP.alt; // next_WP.alt is calculated by the set_next_WP command
|
||||
next_WP.lat = home.lat + 1000; // so we don't have bad calcs
|
||||
next_WP.lng = home.lng + 1000; // so we don't have bad calcs
|
||||
break;
|
||||
|
||||
case CMD_WAYPOINT: // Navigate to Waypoint
|
||||
break;
|
||||
|
||||
case CMD_LAND: // LAND to Waypoint
|
||||
break;
|
||||
|
||||
case CMD_LOITER: // Loiter indefinitely
|
||||
break;
|
||||
|
||||
case CMD_LOITER_N_TURNS: // Loiter N Times
|
||||
loiter_total = next_command.p1 * 360;
|
||||
break;
|
||||
|
||||
case CMD_LOITER_TIME:
|
||||
loiter_time = millis();
|
||||
loiter_time_max = next_command.p1 * 10000; // seconds * 10 * 1000
|
||||
break;
|
||||
|
||||
case CMD_RTL:
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
void process_may()
|
||||
{
|
||||
//Serial.print("process_may cmd# ");
|
||||
//Serial.println(wp_index,DEC);
|
||||
command_may_ID = next_command.id;
|
||||
|
||||
// invalidate command so a new one is loaded
|
||||
// -----------------------------------------
|
||||
next_command.id = 0;
|
||||
|
||||
send_message(SEVERITY_LOW,"<process_may> New may command loaded:");
|
||||
send_message(MSG_COMMAND, wp_index);
|
||||
|
||||
// do the command
|
||||
// --------------
|
||||
switch(command_may_ID){
|
||||
|
||||
case CMD_DELAY: // Navigate to Waypoint
|
||||
delay_start = millis();
|
||||
delay_timeout = next_command.lat;
|
||||
break;
|
||||
|
||||
case CMD_LAND_OPTIONS: // Land this puppy
|
||||
//send_message(SEVERITY_LOW,"Landing");
|
||||
|
||||
// pitch in deg, airspeed m/s, throttle %, track WP 1 or 0
|
||||
landing_pitch = next_command.lng * 100;
|
||||
airspeed_cruise = next_command.alt * 100;
|
||||
throttle_cruise = next_command.lat;
|
||||
landing_distance = next_command.p1;
|
||||
//landing_roll = command.lng;
|
||||
|
||||
SendDebug("MSG: throttle_cruise = ");
|
||||
SendDebugln(throttle_cruise,DEC);
|
||||
break;
|
||||
|
||||
default:
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
void process_now()
|
||||
{
|
||||
const float t5 = 100000.0;
|
||||
//Serial.print("process_now cmd# ");
|
||||
//Serial.println(wp_index,DEC);
|
||||
|
||||
byte id = next_command.id;
|
||||
|
||||
// invalidate command so a new one is loaded
|
||||
// -----------------------------------------
|
||||
next_command.id = 0;
|
||||
|
||||
send_message(SEVERITY_LOW, "<process_now> New now command loaded: ");
|
||||
send_message(MSG_COMMAND, wp_index);
|
||||
|
||||
// do the command
|
||||
// --------------
|
||||
switch(id){
|
||||
|
||||
//case CMD_AP_MODE:
|
||||
//next_command.p1 = constrain(next_command.p1, MANUAL, LAND);
|
||||
//set_mode(next_command.p1);
|
||||
//break;
|
||||
|
||||
case CMD_RESET_INDEX:
|
||||
init_commands();
|
||||
break;
|
||||
|
||||
case CMD_GETVAR_INDEX:
|
||||
//
|
||||
break;
|
||||
|
||||
case CMD_SENDVAR_INDEX:
|
||||
//
|
||||
break;
|
||||
|
||||
case CMD_TELEMETRY:
|
||||
//
|
||||
break;
|
||||
|
||||
case CMD_AIRSPEED_CRUISE:
|
||||
airspeed_cruise = next_command.p1 * 100;
|
||||
// todo save to EEPROM
|
||||
break;
|
||||
|
||||
case CMD_THROTTLE_CRUISE:
|
||||
throttle_cruise = next_command.p1;
|
||||
// todo save to EEPROM
|
||||
break;
|
||||
|
||||
case CMD_RESET_HOME:
|
||||
init_home();
|
||||
break;
|
||||
|
||||
case CMD_KP_GAIN:
|
||||
kp[next_command.p1] = next_command.alt/t5;
|
||||
// todo save to EEPROM
|
||||
break;
|
||||
case CMD_KI_GAIN:
|
||||
ki[next_command.p1] = next_command.alt/t5;
|
||||
// todo save to EEPROM
|
||||
break;
|
||||
case CMD_KD_GAIN:
|
||||
kd[next_command.p1] = next_command.alt/t5;
|
||||
// todo save to EEPROM
|
||||
break;
|
||||
|
||||
case CMD_KI_MAX:
|
||||
integrator_max[next_command.p1] = next_command.alt/t5;
|
||||
// todo save to EEPROM
|
||||
break;
|
||||
|
||||
case CMD_KFF_GAIN:
|
||||
kff[next_command.p1] = next_command.alt/t5;
|
||||
// todo save to EEPROM
|
||||
break;
|
||||
|
||||
case CMD_RADIO_TRIM:
|
||||
radio_trim[next_command.p1] = next_command.alt;
|
||||
save_EEPROM_trims();
|
||||
break;
|
||||
|
||||
case CMD_RADIO_MAX:
|
||||
radio_max[next_command.p1] = next_command.alt;
|
||||
save_EEPROM_radio_minmax();
|
||||
break;
|
||||
|
||||
case CMD_RADIO_MIN:
|
||||
radio_min[next_command.p1] = next_command.alt;
|
||||
save_EEPROM_radio_minmax();
|
||||
break;
|
||||
|
||||
case CMD_REPEAT:
|
||||
new_event(&next_command);
|
||||
break;
|
||||
|
||||
case CMD_SERVO:
|
||||
//Serial.print("CMD_SERVO ");
|
||||
//Serial.print(next_command.p1,DEC);
|
||||
//Serial.print(" PWM: ");
|
||||
//Serial.println(next_command.alt,DEC);
|
||||
APM_RC.OutputCh(next_command.p1, next_command.alt);
|
||||
break;
|
||||
|
||||
case CMD_INDEX:
|
||||
command_must_index = 0;
|
||||
command_may_index = 0;
|
||||
wp_index = next_command.p1 - 1;
|
||||
break;
|
||||
|
||||
case CMD_RELAY:
|
||||
if(next_command.p1 = 0){
|
||||
relay_A();
|
||||
}else{
|
||||
relay_B();
|
||||
}
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
// Verify commands
|
||||
// ---------------
|
||||
void verify_must()
|
||||
{
|
||||
float power;
|
||||
|
||||
switch(command_must_ID) {
|
||||
|
||||
case CMD_TAKEOFF: // Takeoff!
|
||||
//Serial.print("verify_must cmd# ");
|
||||
//Serial.println(command_must_index,DEC);
|
||||
|
||||
if (GPS.ground_speed > 3){
|
||||
if(hold_course == -1){
|
||||
// save our current course to land
|
||||
hold_course = yaw_sensor;
|
||||
}
|
||||
}
|
||||
if(hold_course > -1){
|
||||
// recalc bearing error with hold_course;
|
||||
nav_bearing = hold_course;
|
||||
// recalc bearing error
|
||||
calc_bearing_error();
|
||||
}
|
||||
if (current_loc.alt > (home.alt + takeoff_altitude)) {
|
||||
command_must_index = 0;
|
||||
hold_course = -1;
|
||||
}
|
||||
break;
|
||||
|
||||
|
||||
|
||||
case CMD_LAND:
|
||||
// we don't verify landing - we never go to a new Must command after Land.
|
||||
if ( ((wp_distance > 0) && (wp_distance <= (2*GPS.ground_speed/100))) || (current_loc.alt <= next_WP.alt + 300) )
|
||||
{
|
||||
land_complete = 1; //Set land_complete if we are within 2 seconds distance or within 3 meters altitude
|
||||
if(hold_course == -1){
|
||||
// save our current course to land
|
||||
//hold_course = yaw_sensor;
|
||||
// save the course line of the runway to land
|
||||
hold_course = crosstrack_bearing;
|
||||
}
|
||||
}
|
||||
if(hold_course > -1){
|
||||
// recalc bearing error with hold_course;
|
||||
nav_bearing = hold_course;
|
||||
// recalc bearing error
|
||||
calc_bearing_error();
|
||||
}
|
||||
update_crosstrack();
|
||||
|
||||
break;
|
||||
|
||||
case CMD_WAYPOINT: // reach a waypoint
|
||||
hold_course == -1;
|
||||
update_crosstrack();
|
||||
if ((wp_distance > 0) && (wp_distance <= wp_radius)) {
|
||||
SendDebug("MSG <verify_must: CMD_WAYPOINT> REACHED_WAYPOINT #");
|
||||
SendDebugln(command_must_index,DEC);
|
||||
// clear the command queue;
|
||||
command_must_index = 0;
|
||||
}
|
||||
// add in a more complex case
|
||||
// Doug to do
|
||||
if(loiter_sum > 300){
|
||||
send_message(SEVERITY_MEDIUM,"<verify_must: CMD_WAYPOINT> Missed WP");
|
||||
command_must_index = 0;
|
||||
}
|
||||
break;
|
||||
|
||||
case CMD_LOITER_N_TURNS: // LOITER N times
|
||||
if (wp_distance <= loiter_radius){
|
||||
nav_bearing -= 9000;
|
||||
|
||||
}else if (wp_distance < (loiter_radius + LOITER_RANGE)){
|
||||
power = -((float)(wp_distance - loiter_radius - LOITER_RANGE) / LOITER_RANGE);
|
||||
power = constrain(power, 0, 1);
|
||||
nav_bearing -= power * 9000;
|
||||
|
||||
}else{
|
||||
update_crosstrack();
|
||||
}
|
||||
if(loiter_sum > loiter_total) {
|
||||
loiter_total = 0;
|
||||
send_message(SEVERITY_LOW,"<verify_must: CMD_LOITER_N_TURNS> LOITER orbits complete ");
|
||||
// clear the command queue;
|
||||
command_must_index = 0;
|
||||
}
|
||||
// recalc bearing error
|
||||
nav_bearing = wrap_360(nav_bearing);
|
||||
calc_bearing_error();
|
||||
break;
|
||||
|
||||
case CMD_LOITER_TIME: // loiter N milliseconds
|
||||
|
||||
if (wp_distance <= loiter_radius){
|
||||
nav_bearing -= 9000;
|
||||
|
||||
}else if (wp_distance < (loiter_radius + LOITER_RANGE)){
|
||||
power = -((float)(wp_distance - loiter_radius - LOITER_RANGE) / LOITER_RANGE);
|
||||
power = constrain(power, 0, 1);
|
||||
nav_bearing -= power * 9000;
|
||||
|
||||
}else{
|
||||
update_crosstrack();
|
||||
}
|
||||
|
||||
if ((millis() - loiter_time) > loiter_time_max) {
|
||||
send_message(SEVERITY_LOW,"<verify_must: CMD_LOITER_TIME> LOITER time complete ");
|
||||
command_must_index = 0;
|
||||
}
|
||||
nav_bearing = wrap_360(nav_bearing);
|
||||
// recalc bearing error
|
||||
calc_bearing_error();
|
||||
break;
|
||||
|
||||
case CMD_RTL:
|
||||
if (wp_distance <= 30) {
|
||||
//Serial.println("REACHED_HOME");
|
||||
command_must_index = 0;
|
||||
}
|
||||
break;
|
||||
|
||||
case CMD_LOITER: // Just plain LOITER
|
||||
break;
|
||||
|
||||
default:
|
||||
send_message(SEVERITY_HIGH,"<verify_must: default> No current Must commands");
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
void verify_may()
|
||||
{
|
||||
switch(command_may_ID) {
|
||||
case CMD_DELAY:
|
||||
if ((millis() - delay_start) > delay_timeout){
|
||||
command_may_index = 0;
|
||||
delay_timeout = 0;
|
||||
}
|
||||
|
||||
case CMD_LAND_OPTIONS:
|
||||
if ((wp_distance > 0) && (wp_distance <= landing_distance)) {
|
||||
//Serial.print("XXX REACHED_WAYPOINT #");
|
||||
//Serial.println(command_must_index,DEC);
|
||||
// clear the command queue;
|
||||
command_may_index = 0;
|
||||
}
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
@ -0,0 +1,569 @@
|
||||
// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: t -*-
|
||||
//
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
//
|
||||
// WARNING WARNING WARNING WARNING WARNING WARNING WARNING WARNING WARNING
|
||||
//
|
||||
// DO NOT EDIT this file to adjust your configuration. Create your own
|
||||
// APM_Config.h and use APM_Config.h.example as a reference.
|
||||
//
|
||||
// WARNING WARNING WARNING WARNING WARNING WARNING WARNING WARNING WARNING
|
||||
///
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
//
|
||||
// Default and automatic configuration details.
|
||||
//
|
||||
// Notes for maintainers:
|
||||
//
|
||||
// - Try to keep this file organised in the same order as APM_Config.h.example
|
||||
//
|
||||
|
||||
#include "defines.h"
|
||||
|
||||
///
|
||||
/// DO NOT EDIT THIS INCLUDE - if you want to make a local change, make that
|
||||
/// change in your local copy of APM_Config.h.
|
||||
///
|
||||
#include "APM_Config.h" // <== THIS INCLUDE, DO NOT EDIT IT
|
||||
///
|
||||
/// DO NOT EDIT THIS INCLUDE - if you want to make a local change, make that
|
||||
/// change in your local copy of APM_Config.h.
|
||||
///
|
||||
|
||||
// Just so that it's completely clear...
|
||||
#define ENABLED 1
|
||||
#define DISABLED 0
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// HARDWARE CONFIGURATION AND CONNECTIONS
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// ENABLE_HIL
|
||||
// Hardware in the loop output
|
||||
//
|
||||
#ifndef ENABLE_HIL
|
||||
# define ENABLE_HIL DISABLED
|
||||
#endif
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// GPS_PROTOCOL
|
||||
//
|
||||
#ifndef GPS_PROTOCOL
|
||||
# error XXX
|
||||
# error XXX You must define GPS_PROTOCOL in APM_Config.h
|
||||
# error XXX
|
||||
#endif
|
||||
|
||||
// The X-Plane GCS requires the IMU GPS configuration
|
||||
#if (ENABLE_HIL == ENABLED) && (GPS_PROTOCOL != GPS_PROTOCOL_IMU)
|
||||
# error Must select GPS_PROTOCOL_IMU when configuring for X-Plane or Flight Gear HIL
|
||||
#endif
|
||||
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// AIRSPEED_SENSOR
|
||||
// AIRSPEED_RATIO
|
||||
//
|
||||
#ifndef AIRSPEED_SENSOR
|
||||
# define AIRSPEED_SENSOR DISABLED
|
||||
#endif
|
||||
#ifndef AIRSPEED_RATIO
|
||||
# define AIRSPEED_RATIO 1.9936 // Note - this varies from the value in ArduPilot due to the difference in ADC resolution
|
||||
#endif
|
||||
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// GCS_PROTOCOL
|
||||
// GCS_PORT
|
||||
//
|
||||
#ifndef GCS_PROTOCOL
|
||||
# define GCS_PROTOCOL GCS_PROTOCOL_STANDARD
|
||||
#endif
|
||||
|
||||
#ifndef GCS_PORT
|
||||
# if (GCS_PROTOCOL == GCS_PROTOCOL_STANDARD) || (GCS_PROTOCOL == GCS_PROTOCOL_LEGACY) || (GCS_PROTOCOL == GCS_PROTOCOL_DEBUGTERMINAL)
|
||||
# define GCS_PORT 3
|
||||
# else
|
||||
# define GCS_PORT 0
|
||||
# endif
|
||||
#endif
|
||||
|
||||
#ifndef DEBUGTERMINAL_VERBOSE
|
||||
# define DEBUGTERMINAL_VERBOSE 1 //Set this to 1 to get more detail in DEBUGTERMINAL messages, or 0 to save flash space
|
||||
#endif
|
||||
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// Serial port speeds.
|
||||
//
|
||||
#ifndef SERIAL0_BAUD
|
||||
# define SERIAL0_BAUD 38400
|
||||
#endif
|
||||
#ifndef SERIAL3_BAUD
|
||||
# define SERIAL3_BAUD 115200
|
||||
#endif
|
||||
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// Battery monitoring
|
||||
//
|
||||
#ifndef BATTERY_EVENT
|
||||
# define BATTERY_EVENT DISABLED
|
||||
#endif
|
||||
#ifndef BATTERY_TYPE
|
||||
# define BATTERY_TYPE 0
|
||||
#endif
|
||||
#ifndef LOW_VOLTAGE
|
||||
# define LOW_VOLTAGE 11.4
|
||||
#endif
|
||||
#ifndef VOLT_DIV_RATIO
|
||||
# define VOLT_DIV_RATIO 3.0
|
||||
#endif
|
||||
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// INPUT_VOLTAGE
|
||||
//
|
||||
#ifndef INPUT_VOLTAGE
|
||||
# define INPUT_VOLTAGE 5.0
|
||||
#endif
|
||||
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// RADIO CONFIGURATION
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// Radio channel limits
|
||||
//
|
||||
// Note that these are not called out in APM_Config.h.example as there
|
||||
// is currently no configured behaviour for these channels.
|
||||
//
|
||||
#ifndef CH5_MIN
|
||||
# define CH5_MIN 1000
|
||||
#endif
|
||||
#ifndef CH5_MAX
|
||||
# define CH5_MAX 2000
|
||||
#endif
|
||||
#ifndef CH6_MIN
|
||||
# define CH6_MIN 1000
|
||||
#endif
|
||||
#ifndef CH6_MAX
|
||||
# define CH6_MAX 2000
|
||||
#endif
|
||||
#ifndef CH7_MIN
|
||||
# define CH7_MIN 1000
|
||||
#endif
|
||||
#ifndef CH7_MAX
|
||||
# define CH7_MAX 2000
|
||||
#endif
|
||||
#ifndef CH8_MIN
|
||||
# define CH8_MIN 1000
|
||||
#endif
|
||||
#ifndef CH8_MAX
|
||||
# define CH8_MAX 2000
|
||||
#endif
|
||||
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// FLIGHT_MODE
|
||||
// FLIGHT_MODE_CHANNEL
|
||||
//
|
||||
#ifndef FLIGHT_MODE_CHANNEL
|
||||
# define FLIGHT_MODE_CHANNEL 8
|
||||
#endif
|
||||
#if (FLIGHT_MODE_CHANNEL != 5) && (FLIGHT_MODE_CHANNEL != 6) && (FLIGHT_MODE_CHANNEL != 7) && (FLIGHT_MODE_CHANNEL != 8)
|
||||
# error XXX
|
||||
# error XXX You must set FLIGHT_MODE_CHANNEL to 5, 6, 7 or 8
|
||||
# error XXX
|
||||
#endif
|
||||
|
||||
#if !defined(FLIGHT_MODE_1)
|
||||
# define FLIGHT_MODE_1 FLY_BY_WIRE_A
|
||||
#endif
|
||||
#if !defined(FLIGHT_MODE_2)
|
||||
# define FLIGHT_MODE_2 FLY_BY_WIRE_A
|
||||
#endif
|
||||
#if !defined(FLIGHT_MODE_3)
|
||||
# define FLIGHT_MODE_3 STABILIZE
|
||||
#endif
|
||||
#if !defined(FLIGHT_MODE_4)
|
||||
# define FLIGHT_MODE_4 STABILIZE
|
||||
#endif
|
||||
#if !defined(FLIGHT_MODE_5)
|
||||
# define FLIGHT_MODE_5 MANUAL
|
||||
#endif
|
||||
#if !defined(FLIGHT_MODE_6)
|
||||
# define FLIGHT_MODE_6 MANUAL
|
||||
#endif
|
||||
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// THROTTLE_FAILSAFE
|
||||
// THROTTLE_FS_VALUE
|
||||
// THROTTLE_FAILSAFE_ACTION
|
||||
//
|
||||
#ifndef THROTTLE_FAILSAFE
|
||||
# define THROTTLE_FAILSAFE DISABLED
|
||||
#endif
|
||||
#ifndef THROTTE_FS_VALUE
|
||||
# define THROTTLE_FS_VALUE 975
|
||||
#endif
|
||||
#ifndef THROTTLE_FAILSAFE_ACTION
|
||||
# define THROTTLE_FAILSAFE_ACTION 2
|
||||
#endif
|
||||
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// AUTO_TRIM
|
||||
//
|
||||
#ifndef AUTO_TRIM
|
||||
# define AUTO_TRIM ENABLED
|
||||
#endif
|
||||
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// THROTTLE_REVERSE
|
||||
//
|
||||
#ifndef THROTTLE_REVERSE
|
||||
# define THROTTLE_REVERSE DISABLED
|
||||
#endif
|
||||
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// ENABLE_STICK_MIXING
|
||||
//
|
||||
#ifndef ENABLE_STICK_MIXING
|
||||
# define ENABLE_STICK_MIXING ENABLED
|
||||
#endif
|
||||
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// THROTTLE_OUT
|
||||
//
|
||||
#ifndef THROTTE_OUT
|
||||
# define THROTTLE_OUT ENABLED
|
||||
#endif
|
||||
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// STARTUP BEHAVIOUR
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// GROUND_START_DELAY
|
||||
//
|
||||
#ifndef GROUND_START_DELAY
|
||||
# define GROUND_START_DELAY 0
|
||||
#endif
|
||||
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// ENABLE_AIR_START
|
||||
//
|
||||
#ifndef ENABLE_AIR_START
|
||||
# define ENABLE_AIR_START DISABLED
|
||||
#endif
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// ENABLE REVERSE_SWITCH
|
||||
//
|
||||
#ifndef REVERSE_SWITCH
|
||||
# define REVERSE_SWITCH ENABLED
|
||||
#endif
|
||||
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// FLIGHT AND NAVIGATION CONTROL
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// Altitude measurement and control.
|
||||
//
|
||||
#ifndef AOA
|
||||
# define AOA 100 // XXX still 100ths of a degree
|
||||
#endif
|
||||
#ifndef ALT_EST_GAIN
|
||||
# define ALT_EST_GAIN 0.01
|
||||
#endif
|
||||
#ifndef ALTITUDE_MIX
|
||||
# define ALTITUDE_MIX 0
|
||||
#endif
|
||||
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// AIRSPEED_CRUISE
|
||||
//
|
||||
#ifndef AIRSPEED_CRUISE
|
||||
# define AIRSPEED_CRUISE 10
|
||||
#endif
|
||||
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// FLY_BY_WIRE_B airspeed control
|
||||
//
|
||||
#ifndef AIRSPEED_FBW_MIN
|
||||
# define AIRSPEED_FBW_MIN 6
|
||||
#endif
|
||||
#ifndef AIRSPEED_FBW_MAX
|
||||
# define AIRSPEED_FBW_MAX 30
|
||||
#endif
|
||||
#ifndef THROTTLE_ALT_P
|
||||
# define THROTTLE_ALT_P 0.32
|
||||
#endif
|
||||
#ifndef THROTTLE_ALT_I
|
||||
# define THROTTLE_ALT_I 0.04
|
||||
#endif
|
||||
#ifndef THROTTLE_ALT_D
|
||||
# define THROTTLE_ALT_D 0.0
|
||||
#endif
|
||||
#ifndef THROTTLE_ALT_INT_MAX
|
||||
# define THROTTLE_ALT_INT_MAX 20
|
||||
#endif
|
||||
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// Throttle control
|
||||
//
|
||||
#ifndef THROTTLE_MIN
|
||||
# define THROTTLE_MIN 0
|
||||
#endif
|
||||
#ifndef THROTTLE_CRUISE
|
||||
# define THROTTLE_CRUISE 45
|
||||
#endif
|
||||
#ifndef THROTTLE_MAX
|
||||
# define THROTTLE_MAX 75
|
||||
#endif
|
||||
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// Autopilot control limits
|
||||
//
|
||||
#ifndef HEAD_MAX
|
||||
# define HEAD_MAX 45
|
||||
#endif
|
||||
#ifndef PITCH_MAX
|
||||
# define PITCH_MAX 15
|
||||
#endif
|
||||
#ifndef PITCH_MIN
|
||||
# define PITCH_MIN -25
|
||||
#endif
|
||||
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// Attitude control gains
|
||||
//
|
||||
#ifndef SERVO_ROLL_P
|
||||
# define SERVO_ROLL_P 0.4
|
||||
#endif
|
||||
#ifndef SERVO_ROLL_I
|
||||
# define SERVO_ROLL_I 0.0
|
||||
#endif
|
||||
#ifndef SERVO_ROLL_D
|
||||
# define SERVO_ROLL_D 0.0
|
||||
#endif
|
||||
#ifndef SERVO_ROLL_INT_MAX
|
||||
# define SERVO_ROLL_INT_MAX 5
|
||||
#endif
|
||||
#ifndef ROLL_SLEW_LIMIT
|
||||
# define ROLL_SLEW_LIMIT 0
|
||||
#endif
|
||||
#ifndef SERVO_PITCH_P
|
||||
# define SERVO_PITCH_P 0.6
|
||||
#endif
|
||||
#ifndef SERVO_PITCH_I
|
||||
# define SERVO_PITCH_I 0.0
|
||||
#endif
|
||||
#ifndef SERVO_PITCH_D
|
||||
# define SERVO_PITCH_D 0.0
|
||||
#endif
|
||||
#ifndef SERVO_PITCH_INT_MAX
|
||||
# define SERVO_PITCH_INT_MAX 5
|
||||
#endif
|
||||
#ifndef PITCH_COMP
|
||||
# define PITCH_COMP 0.2
|
||||
#endif
|
||||
#ifndef SERVO_YAW_P
|
||||
# define SERVO_YAW_P 0.5
|
||||
#endif
|
||||
#ifndef SERVO_YAW_I
|
||||
# define SERVO_YAW_I 0.0
|
||||
#endif
|
||||
#ifndef SERVO_YAW_D
|
||||
# define SERVO_YAW_D 0.0
|
||||
#endif
|
||||
#ifndef SERVO_YAW_INT_MAX
|
||||
# define SERVO_YAW_INT_MAX 5
|
||||
#endif
|
||||
#ifndef RUDDER_MIX
|
||||
# define RUDDER_MIX 0.5
|
||||
#endif
|
||||
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// Navigation control gains
|
||||
//
|
||||
#ifndef NAV_ROLL_P
|
||||
# define NAV_ROLL_P 0.7
|
||||
#endif
|
||||
#ifndef NAV_ROLL_I
|
||||
# define NAV_ROLL_I 0.01
|
||||
#endif
|
||||
#ifndef NAV_ROLL_D
|
||||
# define NAV_ROLL_D 0.02
|
||||
#endif
|
||||
#ifndef NAV_ROLL_INT_MAX
|
||||
# define NAV_ROLL_INT_MAX 5
|
||||
#endif
|
||||
#ifndef NAV_PITCH_ASP_P
|
||||
# define NAV_PITCH_ASP_P 0.65
|
||||
#endif
|
||||
#ifndef NAV_PITCH_ASP_I
|
||||
# define NAV_PITCH_ASP_I 0.0
|
||||
#endif
|
||||
#ifndef NAV_PITCH_ASP_D
|
||||
# define NAV_PITCH_ASP_D 0.0
|
||||
#endif
|
||||
#ifndef NAV_PITCH_ASP_INT_MAX
|
||||
# define NAV_PITCH_ASP_INT_MAX 5
|
||||
#endif
|
||||
#ifndef NAV_PITCH_ALT_P
|
||||
# define NAV_PITCH_ALT_P 0.65
|
||||
#endif
|
||||
#ifndef NAV_PITCH_ALT_I
|
||||
# define NAV_PITCH_ALT_I 0.0
|
||||
#endif
|
||||
#ifndef NAV_PITCH_ALT_D
|
||||
# define NAV_PITCH_ALT_D 0.0
|
||||
#endif
|
||||
#ifndef NAV_PITCH_ALT_INT_MAX
|
||||
# define NAV_PITCH_ALT_INT_MAX 5
|
||||
#endif
|
||||
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// Energy/Altitude control gains
|
||||
//
|
||||
#ifndef THROTTLE_TE_P
|
||||
# define THROTTLE_TE_P 0.50
|
||||
#endif
|
||||
#ifndef THROTTLE_TE_I
|
||||
# define THROTTLE_TE_I 0.0
|
||||
#endif
|
||||
#ifndef THROTTLE_TE_D
|
||||
# define THROTTLE_TE_D 0.0
|
||||
#endif
|
||||
#ifndef THROTTLE_TE_INT_MAX
|
||||
# define THROTTLE_TE_INT_MAX 20
|
||||
#endif
|
||||
#ifndef THROTTLE_SLEW_LIMIT
|
||||
# define THROTTLE_SLEW_LIMIT 0
|
||||
#endif
|
||||
#ifndef P_TO_T
|
||||
# define P_TO_T 2.5
|
||||
#endif
|
||||
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// Crosstrack compensation
|
||||
//
|
||||
#ifndef XTRACK_GAIN
|
||||
# define XTRACK_GAIN 0.01
|
||||
#endif
|
||||
#ifndef XTRACK_ENTRY_ANGLE
|
||||
# define XTRACK_ENTRY_ANGLE 30
|
||||
#endif
|
||||
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// DEBUGGING
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// DEBUG_SUBSYSTEM
|
||||
//
|
||||
#ifndef DEBUG_SUBSYSTEM
|
||||
# define DEBUG_SUBSYSTEM 0
|
||||
#endif
|
||||
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// DEBUG_LEVEL
|
||||
//
|
||||
#ifndef DEBUG_LEVEL
|
||||
# define DEBUG_LEVEL SEVERITY_LOW
|
||||
#endif
|
||||
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// Dataflash logging control
|
||||
//
|
||||
#ifndef LOG_ATTITUDE_FAST
|
||||
# define LOG_ATTITUDE_FAST DISABLED
|
||||
#endif
|
||||
#ifndef LOG_ATTITUDE_MED
|
||||
# define LOG_ATTITUDE_MED ENABLED
|
||||
#endif
|
||||
#ifndef LOG_GPS
|
||||
# define LOG_GPS ENABLED
|
||||
#endif
|
||||
#ifndef LOG_PM
|
||||
# define LOG_PM ENABLED
|
||||
#endif
|
||||
#ifndef LOG_CTUN
|
||||
# define LOG_CTUN DISABLED
|
||||
#endif
|
||||
#ifndef LOG_NTUN
|
||||
# define LOG_NTUN DISABLED
|
||||
#endif
|
||||
#ifndef LOG_MODE
|
||||
# define LOG_MODE ENABLED
|
||||
#endif
|
||||
#ifndef LOG_RAW
|
||||
# define LOG_RAW DISABLED
|
||||
#endif
|
||||
#ifndef LOG_CMD
|
||||
# define LOG_CMD ENABLED
|
||||
#endif
|
||||
|
||||
#ifndef DEBUG_PORT
|
||||
# define DEBUG_PORT 0
|
||||
#endif
|
||||
|
||||
#if DEBUG_PORT == 0
|
||||
# define SendDebug Serial.print
|
||||
# define SendDebugln Serial.println
|
||||
#elif DEBUG_PORT == 1
|
||||
# define SendDebug Serial1.print
|
||||
# define SendDebugln Serial1.println
|
||||
#elif DEBUG_PORT == 2
|
||||
# define SendDebug Serial2.print
|
||||
# define SendDebugln Serial2.println
|
||||
#elif DEBUG_PORT == 3
|
||||
# define SendDebug Serial3.print
|
||||
# define SendDebugln Serial3.println
|
||||
#endif
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
// Navigation defaults
|
||||
//
|
||||
#ifndef WP_RADIUS_DEFAULT
|
||||
# define WP_RADIUS_DEFAULT 20
|
||||
#endif
|
||||
#ifndef LOITER_RADIUS_DEFAULT
|
||||
# define LOITER_RADIUS_DEFAULT 30
|
||||
#endif
|
@ -0,0 +1,58 @@
|
||||
void read_control_switch()
|
||||
{
|
||||
byte switchPosition = readSwitch();
|
||||
if (oldSwitchPosition != switchPosition){
|
||||
|
||||
set_mode(flight_modes[switchPosition]);
|
||||
|
||||
oldSwitchPosition = switchPosition;
|
||||
|
||||
// reset navigation integrators
|
||||
// -------------------------
|
||||
reset_I();
|
||||
}
|
||||
}
|
||||
/*
|
||||
byte readSwitch(void){
|
||||
int pulsewidth = APM_RC.InputCh(FLIGHT_MODE_CHANNEL - 1);
|
||||
if (pulsewidth > 1230 && pulsewidth <= 1360) return 1;
|
||||
if (pulsewidth > 1360 && pulsewidth <= 1490) return 2;
|
||||
if (pulsewidth > 1490 && pulsewidth <= 1620) return 3;
|
||||
if (pulsewidth > 1620 && pulsewidth <= 1749) return 4; // Software Manual
|
||||
if (pulsewidth >= 1750) return 5; // Hardware Manual
|
||||
return 0;
|
||||
}
|
||||
*/
|
||||
byte readSwitch(void){
|
||||
int pulsewidth = APM_RC.InputCh(FLIGHT_MODE_CHANNEL - 1);
|
||||
if (pulsewidth > FLIGHT_MODE_5_BOUNDARY) return 5;
|
||||
if (pulsewidth > FLIGHT_MODE_4_BOUNDARY) return 4;
|
||||
if (pulsewidth > FLIGHT_MODE_3_BOUNDARY) return 3;
|
||||
if (pulsewidth > FLIGHT_MODE_2_BOUNDARY) return 2;
|
||||
if (pulsewidth > FLIGHT_MODE_1_BOUNDARY) return 1;
|
||||
return 0;
|
||||
}
|
||||
|
||||
void reset_control_switch()
|
||||
{
|
||||
oldSwitchPosition = 0;
|
||||
read_control_switch();
|
||||
SendDebug("MSG: reset_control_switch");
|
||||
SendDebugln(oldSwitchPosition , DEC);
|
||||
}
|
||||
|
||||
void update_servo_switches()
|
||||
{
|
||||
// up is reverse
|
||||
// up is elevon
|
||||
mix_mode = (PINL & 128) ? 1 : 0;
|
||||
if (mix_mode == 0) {
|
||||
reverse_roll = (PINE & 128) ? 1 : -1;
|
||||
reverse_pitch = (PINE & 64) ? 1 : -1;
|
||||
reverse_rudder = (PINL & 64) ? 1 : -1;
|
||||
} else {
|
||||
reverse_elevons = (PINE & 128) ? 1 : -1;
|
||||
reverse_ch1_elevon = (PINE & 64) ? 1 : -1;
|
||||
reverse_ch2_elevon = (PINL & 64) ? 1 : -1;
|
||||
}
|
||||
}
|
@ -0,0 +1,70 @@
|
||||
#if DEBUG_SUBSYSTEM == 1
|
||||
void debug_subsystem()
|
||||
{
|
||||
Serial.println("GPS Subsystem, RAW OUTPUT");
|
||||
|
||||
while(1){
|
||||
if(Serial1.available() > 0) // Ok, let me see, the buffer is empty?
|
||||
{
|
||||
delay(60); // wait for it all
|
||||
while(Serial1.available() > 0){
|
||||
byte incoming = Serial1.read();
|
||||
//Serial.print(incoming,DEC);
|
||||
Serial.print(incoming, BYTE); // will output Hex values
|
||||
//Serial.print(",");
|
||||
}
|
||||
Serial.println(" ");
|
||||
}
|
||||
|
||||
}
|
||||
}
|
||||
#endif
|
||||
|
||||
#if DEBUG_SUBSYSTEM == 2
|
||||
void debug_subsystem()
|
||||
{
|
||||
Serial.println("Control Switch");
|
||||
|
||||
Serial.print("Control CH ");
|
||||
Serial.println(flight_mode_channel, DEC);
|
||||
|
||||
while(1){
|
||||
delay(20);
|
||||
byte switchPosition = readSwitch();
|
||||
if (oldSwitchPosition != switchPosition){
|
||||
Serial.printf_P(PSTR("Position %d\n"), i, switchPosition);
|
||||
oldSwitchPosition = switchPosition;
|
||||
}
|
||||
}
|
||||
}
|
||||
#endif
|
||||
|
||||
#if DEBUG_SUBSYSTEM == 3
|
||||
void debug_subsystem()
|
||||
{
|
||||
Serial.println("DIP Switch Test");
|
||||
|
||||
while(1){
|
||||
delay(100);
|
||||
update_servo_switches();
|
||||
if (mix_mode == 0) {
|
||||
Serial.print("Mix:standard ");
|
||||
Serial.print("\t reverse_roll: ");
|
||||
Serial.print(reverse_roll, DEC);
|
||||
Serial.print("\t reverse_pitch: ");
|
||||
Serial.print(reverse_pitch, DEC);
|
||||
Serial.print("\t reverse_rudder: ");
|
||||
Serial.println(reverse_rudder, DEC);
|
||||
} else {
|
||||
Serial.print("Mix:elevons ");
|
||||
Serial.print("\t reverse_elevons: ");
|
||||
Serial.print(reverse_elevons, DEC);
|
||||
Serial.print("\t reverse_ch1_elevon: ");
|
||||
Serial.print(reverse_ch1_elevon, DEC);
|
||||
Serial.print("\t reverse_ch2_elevon: ");
|
||||
Serial.println(reverse_ch2_elevon, DEC);
|
||||
}
|
||||
}
|
||||
}
|
||||
#endif
|
||||
|
@ -0,0 +1,345 @@
|
||||
// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: t -*-
|
||||
|
||||
// Internal defines, don't edit and expect things to work
|
||||
// -------------------------------------------------------
|
||||
|
||||
#define DEBUG 0
|
||||
#define LOITER_RANGE 30 // for calculating power outside of loiter radius
|
||||
|
||||
// GPS baud rates
|
||||
// --------------
|
||||
#define NO_GPS 38400
|
||||
#define NMEA_GPS 38400
|
||||
#define EM406_GPS 57600
|
||||
#define UBLOX_GPS 38400
|
||||
#define ARDU_IMU 38400
|
||||
#define MTK_GPS 38400
|
||||
#define SIM_GPS 38400
|
||||
|
||||
// GPS type codes - use the names, not the numbers
|
||||
#define GPS_PROTOCOL_NONE -1
|
||||
#define GPS_PROTOCOL_NMEA 0
|
||||
#define GPS_PROTOCOL_SIRF 1
|
||||
#define GPS_PROTOCOL_UBLOX 2
|
||||
#define GPS_PROTOCOL_IMU 3
|
||||
#define GPS_PROTOCOL_MTK 4
|
||||
|
||||
// Radio channels
|
||||
// Note channels are from 0!
|
||||
//
|
||||
// XXX these should be CH_n defines from RC.h at some point.
|
||||
#define CH_ROLL 0
|
||||
#define CH_PITCH 1
|
||||
#define CH_THROTTLE 2
|
||||
#define CH_RUDDER 3
|
||||
#define CH_1 0
|
||||
#define CH_2 1
|
||||
#define CH_3 2
|
||||
#define CH_4 3
|
||||
#define CH_5 4
|
||||
#define CH_6 5
|
||||
#define CH_7 6
|
||||
#define CH_8 7
|
||||
|
||||
#define WP_START_BYTE 0x130 // where in memory home WP is stored + all other WP
|
||||
#define WP_SIZE 14
|
||||
|
||||
// GCS enumeration
|
||||
#define GCS_PROTOCOL_STANDARD 0 // standard APM protocol
|
||||
#define GCS_PROTOCOL_SPECIAL 1 // special test protocol (?)
|
||||
#define GCS_PROTOCOL_LEGACY 2 // legacy ArduPilot protocol
|
||||
#define GCS_PROTOCOL_XPLANE 3 // X-Plane HIL simulation
|
||||
#define GCS_PROTOCOL_IMU 4 // ArdiPilot IMU output
|
||||
#define GCS_PROTOCOL_JASON 5 // Jason's special secret GCS protocol
|
||||
#define GCS_PROTOCOL_DEBUGTERMINAL 6 //Human-readable debug interface for use with a dumb terminal
|
||||
#define GCS_PROTOCOL_XDIY 7 // X-DIY custom protocol
|
||||
#define GCS_PROTOCOL_NONE -1 // No GCS output
|
||||
|
||||
// PID enumeration
|
||||
// ---------------
|
||||
#define CASE_SERVO_ROLL 0
|
||||
#define CASE_SERVO_PITCH 1
|
||||
#define CASE_SERVO_RUDDER 2
|
||||
#define CASE_NAV_ROLL 3
|
||||
#define CASE_NAV_PITCH_ASP 4
|
||||
#define CASE_NAV_PITCH_ALT 5
|
||||
#define CASE_TE_THROTTLE 6
|
||||
#define CASE_ALT_THROTTLE 7
|
||||
|
||||
// Feedforward cases
|
||||
// ----------------
|
||||
#define CASE_PITCH_COMP 0
|
||||
#define CASE_RUDDER_MIX 1
|
||||
#define CASE_P_TO_T 2
|
||||
|
||||
// Auto Pilot modes
|
||||
// ----------------
|
||||
#define MANUAL 0
|
||||
#define CIRCLE 1 // When flying sans GPS, and we loose the radio, just circle
|
||||
#define STABILIZE 2
|
||||
|
||||
#define FLY_BY_WIRE_A 5 // Fly By Wire A has left stick horizontal => desired roll angle, left stick vertical => desired pitch angle, right stick vertical = manual throttle
|
||||
#define FLY_BY_WIRE_B 6 // Fly By Wire B has left stick horizontal => desired roll angle, left stick vertical => desired pitch angle, right stick vertical => desired airspeed
|
||||
// Fly By Wire B = Fly By Wire A if you have AIRSPEED_SENSOR 0
|
||||
#define AUTO 10
|
||||
#define RTL 11
|
||||
#define LOITER 12
|
||||
#define TAKEOFF 13
|
||||
#define LAND 14
|
||||
|
||||
|
||||
// Command IDs - Must
|
||||
#define CMD_BLANK 0x00 // there is no command stored in the mem location requested
|
||||
#define CMD_WAYPOINT 0x10
|
||||
#define CMD_LOITER 0x11
|
||||
#define CMD_LOITER_N_TURNS 0x12
|
||||
#define CMD_LOITER_TIME 0x13
|
||||
#define CMD_RTL 0x14
|
||||
#define CMD_LAND 0x15
|
||||
#define CMD_TAKEOFF 0x16
|
||||
|
||||
// Command IDs - May
|
||||
#define CMD_DELAY 0x20
|
||||
#define CMD_CLIMB 0x21 // NOT IMPLEMENTED
|
||||
#define CMD_LAND_OPTIONS 0x22 // pitch in deg, airspeed m/s, throttle %, track WP 1 or 0
|
||||
|
||||
// Command IDs - Now
|
||||
//#define CMD_AP_MODE 0x30
|
||||
#define CMD_RESET_INDEX 0x31
|
||||
#define CMD_GOTO_INDEX 0x32 // NOT IMPLEMENTED
|
||||
#define CMD_GETVAR_INDEX 0x33
|
||||
#define CMD_SENDVAR_INDEX 0x34
|
||||
#define CMD_TELEMETRY 0x35
|
||||
|
||||
#define CMD_THROTTLE_CRUISE 0x40
|
||||
#define CMD_AIRSPEED_CRUISE 0x41
|
||||
#define CMD_RESET_HOME 0x44
|
||||
|
||||
#define CMD_KP_GAIN 0x60
|
||||
#define CMD_KI_GAIN 0x61
|
||||
#define CMD_KD_GAIN 0x62
|
||||
#define CMD_KI_MAX 0x63
|
||||
#define CMD_KFF_GAIN 0x64
|
||||
|
||||
#define CMD_RADIO_TRIM 0x70
|
||||
#define CMD_RADIO_MAX 0x71
|
||||
#define CMD_RADIO_MIN 0x72
|
||||
#define CMD_RADIO_MIN 0x72
|
||||
#define CMD_ELEVON_TRIM 0x73
|
||||
|
||||
#define CMD_INDEX 0x75 // sets the current Must index
|
||||
#define CMD_REPEAT 0x80
|
||||
#define CMD_RELAY 0x81
|
||||
#define CMD_SERVO 0x82 // move servo N to PWM value
|
||||
|
||||
//repeating events
|
||||
#define NO_REPEAT 0
|
||||
#define CH_4_TOGGLE 1
|
||||
#define CH_5_TOGGLE 2
|
||||
#define CH_6_TOGGLE 3
|
||||
#define CH_7_TOGGLE 4
|
||||
#define RELAY_TOGGLE 5
|
||||
#define STOP_REPEAT 10
|
||||
|
||||
// GCS Message ID's
|
||||
#define MSG_ACKNOWLEDGE 0x00
|
||||
#define MSG_HEARTBEAT 0x01
|
||||
#define MSG_ATTITUDE 0x02
|
||||
#define MSG_LOCATION 0x03
|
||||
#define MSG_PRESSURE 0x04
|
||||
#define MSG_STATUS_TEXT 0x05
|
||||
#define MSG_PERF_REPORT 0x06
|
||||
#define MSG_COMMAND 0x22
|
||||
#define MSG_VALUE 0x32
|
||||
#define MSG_PID 0x42
|
||||
#define MSG_TRIMS 0x50
|
||||
#define MSG_MINS 0x51
|
||||
#define MSG_MAXS 0x52
|
||||
#define MSG_IMU_OUT 0x53
|
||||
|
||||
#define SEVERITY_LOW 1
|
||||
#define SEVERITY_MEDIUM 2
|
||||
#define SEVERITY_HIGH 3
|
||||
#define SEVERITY_CRITICAL 4
|
||||
|
||||
// Logging parameters
|
||||
#define LOG_ATTITUDE_MSG 0x01
|
||||
#define LOG_GPS_MSG 0x02
|
||||
#define LOG_MODE_MSG 0X03
|
||||
#define LOG_CONTROL_TUNING_MSG 0X04
|
||||
#define LOG_NAV_TUNING_MSG 0X05
|
||||
#define LOG_PERFORMANCE_MSG 0X06
|
||||
#define LOG_RAW_MSG 0x07
|
||||
#define LOG_CMD_MSG 0x08
|
||||
#define LOG_STARTUP_MSG 0x09
|
||||
#define TYPE_AIRSTART_MSG 0x00
|
||||
#define TYPE_GROUNDSTART_MSG 0x01
|
||||
|
||||
#define MASK_LOG_ATTITUDE_FAST 0
|
||||
#define MASK_LOG_ATTITUDE_MED 2
|
||||
#define MASK_LOG_GPS 4
|
||||
#define MASK_LOG_PM 8
|
||||
#define MASK_LOG_CTUN 16
|
||||
#define MASK_LOG_NTUN 32
|
||||
#define MASK_LOG_MODE 64
|
||||
#define MASK_LOG_RAW 128
|
||||
#define MASK_LOG_CMD 256
|
||||
|
||||
// Yaw modes
|
||||
#define YAW_MODE_COORDINATE_TURNS 0
|
||||
#define YAW_MODE_HOLD_HEADING 1
|
||||
#define YAW_MODE_SLIP 2
|
||||
|
||||
// Waypoint Modes
|
||||
// ----------------
|
||||
#define ABS_WP 0
|
||||
#define REL_WP 1
|
||||
|
||||
// Command Queues
|
||||
// ---------------
|
||||
#define COMMAND_MUST 0
|
||||
#define COMMAND_MAY 1
|
||||
#define COMMAND_NOW 2
|
||||
|
||||
// Events
|
||||
// ------
|
||||
#define EVENT_WILL_REACH_WAYPOINT 1
|
||||
#define EVENT_SET_NEW_WAYPOINT_INDEX 2
|
||||
#define EVENT_LOADED_WAYPOINT 3
|
||||
#define EVENT_LOOP 4
|
||||
|
||||
//GPS_fix
|
||||
#define VALID_GPS 0x00
|
||||
#define BAD_GPS 0x01
|
||||
#define FAILED_GPS 0x03
|
||||
|
||||
|
||||
|
||||
#define BATTERY_VOLTAGE(x) (x*(INPUT_VOLTAGE/1024.0))*VOLT_DIV_RATIO
|
||||
|
||||
#define AIRSPEED_CH 7 // The external ADC channel for the airspeed sensor
|
||||
#define BATTERY_PIN1 0 // These are the pins for the voltage dividers
|
||||
#define BATTERY_PIN2 1
|
||||
#define BATTERY_PIN3 2
|
||||
#define BATTERY_PIN4 3
|
||||
#define RELAY_PIN 47
|
||||
|
||||
// Hardware Parameters
|
||||
#define SLIDE_SWITCH_PIN 40
|
||||
#define PUSHBUTTON_PIN 41
|
||||
|
||||
#define A_LED_PIN 37 //36 = B, 37 = A, 35 = C
|
||||
#define B_LED_PIN 36
|
||||
#define C_LED_PIN 35
|
||||
|
||||
#define HOLD_ALT_ABOVE_HOME 8 // bitmask value
|
||||
|
||||
// IMU Parameters
|
||||
|
||||
#define ADC_CONSTRAINT 900
|
||||
#define TRUE 1
|
||||
#define FALSE 0
|
||||
#define ADC_WARM_CYCLES 200
|
||||
#define SPEEDFILT 400 // centimeters/second
|
||||
|
||||
#define GYRO_TEMP_CH 3 // The ADC channel reading the gyro temperature
|
||||
|
||||
// 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 : 418
|
||||
#define GRAVITY 418 //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 : 0.4026, ?, 0.4192
|
||||
#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
|
||||
|
||||
#define Kp_ROLLPITCH 0.0014 // Pitch&Roll Drift Correction Proportional Gain
|
||||
#define Ki_ROLLPITCH 0.0000003 // Pitch&Roll Drift Correction Integrator Gain
|
||||
#define Kp_YAW 0.8 // Yaw Drift Correction Porportional Gain
|
||||
#define Ki_YAW 0.00004 // Yaw Drift CorrectionIntegrator Gain
|
||||
|
||||
/*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
|
||||
|
||||
|
||||
#define EEPROM_MAX_ADDR 4096
|
||||
|
||||
// Radio setup
|
||||
#define EE_TRIM 0x00
|
||||
#define EE_MIN 0x10
|
||||
#define EE_MAX 0x20
|
||||
#define EE_ELEVON1_TRIM 0x30
|
||||
#define EE_ELEVON2_TRIM 0x32
|
||||
|
||||
// user gains
|
||||
#define EE_XTRACK_GAIN 0x34
|
||||
#define EE_XTRACK_ANGLE 0x36
|
||||
#define EE_ALT_MIX 0x3B
|
||||
#define EE_HEAD_MAX 0x38
|
||||
#define EE_PITCH_MAX 0x39
|
||||
#define EE_PITCH_MIN 0x3A
|
||||
#define EE_KP 0x40
|
||||
#define EE_KI 0x60
|
||||
#define EE_KD 0x80
|
||||
#define EE_IMAX 0xA0
|
||||
#define EE_KFF 0xC0
|
||||
#define EE_AN_OFFSET 0xE0
|
||||
|
||||
//mission specific
|
||||
#define EE_CONFIG 0X0F8
|
||||
#define EE_WP_MODE 0x0F9
|
||||
#define EE_YAW_MODE 0x0FA // not used
|
||||
#define EE_WP_TOTAL 0x0FB
|
||||
#define EE_WP_INDEX 0x0FC
|
||||
#define EE_WP_RADIUS 0x0FD
|
||||
#define EE_LOITER_RADIUS 0x0FE
|
||||
#define EE_ALT_HOLD_HOME 0x0FF
|
||||
|
||||
// user configs
|
||||
#define EE_AIRSPEED_CRUISE 0x103
|
||||
#define EE_AIRSPEED_RATIO 0x104
|
||||
#define EE_AIRSPEED_FBW_MIN 0x108
|
||||
#define EE_AIRSPEED_FBW_MAX 0x109
|
||||
#define EE_THROTTLE_MIN 0x10A
|
||||
#define EE_THROTTLE_CRUISE 0x10B
|
||||
#define EE_THROTTLE_MAX 0x10C
|
||||
#define EE_THROTTLE_FAILSAFE 0x10D
|
||||
#define EE_THROTTLE_FS_VALUE 0x10E
|
||||
#define EE_THROTTLE_FAILSAFE_ACTION 0x110
|
||||
#define EE_FLIGHT_MODE_CHANNEL 0x112
|
||||
#define EE_AUTO_TRIM 0x113
|
||||
#define EE_LOG_BITMASK 0x114
|
||||
#define EE_REVERSE_SWITCH 0x120
|
||||
#define EE_FLIGHT_MODES 0x121
|
||||
|
||||
// sensors
|
||||
#define EE_ABS_PRESS_GND 0x116
|
||||
#define EE_GND_TEMP 0x11A
|
||||
#define EE_GND_ALT 0x11C
|
||||
#define EE_AP_OFFSET 0x11E
|
||||
|
||||
// log
|
||||
#define EE_LAST_LOG_PAGE 0xE00
|
||||
#define EE_LAST_LOG_NUM 0xE02
|
||||
#define EE_LOG_1_START 0xE04
|
||||
|
||||
// bits in log_bitmask
|
||||
#define LOGBIT_ATTITUDE_FAST (1<<0)
|
||||
#define LOGBIT_ATTITUDE_MED (1<<1)
|
||||
#define LOGBIT_GPS (1<<2)
|
||||
#define LOGBIT_PM (1<<3)
|
||||
#define LOGBIT_CTUN (1<<4)
|
||||
#define LOGBIT_NTUN (1<<5)
|
||||
#define LOGBIT_MODE (1<<6)
|
||||
#define LOGBIT_RAW (1<<7)
|
||||
#define LOGBIT_CMD (1<<8)
|
||||
|
@ -0,0 +1,226 @@
|
||||
/*
|
||||
This event will be called when the failsafe changes
|
||||
boolean failsafe reflects the current state
|
||||
*/
|
||||
void failsafe_event()
|
||||
{
|
||||
if (failsafe == true){
|
||||
|
||||
// This is how to handle a failsafe.
|
||||
switch(control_mode)
|
||||
{
|
||||
case MANUAL: // First position
|
||||
set_mode(STABILIZE);
|
||||
break;
|
||||
|
||||
case STABILIZE:
|
||||
case FLY_BY_WIRE_A: // middle position
|
||||
case FLY_BY_WIRE_B: // middle position
|
||||
set_mode(RTL);
|
||||
throttle_cruise = THROTTLE_CRUISE;
|
||||
|
||||
case CIRCLE: // middle position
|
||||
break;
|
||||
|
||||
case AUTO: // middle position
|
||||
case LOITER: // middle position
|
||||
if (throttle_failsafe_action == 1){
|
||||
set_mode(RTL);
|
||||
throttle_cruise = THROTTLE_CRUISE;
|
||||
}
|
||||
// 2 = Stay in AUTO and ignore failsafe
|
||||
break;
|
||||
|
||||
case RTL: // middle position
|
||||
break;
|
||||
|
||||
default:
|
||||
set_mode(RTL);
|
||||
throttle_cruise = THROTTLE_CRUISE;
|
||||
break;
|
||||
}
|
||||
}else{
|
||||
reset_I();
|
||||
}
|
||||
}
|
||||
|
||||
void low_battery_event(void)
|
||||
{
|
||||
send_message(SEVERITY_HIGH,"Low Battery!");
|
||||
set_mode(RTL);
|
||||
throttle_cruise = THROTTLE_CRUISE;
|
||||
}
|
||||
|
||||
|
||||
/*
|
||||
4 simultaneous events
|
||||
int event_original - original time in seconds
|
||||
int event_countdown - count down to zero
|
||||
byte event_countdown - ID for what to do
|
||||
byte event_countdown - how many times to loop, 0 = indefinite
|
||||
byte event_value - specific information for an event like PWM value
|
||||
byte counterEvent - undo the event if nescessary
|
||||
|
||||
count down each one
|
||||
|
||||
|
||||
new event
|
||||
undo_event
|
||||
*/
|
||||
|
||||
void new_event(struct Location *cmd)
|
||||
{
|
||||
SendDebug("New Event Loaded ");
|
||||
SendDebugln(cmd->p1,DEC);
|
||||
|
||||
|
||||
if(cmd->p1 == STOP_REPEAT){
|
||||
SendDebugln("STOP repeat ");
|
||||
event_id = NO_REPEAT;
|
||||
event_timer = -1;
|
||||
undo_event = 0;
|
||||
event_value = 0;
|
||||
event_delay = 0;
|
||||
event_repeat = 0; // convert seconds to millis
|
||||
event_undo_value = 0;
|
||||
repeat_forever = 0;
|
||||
}else{
|
||||
// reset the event timer
|
||||
event_timer = millis();
|
||||
event_id = cmd->p1;
|
||||
event_value = cmd->alt;
|
||||
event_delay = cmd->lat;
|
||||
event_repeat = cmd->lng; // convert seconds to millis
|
||||
event_undo_value = 0;
|
||||
repeat_forever = (event_repeat == 0) ? 1:0;
|
||||
}
|
||||
|
||||
/*
|
||||
Serial.print("event_id: ");
|
||||
Serial.println(event_id,DEC);
|
||||
Serial.print("event_value: ");
|
||||
Serial.println(event_value,DEC);
|
||||
Serial.print("event_delay: ");
|
||||
Serial.println(event_delay,DEC);
|
||||
Serial.print("event_repeat: ");
|
||||
Serial.println(event_repeat,DEC);
|
||||
Serial.print("event_undo_value: ");
|
||||
Serial.println(event_undo_value,DEC);
|
||||
Serial.print("repeat_forever: ");
|
||||
Serial.println(repeat_forever,DEC);
|
||||
Serial.print("Event_timer: ");
|
||||
Serial.println(event_timer,DEC);
|
||||
*/
|
||||
perform_event();
|
||||
}
|
||||
|
||||
void perform_event()
|
||||
{
|
||||
if (event_repeat > 0){
|
||||
event_repeat --;
|
||||
}
|
||||
switch(event_id) {
|
||||
case CH_4_TOGGLE:
|
||||
event_undo_value = readOutputCh(4);
|
||||
|
||||
APM_RC.OutputCh(4, event_value); // send to Servos
|
||||
undo_event = 2;
|
||||
break;
|
||||
case CH_5_TOGGLE:
|
||||
event_undo_value = readOutputCh(5);
|
||||
APM_RC.OutputCh(5, event_value); // send to Servos
|
||||
undo_event = 2;
|
||||
break;
|
||||
case CH_6_TOGGLE:
|
||||
event_undo_value = readOutputCh(6);
|
||||
APM_RC.OutputCh(6, event_value); // send to Servos
|
||||
undo_event = 2;
|
||||
break;
|
||||
case CH_7_TOGGLE:
|
||||
event_undo_value = readOutputCh(7);
|
||||
APM_RC.OutputCh(7, event_value); // send to Servos
|
||||
undo_event = 2;
|
||||
event_undo_value = 1;
|
||||
break;
|
||||
case RELAY_TOGGLE:
|
||||
|
||||
event_undo_value = PORTL & B00000100 ? 1:0;
|
||||
if(event_undo_value == 1){
|
||||
relay_A();
|
||||
}else{
|
||||
relay_B();
|
||||
}
|
||||
SendDebug("toggle relay ");
|
||||
SendDebugln(PORTL,BIN);
|
||||
undo_event = 2;
|
||||
break;
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
void relay_A()
|
||||
{
|
||||
PORTL |= B00000100;
|
||||
}
|
||||
|
||||
void relay_B()
|
||||
{
|
||||
PORTL ^= B00000100;
|
||||
}
|
||||
|
||||
void update_events()
|
||||
{
|
||||
// repeating events
|
||||
if(undo_event == 1){
|
||||
perform_event_undo();
|
||||
undo_event = 0;
|
||||
}else if(undo_event > 1){
|
||||
undo_event --;
|
||||
}
|
||||
|
||||
if(event_timer == -1)
|
||||
return;
|
||||
|
||||
if((millis() - event_timer) > event_delay){
|
||||
perform_event();
|
||||
|
||||
if(event_repeat > 0 || repeat_forever == 1){
|
||||
event_repeat--;
|
||||
event_timer = millis();
|
||||
}else{
|
||||
event_timer = -1;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void perform_event_undo()
|
||||
{
|
||||
switch(event_id) {
|
||||
case CH_4_TOGGLE:
|
||||
APM_RC.OutputCh(4, event_undo_value); // send to Servos
|
||||
break;
|
||||
|
||||
case CH_5_TOGGLE:
|
||||
APM_RC.OutputCh(5, event_undo_value); // send to Servos
|
||||
break;
|
||||
|
||||
case CH_6_TOGGLE:
|
||||
APM_RC.OutputCh(6, event_undo_value); // send to Servos
|
||||
break;
|
||||
|
||||
case CH_7_TOGGLE:
|
||||
APM_RC.OutputCh(7, event_undo_value); // send to Servos
|
||||
break;
|
||||
|
||||
case RELAY_TOGGLE:
|
||||
|
||||
if(event_undo_value == 1){
|
||||
relay_A();
|
||||
}else{
|
||||
relay_B();
|
||||
}
|
||||
SendDebug("untoggle relay ");
|
||||
SendDebugln(PORTL,BIN);
|
||||
break;
|
||||
}
|
||||
}
|
@ -0,0 +1,230 @@
|
||||
// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: t -*-
|
||||
|
||||
//****************************************************************
|
||||
// Function that will calculate the desired direction to fly and altitude error
|
||||
//****************************************************************
|
||||
void navigate()
|
||||
{
|
||||
// do not navigate with corrupt data
|
||||
// ---------------------------------
|
||||
if (GPS.fix == 0)
|
||||
{
|
||||
GPS.new_data = false;
|
||||
return;
|
||||
}
|
||||
if(next_WP.lat == 0){
|
||||
return;
|
||||
}
|
||||
|
||||
// We only perform most nav computations if we have new gps data to work with
|
||||
// --------------------------------------------------------------------------
|
||||
if(GPS.new_data){
|
||||
GPS.new_data = false;
|
||||
|
||||
// target_bearing is where we should be heading
|
||||
// --------------------------------------------
|
||||
target_bearing = get_bearing(¤t_loc, &next_WP);
|
||||
|
||||
// nav_bearing will includes xtrac correction
|
||||
// ------------------------------------------
|
||||
nav_bearing = target_bearing;
|
||||
|
||||
// waypoint distance from plane
|
||||
// ----------------------------
|
||||
wp_distance = getDistance(¤t_loc, &next_WP);
|
||||
|
||||
if (wp_distance < 0){
|
||||
send_message(SEVERITY_HIGH,"<navigate> WP error - distance < 0");
|
||||
//Serial.println(wp_distance,DEC);
|
||||
//print_current_waypoints();
|
||||
return;
|
||||
}
|
||||
|
||||
// check if we have missed the WP
|
||||
loiter_delta = (target_bearing - old_target_bearing)/100;
|
||||
|
||||
// reset the old value
|
||||
old_target_bearing = target_bearing;
|
||||
|
||||
// wrap values
|
||||
if (loiter_delta > 180) loiter_delta -= 360;
|
||||
if (loiter_delta < -180) loiter_delta += 360;
|
||||
loiter_sum += abs(loiter_delta);
|
||||
|
||||
calc_bearing_error();
|
||||
|
||||
// control mode specific updates to nav_bearing
|
||||
update_navigation();
|
||||
}
|
||||
}
|
||||
|
||||
void update_navigation()
|
||||
{
|
||||
// wp_distance is in ACTUAL meters, not the *100 meters we get from the GPS
|
||||
// ------------------------------------------------------------------------
|
||||
|
||||
// distance and bearing calcs only
|
||||
if(control_mode == AUTO){
|
||||
verify_must();
|
||||
verify_may();
|
||||
}else{
|
||||
|
||||
switch(control_mode){
|
||||
case LOITER:
|
||||
float power;
|
||||
if (wp_distance <= loiter_radius){
|
||||
nav_bearing -= 9000;
|
||||
|
||||
}else if (wp_distance < (loiter_radius + LOITER_RANGE)){
|
||||
power = -((float)(wp_distance - loiter_radius - LOITER_RANGE) / LOITER_RANGE);
|
||||
power = constrain(power, 0, 1);
|
||||
nav_bearing -= power * 9000;
|
||||
|
||||
}else{
|
||||
update_crosstrack();
|
||||
}
|
||||
nav_bearing = wrap_360(nav_bearing);
|
||||
calc_bearing_error();
|
||||
break;
|
||||
|
||||
case RTL:
|
||||
if(wp_distance <= (loiter_radius + 10)) { // + 10 meters
|
||||
set_mode(LOITER);
|
||||
}else{
|
||||
update_crosstrack();
|
||||
}
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
/*
|
||||
Disabled for now
|
||||
void calc_distance_error()
|
||||
{
|
||||
//distance_estimate += (float)GPS.ground_speed * .0002 * cos(radians(bearing_error * .01));
|
||||
//distance_estimate -= DST_EST_GAIN * (float)(distance_estimate - GPS_wp_distance);
|
||||
//wp_distance = max(distance_estimate,10);
|
||||
}
|
||||
*/
|
||||
|
||||
void calc_airspeed_errors()
|
||||
{
|
||||
airspeed_error = airspeed_cruise - airspeed;
|
||||
airspeed_energy_error = (long)(((long)airspeed_cruise * (long)airspeed_cruise) - ((long)airspeed * (long)airspeed))/20000; //Changed 0.00005f * to / 20000 to avoid floating point calculation
|
||||
}
|
||||
|
||||
void calc_bearing_error()
|
||||
{
|
||||
bearing_error = nav_bearing - yaw_sensor;
|
||||
bearing_error = wrap_180(bearing_error);
|
||||
}
|
||||
|
||||
void calc_altitude_error()
|
||||
{
|
||||
// limit climb rates
|
||||
target_altitude = next_WP.alt - ((float)((wp_distance -30) * offset_altitude) / (float)(wp_totalDistance - 30));
|
||||
if(prev_WP.alt > next_WP.alt){
|
||||
target_altitude = constrain(target_altitude, next_WP.alt, prev_WP.alt);
|
||||
}else{
|
||||
target_altitude = constrain(target_altitude, prev_WP.alt, next_WP.alt);
|
||||
}
|
||||
|
||||
/*
|
||||
// Disabled for now
|
||||
#if AIRSPEED_SENSOR == 1
|
||||
// special thanks to Ryan Beall for this one
|
||||
float pitch_angle = pitch_sensor - AOA; // pitch_angle = pitch sensor - angle of attack of your plane at level *100 (50 = .5°)
|
||||
pitch_angle = constrain(pitch_angle, -2000, 2000);
|
||||
float scale = sin(radians(pitch_angle * .01));
|
||||
altitude_estimate += (float)airspeed * .0002 * scale;
|
||||
altitude_estimate -= ALT_EST_GAIN * (float)(altitude_estimate - current_loc.alt);
|
||||
|
||||
// compute altitude error for throttle control
|
||||
altitude_error = target_altitude - altitude_estimate;
|
||||
#else
|
||||
altitude_error = target_altitude - current_loc.alt;
|
||||
#endif
|
||||
*/
|
||||
|
||||
altitude_error = target_altitude - current_loc.alt;
|
||||
}
|
||||
|
||||
|
||||
long wrap_360(long error)
|
||||
{
|
||||
if (error > 36000) error -= 36000;
|
||||
if (error < 0) error += 36000;
|
||||
return error;
|
||||
}
|
||||
|
||||
long wrap_180(long error)
|
||||
{
|
||||
if (error > 18000) error -= 36000;
|
||||
if (error < -18000) error += 36000;
|
||||
return error;
|
||||
}
|
||||
|
||||
/*
|
||||
// disabled for now
|
||||
void update_loiter()
|
||||
{
|
||||
loiter_delta = (target_bearing - old_target_bearing)/100;
|
||||
// reset the old value
|
||||
old_target_bearing = target_bearing;
|
||||
// wrap values
|
||||
if (loiter_delta > 170) loiter_delta -= 360;
|
||||
if (loiter_delta < -170) loiter_delta += 360;
|
||||
loiter_sum += loiter_delta;
|
||||
}*/
|
||||
|
||||
void update_crosstrack(void)
|
||||
{
|
||||
// Crosstrack Error
|
||||
// ----------------
|
||||
if (abs(target_bearing - crosstrack_bearing) < 4500) { // If we are too far off or too close we don't do track following
|
||||
crosstrack_error = sin(radians((target_bearing - crosstrack_bearing)/100)) * wp_distance; // Meters we are off track line
|
||||
nav_bearing += constrain(crosstrack_error * x_track_gain, -x_track_angle, x_track_angle);
|
||||
nav_bearing = wrap_360(nav_bearing);
|
||||
}
|
||||
}
|
||||
|
||||
void reset_crosstrack()
|
||||
{
|
||||
crosstrack_bearing = get_bearing(¤t_loc, &next_WP); // Used for track following
|
||||
}
|
||||
|
||||
int get_altitude_above_home(void)
|
||||
{
|
||||
// This is the altitude above the home location
|
||||
// The GPS gives us altitude at Sea Level
|
||||
// if you slope soar, you should see a negative number sometimes
|
||||
// -------------------------------------------------------------
|
||||
return current_loc.alt - home.alt;
|
||||
}
|
||||
|
||||
long getDistance(struct Location *loc1, struct Location *loc2)
|
||||
{
|
||||
if(loc1->lat == 0 || loc1->lng == 0)
|
||||
return -1;
|
||||
if(loc2->lat == 0 || loc2->lng == 0)
|
||||
return -1;
|
||||
float dlat = (float)(loc2->lat - loc1->lat);
|
||||
float dlong = ((float)(loc2->lng - loc1->lng)) * scaleLongDown;
|
||||
return sqrt(sq(dlat) + sq(dlong)) * .01113195;
|
||||
}
|
||||
|
||||
long get_alt_distance(struct Location *loc1, struct Location *loc2)
|
||||
{
|
||||
return abs(loc1->alt - loc2->alt);
|
||||
}
|
||||
|
||||
long get_bearing(struct Location *loc1, struct Location *loc2)
|
||||
{
|
||||
long off_x = loc2->lng - loc1->lng;
|
||||
long off_y = (loc2->lat - loc1->lat) * scaleLongUp;
|
||||
long bearing = 9000 + atan2(-off_y, off_x) * 5729.57795;
|
||||
if (bearing < 0) bearing += 36000;
|
||||
return bearing;
|
||||
}
|
@ -0,0 +1,180 @@
|
||||
//Function that will read the radio data, limit servos and trigger a failsafe
|
||||
// ----------------------------------------------------------------------------
|
||||
byte failsafeCounter = 0; // we wait a second to take over the throttle and send the plane circling
|
||||
|
||||
void read_radio()
|
||||
{
|
||||
ch1_temp = APM_RC.InputCh(CH_ROLL);
|
||||
ch2_temp = APM_RC.InputCh(CH_PITCH);
|
||||
|
||||
if(mix_mode == 0){
|
||||
radio_in[CH_ROLL] = ch1_temp;
|
||||
radio_in[CH_PITCH] = ch2_temp;
|
||||
}else{
|
||||
radio_in[CH_ROLL] = reverse_elevons * (reverse_ch2_elevon * (ch2_temp - elevon2_trim) - reverse_ch1_elevon * (ch1_temp - elevon1_trim)) / 2 + 1500;
|
||||
radio_in[CH_PITCH] = (reverse_ch2_elevon * (ch2_temp - elevon2_trim) + reverse_ch1_elevon * (ch1_temp - elevon1_trim)) / 2 + 1500;
|
||||
}
|
||||
|
||||
for (int y = 2; y < 8; y++)
|
||||
radio_in[y] = APM_RC.InputCh(y);
|
||||
|
||||
#if THROTTLE_REVERSE == 1
|
||||
radio_in[CH_THROTTLE] = radio_max[CH_THROTTLE] + radio_min[CH_THROTTLE] - radio_in[CH_THROTTLE];
|
||||
#endif
|
||||
|
||||
throttle_failsafe(radio_in[CH_THROTTLE]);
|
||||
servo_out[CH_THROTTLE] = ((float)(radio_in[CH_THROTTLE] - radio_min[CH_THROTTLE]) / (float)(radio_max[CH_THROTTLE] - radio_min[CH_THROTTLE])) * 100;
|
||||
servo_out[CH_THROTTLE] = constrain(servo_out[CH_THROTTLE], 0, 100);
|
||||
}
|
||||
|
||||
void throttle_failsafe(int pwm)
|
||||
{
|
||||
if(throttle_failsafe_enabled == 0)
|
||||
return;
|
||||
|
||||
//check for failsafe and debounce funky reads
|
||||
// ------------------------------------------
|
||||
if (pwm < throttle_failsafe_value){
|
||||
// we detect a failsafe from radio
|
||||
// throttle has dropped below the mark
|
||||
failsafeCounter++;
|
||||
if (failsafeCounter == 9){
|
||||
SendDebug("MSG FS ON ");
|
||||
SendDebugln(pwm, DEC);
|
||||
}else if(failsafeCounter == 10) {
|
||||
ch3_failsafe = true;
|
||||
//set_failsafe(true);
|
||||
//failsafeCounter = 10;
|
||||
}else if (failsafeCounter > 10){
|
||||
failsafeCounter = 11;
|
||||
}
|
||||
|
||||
}else if(failsafeCounter > 0){
|
||||
// we are no longer in failsafe condition
|
||||
// but we need to recover quickly
|
||||
failsafeCounter--;
|
||||
if (failsafeCounter > 3){
|
||||
failsafeCounter = 3;
|
||||
}
|
||||
if (failsafeCounter == 1){
|
||||
SendDebug("MSG FS OFF ");
|
||||
SendDebugln(pwm, DEC);
|
||||
}else if(failsafeCounter == 0) {
|
||||
ch3_failsafe = false;
|
||||
//set_failsafe(false);
|
||||
//failsafeCounter = -1;
|
||||
}else if (failsafeCounter <0){
|
||||
failsafeCounter = -1;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void trim_control_surfaces()
|
||||
{
|
||||
// Store control surface trim values
|
||||
// ---------------------------------
|
||||
if(mix_mode == 0){
|
||||
radio_trim[CH_ROLL] = radio_in[CH_ROLL];
|
||||
radio_trim[CH_PITCH] = radio_in[CH_PITCH];
|
||||
radio_trim[CH_RUDDER] = radio_in[CH_RUDDER];
|
||||
}else{
|
||||
elevon1_trim = ch1_temp;
|
||||
elevon2_trim = ch2_temp;
|
||||
//Recompute values here using new values for elevon1_trim and elevon2_trim
|
||||
//We cannot use radio_in[CH_ROLL] and radio_in[CH_PITCH] values from read_radio() because the elevon trim values have changed
|
||||
radio_trim[CH_ROLL] = 1500;
|
||||
radio_trim[CH_PITCH] = 1500;
|
||||
}
|
||||
// disabled for now
|
||||
//save_EEPROM_trims();
|
||||
}
|
||||
|
||||
void trim_radio()
|
||||
{
|
||||
for (int y = 0; y < 50; y++) {
|
||||
read_radio();
|
||||
}
|
||||
|
||||
// Store the trim values
|
||||
// ---------------------
|
||||
if(mix_mode == 0){
|
||||
for (int y = 0; y < 8; y++) radio_trim[y] = radio_in[y];
|
||||
}else{
|
||||
elevon1_trim = ch1_temp;
|
||||
elevon2_trim = ch2_temp;
|
||||
radio_trim[CH_ROLL] = 1500;
|
||||
radio_trim[CH_PITCH] = 1500;
|
||||
for (int y = 2; y < 8; y++) radio_trim[y] = radio_in[y];
|
||||
}
|
||||
save_EEPROM_trims();
|
||||
}
|
||||
|
||||
|
||||
#if SET_RADIO_LIMITS == 1
|
||||
void read_radio_limits()
|
||||
{
|
||||
// set initial servo limits for calibration routine
|
||||
// -------------------------------------------------
|
||||
radio_min[CH_ROLL] = radio_in[CH_ROLL] - 150;
|
||||
radio_max[CH_ROLL] = radio_in[CH_ROLL] + 150;
|
||||
|
||||
radio_min[CH_PITCH] = radio_in[CH_PITCH] - 150;
|
||||
radio_max[CH_PITCH] = radio_in[CH_PITCH] + 150;
|
||||
|
||||
// vars for the radio config routine
|
||||
// ---------------------------------
|
||||
int counter = 0;
|
||||
long reminder;
|
||||
reminder = millis() - 10000;
|
||||
|
||||
// Allows user to set stick limits and calibrate the IR
|
||||
// ----------------------------------------------------
|
||||
while(counter < 50){
|
||||
|
||||
if (millis() - reminder >= 10000) { // Remind user every 10 seconds what is going on
|
||||
send_message(SEVERITY_LOW,"Reading radio limits:");
|
||||
send_message(SEVERITY_LOW,"Move sticks to: upper right and lower Left");
|
||||
send_message(SEVERITY_LOW,"To Continue, hold the stick in the corner for 2 seconds.");
|
||||
send_message(SEVERITY_LOW," ");
|
||||
//print_radio();
|
||||
demo_servos(1);
|
||||
reminder = millis();
|
||||
}
|
||||
|
||||
delay(40);
|
||||
read_radio();
|
||||
|
||||
// AutoSet servo limits
|
||||
// --------------------
|
||||
if (radio_in[CH_ROLL] > 1000 && radio_in[CH_ROLL] < 2000){
|
||||
radio_min[CH_ROLL] = min(radio_in[CH_ROLL], radio_min[CH_ROLL]);
|
||||
radio_max[CH_ROLL] = max(radio_in[CH_ROLL], radio_max[CH_ROLL]);
|
||||
}
|
||||
|
||||
if (radio_in[CH_PITCH] > 1000 && radio_in[CH_PITCH]< 2000){
|
||||
radio_min[CH_PITCH] = min(radio_in[CH_PITCH], radio_min[CH_PITCH]);
|
||||
radio_max[CH_PITCH] = max(radio_in[CH_PITCH], radio_max[CH_PITCH]);
|
||||
}
|
||||
if(radio_in[CH_PITCH] < (radio_min[CH_PITCH] + 30) || radio_in[CH_PITCH] > (radio_max[CH_PITCH] -30)){
|
||||
SendDebug(".");
|
||||
counter++;
|
||||
}else{
|
||||
if (counter > 0)
|
||||
counter--;
|
||||
}
|
||||
}
|
||||
|
||||
// contstrain min values
|
||||
// ---------------------
|
||||
radio_min[CH_ROLL] = constrain(radio_min[CH_ROLL], 800, 2200);
|
||||
radio_max[CH_ROLL] = constrain(radio_max[CH_ROLL], 800, 2200);
|
||||
radio_min[CH_PITCH] = constrain(radio_min[CH_PITCH], 800, 2200);
|
||||
radio_max[CH_PITCH] = constrain(radio_max[CH_PITCH], 800, 2200);
|
||||
|
||||
SendDebugln(" ");
|
||||
}
|
||||
#endif
|
||||
|
||||
|
||||
|
||||
|
@ -0,0 +1,65 @@
|
||||
void ReadSCP1000(void) {}
|
||||
|
||||
|
||||
void read_airpressure(void){
|
||||
double x;
|
||||
|
||||
APM_BMP085.Read(); // Get new data from absolute pressure sensor
|
||||
abs_press = APM_BMP085.Press;
|
||||
abs_press_filt = (abs_press); // + 2l * abs_press_filt) / 3l; // Light filtering
|
||||
//temperature = (temperature * 9 + temp_unfilt) / 10; We will just use the ground temp for the altitude calculation
|
||||
|
||||
double p = (double)abs_press_gnd / (double)abs_press_filt;
|
||||
double temp = (float)ground_temperature / 10.f + 273.15f;
|
||||
x = log(p) * temp * 29271.267f;
|
||||
//x = log(p) * temp * 29.271267 * 1000;
|
||||
press_alt = (long)(x / 10) + ground_alt; // Pressure altitude in centimeters
|
||||
// Need to add comments for theory.....
|
||||
}
|
||||
|
||||
// in M/S * 100
|
||||
void read_airspeed(void)
|
||||
{
|
||||
#if GPS_PROTOCOL != GPS_PROTOCOL_IMU // Xplane will supply the airspeed
|
||||
airpressure_raw = ((float)APM_ADC.Ch(AIRSPEED_CH) * .25) + (airpressure_raw * .75);
|
||||
airpressure = (int)airpressure_raw - airpressure_offset;
|
||||
airpressure = max(airpressure, 0);
|
||||
airspeed = sqrt((float)airpressure * airspeed_ratio) * 100;
|
||||
#endif
|
||||
|
||||
calc_airspeed_errors();
|
||||
}
|
||||
|
||||
#if BATTERY_EVENT == 1
|
||||
void read_battery(void)
|
||||
{
|
||||
battery_voltage1 = BATTERY_VOLTAGE(analogRead(BATTERY_PIN1)) * .1 + battery_voltage1 * .9;
|
||||
battery_voltage2 = BATTERY_VOLTAGE(analogRead(BATTERY_PIN2)) * .1 + battery_voltage2 * .9;
|
||||
battery_voltage3 = BATTERY_VOLTAGE(analogRead(BATTERY_PIN3)) * .1 + battery_voltage3 * .9;
|
||||
battery_voltage4 = BATTERY_VOLTAGE(analogRead(BATTERY_PIN4)) * .1 + battery_voltage4 * .9;
|
||||
|
||||
#if BATTERY_TYPE == 0
|
||||
if(battery_voltage3 < LOW_VOLTAGE)
|
||||
low_battery_event();
|
||||
battery_voltage = battery_voltage3; // set total battery voltage, for telemetry stream
|
||||
#endif
|
||||
|
||||
#if BATTERY_TYPE == 1
|
||||
if(battery_voltage4 < LOW_VOLTAGE)
|
||||
low_battery_event();
|
||||
battery_voltage = battery_voltage4; // set total battery voltage, for telemetry stream
|
||||
#endif
|
||||
}
|
||||
#endif
|
||||
|
||||
void zero_airspeed(void)
|
||||
{
|
||||
airpressure_raw = (float)APM_ADC.Ch(AIRSPEED_CH);
|
||||
for(int c = 0; c < 50; c++){
|
||||
delay(20);
|
||||
airpressure_raw = (airpressure_raw * .90) + ((float)APM_ADC.Ch(AIRSPEED_CH) * .10);
|
||||
}
|
||||
airpressure_offset = airpressure_raw;
|
||||
}
|
||||
|
||||
|
@ -0,0 +1,410 @@
|
||||
// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: t -*-
|
||||
|
||||
// Functions called from the setup menu
|
||||
static int8_t setup_radio(uint8_t argc, const Menu::arg *argv);
|
||||
static int8_t setup_show(uint8_t argc, const Menu::arg *argv);
|
||||
static int8_t setup_factory(uint8_t argc, const Menu::arg *argv);
|
||||
static int8_t setup_flightmodes(uint8_t argc, const Menu::arg *argv);
|
||||
|
||||
// Command/function table for the setup menu
|
||||
const struct Menu::command setup_menu_commands[] PROGMEM = {
|
||||
// command function called
|
||||
// ======= ===============
|
||||
{"reset", setup_factory},
|
||||
{"radio", setup_radio},
|
||||
{"modes", setup_flightmodes},
|
||||
{"show", setup_show},
|
||||
};
|
||||
|
||||
// Create the setup menu object.
|
||||
MENU(setup_menu, "setup", setup_menu_commands);
|
||||
|
||||
// Called from the top-level menu to run the setup menu.
|
||||
int8_t
|
||||
setup_mode(uint8_t argc, const Menu::arg *argv)
|
||||
{
|
||||
// Give the user some guidance
|
||||
Serial.printf_P(PSTR("Setup Mode\n"
|
||||
"\n"
|
||||
"IMPORTANT: if you have not previously set this system up, use the\n"
|
||||
"'reset' command to initialize the EEPROM to sensible default values\n"
|
||||
"and then the 'radio' command to configure for your radio.\n"
|
||||
"\n"));
|
||||
|
||||
// Run the setup menu. When the menu exits, we will return to the main menu.
|
||||
setup_menu.run();
|
||||
}
|
||||
|
||||
// Print the current configuration.
|
||||
// Called by the setup menu 'show' command.
|
||||
static int8_t
|
||||
setup_show(uint8_t argc, const Menu::arg *argv)
|
||||
{
|
||||
uint8_t i;
|
||||
|
||||
Serial.printf_P(PSTR("\nRadio:\n"));
|
||||
read_EEPROM_radio_minmax();
|
||||
for(i = 0; i < 8; i++)
|
||||
Serial.printf_P(PSTR("CH%d: %d | %d\n"), i + 1, radio_min[i], radio_max[i]);
|
||||
|
||||
Serial.printf_P(PSTR("\nGains:\n"));
|
||||
read_EEPROM_gains();
|
||||
|
||||
for(i = 0; i < 8; i++){
|
||||
Serial.printf_P(PSTR("P,I,D,iMax:"));
|
||||
Serial.print(kp[i],3);
|
||||
Serial.print(",");
|
||||
Serial.print(ki[i],3);
|
||||
Serial.print(",");
|
||||
Serial.print(kd[i],3);
|
||||
Serial.print(",");
|
||||
Serial.println(integrator_max[i]/100,DEC);
|
||||
}
|
||||
|
||||
Serial.printf_P(PSTR("kff:"));
|
||||
Serial.print(kff[0],3);
|
||||
Serial.print(",");
|
||||
Serial.print(kff[1],3);
|
||||
Serial.print(",");
|
||||
Serial.println(kff[2],3);
|
||||
|
||||
Serial.printf_P(PSTR("XTRACK_GAIN:"));
|
||||
Serial.println(x_track_gain,DEC);
|
||||
|
||||
Serial.printf_P(PSTR("XTRACK_ENTRY_ANGLE: %d\n"), x_track_angle);
|
||||
Serial.printf_P(PSTR("HEAD_MAX: %d\n"), head_max);
|
||||
Serial.printf_P(PSTR("PITCH_MAX: %d\n"), pitch_max);
|
||||
Serial.printf_P(PSTR("PITCH_MIN: %d\n"), pitch_min);
|
||||
|
||||
read_user_configs();
|
||||
Serial.printf_P(PSTR("\nUser config:\n"));
|
||||
Serial.printf_P(PSTR("airspeed_cruise: %d\n"), airspeed_cruise);
|
||||
Serial.printf_P(PSTR("airspeed_fbw_min: %d\n"), airspeed_fbw_min);
|
||||
Serial.printf_P(PSTR("airspeed_fbw_max: %d\n"), airspeed_fbw_max);
|
||||
Serial.printf_P(PSTR("airspeed_ratio: "));
|
||||
Serial.println(airspeed_ratio, 4);
|
||||
|
||||
Serial.printf_P(PSTR("throttle_min: %d\n"), throttle_min);
|
||||
Serial.printf_P(PSTR("throttle_max: %d\n"), throttle_max);
|
||||
Serial.printf_P(PSTR("throttle_cruise: %d\n"), throttle_cruise);
|
||||
Serial.printf_P(PSTR("throttle_failsafe_enabled: %d\n"), throttle_failsafe_enabled);
|
||||
Serial.printf_P(PSTR("throttle_failsafe_value: %d\n"), throttle_failsafe_value);
|
||||
Serial.printf_P(PSTR("flight_mode_channel: %d\n"), flight_mode_channel+1); //Add 1 to flight_mode_channel to change 0-based channels to 1-based channels
|
||||
Serial.printf_P(PSTR("auto_trim: %d\n"), auto_trim);
|
||||
Serial.printf_P(PSTR("log_bitmask: %d\n\n"), log_bitmask);
|
||||
//Serial.printf_P(PSTR("Switch settings:\n"));
|
||||
//for(i = 0; i < 6; i++){
|
||||
// print_switch(i+1, flight_modes[i]);
|
||||
//}
|
||||
|
||||
return(0);
|
||||
}
|
||||
|
||||
// Initialise the EEPROM to 'factory' settings (mostly defined in APM_Config.h or via defaults).
|
||||
// Called by the setup menu 'factoryreset' command.
|
||||
static int8_t
|
||||
setup_factory(uint8_t argc, const Menu::arg *argv)
|
||||
{
|
||||
uint8_t i;
|
||||
int c;
|
||||
|
||||
Serial.printf_P(PSTR("\nType 'Y' and hit Enter to perform factory reset, any other key to abort: "));
|
||||
|
||||
do {
|
||||
c = Serial.read();
|
||||
} while (-1 == c);
|
||||
|
||||
if (('y' != c) && ('Y' != c))
|
||||
return(-1);
|
||||
|
||||
Serial.printf_P(PSTR("\nFACTORY RESET\n\n"));
|
||||
|
||||
wp_radius = WP_RADIUS_DEFAULT;
|
||||
loiter_radius = LOITER_RADIUS_DEFAULT;
|
||||
|
||||
save_EEPROM_waypoint_info();
|
||||
|
||||
radio_min[CH_1] = 0;
|
||||
radio_min[CH_2] = 0;
|
||||
radio_min[CH_3] = 0;
|
||||
radio_min[CH_4] = 0;
|
||||
radio_min[CH_5] = CH5_MIN;
|
||||
radio_min[CH_6] = CH6_MIN;
|
||||
radio_min[CH_7] = CH7_MIN;
|
||||
radio_min[CH_8] = CH8_MIN;
|
||||
|
||||
radio_max[CH_1] = 0;
|
||||
radio_max[CH_2] = 0;
|
||||
radio_max[CH_3] = 0;
|
||||
radio_max[CH_4] = 0;
|
||||
radio_max[CH_5] = CH5_MAX;
|
||||
radio_max[CH_6] = CH6_MAX;
|
||||
radio_max[CH_7] = CH7_MAX;
|
||||
radio_max[CH_8] = CH8_MAX;
|
||||
|
||||
save_EEPROM_radio_minmax();
|
||||
|
||||
kp[0] = SERVO_ROLL_P;
|
||||
kp[1] = SERVO_PITCH_P;
|
||||
kp[2] = SERVO_YAW_P;
|
||||
kp[3] = NAV_ROLL_P;
|
||||
kp[4] = NAV_PITCH_ASP_P;
|
||||
kp[5] = NAV_PITCH_ALT_P;
|
||||
kp[6] = THROTTLE_TE_P;
|
||||
kp[7] = THROTTLE_ALT_P;
|
||||
|
||||
ki[0] = SERVO_ROLL_I;
|
||||
ki[1] = SERVO_PITCH_I;
|
||||
ki[2] = SERVO_YAW_I;
|
||||
ki[3] = NAV_ROLL_I;
|
||||
ki[4] = NAV_PITCH_ASP_I;
|
||||
ki[5] = NAV_PITCH_ALT_I;
|
||||
ki[6] = THROTTLE_TE_I;
|
||||
ki[7] = THROTTLE_ALT_I;
|
||||
|
||||
kd[0] = SERVO_ROLL_D;
|
||||
kd[1] = SERVO_PITCH_D;
|
||||
kd[2] = SERVO_YAW_D;
|
||||
kd[3] = NAV_ROLL_D;
|
||||
kd[4] = NAV_PITCH_ASP_D;
|
||||
kd[5] = NAV_PITCH_ALT_D;
|
||||
kd[6] = THROTTLE_TE_D;
|
||||
kd[7] = THROTTLE_ALT_D;
|
||||
|
||||
integrator_max[0] = SERVO_ROLL_INT_MAX;
|
||||
integrator_max[1] = SERVO_PITCH_INT_MAX;
|
||||
integrator_max[2] = SERVO_YAW_INT_MAX;
|
||||
integrator_max[3] = NAV_ROLL_INT_MAX;
|
||||
integrator_max[4] = NAV_PITCH_ASP_INT_MAX;
|
||||
integrator_max[5] = NAV_PITCH_ALT_INT_MAX;
|
||||
integrator_max[6] = THROTTLE_TE_INT_MAX;
|
||||
integrator_max[7] = THROTTLE_ALT_INT_MAX;
|
||||
|
||||
kff[0] = PITCH_COMP;
|
||||
kff[1] = RUDDER_MIX;
|
||||
kff[2] = P_TO_T;
|
||||
|
||||
for(i = 0; i < 8; i++){
|
||||
// scale input to deg * 100;
|
||||
integrator_max[i] *= 100;
|
||||
}
|
||||
x_track_gain = XTRACK_GAIN * 100;
|
||||
x_track_angle = XTRACK_ENTRY_ANGLE * 100;
|
||||
altitude_mix = ALTITUDE_MIX;
|
||||
|
||||
head_max = HEAD_MAX * 100;
|
||||
pitch_max = PITCH_MAX * 100;
|
||||
pitch_min = PITCH_MIN * 100;
|
||||
|
||||
save_EEPROM_gains();
|
||||
|
||||
airspeed_cruise = AIRSPEED_CRUISE*100;
|
||||
airspeed_fbw_min = AIRSPEED_FBW_MIN;
|
||||
airspeed_fbw_max = AIRSPEED_FBW_MAX;
|
||||
airspeed_ratio = AIRSPEED_RATIO;
|
||||
throttle_min = THROTTLE_MIN;
|
||||
throttle_max = THROTTLE_MAX;
|
||||
throttle_cruise = THROTTLE_CRUISE;
|
||||
throttle_failsafe_enabled = THROTTLE_FAILSAFE;
|
||||
throttle_failsafe_action = THROTTLE_FAILSAFE_ACTION;
|
||||
throttle_failsafe_value = THROTTLE_FS_VALUE;
|
||||
//flight_mode_channel = FLIGHT_MODE_CHANNEL - 1;
|
||||
auto_trim = AUTO_TRIM;
|
||||
|
||||
flight_modes[0] = FLIGHT_MODE_1;
|
||||
flight_modes[1] = FLIGHT_MODE_2;
|
||||
flight_modes[2] = FLIGHT_MODE_3;
|
||||
flight_modes[3] = FLIGHT_MODE_4;
|
||||
flight_modes[4] = FLIGHT_MODE_5;
|
||||
flight_modes[5] = FLIGHT_MODE_6;
|
||||
save_EEPROM_flight_modes();
|
||||
|
||||
// convenience macro for testing LOG_* and setting LOGBIT_*
|
||||
#define LOGBIT(_s) (LOG_ ## _s ? LOGBIT_ ## _s : 0)
|
||||
log_bitmask =
|
||||
LOGBIT(ATTITUDE_FAST) |
|
||||
LOGBIT(ATTITUDE_MED) |
|
||||
LOGBIT(GPS) |
|
||||
LOGBIT(PM) |
|
||||
LOGBIT(CTUN) |
|
||||
LOGBIT(NTUN) |
|
||||
LOGBIT(MODE) |
|
||||
LOGBIT(RAW) |
|
||||
LOGBIT(CMD);
|
||||
#undef LOGBIT
|
||||
|
||||
save_user_configs();
|
||||
|
||||
return(0);
|
||||
}
|
||||
|
||||
// Perform radio setup.
|
||||
// Called by the setup menu 'radio' command.
|
||||
static int8_t
|
||||
setup_radio(uint8_t argc, const Menu::arg *argv)
|
||||
{
|
||||
Serial.println("\n\nRadio Setup:");
|
||||
uint8_t i;
|
||||
|
||||
for(i = 0; i < 100;i++){
|
||||
delay(20);
|
||||
read_radio();
|
||||
}
|
||||
|
||||
if(radio_in[CH_ROLL] < 500){
|
||||
while(1){
|
||||
Serial.print("Radio error");
|
||||
delay(1000);
|
||||
// stop here
|
||||
}
|
||||
}
|
||||
|
||||
for(i = 0; i < 4; i++){
|
||||
radio_min[i] = radio_in[i];
|
||||
radio_max[i] = radio_in[i];
|
||||
}
|
||||
|
||||
Serial.printf_P(PSTR("\nMove both sticks to each corner. Hit Enter to save: "));
|
||||
while(1){
|
||||
|
||||
delay(20);
|
||||
// Filters radio input - adjust filters in the radio.pde file
|
||||
// ----------------------------------------------------------
|
||||
read_radio();
|
||||
|
||||
for(i = 0; i < 4; i++){
|
||||
radio_min[i] = min(radio_min[i], radio_in[i]);
|
||||
radio_max[i] = max(radio_max[i], radio_in[i]);
|
||||
}
|
||||
|
||||
if(Serial.available() > 0){
|
||||
Serial.flush();
|
||||
Serial.println("Saving:");
|
||||
|
||||
save_EEPROM_radio_minmax();
|
||||
delay(100);
|
||||
read_EEPROM_radio_minmax();
|
||||
|
||||
for(i = 0; i < 8; i++)
|
||||
Serial.printf_P(PSTR("CH%d: %d | %d\n"), i + 1, radio_min[i], radio_max[i]);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
return(0);
|
||||
}
|
||||
|
||||
|
||||
static int8_t
|
||||
setup_flightmodes(uint8_t argc, const Menu::arg *argv)
|
||||
{
|
||||
byte switchPosition, oldSwitchPosition, mode;
|
||||
|
||||
Serial.printf_P(PSTR("\nMove RC toggle switch to each position to edit, move aileron stick to select modes."));
|
||||
print_hit_enter();
|
||||
trim_radio();
|
||||
|
||||
while(1){
|
||||
delay(20);
|
||||
read_radio();
|
||||
switchPosition = readSwitch();
|
||||
|
||||
|
||||
// look for control switch change
|
||||
if (oldSwitchPosition != switchPosition){
|
||||
|
||||
// Override position 5
|
||||
if(switchPosition > 4){
|
||||
mode = flight_modes[switchPosition] = MANUAL;
|
||||
}else{
|
||||
// update our current mode
|
||||
mode = flight_modes[switchPosition];
|
||||
}
|
||||
|
||||
// update the user
|
||||
print_switch(switchPosition, mode);
|
||||
|
||||
// Remember switch position
|
||||
oldSwitchPosition = switchPosition;
|
||||
}
|
||||
|
||||
// look for stick input
|
||||
if (radio_input_switch() == true){
|
||||
switch(mode){
|
||||
case MANUAL:
|
||||
mode = STABILIZE;
|
||||
break;
|
||||
|
||||
case STABILIZE:
|
||||
mode = FLY_BY_WIRE_A;
|
||||
break;
|
||||
|
||||
case FLY_BY_WIRE_A:
|
||||
mode = FLY_BY_WIRE_B;
|
||||
break;
|
||||
|
||||
case FLY_BY_WIRE_B:
|
||||
mode = AUTO;
|
||||
break;
|
||||
|
||||
case AUTO:
|
||||
mode = RTL;
|
||||
break;
|
||||
|
||||
case RTL:
|
||||
mode = LOITER;
|
||||
break;
|
||||
|
||||
case LOITER:
|
||||
mode = MANUAL;
|
||||
break;
|
||||
|
||||
default:
|
||||
mode = MANUAL;
|
||||
break;
|
||||
}
|
||||
|
||||
// Override position 5
|
||||
if(switchPosition > 4)
|
||||
mode = MANUAL;
|
||||
|
||||
// save new mode
|
||||
flight_modes[switchPosition] = mode;
|
||||
|
||||
// print new mode
|
||||
print_switch(switchPosition, mode);
|
||||
}
|
||||
|
||||
// escape hatch
|
||||
if(Serial.available() > 0){
|
||||
save_EEPROM_flight_modes();
|
||||
return (0);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void
|
||||
print_switch(byte p, byte m)
|
||||
{
|
||||
Serial.printf_P(PSTR("Pos %d: "),p);
|
||||
Serial.println(flight_mode_strings[m]);
|
||||
}
|
||||
|
||||
boolean
|
||||
radio_input_switch(void)
|
||||
{
|
||||
static byte bouncer;
|
||||
|
||||
if (abs(radio_in[0] - radio_trim[0]) > 200)
|
||||
bouncer = 10;
|
||||
|
||||
if (bouncer > 0)
|
||||
bouncer--;
|
||||
|
||||
if (bouncer == 1){
|
||||
return true;
|
||||
}else{
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
@ -0,0 +1,491 @@
|
||||
// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: t -*-
|
||||
/*****************************************************************************
|
||||
The init_ardupilot function processes everything we need for an in - air restart
|
||||
We will determine later if we are actually on the ground and process a
|
||||
ground start in that case.
|
||||
|
||||
*****************************************************************************/
|
||||
|
||||
// Functions called from the top-level menu
|
||||
extern int8_t process_logs(uint8_t argc, const Menu::arg *argv); // in Log.pde
|
||||
extern int8_t setup_mode(uint8_t argc, const Menu::arg *argv); // in setup.pde
|
||||
extern int8_t test_mode(uint8_t argc, const Menu::arg *argv); // in test.cpp
|
||||
|
||||
// This is the help function
|
||||
// PSTR is an AVR macro to read strings from flash memory
|
||||
// printf_P is a version of print_f that reads from flash memory
|
||||
static int8_t main_menu_help(uint8_t argc, const Menu::arg *argv)
|
||||
{
|
||||
Serial.printf_P(PSTR("Commands:\n"
|
||||
" logs log readback/setup mode\n"
|
||||
" setup setup mode\n"
|
||||
" test test mode\n"
|
||||
"\n"
|
||||
"Move the slide switch and reset to FLY.\n"
|
||||
"\n"));
|
||||
return(0);
|
||||
}
|
||||
|
||||
// Command/function table for the top-level menu.
|
||||
const struct Menu::command main_menu_commands[] PROGMEM = {
|
||||
// command function called
|
||||
// ======= ===============
|
||||
{"logs", process_logs},
|
||||
{"setup", setup_mode},
|
||||
{"test", test_mode},
|
||||
{"help", main_menu_help}
|
||||
};
|
||||
|
||||
// Create the top-level menu object.
|
||||
MENU(main_menu, "ArduPilotMega", main_menu_commands);
|
||||
|
||||
void init_ardupilot()
|
||||
{
|
||||
|
||||
byte last_log_num;
|
||||
int last_log_start;
|
||||
int last_log_end;
|
||||
|
||||
// Console serial port
|
||||
//
|
||||
// The console port buffers are defined to be sufficiently large to support
|
||||
// the console's use as a logging device, optionally as the GPS port when
|
||||
// GPS_PROTOCOL_IMU is selected, and as the telemetry port.
|
||||
//
|
||||
// XXX This could be optimised to reduce the buffer sizes in the cases
|
||||
// where they are not otherwise required.
|
||||
//
|
||||
Serial.begin(SERIAL0_BAUD, 128, 128);
|
||||
|
||||
// GPS serial port.
|
||||
//
|
||||
// Not used if the IMU/X-Plane GPS is in use.
|
||||
//
|
||||
// XXX currently the EM406 (SiRF receiver) is nominally configured
|
||||
// at 57600, however it's not been supported to date. We should
|
||||
// probably standardise on 38400.
|
||||
//
|
||||
// XXX the 128 byte receive buffer may be too small for NMEA, depending
|
||||
// on the message set configured.
|
||||
//
|
||||
#if GPS_PROTOCOL != GPS_PROTOCOL_IMU
|
||||
Serial1.begin(38400, 128, 16);
|
||||
#endif
|
||||
|
||||
// Telemetry port.
|
||||
//
|
||||
// Not used if telemetry is going to the console.
|
||||
//
|
||||
// XXX for unidirectional protocols, we could (should) minimize
|
||||
// the receive buffer, and the transmit buffer could also be
|
||||
// shrunk for protocols that don't send large messages.
|
||||
//
|
||||
#if GCS_PORT == 3
|
||||
Serial3.begin(SERIAL3_BAUD, 128, 128);
|
||||
#endif
|
||||
|
||||
Serial.printf_P(PSTR("\n\n"
|
||||
"Init ArduPilotMega X-DIY\n\n"
|
||||
#if TELEMETRY_PORT == 3
|
||||
"Telemetry is on the xbee port\n"
|
||||
#endif
|
||||
"freeRAM: %d\n"), freeRAM());
|
||||
|
||||
APM_RC.Init(); // APM Radio initialization
|
||||
read_EEPROM_startup(); // Read critical config information to start
|
||||
|
||||
APM_ADC.Init(); // APM ADC library initialization
|
||||
APM_BMP085.Init(); // APM Abs Pressure sensor initialization
|
||||
DataFlash.Init(); // DataFlash log initialization
|
||||
GPS.init(); // GPS Initialization
|
||||
|
||||
#if MAGNETOMETER == ENABLED
|
||||
APM_Compass.Init(); // I2C initialization
|
||||
#endif
|
||||
|
||||
APM_RC.OutputCh(CH_ROLL, radio_trim[CH_ROLL]); // Initialization of servo outputs
|
||||
APM_RC.OutputCh(CH_PITCH, radio_trim[CH_PITCH]);
|
||||
APM_RC.OutputCh(CH_THROTTLE, radio_trim[CH_THROTTLE]);
|
||||
APM_RC.OutputCh(CH_RUDDER, radio_trim[CH_RUDDER]);
|
||||
|
||||
pinMode(C_LED_PIN, OUTPUT); // GPS status LED
|
||||
pinMode(A_LED_PIN, OUTPUT); // GPS status LED
|
||||
pinMode(B_LED_PIN, OUTPUT); // GPS status LED
|
||||
pinMode(SLIDE_SWITCH_PIN, INPUT); // To enter interactive mode
|
||||
pinMode(PUSHBUTTON_PIN, INPUT); // unused
|
||||
|
||||
|
||||
// If the switch is in 'menu' mode, run the main menu.
|
||||
//
|
||||
// Since we can't be sure that the setup or test mode won't leave
|
||||
// the system in an odd state, we don't let the user exit the top
|
||||
// menu; they must reset in order to fly.
|
||||
//
|
||||
if (digitalRead(SLIDE_SWITCH_PIN) == 0) {
|
||||
digitalWrite(A_LED_PIN,HIGH); // turn on setup-mode LED
|
||||
Serial.printf_P(PSTR("\n"
|
||||
"Entering interactive setup mode...\n"
|
||||
"\n"
|
||||
"If using the Arduino Serial Monitor, ensure Line Ending is set to Carriage Return.\n"
|
||||
"Type 'help' to list commands, 'exit' to leave a submenu.\n"
|
||||
"Visit the 'setup' menu for first-time configuration.\n"));
|
||||
for (;;) {
|
||||
Serial.printf_P(PSTR("\n"
|
||||
"Move the slide switch and reset to FLY.\n"
|
||||
"\n"));
|
||||
main_menu.run();
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
if(log_bitmask > 0){
|
||||
// Here we will check on the length of the last log
|
||||
// We don't want to create a bunch of little logs due to powering on and off
|
||||
last_log_num = eeprom_read_byte((uint8_t *) EE_LAST_LOG_NUM);
|
||||
last_log_start = eeprom_read_word((uint16_t *) (EE_LOG_1_START+(last_log_num - 1) * 0x02));
|
||||
last_log_end = eeprom_read_word((uint16_t *) EE_LAST_LOG_PAGE);
|
||||
|
||||
if(last_log_num == 0) {
|
||||
// The log space is empty. Start a write session on page 1
|
||||
DataFlash.StartWrite(1);
|
||||
eeprom_write_byte((uint8_t *) EE_LAST_LOG_NUM, (1));
|
||||
eeprom_write_word((uint16_t *) EE_LOG_1_START, (1));
|
||||
|
||||
} else if (last_log_end <= last_log_start + 10) {
|
||||
// The last log is small. We consider it junk. Overwrite it.
|
||||
DataFlash.StartWrite(last_log_start);
|
||||
|
||||
} else {
|
||||
// The last log is valid. Start a new log
|
||||
if(last_log_num >= 19) {
|
||||
Serial.println("Number of log files exceeds max. Log 19 will be overwritten.");
|
||||
last_log_num --;
|
||||
}
|
||||
DataFlash.StartWrite(last_log_end + 1);
|
||||
eeprom_write_byte((uint8_t *) EE_LAST_LOG_NUM, (last_log_num + 1));
|
||||
eeprom_write_word((uint16_t *) (EE_LOG_1_START+(last_log_num)*0x02), (last_log_end + 1));
|
||||
}
|
||||
}
|
||||
|
||||
// read in the flight switches
|
||||
update_servo_switches();
|
||||
|
||||
if(DEBUG_SUBSYSTEM > 0){
|
||||
debug_subsystem();
|
||||
|
||||
} else if (ENABLE_AIR_START == 1) {
|
||||
// Perform an air start and get back to flying
|
||||
send_message(SEVERITY_LOW,"<init_ardupilot> AIR START");
|
||||
|
||||
// Get necessary data from EEPROM
|
||||
//----------------
|
||||
read_EEPROM_airstart_critical();
|
||||
|
||||
// This delay is important for the APM_RC library to work. We need some time for the comm between the 328 and 1280 to be established.
|
||||
int old_pulse = 0;
|
||||
while (millis()<=1000 && (abs(old_pulse - APM_RC.InputCh(flight_mode_channel)) > 5 || APM_RC.InputCh(flight_mode_channel) == 1000 || APM_RC.InputCh(flight_mode_channel) == 1200)) {
|
||||
old_pulse = APM_RC.InputCh(flight_mode_channel);
|
||||
delay(25);
|
||||
}
|
||||
if (log_bitmask & MASK_LOG_CMD)
|
||||
Log_Write_Startup(TYPE_AIRSTART_MSG);
|
||||
reload_commands(); // Get set to resume AUTO from where we left off
|
||||
|
||||
}else {
|
||||
startup_ground();
|
||||
if (log_bitmask & MASK_LOG_CMD)
|
||||
Log_Write_Startup(TYPE_GROUNDSTART_MSG);
|
||||
}
|
||||
|
||||
// set the correct flight mode
|
||||
// ---------------------------
|
||||
reset_control_switch();
|
||||
}
|
||||
|
||||
//********************************************************************************
|
||||
//This function does all the calibrations, etc. that we need during a ground start
|
||||
//********************************************************************************
|
||||
void startup_ground(void)
|
||||
{
|
||||
send_message(SEVERITY_LOW,"<startup_ground> GROUND START");
|
||||
|
||||
#if(GROUND_START_DELAY > 0)
|
||||
send_message(SEVERITY_LOW,"<startup_ground> With Delay");
|
||||
delay(GROUND_START_DELAY * 1000);
|
||||
#endif
|
||||
|
||||
// Output waypoints for confirmation
|
||||
// --------------------------------
|
||||
for(int i = 1; i < wp_total + 1; i++) {
|
||||
send_message(MSG_COMMAND, i);
|
||||
}
|
||||
|
||||
// Makes the servos wiggle
|
||||
// step 1 = 1 wiggle
|
||||
// -----------------------
|
||||
demo_servos(1);
|
||||
|
||||
//IMU ground start
|
||||
//------------------------
|
||||
#if GPS_PROTOCOL != GPS_PROTOCOL_IMU
|
||||
startup_IMU_ground();
|
||||
#endif
|
||||
|
||||
|
||||
// read the radio to set trims
|
||||
// ---------------------------
|
||||
trim_radio();
|
||||
|
||||
#if AIRSPEED_SENSOR == 1
|
||||
// initialize airspeed sensor
|
||||
// --------------------------
|
||||
zero_airspeed();
|
||||
send_message(SEVERITY_LOW,"<startup_ground> zero airspeed calibrated");
|
||||
#else
|
||||
send_message(SEVERITY_LOW,"<startup_ground> NO airspeed");
|
||||
#endif
|
||||
|
||||
// Save the settings for in-air restart
|
||||
// ------------------------------------
|
||||
save_EEPROM_groundstart();
|
||||
|
||||
// initialize commands
|
||||
// -------------------
|
||||
init_commands();
|
||||
|
||||
}
|
||||
|
||||
void set_mode(byte mode)
|
||||
{
|
||||
if(control_mode == mode){
|
||||
// don't switch modes if we are already in the correct mode.
|
||||
return;
|
||||
}
|
||||
if(auto_trim > 0 || control_mode == MANUAL)
|
||||
trim_control_surfaces();
|
||||
|
||||
control_mode = mode;
|
||||
crash_timer = 0;
|
||||
|
||||
switch(control_mode)
|
||||
{
|
||||
case MANUAL:
|
||||
break;
|
||||
|
||||
case STABILIZE:
|
||||
break;
|
||||
|
||||
case FLY_BY_WIRE_A:
|
||||
break;
|
||||
|
||||
case FLY_BY_WIRE_B:
|
||||
break;
|
||||
|
||||
case AUTO:
|
||||
update_auto();
|
||||
break;
|
||||
|
||||
case RTL:
|
||||
return_to_launch();
|
||||
break;
|
||||
|
||||
case LOITER:
|
||||
loiter_at_location();
|
||||
break;
|
||||
|
||||
case TAKEOFF:
|
||||
break;
|
||||
|
||||
case LAND:
|
||||
break;
|
||||
|
||||
default:
|
||||
return_to_launch();
|
||||
break;
|
||||
}
|
||||
|
||||
// output control mode to the ground station
|
||||
send_message(MSG_HEARTBEAT);
|
||||
|
||||
if (log_bitmask & MASK_LOG_MODE)
|
||||
Log_Write_Mode(control_mode);
|
||||
}
|
||||
|
||||
void set_failsafe(boolean mode)
|
||||
{
|
||||
// only act on changes
|
||||
// -------------------
|
||||
if(failsafe != mode){
|
||||
|
||||
// store the value so we don't trip the gate twice
|
||||
// -----------------------------------------------
|
||||
failsafe = mode;
|
||||
|
||||
if (failsafe == false){
|
||||
// We're back in radio contact
|
||||
// ---------------------------
|
||||
|
||||
// re-read the switch so we can return to our preferred mode
|
||||
reset_control_switch();
|
||||
|
||||
// Reset control integrators
|
||||
// ---------------------
|
||||
reset_I();
|
||||
|
||||
}else{
|
||||
// We've lost radio contact
|
||||
// ------------------------
|
||||
// nothing to do right now
|
||||
}
|
||||
|
||||
// Let the user know what's up so they can override the behavior
|
||||
// -------------------------------------------------------------
|
||||
failsafe_event();
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
void startup_IMU_ground(void)
|
||||
{
|
||||
uint16_t store = 0;
|
||||
int flashcount = 0;
|
||||
// 12345678901234567890123456789012
|
||||
jeti_status(" **** INIT **** Warming up ADC");
|
||||
jeti_update();
|
||||
SendDebugln("<startup_IMU_ground> Warming up ADC...");
|
||||
|
||||
for(int c = 0; c < ADC_WARM_CYCLES; c++)
|
||||
{
|
||||
digitalWrite(C_LED_PIN, LOW);
|
||||
digitalWrite(A_LED_PIN, HIGH);
|
||||
digitalWrite(B_LED_PIN, LOW);
|
||||
delay(50);
|
||||
Read_adc_raw();
|
||||
digitalWrite(C_LED_PIN, HIGH);
|
||||
digitalWrite(A_LED_PIN, LOW);
|
||||
digitalWrite(B_LED_PIN, HIGH);
|
||||
delay(50);
|
||||
}
|
||||
|
||||
// Makes the servos wiggle twice - about to begin IMU calibration - HOLD LEVEL AND STILL!!
|
||||
// -----------------------
|
||||
demo_servos(2);
|
||||
SendDebugln("<startup_IMU_ground> Beginning IMU calibration; do not move plane");
|
||||
// 12345678901234567890123456789012
|
||||
jeti_status(" **** INIT **** Do not move!!!");
|
||||
jeti_update();
|
||||
|
||||
|
||||
for(int y = 0; y <= 5; y++){ // Read first initial ADC values for offset.
|
||||
AN_OFFSET[y] = AN[y];
|
||||
}
|
||||
|
||||
APM_BMP085.Read(); // Get initial data from absolute pressure sensor
|
||||
abs_press_gnd = APM_BMP085.Press;
|
||||
ground_temperature = APM_BMP085.Temp;
|
||||
delay(20);
|
||||
// ***********
|
||||
|
||||
for(int i = 0; i < 400; i++){ // We take some readings...
|
||||
Read_adc_raw();
|
||||
for(int y = 0; y <= 5; y++) // Read initial ADC values for offset (averaging).
|
||||
AN_OFFSET[y] = AN_OFFSET[y] * 0.9 + AN[y] * 0.1;
|
||||
|
||||
APM_BMP085.Read(); // Get initial data from absolute pressure sensor
|
||||
abs_press_gnd = (abs_press_gnd * 9l + APM_BMP085.Press) / 10l;
|
||||
ground_temperature = (ground_temperature * 9 + APM_BMP085.Temp) / 10;
|
||||
|
||||
delay(20);
|
||||
if(flashcount == 5) {
|
||||
digitalWrite(C_LED_PIN, LOW);
|
||||
digitalWrite(A_LED_PIN, HIGH);
|
||||
digitalWrite(B_LED_PIN, LOW);
|
||||
}
|
||||
if(flashcount >= 10) {
|
||||
flashcount = 0;
|
||||
digitalWrite(C_LED_PIN, HIGH);
|
||||
digitalWrite(A_LED_PIN, LOW);
|
||||
digitalWrite(B_LED_PIN, HIGH);
|
||||
}
|
||||
flashcount++;
|
||||
|
||||
}
|
||||
SendDebugln(" <startup_IMU_ground> Calibration complete.");
|
||||
digitalWrite(B_LED_PIN, HIGH); // Set LED B high to indicate IMU ready
|
||||
digitalWrite(A_LED_PIN, LOW);
|
||||
digitalWrite(C_LED_PIN, LOW);
|
||||
|
||||
AN_OFFSET[5] -= GRAVITY * SENSOR_SIGN[5];
|
||||
/*
|
||||
Serial.print ("Offsets ");
|
||||
Serial.print (AN_OFFSET[0]);
|
||||
Serial.print (" ");
|
||||
Serial.print (AN_OFFSET[1]);
|
||||
Serial.print (" ");
|
||||
Serial.print (AN_OFFSET[2]);
|
||||
Serial.print (" ");
|
||||
Serial.print (AN_OFFSET[3]);
|
||||
Serial.print (" ");
|
||||
Serial.print (AN_OFFSET[4]);
|
||||
Serial.print (" ");
|
||||
Serial.println (AN_OFFSET[5]);
|
||||
*/
|
||||
}
|
||||
|
||||
|
||||
void update_GPS_light(void)
|
||||
{
|
||||
// GPS LED on if we have a fix or Blink GPS LED if we are receiving data
|
||||
// ---------------------------------------------------------------------
|
||||
switch (GPS.status()) {
|
||||
case(2):
|
||||
digitalWrite(C_LED_PIN, HIGH); //Turn LED C on when gps has valid fix.
|
||||
break;
|
||||
|
||||
case(1):
|
||||
if (GPS.valid_read == true){
|
||||
GPS_light = !GPS_light; // Toggle light on and off to indicate gps messages being received, but no GPS fix lock
|
||||
if (GPS_light){
|
||||
digitalWrite(C_LED_PIN, LOW);
|
||||
} else {
|
||||
digitalWrite(C_LED_PIN, HIGH);
|
||||
}
|
||||
GPS.valid_read = false;
|
||||
}
|
||||
break;
|
||||
|
||||
default:
|
||||
digitalWrite(C_LED_PIN, LOW);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
|
||||
void resetPerfData(void) {
|
||||
mainLoop_count = 0;
|
||||
G_Dt_max = 0;
|
||||
gyro_sat_count = 0;
|
||||
adc_constraints = 0;
|
||||
renorm_sqrt_count = 0;
|
||||
renorm_blowup_count = 0;
|
||||
gps_fix_count = 0;
|
||||
perf_mon_timer = millis();
|
||||
}
|
||||
|
||||
|
||||
/* This function gets the current value of the heap and stack pointers.
|
||||
* The stack pointer starts at the top of RAM and grows downwards. The heap pointer
|
||||
* starts just above the static variables etc. and grows upwards. SP should always
|
||||
* be larger than HP or you'll be in big trouble! The smaller the gap, the more
|
||||
* careful you need to be. Julian Gall 6 - Feb - 2009.
|
||||
*/
|
||||
unsigned long freeRAM() {
|
||||
uint8_t * heapptr, * stackptr;
|
||||
stackptr = (uint8_t *)malloc(4); // use stackptr temporarily
|
||||
heapptr = stackptr; // save value of heap pointer
|
||||
free(stackptr); // free up the memory again (sets stackptr to 0)
|
||||
stackptr = (uint8_t *)(SP); // save value of stack pointer
|
||||
return stackptr - heapptr;
|
||||
}
|
||||
|
@ -0,0 +1,433 @@
|
||||
// These are function definitions so the Menu can be constructed before the functions
|
||||
// are defined below. Order matters to the compiler.
|
||||
static int8_t test_radio(uint8_t argc, const Menu::arg *argv);
|
||||
static int8_t test_gps(uint8_t argc, const Menu::arg *argv);
|
||||
static int8_t test_imu(uint8_t argc, const Menu::arg *argv);
|
||||
static int8_t test_gyro(uint8_t argc, const Menu::arg *argv);
|
||||
static int8_t test_battery(uint8_t argc, const Menu::arg *argv);
|
||||
static int8_t test_relay(uint8_t argc, const Menu::arg *argv);
|
||||
static int8_t test_wp(uint8_t argc, const Menu::arg *argv);
|
||||
static int8_t test_airspeed(uint8_t argc, const Menu::arg *argv);
|
||||
static int8_t test_pressure(uint8_t argc, const Menu::arg *argv);
|
||||
static int8_t test_mag(uint8_t argc, const Menu::arg *argv);
|
||||
static int8_t test_xbee(uint8_t argc, const Menu::arg *argv);
|
||||
static int8_t test_eedump(uint8_t argc, const Menu::arg *argv);
|
||||
static int8_t test_jeti(uint8_t argc, const Menu::arg *argv);
|
||||
|
||||
// This is the help function
|
||||
// PSTR is an AVR macro to read strings from flash memory
|
||||
// printf_P is a version of print_f that reads from flash memory
|
||||
/*static int8_t help_test(uint8_t argc, const Menu::arg *argv)
|
||||
{
|
||||
Serial.printf_P(PSTR("\n"
|
||||
"Commands:\n"
|
||||
" radio\n"
|
||||
" servos\n"
|
||||
" gps\n"
|
||||
" imu\n"
|
||||
" battery\n"
|
||||
"\n"));
|
||||
}*/
|
||||
|
||||
// Creates a constant array of structs representing menu options
|
||||
// and stores them in Flash memory, not RAM.
|
||||
// User enters the string in the console to call the functions on the right.
|
||||
// See class Menu in AP_Coommon for implementation details
|
||||
const struct Menu::command test_menu_commands[] PROGMEM = {
|
||||
{"radio", test_radio},
|
||||
{"gps", test_gps},
|
||||
{"imu", test_imu},
|
||||
{"gyro", test_gyro},
|
||||
{"battery", test_battery},
|
||||
{"relay", test_relay},
|
||||
{"waypoints", test_wp},
|
||||
{"airspeed", test_airspeed},
|
||||
{"airpressure", test_pressure},
|
||||
{"compass", test_mag},
|
||||
{"xbee", test_xbee},
|
||||
{"eedump", test_eedump},
|
||||
{"jeti", test_jeti},
|
||||
};
|
||||
|
||||
// A Macro to create the Menu
|
||||
MENU(test_menu, "test", test_menu_commands);
|
||||
|
||||
int8_t
|
||||
test_mode(uint8_t argc, const Menu::arg *argv)
|
||||
{
|
||||
Serial.printf_P(PSTR("Test Mode\n\n"));
|
||||
test_menu.run();
|
||||
}
|
||||
|
||||
static int8_t
|
||||
test_eedump(uint8_t argc, const Menu::arg *argv)
|
||||
{
|
||||
int i, j;
|
||||
|
||||
// hexdump the EEPROM
|
||||
for (i = 0; i < EEPROM_MAX_ADDR; i += 16) {
|
||||
Serial.printf_P(PSTR("%04x:"), i);
|
||||
for (j = 0; j < 16; j++)
|
||||
Serial.printf_P(PSTR(" %02x"), eeprom_read_byte((const uint8_t *)(i + j)));
|
||||
Serial.println();
|
||||
}
|
||||
return(0);
|
||||
}
|
||||
|
||||
static int8_t
|
||||
test_radio(uint8_t argc, const Menu::arg *argv)
|
||||
{
|
||||
print_hit_enter();
|
||||
delay(1000);
|
||||
|
||||
#if THROTTLE_REVERSE == 1
|
||||
Serial.println("Throttle is reversed in config: ");
|
||||
delay(1000);
|
||||
#endif
|
||||
|
||||
// read the radio to set trims
|
||||
// ---------------------------
|
||||
trim_radio();
|
||||
|
||||
while(1){
|
||||
delay(20);
|
||||
// Filters radio input - adjust filters in the radio.pde file
|
||||
// ----------------------------------------------------------
|
||||
read_radio();
|
||||
update_servo_switches();
|
||||
|
||||
servo_out[CH_ROLL] = reverse_roll * (radio_in[CH_ROLL] - radio_trim[CH_ROLL]) * 9;
|
||||
servo_out[CH_PITCH] = reverse_pitch * (radio_in[CH_PITCH] - radio_trim[CH_PITCH]) * 9;
|
||||
servo_out[CH_RUDDER] = reverse_rudder * (radio_in[CH_RUDDER] - radio_trim[CH_RUDDER]) * 9;
|
||||
|
||||
// write out the servo PWM values
|
||||
// ------------------------------
|
||||
set_servos_4();
|
||||
|
||||
|
||||
for(int y = 4; y < 8; y++) {
|
||||
radio_out[y] = constrain(radio_in[y], radio_min[y], radio_max[y]);
|
||||
APM_RC.OutputCh(y, radio_out[y]); // send to Servos
|
||||
}
|
||||
Serial.printf_P(PSTR("IN: 1: %d\t2: %d\t3: %d\t4: %d\t5: %d\t6: %d\t7: %d\t8: %d\t"), radio_in[CH_1], radio_in[CH_2], radio_in[CH_3], radio_in[CH_4], radio_in[CH_5], radio_in[CH_6], radio_in[CH_7], radio_in[CH_8]);
|
||||
Serial.printf_P(PSTR("OUT 1: %d\t2: %d\t3: %d\t4: %d\n"), (servo_out[CH_ROLL]/100), (servo_out[CH_PITCH]/100), servo_out[CH_THROTTLE], (servo_out[CH_RUDDER]/100));
|
||||
|
||||
if(Serial.available() > 0){
|
||||
return (0);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
static int8_t
|
||||
test_gps(uint8_t argc, const Menu::arg *argv)
|
||||
{
|
||||
print_hit_enter();
|
||||
delay(1000);
|
||||
|
||||
while(1){
|
||||
delay(333);
|
||||
|
||||
// Blink GPS LED if we don't have a fix
|
||||
// ------------------------------------
|
||||
update_GPS_light();
|
||||
|
||||
GPS.update();
|
||||
|
||||
if (GPS.new_data){
|
||||
Serial.print("Lat:");
|
||||
Serial.print((float)GPS.latitude/10000000, 10);
|
||||
Serial.print(" Lon:");
|
||||
Serial.print((float)GPS.longitude/10000000, 10);
|
||||
Serial.printf_P(PSTR("alt %dm, #sats: %d\n"), GPS.altitude/100, GPS.num_sats);
|
||||
}else{
|
||||
Serial.print(".");
|
||||
}
|
||||
if(Serial.available() > 0){
|
||||
return (0);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
static int8_t
|
||||
test_imu(uint8_t argc, const Menu::arg *argv)
|
||||
{
|
||||
Serial.printf_P(PSTR("Calibrating."));
|
||||
startup_IMU_ground();
|
||||
print_hit_enter();
|
||||
delay(1000);
|
||||
|
||||
while(1){
|
||||
delay(20);
|
||||
read_AHRS();
|
||||
|
||||
// We are using the IMU
|
||||
// ---------------------
|
||||
Serial.printf_P(PSTR("r: %d\tp: %d\t y: %d\n"), ((int)roll_sensor/100), ((int)pitch_sensor/100), ((uint16_t)yaw_sensor/100));
|
||||
|
||||
if(Serial.available() > 0){
|
||||
return (0);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
static int8_t
|
||||
test_battery(uint8_t argc, const Menu::arg *argv)
|
||||
{
|
||||
#if BATTERY_EVENT == 1
|
||||
for (int i = 0; i < 20; i++){
|
||||
delay(20);
|
||||
read_battery();
|
||||
}
|
||||
Serial.printf_P(PSTR("Volts: 1:"));
|
||||
Serial.print(battery_voltage1, 4);
|
||||
Serial.print(" 2:");
|
||||
Serial.print(battery_voltage2, 4);
|
||||
Serial.print(" 3:");
|
||||
Serial.print(battery_voltage3, 4);
|
||||
Serial.print(" 4:");
|
||||
Serial.println(battery_voltage4, 4);
|
||||
#else
|
||||
Serial.printf_P(PSTR("Not enabled\n"));
|
||||
|
||||
#endif
|
||||
return (0);
|
||||
}
|
||||
|
||||
|
||||
static int8_t
|
||||
test_relay(uint8_t argc, const Menu::arg *argv)
|
||||
{
|
||||
print_hit_enter();
|
||||
delay(1000);
|
||||
//DDRL |= B00000100;
|
||||
|
||||
while(1){
|
||||
Serial.println(int DDRL);
|
||||
Serial.println(int PORTL);
|
||||
Serial.println("Relay A");
|
||||
relay_A();
|
||||
delay(500);
|
||||
if(Serial.available() > 0){
|
||||
return (0);
|
||||
}
|
||||
|
||||
Serial.println("Relay B");
|
||||
relay_B();
|
||||
delay(500);
|
||||
if(Serial.available() > 0){
|
||||
return (0);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
static int8_t
|
||||
test_gyro(uint8_t argc, const Menu::arg *argv)
|
||||
{
|
||||
print_hit_enter();
|
||||
delay(1000);
|
||||
Serial.printf_P(PSTR("Gyro | Accel\n"));
|
||||
delay(200);
|
||||
|
||||
while(1){
|
||||
for (int i = 0; i < 6; i++) {
|
||||
AN[i] = APM_ADC.Ch(sensors[i]);
|
||||
}
|
||||
Serial.printf_P(PSTR("%d\t%d\t%d\t|\t%d\t%d\t%d\n"), (int)AN[0], (int)AN[1], (int)AN[2], (int)AN[3], (int)AN[4], (int)AN[5]);
|
||||
delay(100);
|
||||
|
||||
if(Serial.available() > 0){
|
||||
return (0);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
static int8_t
|
||||
test_wp(uint8_t argc, const Menu::arg *argv)
|
||||
{
|
||||
delay(1000);
|
||||
ground_alt = 0;
|
||||
read_EEPROM_waypoint_info();
|
||||
|
||||
uint8_t options = eeprom_read_byte((const uint8_t *) EE_CONFIG);
|
||||
int32_t hold = eeprom_read_dword((const uint32_t *) EE_ALT_HOLD_HOME);
|
||||
|
||||
// save the alitude above home option
|
||||
if(options & HOLD_ALT_ABOVE_HOME){
|
||||
Serial.printf_P(PSTR("Hold altitude of %dm\n"), hold/100);
|
||||
}else{
|
||||
Serial.printf_P(PSTR("Hold current altitude\n"));
|
||||
}
|
||||
|
||||
Serial.printf_P(PSTR("%d waypoints\n"), wp_total);
|
||||
Serial.printf_P(PSTR("Hit radius: %d\n"), wp_radius);
|
||||
Serial.printf_P(PSTR("Loiter radius: %d\n\n"), loiter_radius);
|
||||
|
||||
for(byte i = 0; i <= wp_total; i++){
|
||||
struct Location temp = get_wp_with_index(i);
|
||||
print_waypoint(&temp, i);
|
||||
}
|
||||
|
||||
return (0);
|
||||
}
|
||||
|
||||
|
||||
static int8_t
|
||||
test_airspeed(uint8_t argc, const Menu::arg *argv)
|
||||
{
|
||||
#if AIRSPEED_SENSOR == DISABLED
|
||||
Serial.printf_P(PSTR("airspeed disabled\n"));
|
||||
return (0);
|
||||
#else
|
||||
print_hit_enter();
|
||||
zero_airspeed();
|
||||
|
||||
while(1){
|
||||
delay(20);
|
||||
read_airspeed();
|
||||
Serial.printf_P(PSTR("%dm/s\n"),airspeed/100);
|
||||
|
||||
if(Serial.available() > 0){
|
||||
return (0);
|
||||
}
|
||||
}
|
||||
#endif
|
||||
}
|
||||
|
||||
static int8_t
|
||||
test_xbee(uint8_t argc, const Menu::arg *argv)
|
||||
{
|
||||
print_hit_enter();
|
||||
delay(1000);
|
||||
Serial.printf_P(PSTR("Begin XBee X-CTU Range and RSSI Test:\n"));
|
||||
while(1){
|
||||
delay(250);
|
||||
// Timeout set high enough for X-CTU RSSI Calc over XBee @ 115200
|
||||
//Serial3.printf_P(PSTR("0123456789:;<=>?@ABCDEFGHIJKLMNO\n"));
|
||||
//Serial.print("X");
|
||||
// Default 32bit data from X-CTU Range Test
|
||||
if(Serial.available() > 0){
|
||||
return (0);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
static int8_t
|
||||
test_pressure(uint8_t argc, const Menu::arg *argv)
|
||||
{
|
||||
uint32_t sum = 0;
|
||||
Serial.printf_P(PSTR("Uncalibrated Abs Airpressure\n"));
|
||||
Serial.printf_P(PSTR("Note - the altitude displayed is relative to the start of this test\n"));
|
||||
print_hit_enter();
|
||||
Serial.printf_P(PSTR("\nCalibrating....\n"));
|
||||
for (int i=1;i<301;i++) {
|
||||
read_airpressure();
|
||||
if(i>200)sum += abs_press_filt;
|
||||
delay(10);
|
||||
}
|
||||
abs_press_gnd = (double)sum/100.0;
|
||||
|
||||
ground_alt = 0;
|
||||
while(1){
|
||||
delay(100);
|
||||
read_airpressure();
|
||||
//Serial.printf_P(PSTR("Alt: %dm, Raw: %d\n"), press_alt / 100, abs_press_filt); // Someone needs to fix the formatting here for long integers
|
||||
Serial.print("Altitude: ");
|
||||
Serial.print(press_alt/100.0,2);
|
||||
Serial.print(" Raw pressure value: ");
|
||||
Serial.println(abs_press_filt);
|
||||
if(Serial.available() > 0){
|
||||
return (0);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
static int8_t
|
||||
test_mag(uint8_t argc, const Menu::arg *argv)
|
||||
{
|
||||
#if MAGNETOMETER == DISABLED
|
||||
Serial.printf_P(PSTR("Compass disabled\n"));
|
||||
return (0);
|
||||
#else
|
||||
print_hit_enter();
|
||||
|
||||
while(1){
|
||||
delay(250);
|
||||
APM_Compass.Read();
|
||||
APM_Compass.Calculate(0,0);
|
||||
Serial.printf_P(PSTR("Heading: ("));
|
||||
Serial.print(ToDeg(APM_Compass.Heading));
|
||||
Serial.printf_P(PSTR(") XYZ: ("));
|
||||
Serial.print(APM_Compass.Mag_X);
|
||||
Serial.print(comma);
|
||||
Serial.print(APM_Compass.Mag_Y);
|
||||
Serial.print(comma);
|
||||
Serial.print(APM_Compass.Mag_Z);
|
||||
Serial.println(")");
|
||||
if(Serial.available() > 0){
|
||||
return (0);
|
||||
}
|
||||
}
|
||||
#endif
|
||||
}
|
||||
|
||||
static int8_t
|
||||
test_jeti(uint8_t argc, const Menu::arg *argv)
|
||||
{
|
||||
uint8_t switchPosition, m;
|
||||
print_hit_enter();
|
||||
|
||||
JB.begin();
|
||||
//JB.print(" X-DIY");
|
||||
//JB.print(" Test mode");
|
||||
//JB.checkvalue(20000);
|
||||
Serial.print("Jeti Box test: press any buttons\n\n");
|
||||
while(1){
|
||||
delay(200);
|
||||
//Serial.println(JB.checkvalue(0));
|
||||
read_radio();
|
||||
switchPosition = readSwitch();
|
||||
m = flight_modes[switchPosition];
|
||||
JB.clear(1);
|
||||
JB.print("Mode: ");
|
||||
JB.print(flight_mode_strings[m]);
|
||||
JB.clear(2);
|
||||
switch (JB.readbuttons())
|
||||
{
|
||||
case JB_key_right:
|
||||
JB.print("rechts");
|
||||
Serial.print("rechts\n");
|
||||
break;
|
||||
|
||||
case JB_key_left:
|
||||
JB.print("links");
|
||||
Serial.print("links\n");
|
||||
break;
|
||||
|
||||
case JB_key_up:
|
||||
JB.print("hoch");
|
||||
Serial.print("hoch\n");
|
||||
break;
|
||||
|
||||
case JB_key_down:
|
||||
JB.print("runter");
|
||||
Serial.print("runter\n");
|
||||
break;
|
||||
|
||||
case JB_key_left | JB_key_right:
|
||||
JB.print("links-rechts");
|
||||
Serial.print("links-rechts\n");
|
||||
break;
|
||||
}
|
||||
|
||||
if(Serial.available() > 0){
|
||||
return (0);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void print_hit_enter()
|
||||
{
|
||||
Serial.printf_P(PSTR("Hit Enter to exit.\n\n"));
|
||||
}
|
@ -0,0 +1,149 @@
|
||||
/*
|
||||
APM_ADC.cpp - ADC ADS7844 Library for Ardupilot Mega
|
||||
Code by Jordi Muñoz and Jose Julio. DIYDrones.com
|
||||
|
||||
Modified by John Ihlein 6/19/2010 to:
|
||||
1)Prevent overflow of adc_counter when more than 8 samples collected between reads. Probably
|
||||
only an issue on initial read of ADC at program start.
|
||||
2)Reorder analog read order as follows:
|
||||
p, q, r, ax, ay, az
|
||||
|
||||
This library is free software; you can redistribute it and/or
|
||||
modify it under the terms of the GNU Lesser General Public
|
||||
License as published by the Free Software Foundation; either
|
||||
version 2.1 of the License, or (at your option) any later version.
|
||||
|
||||
External ADC ADS7844 is connected via Serial port 2 (in SPI mode)
|
||||
TXD2 = MOSI = pin PH1
|
||||
RXD2 = MISO = pin PH0
|
||||
XCK2 = SCK = pin PH2
|
||||
Chip Select pin is PC4 (33) [PH6 (9)]
|
||||
We are using the 16 clocks per conversion timming to increase efficiency (fast)
|
||||
The sampling frequency is 400Hz (Timer2 overflow interrupt)
|
||||
So if our loop is at 50Hz, our needed sampling freq should be 100Hz, so
|
||||
we have an 4x oversampling and averaging.
|
||||
|
||||
Methods:
|
||||
Init() : Initialization of interrupts an Timers (Timer2 overflow interrupt)
|
||||
Ch(ch_num) : Return the ADC channel value
|
||||
|
||||
// HJI - Input definitions. USB connector assumed to be on the left, Rx and servo
|
||||
// connector pins to the rear. IMU shield components facing up. These are board
|
||||
// referenced sensor inputs, not device referenced.
|
||||
On Ardupilot Mega Hardware, oriented as described above:
|
||||
Channel 0 : yaw rate, r
|
||||
Channel 1 : roll rate, p
|
||||
Channel 2 : pitch rate, q
|
||||
Channel 3 : x/y gyro temperature
|
||||
Channel 4 : x acceleration, aX
|
||||
Channel 5 : y acceleration, aY
|
||||
Channel 6 : z acceleration, aZ
|
||||
Channel 7 : Differential pressure sensor port
|
||||
|
||||
*/
|
||||
extern "C" {
|
||||
// AVR LibC Includes
|
||||
#include <inttypes.h>
|
||||
#include <avr/interrupt.h>
|
||||
#include "WConstants.h"
|
||||
}
|
||||
|
||||
#include "APM_ADC.h"
|
||||
|
||||
|
||||
// Commands for reading ADC channels on ADS7844
|
||||
const unsigned char adc_cmd[9]= { 0x87, 0xC7, 0x97, 0xD7, 0xA7, 0xE7, 0xB7, 0xF7, 0x00 };
|
||||
volatile long adc_value[8] = { 0, 0, 0, 0, 0, 0, 0, 0 };
|
||||
volatile unsigned char adc_counter[8] = { 0, 0, 0, 0, 0, 0, 0, 0 };
|
||||
|
||||
unsigned char ADC_SPI_transfer(unsigned char data)
|
||||
{
|
||||
/* Wait for empty transmit buffer */
|
||||
while ( !( UCSR2A & (1<<UDRE2)) );
|
||||
/* Put data into buffer, sends the data */
|
||||
UDR2 = data;
|
||||
/* Wait for data to be received */
|
||||
while ( !(UCSR2A & (1<<RXC2)) );
|
||||
/* Get and return received data from buffer */
|
||||
return UDR2;
|
||||
}
|
||||
|
||||
|
||||
ISR (TIMER2_OVF_vect)
|
||||
{
|
||||
uint8_t ch;
|
||||
unsigned int adc_tmp;
|
||||
|
||||
//bit_set(PORTL,6); // To test performance
|
||||
bit_clear(PORTC,4); // Enable Chip Select (PIN PC4)
|
||||
ADC_SPI_transfer(adc_cmd[0]); // Command to read the first channel
|
||||
for (ch=0;ch<8;ch++)
|
||||
{
|
||||
if (adc_counter[ch] >= 17) // HJI - Added this to prevent
|
||||
{ // overflow of adc_value
|
||||
adc_value[ch] = 0;
|
||||
adc_counter[ch] = 0;
|
||||
}
|
||||
adc_tmp = ADC_SPI_transfer(0)<<8; // Read first byte
|
||||
adc_tmp |= ADC_SPI_transfer(adc_cmd[ch+1]); // Read second byte and send next command
|
||||
adc_value[ch] += adc_tmp>>3; // Shift to 12 bits
|
||||
adc_counter[ch]++; // Number of samples
|
||||
}
|
||||
bit_set(PORTC,4); // Disable Chip Select (PIN PC4)
|
||||
//bit_clear(PORTL,6); // To test performance
|
||||
TCNT2 = 104; // 400 Hz
|
||||
}
|
||||
|
||||
|
||||
// Constructors ////////////////////////////////////////////////////////////////
|
||||
APM_ADC_Class::APM_ADC_Class()
|
||||
{
|
||||
}
|
||||
|
||||
// Public Methods //////////////////////////////////////////////////////////////
|
||||
void APM_ADC_Class::Init(void)
|
||||
{
|
||||
unsigned char tmp;
|
||||
|
||||
pinMode(ADC_CHIP_SELECT,OUTPUT);
|
||||
|
||||
digitalWrite(ADC_CHIP_SELECT,HIGH); // Disable device (Chip select is active low)
|
||||
|
||||
// Setup Serial Port2 in SPI mode
|
||||
UBRR2 = 0;
|
||||
DDRH |= (1<<PH2); // SPI clock XCK2 (PH2) as output. This enable SPI Master mode
|
||||
// Set MSPI mode of operation and SPI data mode 0.
|
||||
UCSR2C = (1<<UMSEL21)|(1<<UMSEL20); //|(0<<UCPHA2)|(0<<UCPOL2);
|
||||
// Enable receiver and transmitter.
|
||||
UCSR2B = (1<<RXEN2)|(1<<TXEN2);
|
||||
// Set Baud rate
|
||||
UBRR2 = 2; // SPI clock running at 2.6MHz
|
||||
|
||||
|
||||
// Enable Timer2 Overflow interrupt to capture ADC data
|
||||
TIMSK2 = 0; // Disable interrupts
|
||||
TCCR2A = 0; // normal counting mode
|
||||
TCCR2B = _BV(CS21)|_BV(CS22); // Set prescaler of 256
|
||||
TCNT2 = 0;
|
||||
TIFR2 = _BV(TOV2); // clear pending interrupts;
|
||||
TIMSK2 = _BV(TOIE2) ; // enable the overflow interrupt
|
||||
}
|
||||
|
||||
// Read one channel value
|
||||
int APM_ADC_Class::Ch(unsigned char ch_num)
|
||||
{
|
||||
int result;
|
||||
|
||||
cli(); // We stop interrupts to read the variables
|
||||
if (adc_counter[ch_num]>0)
|
||||
result = adc_value[ch_num]/adc_counter[ch_num];
|
||||
else
|
||||
result = 0;
|
||||
adc_value[ch_num] = 0; // Initialize for next reading
|
||||
adc_counter[ch_num] = 0;
|
||||
sei();
|
||||
return(result);
|
||||
}
|
||||
|
||||
// make one instance for the user to use
|
||||
APM_ADC_Class APM_ADC;
|
@ -0,0 +1,24 @@
|
||||
#ifndef APM_ADC_h
|
||||
#define APM_ADC_h
|
||||
|
||||
#define bit_set(p,m) ((p) |= ( 1<<m))
|
||||
#define bit_clear(p,m) ((p) &= ~(1<<m))
|
||||
|
||||
// We use Serial Port 2 in SPI Mode
|
||||
#define ADC_DATAOUT 51 // MOSI
|
||||
#define ADC_DATAIN 50 // MISO
|
||||
#define ADC_SPICLOCK 52 // SCK
|
||||
#define ADC_CHIP_SELECT 33 // PC4 9 // PH6 Puerto:0x08 Bit mask : 0x40
|
||||
|
||||
class APM_ADC_Class
|
||||
{
|
||||
private:
|
||||
public:
|
||||
APM_ADC_Class(); // Constructor
|
||||
void Init();
|
||||
int Ch(unsigned char ch_num);
|
||||
};
|
||||
|
||||
extern APM_ADC_Class APM_ADC;
|
||||
|
||||
#endif
|
@ -0,0 +1,33 @@
|
||||
/*
|
||||
Example of APM_ADC library.
|
||||
Code by Jordi Muñoz and Jose Julio. DIYDrones.com
|
||||
*/
|
||||
|
||||
#include <APM_ADC.h> // ArduPilot Mega ADC Library
|
||||
|
||||
unsigned long timer;
|
||||
|
||||
void setup()
|
||||
{
|
||||
APM_ADC.Init(); // APM ADC initialization
|
||||
Serial.begin(57600);
|
||||
Serial.println("ArduPilot Mega ADC library test");
|
||||
delay(1000);
|
||||
timer = millis();
|
||||
}
|
||||
|
||||
void loop()
|
||||
{
|
||||
int ch;
|
||||
|
||||
if((millis()- timer) > 20)
|
||||
{
|
||||
timer = millis();
|
||||
for (ch=0;ch<7;ch++)
|
||||
{
|
||||
Serial.print(APM_ADC.Ch(ch));
|
||||
Serial.print(",");
|
||||
}
|
||||
Serial.println();
|
||||
}
|
||||
}
|
@ -0,0 +1,3 @@
|
||||
APM_ADC KEYWORD1
|
||||
Init KEYWORD2
|
||||
Ch KEYWORD2
|
@ -0,0 +1,243 @@
|
||||
/*
|
||||
APM_BMP085.cpp - Arduino Library for BMP085 absolute pressure sensor
|
||||
Code by Jordi Muñoz and Jose Julio. DIYDrones.com
|
||||
|
||||
This library is free software; you can redistribute it and/or
|
||||
modify it under the terms of the GNU Lesser General Public
|
||||
License as published by the Free Software Foundation; either
|
||||
version 2.1 of the License, or (at your option) any later version.
|
||||
|
||||
Sensor is conected to I2C port
|
||||
Sensor End of Conversion (EOC) pin is PC7 (30)
|
||||
|
||||
Variables:
|
||||
RawTemp : Raw temperature data
|
||||
RawPress : Raw pressure data
|
||||
|
||||
Temp : Calculated temperature (in 0.1ºC units)
|
||||
Press : Calculated pressure (in Pa units)
|
||||
|
||||
Methods:
|
||||
Init() : Initialization of I2C and read sensor calibration data
|
||||
Read() : Read sensor data and calculate Temperature and Pressure
|
||||
This function is optimized so the main host don´t need to wait
|
||||
You can call this function in your main loop
|
||||
It returns a 1 if there are new data.
|
||||
|
||||
Internal functions:
|
||||
Command_ReadTemp(): Send commando to read temperature
|
||||
Command_ReadPress(): Send commando to read Pressure
|
||||
ReadTemp() : Read temp register
|
||||
ReadPress() : Read press register
|
||||
Calculate() : Calculate Temperature and Pressure in real units
|
||||
|
||||
|
||||
*/
|
||||
extern "C" {
|
||||
// AVR LibC Includes
|
||||
#include <inttypes.h>
|
||||
#include <avr/interrupt.h>
|
||||
#include "WConstants.h"
|
||||
}
|
||||
|
||||
#include <Wire.h>
|
||||
#include "APM_BMP085.h"
|
||||
|
||||
#define BMP085_ADDRESS 0x77 //(0xEE >> 1)
|
||||
#define BMP085_EOC 30 // End of conversion pin PC7
|
||||
|
||||
// Constructors ////////////////////////////////////////////////////////////////
|
||||
APM_BMP085_Class::APM_BMP085_Class()
|
||||
{
|
||||
}
|
||||
|
||||
// Public Methods //////////////////////////////////////////////////////////////
|
||||
void APM_BMP085_Class::Init(void)
|
||||
{
|
||||
unsigned char tmp;
|
||||
byte buff[22];
|
||||
int i=0;
|
||||
|
||||
pinMode(BMP085_EOC,INPUT); // End Of Conversion (PC7) input
|
||||
Wire.begin();
|
||||
oss = 3; // Over Sampling setting 3 = High resolution
|
||||
BMP085_State = 0; // Initial state
|
||||
|
||||
// We read the calibration data registers
|
||||
Wire.beginTransmission(BMP085_ADDRESS);
|
||||
Wire.send(0xAA);
|
||||
Wire.endTransmission();
|
||||
|
||||
Wire.requestFrom(BMP085_ADDRESS, 22);
|
||||
//Wire.endTransmission();
|
||||
while(Wire.available())
|
||||
{
|
||||
buff[i] = Wire.receive(); // receive one byte
|
||||
i++;
|
||||
}
|
||||
ac1 = ((int)buff[0] << 8) | buff[1];
|
||||
ac2 = ((int)buff[2] << 8) | buff[3];
|
||||
ac3 = ((int)buff[4] << 8) | buff[5];
|
||||
ac4 = ((int)buff[6] << 8) | buff[7];
|
||||
ac5 = ((int)buff[8] << 8) | buff[9];
|
||||
ac6 = ((int)buff[10] << 8) | buff[11];
|
||||
b1 = ((int)buff[12] << 8) | buff[13];
|
||||
b2 = ((int)buff[14] << 8) | buff[15];
|
||||
mb = ((int)buff[16] << 8) | buff[17];
|
||||
mc = ((int)buff[18] << 8) | buff[19];
|
||||
md = ((int)buff[20] << 8) | buff[21];
|
||||
|
||||
//Send a command to read Temp
|
||||
Command_ReadTemp();
|
||||
BMP085_State=1;
|
||||
}
|
||||
|
||||
|
||||
// Read the sensor. This is a state machine
|
||||
// We read one time Temperature (state=1) and then 4 times Pressure (states 2-5)
|
||||
uint8_t APM_BMP085_Class::Read()
|
||||
{
|
||||
uint8_t result=0;
|
||||
|
||||
if (BMP085_State==1)
|
||||
{
|
||||
if (digitalRead(BMP085_EOC))
|
||||
{
|
||||
ReadTemp(); // On state 1 we read temp
|
||||
BMP085_State++;
|
||||
Command_ReadPress();
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
if (BMP085_State==5)
|
||||
{
|
||||
if (digitalRead(BMP085_EOC))
|
||||
{
|
||||
ReadPress();
|
||||
Calculate();
|
||||
BMP085_State = 1; // Start again from state=1
|
||||
Command_ReadTemp(); // Read Temp
|
||||
result=1; // New pressure reading
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
if (digitalRead(BMP085_EOC))
|
||||
{
|
||||
ReadPress();
|
||||
Calculate();
|
||||
BMP085_State++;
|
||||
Command_ReadPress();
|
||||
result=1; // New pressure reading
|
||||
}
|
||||
}
|
||||
}
|
||||
return(result);
|
||||
}
|
||||
|
||||
|
||||
// Send command to Read Pressure
|
||||
void APM_BMP085_Class::Command_ReadPress()
|
||||
{
|
||||
Wire.beginTransmission(BMP085_ADDRESS);
|
||||
Wire.send(0xF4);
|
||||
Wire.send(0x34+(oss<<6)); //write_register(0xF4,0x34+(oversampling_setting<<6));
|
||||
Wire.endTransmission();
|
||||
}
|
||||
|
||||
// Read Raw Pressure values
|
||||
void APM_BMP085_Class::ReadPress()
|
||||
{
|
||||
byte msb;
|
||||
byte lsb;
|
||||
byte xlsb;
|
||||
|
||||
Wire.beginTransmission(BMP085_ADDRESS);
|
||||
Wire.send(0xF6);
|
||||
Wire.endTransmission();
|
||||
|
||||
Wire.requestFrom(BMP085_ADDRESS, 3); // read a byte
|
||||
while(!Wire.available()) {
|
||||
// waiting
|
||||
}
|
||||
msb = Wire.receive();
|
||||
while(!Wire.available()) {
|
||||
// waiting
|
||||
}
|
||||
lsb = Wire.receive();
|
||||
while(!Wire.available()) {
|
||||
// waiting
|
||||
}
|
||||
xlsb = Wire.receive();
|
||||
RawPress = (((long)msb<<16) | ((long)lsb<<8) | ((long)xlsb)) >> (8-oss);
|
||||
}
|
||||
|
||||
// Send Command to Read Temperature
|
||||
void APM_BMP085_Class::Command_ReadTemp()
|
||||
{
|
||||
Wire.beginTransmission(BMP085_ADDRESS);
|
||||
Wire.send(0xF4);
|
||||
Wire.send(0x2E);
|
||||
Wire.endTransmission();
|
||||
}
|
||||
|
||||
// Read Raw Temperature values
|
||||
void APM_BMP085_Class::ReadTemp()
|
||||
{
|
||||
byte tmp;
|
||||
|
||||
Wire.beginTransmission(BMP085_ADDRESS);
|
||||
Wire.send(0xF6);
|
||||
Wire.endTransmission();
|
||||
|
||||
Wire.beginTransmission(BMP085_ADDRESS);
|
||||
Wire.requestFrom(BMP085_ADDRESS,2);
|
||||
while(!Wire.available()); // wait
|
||||
RawTemp = Wire.receive();
|
||||
while(!Wire.available()); // wait
|
||||
tmp = Wire.receive();
|
||||
RawTemp = RawTemp<<8 | tmp;
|
||||
}
|
||||
|
||||
// Calculate Temperature and Pressure in real units.
|
||||
void APM_BMP085_Class::Calculate()
|
||||
{
|
||||
long x1, x2, x3, b3, b5, b6, p;
|
||||
unsigned long b4, b7;
|
||||
int32_t tmp;
|
||||
|
||||
// See Datasheet page 13 for this formulas
|
||||
// Based also on Jee Labs BMP085 example code. Thanks for share.
|
||||
// Temperature calculations
|
||||
x1 = ((long)RawTemp - ac6) * ac5 >> 15;
|
||||
x2 = ((long) mc << 11) / (x1 + md);
|
||||
b5 = x1 + x2;
|
||||
Temp = (b5 + 8) >> 4;
|
||||
|
||||
// Pressure calculations
|
||||
b6 = b5 - 4000;
|
||||
x1 = (b2 * (b6 * b6 >> 12)) >> 11;
|
||||
x2 = ac2 * b6 >> 11;
|
||||
x3 = x1 + x2;
|
||||
//b3 = (((int32_t) ac1 * 4 + x3)<<oss + 2) >> 2; // BAD
|
||||
//b3 = ((int32_t) ac1 * 4 + x3 + 2) >> 2; //OK for oss=0
|
||||
tmp = ac1;
|
||||
tmp = (tmp*4 + x3)<<oss;
|
||||
b3 = (tmp+2)/4;
|
||||
x1 = ac3 * b6 >> 13;
|
||||
x2 = (b1 * (b6 * b6 >> 12)) >> 16;
|
||||
x3 = ((x1 + x2) + 2) >> 2;
|
||||
b4 = (ac4 * (uint32_t) (x3 + 32768)) >> 15;
|
||||
b7 = ((uint32_t) RawPress - b3) * (50000 >> oss);
|
||||
p = b7 < 0x80000000 ? (b7 * 2) / b4 : (b7 / b4) * 2;
|
||||
|
||||
x1 = (p >> 8) * (p >> 8);
|
||||
x1 = (x1 * 3038) >> 16;
|
||||
x2 = (-7357 * p) >> 16;
|
||||
Press = p + ((x1 + x2 + 3791) >> 4);
|
||||
}
|
||||
|
||||
|
||||
// make one instance for the user to use
|
||||
APM_BMP085_Class APM_BMP085;
|
@ -0,0 +1,34 @@
|
||||
#ifndef APM_BMP085_h
|
||||
#define APM_BMP085_h
|
||||
|
||||
|
||||
class APM_BMP085_Class
|
||||
{
|
||||
private:
|
||||
// State machine
|
||||
uint8_t BMP085_State;
|
||||
// Internal calibration registers
|
||||
int16_t ac1, ac2, ac3, b1, b2, mb, mc, md;
|
||||
uint16_t ac4, ac5, ac6;
|
||||
void Command_ReadPress();
|
||||
void Command_ReadTemp();
|
||||
void ReadPress();
|
||||
void ReadTemp();
|
||||
void Calculate();
|
||||
public:
|
||||
int32_t RawPress;
|
||||
int32_t RawTemp;
|
||||
int16_t Temp;
|
||||
int32_t Press;
|
||||
//int Altitude;
|
||||
uint8_t oss;
|
||||
//int32_t Press0; // Pressure at sea level
|
||||
|
||||
APM_BMP085_Class(); // Constructor
|
||||
void Init();
|
||||
uint8_t Read();
|
||||
};
|
||||
|
||||
extern APM_BMP085_Class APM_BMP085;
|
||||
|
||||
#endif
|
@ -0,0 +1,41 @@
|
||||
/*
|
||||
Example of APM_BMP085 (absolute pressure sensor) library.
|
||||
Code by Jordi Muñoz and Jose Julio. DIYDrones.com
|
||||
*/
|
||||
|
||||
#include <Wire.h>
|
||||
#include <APM_BMP085.h> // ArduPilot Mega BMP085 Library
|
||||
|
||||
unsigned long timer;
|
||||
|
||||
void setup()
|
||||
{
|
||||
APM_BMP085.Init(); // APM ADC initialization
|
||||
Serial.begin(57600);
|
||||
Serial.println("ArduPilot Mega BMP085 library test");
|
||||
delay(1000);
|
||||
timer = millis();
|
||||
}
|
||||
|
||||
void loop()
|
||||
{
|
||||
int ch;
|
||||
float tmp_float;
|
||||
float Altitude;
|
||||
|
||||
if((millis()- timer) > 50)
|
||||
{
|
||||
timer=millis();
|
||||
APM_BMP085.Read();
|
||||
Serial.print("Pressure:");
|
||||
Serial.print(APM_BMP085.Press);
|
||||
Serial.print(" Temperature:");
|
||||
Serial.print(APM_BMP085.Temp/10.0);
|
||||
Serial.print(" Altitude:");
|
||||
tmp_float = (APM_BMP085.Press/101325.0);
|
||||
tmp_float = pow(tmp_float,0.190295);
|
||||
Altitude = 44330*(1.0-tmp_float);
|
||||
Serial.print(Altitude);
|
||||
Serial.println();
|
||||
}
|
||||
}
|
@ -0,0 +1,5 @@
|
||||
APM_BMP085 KEYWORD1
|
||||
Init KEYWORD2
|
||||
Read KEYWORD2
|
||||
Press KEYWORD2
|
||||
Temp KEYWORD2
|
@ -0,0 +1,206 @@
|
||||
// -*- Mode: C++; c-basic-offset: 8; indent-tabs-mode: nil -*-
|
||||
//
|
||||
// Copyright (c) 2010 Michael Smith. All rights reserved.
|
||||
//
|
||||
// Redistribution and use in source and binary forms, with or without
|
||||
// modification, are permitted provided that the following conditions
|
||||
// are met:
|
||||
// 1. Redistributions of source code must retain the above copyright
|
||||
// notice, this list of conditions and the following disclaimer.
|
||||
// 2. Redistributions in binary form must reproduce the above copyright
|
||||
// notice, this list of conditions and the following disclaimer in the
|
||||
// documentation and/or other materials provided with the distribution.
|
||||
//
|
||||
// THIS SOFTWARE IS PROVIDED BY THE AUTHOR 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 AUTHOR 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.
|
||||
|
||||
/// @file BinComm.cpp
|
||||
/// @brief Implementation of the ArduPilot Mega binary communications
|
||||
/// library.
|
||||
|
||||
#include "APM_BinComm.h"
|
||||
#include "WProgram.h"
|
||||
|
||||
/// @name decoder state machine phases
|
||||
//@{
|
||||
#define DEC_WAIT_P1 0
|
||||
#define DEC_WAIT_P2 1
|
||||
#define DEC_WAIT_HEADER 2
|
||||
#define DEC_WAIT_MESSAGE 3
|
||||
#define DEC_WAIT_SUM_A 4
|
||||
#define DEC_WAIT_SUM_B 5
|
||||
//@}
|
||||
|
||||
|
||||
/// inter-byte timeout for decode (ms)
|
||||
#define DEC_MESSAGE_TIMEOUT 100
|
||||
|
||||
BinComm::BinComm(const BinComm::MessageHandler *handlerTable,
|
||||
Stream *interface) :
|
||||
_handlerTable(handlerTable),
|
||||
_interface(interface)
|
||||
{
|
||||
};
|
||||
|
||||
void
|
||||
BinComm::_sendMessage(void)
|
||||
{
|
||||
uint8_t bytesToSend;
|
||||
uint8_t sumA, sumB;
|
||||
const uint8_t *p;
|
||||
|
||||
// send the preamble first
|
||||
_interface->write((uint8_t)MSG_PREAMBLE_1);
|
||||
_interface->write((uint8_t)MSG_PREAMBLE_2);
|
||||
|
||||
// set up to send the payload
|
||||
bytesToSend = _encodeBuf.header.length + sizeof(_encodeBuf.header);
|
||||
sumA = sumB = 0;
|
||||
p = (const uint8_t *)&_encodeBuf;
|
||||
|
||||
// send message bytes and compute checksum on the fly
|
||||
while (bytesToSend--) {
|
||||
sumA += *p;
|
||||
sumB += sumA;
|
||||
_interface->write(*p++);
|
||||
}
|
||||
|
||||
// send the checksum
|
||||
_interface->write(sumA);
|
||||
_interface->write(sumB);
|
||||
}
|
||||
|
||||
void
|
||||
BinComm::update(void)
|
||||
{
|
||||
uint8_t count;
|
||||
|
||||
// Ensure that we don't spend too long here by only processing
|
||||
// the bytes that were available when we started.
|
||||
//
|
||||
// XXX we might want to further constrain this count
|
||||
count = _interface->available();
|
||||
while (count--)
|
||||
_decode(_interface->read());
|
||||
}
|
||||
|
||||
void
|
||||
BinComm::_decode(uint8_t inByte)
|
||||
{
|
||||
uint8_t tableIndex;
|
||||
|
||||
// handle inter-byte timeouts (resync after link loss, etc.)
|
||||
//
|
||||
if ((millis() - _lastReceived) > DEC_MESSAGE_TIMEOUT)
|
||||
_decodePhase = DEC_WAIT_P1;
|
||||
|
||||
// run the decode state machine
|
||||
//
|
||||
switch (_decodePhase) {
|
||||
|
||||
// Preamble detection
|
||||
//
|
||||
// Note the fallthrough from P2 to P1 deals with the case where
|
||||
// we see 0x34, 0x34, 0x44 where the first 0x34 is garbage or
|
||||
// a SUM_B byte we never looked at.
|
||||
//
|
||||
case DEC_WAIT_P2:
|
||||
if (MSG_PREAMBLE_2 == inByte) {
|
||||
_decodePhase++;
|
||||
|
||||
// prepare for the header
|
||||
_bytesIn = 0;
|
||||
_bytesExpected = sizeof(MessageHeader);
|
||||
|
||||
// intialise the checksum accumulators
|
||||
_sumA = _sumB = 0;
|
||||
|
||||
break;
|
||||
}
|
||||
_decodePhase = DEC_WAIT_P1;
|
||||
// FALLTHROUGH
|
||||
case DEC_WAIT_P1:
|
||||
if (MSG_PREAMBLE_1 == inByte) {
|
||||
_decodePhase++;
|
||||
}
|
||||
break;
|
||||
|
||||
// receiving the header
|
||||
//
|
||||
case DEC_WAIT_HEADER:
|
||||
// do checksum accumulation
|
||||
_sumA += inByte;
|
||||
_sumB += _sumA;
|
||||
|
||||
// store the byte
|
||||
_decodeBuf.bytes[_bytesIn++] = inByte;
|
||||
|
||||
// check for complete header received
|
||||
if (_bytesIn == _bytesExpected) {
|
||||
_decodePhase++;
|
||||
|
||||
// prepare for the payload
|
||||
// variable-length data?
|
||||
_bytesIn = 0;
|
||||
_bytesExpected = _decodeBuf.header.length;
|
||||
_messageID = _decodeBuf.header.messageID;
|
||||
_messageVersion = _decodeBuf.header.messageVersion;
|
||||
|
||||
// sanity check to avoid buffer overflow - revert back to waiting
|
||||
if (_bytesExpected > sizeof(_decodeBuf))
|
||||
_decodePhase = DEC_WAIT_P1;
|
||||
}
|
||||
break;
|
||||
|
||||
// receiving payload data
|
||||
//
|
||||
case DEC_WAIT_MESSAGE:
|
||||
// do checksum accumulation
|
||||
_sumA += inByte;
|
||||
_sumB += _sumA;
|
||||
|
||||
// store the byte
|
||||
_decodeBuf.bytes[_bytesIn++] = inByte;
|
||||
|
||||
// check for complete payload received
|
||||
if (_bytesIn == _bytesExpected) {
|
||||
_decodePhase++;
|
||||
}
|
||||
break;
|
||||
|
||||
// waiting for the checksum bytes
|
||||
//
|
||||
case DEC_WAIT_SUM_A:
|
||||
if (inByte != _sumA) {
|
||||
badMessagesReceived++;
|
||||
_decodePhase = DEC_WAIT_P1;
|
||||
} else {
|
||||
_decodePhase++;
|
||||
}
|
||||
break;
|
||||
case DEC_WAIT_SUM_B:
|
||||
if (inByte == _sumB) {
|
||||
// if we got this far, we have a message
|
||||
messagesReceived++;
|
||||
|
||||
// call any handler interested in this message
|
||||
for (tableIndex = 0; MSG_NULL != _handlerTable[tableIndex].messageID; tableIndex++)
|
||||
if ((_handlerTable[tableIndex].messageID == _messageID) ||
|
||||
(_handlerTable[tableIndex].messageID == MSG_ANY))
|
||||
_handlerTable[tableIndex].handler(_handlerTable[tableIndex].arg, _messageID, _messageVersion, &_decodeBuf);
|
||||
} else {
|
||||
badMessagesReceived++;
|
||||
}
|
||||
_decodePhase = DEC_WAIT_P1;
|
||||
break;
|
||||
}
|
||||
}
|
@ -0,0 +1,267 @@
|
||||
// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: t -*-
|
||||
//
|
||||
// Copyright (c) 2010 Michael Smith. All rights reserved.
|
||||
//
|
||||
// Redistribution and use in source and binary forms, with or without
|
||||
// modification, are permitted provided that the following conditions
|
||||
// are met:
|
||||
// 1. Redistributions of source code must retain the above copyright
|
||||
// notice, this list of conditions and the following disclaimer.
|
||||
// 2. Redistributions in binary form must reproduce the above copyright
|
||||
// notice, this list of conditions and the following disclaimer in the
|
||||
// documentation and/or other materials provided with the distribution.
|
||||
//
|
||||
// THIS SOFTWARE IS PROVIDED BY THE AUTHOR 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 AUTHOR 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.
|
||||
|
||||
/// @file BinComm.h
|
||||
/// @brief Definitions for the ArduPilot Mega binary communications
|
||||
/// library.
|
||||
|
||||
#ifndef APM_BinComm_h
|
||||
#define APM_BinComm_h
|
||||
|
||||
#include <string.h>
|
||||
#include <inttypes.h>
|
||||
#include "WProgram.h"
|
||||
|
||||
///
|
||||
/// @class BinComm
|
||||
/// @brief Class providing protocol en/decoding services for the ArduPilot
|
||||
/// Mega binary telemetry protocol.
|
||||
///
|
||||
class BinComm {
|
||||
public:
|
||||
struct MessageHandler;
|
||||
|
||||
//////////////////////////////////////////////////////////////////////
|
||||
/// Constructor.
|
||||
///
|
||||
/// @param handlerTable Array of callout functions to which
|
||||
/// received messages will be sent. More than
|
||||
/// one handler for a given messageID may be
|
||||
/// registered; handlers are called in the order
|
||||
/// they appear in the table.
|
||||
///
|
||||
/// @param interface The stream that will be used
|
||||
/// for telemetry communications.
|
||||
///
|
||||
BinComm(const MessageHandler *handlerTable,
|
||||
Stream *interface);
|
||||
|
||||
private:
|
||||
/// OTA message header
|
||||
struct MessageHeader {
|
||||
uint8_t length;
|
||||
uint8_t messageID;
|
||||
uint8_t messageVersion;
|
||||
};
|
||||
|
||||
/// Incoming header/packet buffer
|
||||
/// XXX we could make this smaller
|
||||
union {
|
||||
uint8_t bytes[0];
|
||||
MessageHeader header;
|
||||
uint8_t payload[256];
|
||||
} _decodeBuf;
|
||||
|
||||
/// Outgoing header/packet buffer
|
||||
/// XXX we could make this smaller
|
||||
struct {
|
||||
MessageHeader header;
|
||||
uint8_t payload[256 - sizeof(MessageHeader)];
|
||||
} _encodeBuf;
|
||||
|
||||
|
||||
//////////////////////////////////////////////////////////////////////
|
||||
/// @name Message pack/unpack utility functions
|
||||
///
|
||||
//@{
|
||||
inline void _pack(uint8_t *&ptr, const uint8_t x) { *(uint8_t *)ptr = x; ptr += sizeof(x); }
|
||||
inline void _pack(uint8_t *&ptr, const uint16_t x) { *(uint16_t *)ptr = x; ptr += sizeof(x); }
|
||||
inline void _pack(uint8_t *&ptr, const int16_t x) { *(int16_t *)ptr = x; ptr += sizeof(x); }
|
||||
inline void _pack(uint8_t *&ptr, const uint32_t x) { *(uint32_t *)ptr = x; ptr += sizeof(x); }
|
||||
inline void _pack(uint8_t *&ptr, const int32_t x) { *(int32_t *)ptr = x; ptr += sizeof(x); }
|
||||
|
||||
inline void _pack(uint8_t *&ptr, const char *msg, uint8_t size) { strlcpy((char *)ptr, msg, size); ptr += size; }
|
||||
inline void _pack(uint8_t *&ptr, const uint8_t *values, uint8_t count) { memcpy(ptr, values, count); ptr += count; }
|
||||
inline void _pack(uint8_t *&ptr, const uint16_t *values, uint8_t count) { memcpy(ptr, values, count * 2); ptr += count * 2; }
|
||||
|
||||
inline void _unpack(uint8_t *&ptr, uint8_t &x) { x = *(uint8_t *)ptr; ptr += sizeof(x); }
|
||||
inline void _unpack(uint8_t *&ptr, uint16_t &x) { x = *(uint16_t *)ptr; ptr += sizeof(x); }
|
||||
inline void _unpack(uint8_t *&ptr, int16_t &x) { x = *(int16_t *)ptr; ptr += sizeof(x); }
|
||||
inline void _unpack(uint8_t *&ptr, uint32_t &x) { x = *(uint32_t *)ptr; ptr += sizeof(x); }
|
||||
inline void _unpack(uint8_t *&ptr, int32_t &x) { x = *(int32_t *)ptr; ptr += sizeof(x); }
|
||||
|
||||
inline void _unpack(uint8_t *&ptr, char *msg, uint8_t size) { strlcpy(msg, (char *)ptr, size); ptr += size; }
|
||||
inline void _unpack(uint8_t *&ptr, uint8_t *values, uint8_t count) { memcpy(values, ptr, count); ptr += count; }
|
||||
inline void _unpack(uint8_t *&ptr, uint16_t *values, uint8_t count) { memcpy(values, ptr, count * 2); ptr += count * 2; }
|
||||
//@}
|
||||
|
||||
public:
|
||||
//////////////////////////////////////////////////////////////////////
|
||||
/// @name Protocol definition
|
||||
///
|
||||
/// The protocol definition, including structures describing messages,
|
||||
/// MessageID values and helper functions for packing messages are
|
||||
/// automatically generated.
|
||||
//@{
|
||||
#include "protocol/protocol.h"
|
||||
//@}
|
||||
|
||||
//////////////////////////////////////////////////////////////////////
|
||||
/// @name Protocol magic numbers
|
||||
///
|
||||
/// @note The MessageID enum is automatically generated and thus not described here.
|
||||
///
|
||||
//@{
|
||||
|
||||
/// Variables defined
|
||||
/// XXX these should probably be handled by the database/MIB?
|
||||
enum variableID {
|
||||
MSG_VAR_ROLL_MODE = 0x00,
|
||||
MSG_VAR_PITCH_MODE = 0x01,
|
||||
MSG_VAR_THROTTLE_MODE = 0x02,
|
||||
MSG_VAR_YAW_MODE = 0x03,
|
||||
MSG_VAR_ELEVON_TRIM_1 = 0x04,
|
||||
MSG_VAR_ELEVON_TRIM_2 = 0x05,
|
||||
|
||||
MSG_VAR_INTEGRATOR_0 = 0x10,
|
||||
MSG_VAR_INTEGRATOR_1 = 0x11,
|
||||
MSG_VAR_INTEGRATOR_2 = 0x12,
|
||||
MSG_VAR_INTEGRATOR_3 = 0x13,
|
||||
MSG_VAR_INTEGRATOR_4 = 0x14,
|
||||
MSG_VAR_INTEGRATOR_5 = 0x15,
|
||||
MSG_VAR_INTEGRATOR_6 = 0x16,
|
||||
MSG_VAR_INTEGRATOR_7 = 0x17,
|
||||
|
||||
MSG_VAR_KFF_0 = 0x1a,
|
||||
MSG_VAR_KFF_1 = 0x1b,
|
||||
MSG_VAR_KFF_2 = 0x1c,
|
||||
|
||||
MSG_VAR_TARGET_BEARING = 0x20,
|
||||
MSG_VAR_NAV_BEARING = 0x21,
|
||||
MSG_VAR_BEARING_ERROR = 0x22,
|
||||
MSG_VAR_CROSSTRACK_BEARING = 0x23,
|
||||
MSG_VAR_CROSSTRACK_ERROR = 0x24,
|
||||
MSG_VAR_ALTITUDE_ERROR = 0x25,
|
||||
MSG_VAR_WP_RADIUS = 0x26,
|
||||
MSG_VAR_LOITER_RADIUS = 0x27,
|
||||
MSG_VAR_WP_MODE = 0x28,
|
||||
MSG_VAR_LOOP_COMMANDS = 0x29,
|
||||
MSG_VAR_NAV_GAIN_SCALER = 0x2a,
|
||||
};
|
||||
|
||||
/// PID sets defined
|
||||
enum PIDSet {
|
||||
MSG_SERVO_ROLL = 0,
|
||||
MSG_SERVO_PITCH = 1,
|
||||
MSG_SERVO_RUDDER = 2,
|
||||
MSG_SERVO_NAV_ROLL = 3,
|
||||
MSG_SERVO_NAV_PITCH_ASP = 4,
|
||||
MSG_SERVO_NAV_PITCH_ALT = 5,
|
||||
MSG_SERVO_TE_THROTTLE = 6,
|
||||
MSG_SERVO_ALT_THROTTLE = 7,
|
||||
MSG_SERVO_ELEVATOR = 8 // Added by Randy
|
||||
};
|
||||
|
||||
//@}
|
||||
|
||||
//////////////////////////////////////////////////////////////////////
|
||||
/// Message reception callout descriptor
|
||||
///
|
||||
/// An array of these handlers is passed to the constructor to delegate
|
||||
/// processing for received messages.
|
||||
///
|
||||
struct MessageHandler {
|
||||
MessageID messageID; ///< messageID for which the handler will be called
|
||||
void (* handler)(void *arg,
|
||||
uint8_t messageId,
|
||||
uint8_t messageVersion,
|
||||
void *messageData); ///< function to be called
|
||||
void *arg; ///< argument passed to function
|
||||
};
|
||||
|
||||
//////////////////////////////////////////////////////////////////////
|
||||
/// @name Decoder interface
|
||||
//@{
|
||||
|
||||
/// Consume bytes from the interface and feed them to the decoder.
|
||||
///
|
||||
/// If a packet is completed, then any callbacks associated
|
||||
/// with the packet's messageID will be called.
|
||||
///
|
||||
/// If no bytes are passed to the decoder for a period determined
|
||||
/// by DEC_MESSAGE_TIMEOUT, the decode state machine will reset
|
||||
/// before processing the next byte. This can help re-synchronise
|
||||
/// after a link loss or in-flight failure.
|
||||
///
|
||||
|
||||
void update(void);
|
||||
|
||||
uint32_t messagesReceived; ///< statistics
|
||||
uint32_t badMessagesReceived; ///< statistics
|
||||
|
||||
//@}
|
||||
|
||||
//////////////////////////////////////////////////////////////////////
|
||||
/// @name Encoder interface
|
||||
///
|
||||
/// Messages are normally encoded and sent using the
|
||||
/// send_msg_* functions defined in protocol/protocol.h.
|
||||
/// For each message type MSG_* there is a corresponding send_msg_*
|
||||
/// function which will construct and transmit the message.
|
||||
///
|
||||
//@{
|
||||
uint32_t messagesSent; ///< statistics
|
||||
//@}
|
||||
|
||||
|
||||
private:
|
||||
const MessageHandler *_handlerTable; ///< callout table
|
||||
Stream *_interface; ///< Serial port we send/receive using.
|
||||
|
||||
/// Various magic numbers
|
||||
enum MagicNumbers {
|
||||
MSG_PREAMBLE_1 = 0x34,
|
||||
MSG_PREAMBLE_2 = 0x44,
|
||||
MSG_VERSION_1 = 1,
|
||||
MSG_VARIABLE_LENGTH = 0xff
|
||||
};
|
||||
|
||||
//////////////////////////////////////////////////////////////////////
|
||||
/// @name Decoder state
|
||||
//@{
|
||||
uint8_t _decodePhase; ///< decoder state machine phase
|
||||
uint8_t _bytesIn; ///< bytes received in the current phase
|
||||
uint8_t _bytesExpected; ///< bytes expected in the current phase
|
||||
uint8_t _sumA; ///< sum of incoming bytes
|
||||
uint8_t _sumB; ///< sum of _sumA values
|
||||
|
||||
uint8_t _messageID; ///< messageID from the packet being received
|
||||
uint8_t _messageVersion;///< messageVersion from the packet being received
|
||||
|
||||
unsigned long _lastReceived; ///< timestamp of last byte reception
|
||||
//@}
|
||||
|
||||
/// Decoder state machine.
|
||||
///
|
||||
/// @param inByte The byte to process.
|
||||
///
|
||||
void _decode(uint8_t inByte);
|
||||
|
||||
/// Send the packet in the encode buffer.
|
||||
///
|
||||
void _sendMessage(void);
|
||||
};
|
||||
|
||||
#endif // BinComm_h
|
@ -0,0 +1,6 @@
|
||||
APM_BinComm KEYWORD1
|
||||
update KEYWORD2
|
||||
messagesReceived KEYWORD2
|
||||
badMessagesReceived KEYWORD2
|
||||
messagesSent KEYWORD2
|
||||
|
@ -0,0 +1,224 @@
|
||||
#
|
||||
# Message definitions for the ArduPilot Mega binary communications protocol.
|
||||
#
|
||||
# Process this file using:
|
||||
#
|
||||
# awk -f protogen.awk protocol.def > protocol.h
|
||||
#
|
||||
# Messages are declared with
|
||||
#
|
||||
# message <MESSAGE_ID> <MESSAGE_NAME>
|
||||
#
|
||||
# <MESSAGE_NAME> is a valid member of BinComm::MessageID.
|
||||
# <MESSAGE_ID> is the message ID byte
|
||||
#
|
||||
# Following a message declaration the fields of the message are
|
||||
# defined in the format:
|
||||
#
|
||||
# <TYPE> <NAME> [<COUNT>]
|
||||
#
|
||||
# <TYPE> is a C type corresponding to the field. The type must be a single
|
||||
# word, either an integer from <inttypes.h> or "char".
|
||||
# <NAME> is the name of the field; it should be unique within the message
|
||||
# <COUNT> is an optional array count for fields that are arrays
|
||||
#
|
||||
|
||||
#
|
||||
# Acknowledge message
|
||||
#
|
||||
message 0x00 MSG_ACKNOWLEDGE
|
||||
uint8_t msgID
|
||||
uint16_t msgSum
|
||||
|
||||
#
|
||||
# System heartbeat
|
||||
#
|
||||
message 0x01 MSG_HEARTBEAT
|
||||
uint8_t flightMode
|
||||
uint16_t timeStamp
|
||||
uint16_t batteryVoltage
|
||||
uint16_t commandIndex
|
||||
|
||||
#
|
||||
# Attitude report
|
||||
#
|
||||
message 0x02 MSG_ATTITUDE
|
||||
int16_t roll
|
||||
int16_t pitch
|
||||
int16_t yaw
|
||||
|
||||
#
|
||||
# Location report
|
||||
#
|
||||
message 0x03 MSG_LOCATION
|
||||
int32_t latitude
|
||||
int32_t longitude
|
||||
int16_t altitude
|
||||
int16_t groundSpeed
|
||||
int16_t groundCourse
|
||||
uint16_t timeOfWeek
|
||||
|
||||
#
|
||||
# Optional pressure-based location report
|
||||
#
|
||||
message 0x04 MSG_PRESSURE
|
||||
uint16_t pressureAltitude
|
||||
uint16_t airSpeed
|
||||
|
||||
#
|
||||
# Text status message
|
||||
#
|
||||
message 0x05 MSG_STATUS_TEXT
|
||||
uint8_t severity
|
||||
char text 50
|
||||
|
||||
#
|
||||
# Algorithm performance report
|
||||
#
|
||||
message 0x06 MSG_PERF_REPORT
|
||||
uint32_t interval
|
||||
uint16_t mainLoopCycles
|
||||
uint8_t mainLoopTime
|
||||
uint8_t gyroSaturationCount
|
||||
uint8_t adcConstraintCount
|
||||
uint16_t imuHealth
|
||||
uint16_t gcsMessageCount
|
||||
|
||||
#
|
||||
# System version messages
|
||||
#
|
||||
message 0x07 MSG_VERSION_REQUEST
|
||||
uint8_t systemType
|
||||
uint8_t systemID
|
||||
|
||||
message 0x08 MSG_VERSION
|
||||
uint8_t systemType
|
||||
uint8_t systemID
|
||||
uint8_t firmwareVersion 3
|
||||
|
||||
#
|
||||
# Flight command operations
|
||||
#
|
||||
message 0x20 MSG_COMMAND_REQUEST
|
||||
uint16_t UNSPECIFIED
|
||||
|
||||
message 0x21 MSG_COMMAND_UPLOAD
|
||||
uint8_t action
|
||||
uint16_t itemNumber
|
||||
int listLength
|
||||
uint8_t commandID
|
||||
uint8_t p1
|
||||
uint16_t p2
|
||||
uint32_t p3
|
||||
uint32_t p4
|
||||
|
||||
message 0x22 MSG_COMMAND_LIST
|
||||
int itemNumber
|
||||
int listLength
|
||||
uint8_t commandID
|
||||
uint8_t p1
|
||||
uint16_t p2
|
||||
uint32_t p3
|
||||
uint32_t p4
|
||||
|
||||
message 0x23 MSG_COMMAND_MODE_CHANGE
|
||||
uint16_t UNSPECIFIED
|
||||
|
||||
#
|
||||
# Parameter operations
|
||||
#
|
||||
message 0x30 MSG_VALUE_REQUEST
|
||||
uint8_t valueID
|
||||
uint8_t broadcast
|
||||
|
||||
|
||||
message 0x31 MSG_VALUE_SET
|
||||
uint8_t valueID
|
||||
uint32_t value
|
||||
|
||||
message 0x32 MSG_VALUE
|
||||
uint8_t valueID
|
||||
uint32_t value
|
||||
|
||||
#
|
||||
# PID adjustments
|
||||
#
|
||||
message 0x40 MSG_PID_REQUEST
|
||||
uint8_t pidSet
|
||||
|
||||
message 0x41 MSG_PID_SET
|
||||
uint8_t pidSet
|
||||
int32_t p
|
||||
int32_t i
|
||||
int32_t d
|
||||
int16_t integratorMax
|
||||
|
||||
message 0x42 MSG_PID
|
||||
uint8_t pidSet
|
||||
int32_t p
|
||||
int32_t i
|
||||
int32_t d
|
||||
int16_t integratorMax
|
||||
|
||||
|
||||
#
|
||||
# Radio settings and values
|
||||
#
|
||||
message 0x50 MSG_TRIM_STARTUP
|
||||
uint16_t value 8
|
||||
|
||||
message 0x51 MSG_TRIM_MIN
|
||||
uint16_t value 8
|
||||
|
||||
message 0x52 MSG_TRIM_MAX
|
||||
uint16_t value 8
|
||||
|
||||
message 0x53 MSG_SERVOS
|
||||
int16_t ch1
|
||||
int16_t ch2
|
||||
int16_t ch3
|
||||
int16_t ch4
|
||||
int16_t ch5
|
||||
int16_t ch6
|
||||
int16_t ch7
|
||||
int16_t ch8
|
||||
|
||||
#
|
||||
# Direct sensor access
|
||||
#
|
||||
message 0x60 MSG_SENSOR
|
||||
uint16_t UNSPECIFIED
|
||||
|
||||
#
|
||||
# Simulation-related messages
|
||||
#
|
||||
message 0x70 MSG_SIM
|
||||
uint16_t UNSPECIFIED
|
||||
|
||||
#
|
||||
# Direct I/O pin control
|
||||
#
|
||||
message 0x80 MSG_PIN_REQUEST
|
||||
uint16_t UNSPECIFIED
|
||||
|
||||
message 0x81 MSG_PIN_SET
|
||||
uint16_t UNSPECIFIED
|
||||
|
||||
#
|
||||
# Dataflash operations
|
||||
#
|
||||
message 0x90 MSG_DATAFLASH_REQUEST
|
||||
uint16_t UNSPECIFIED
|
||||
|
||||
message 0x91 MSG_DATAFLASH_SET
|
||||
uint16_t UNSPECIFIED
|
||||
|
||||
#
|
||||
# EEPROM operations
|
||||
#
|
||||
message 0xa0 MSG_EEPROM_REQUEST
|
||||
uint16_t UNSPECIFIED
|
||||
|
||||
message 0xa1 MSG_EEPROM_SET
|
||||
uint16_t UNSPECIFIED
|
||||
|
File diff suppressed because it is too large
Load Diff
@ -0,0 +1,174 @@
|
||||
#
|
||||
# Process the protocol specification and emit functions to pack and unpack buffers.
|
||||
#
|
||||
# See protocol.def for a description of the definition format.
|
||||
#
|
||||
|
||||
BEGIN {
|
||||
printf("//\n// THIS FILE WAS AUTOMATICALLY GENERATED - DO NOT EDIT\n//\n")
|
||||
printf("/// @file protocol.h\n")
|
||||
printf("#pragma pack(1)\n");
|
||||
|
||||
currentMessage = ""
|
||||
}
|
||||
|
||||
END {
|
||||
# finalise the last message definition
|
||||
EMIT_MESSAGE()
|
||||
|
||||
#
|
||||
# emit the MessageID enum
|
||||
#
|
||||
# XXX it would be elegant to sort the array here, but not
|
||||
# everyone has GNU awk.
|
||||
#
|
||||
printf("\n//////////////////////////////////////////////////////////////////////\n")
|
||||
printf("/// Message ID values\n")
|
||||
printf("enum MessageID {\n")
|
||||
for (opcode in opcodes) {
|
||||
printf("\t%s = 0x%x,\n", opcodes[opcode], opcode)
|
||||
}
|
||||
printf("\tMSG_ANY = 0xfe,\n")
|
||||
printf("\tMSG_NULL = 0xff\n")
|
||||
printf("};\n")
|
||||
|
||||
printf("#pragma pack(pop)\n")
|
||||
}
|
||||
|
||||
#
|
||||
# Emit definitions for one message
|
||||
#
|
||||
function EMIT_MESSAGE(payloadSize)
|
||||
{
|
||||
if (currentMessage != "") {
|
||||
printf("\n//////////////////////////////////////////////////////////////////////\n")
|
||||
printf("/// @name %s \n//@{\n\n", currentMessage)
|
||||
|
||||
#
|
||||
# emit a structure defining the message payload
|
||||
#
|
||||
printf("/// Structure describing the payload section of the %s message\n", currentMessage)
|
||||
printf("struct %s {\n", tolower(currentMessage))
|
||||
for (i = 0; i < fieldCount; i++) {
|
||||
printf("\t%s %s", types[i], names[i])
|
||||
if (counts[i])
|
||||
printf("[%s]", counts[i])
|
||||
printf(";\n")
|
||||
}
|
||||
printf("};\n\n")
|
||||
|
||||
#
|
||||
# emit a routine to pack the message payload from a set of variables and send it
|
||||
#
|
||||
printf("/// Send a %s message\n", currentMessage)
|
||||
printf("inline void\nsend_%s(\n", tolower(currentMessage))
|
||||
for (i = 0; i < fieldCount; i++) {
|
||||
if (counts[i]) {
|
||||
printf("\tconst %s (&%s)[%d]", types[i], names[i], counts[i])
|
||||
} else {
|
||||
printf("\tconst %s %s", types[i], names[i])
|
||||
}
|
||||
if (i < (fieldCount -1))
|
||||
printf(",\n");
|
||||
}
|
||||
printf(")\n{\n")
|
||||
printf("\tuint8_t *__p = &_encodeBuf.payload[0];\n")
|
||||
payloadSize = 0;
|
||||
for (i = 0; i < fieldCount; i++) {
|
||||
if (counts[i]) {
|
||||
printf("\t_pack(__p, %s, %s);\n", names[i], counts[i])
|
||||
payloadSize += sizes[i] * counts[i]
|
||||
} else {
|
||||
printf("\t_pack(__p, %s);\n", names[i])
|
||||
payloadSize += sizes[i]
|
||||
}
|
||||
}
|
||||
printf("\t_encodeBuf.header.length = %s;\n", payloadSize)
|
||||
printf("\t_encodeBuf.header.messageID = %s;\n", currentMessage)
|
||||
printf("\t_encodeBuf.header.messageVersion = MSG_VERSION_1;\n")
|
||||
printf("\t_sendMessage();\n")
|
||||
printf("};\n\n")
|
||||
|
||||
#
|
||||
# emit a routine to unpack the current message into a set of variables
|
||||
#
|
||||
printf("/// Unpack a %s message\n", currentMessage)
|
||||
printf("inline void\nunpack_%s(\n", tolower(currentMessage))
|
||||
for (i = 0; i < fieldCount; i++) {
|
||||
if (counts[i]) {
|
||||
printf("\t%s (&%s)[%d]", types[i], names[i], counts[i])
|
||||
} else {
|
||||
printf("\t%s &%s", types[i], names[i])
|
||||
}
|
||||
if (i < (fieldCount -1))
|
||||
printf(",\n");
|
||||
}
|
||||
printf(")\n{\n")
|
||||
printf("\tuint8_t *__p = &_decodeBuf.payload[0];\n")
|
||||
for (i = 0; i < fieldCount; i++) {
|
||||
if (counts[i]) {
|
||||
printf("\t_unpack(__p, %s, %s);\n", names[i], counts[i])
|
||||
payloadSize += sizes[i] * counts[i]
|
||||
} else {
|
||||
printf("\t_unpack(__p, %s);\n", names[i])
|
||||
payloadSize += sizes[i]
|
||||
}
|
||||
}
|
||||
printf("};\n")
|
||||
|
||||
# close the Doxygen group
|
||||
printf("//@}\n")
|
||||
}
|
||||
}
|
||||
|
||||
# skip lines containing comments
|
||||
$1=="#" {
|
||||
next
|
||||
}
|
||||
|
||||
#
|
||||
# process a new message declaration
|
||||
#
|
||||
$1=="message" {
|
||||
|
||||
# emit any previous message
|
||||
EMIT_MESSAGE()
|
||||
|
||||
# save the current opcode and message name
|
||||
currentOpcode = $2
|
||||
currentMessage = $3
|
||||
opcodes[$2] = $3
|
||||
|
||||
# set up for the coming fields
|
||||
fieldCount = 0
|
||||
delete types
|
||||
delete names
|
||||
delete sizes
|
||||
delete counts
|
||||
|
||||
next
|
||||
}
|
||||
|
||||
#
|
||||
# process a field inside a message definition
|
||||
#
|
||||
NF >= 2 {
|
||||
|
||||
# save the field definition
|
||||
types[fieldCount] = $1
|
||||
names[fieldCount] = $2
|
||||
|
||||
# guess the field size, note that we only support <inttypes.h> and "char"
|
||||
sizes[fieldCount] = 1
|
||||
if ($1 ~ ".*16.*")
|
||||
sizes[fieldCount] = 2
|
||||
if ($1 ~ ".*32.*")
|
||||
sizes[fieldCount] = 4
|
||||
|
||||
# if an array size was supplied, save it
|
||||
if (NF >= 3) {
|
||||
counts[fieldCount] = $3
|
||||
}
|
||||
|
||||
fieldCount++
|
||||
}
|
@ -0,0 +1,14 @@
|
||||
|
||||
PROG := BinCommTest
|
||||
SRCS := test.cpp ../APM_BinComm.cpp
|
||||
OBJS := $(SRCS:.cpp=.o)
|
||||
|
||||
BinCommTest: $(OBJS)
|
||||
c++ -g -o $@ $(OBJS)
|
||||
|
||||
.cpp.o:
|
||||
@echo C++ $< -> $@
|
||||
c++ -g -c -I. -o $@ $<
|
||||
|
||||
clean:
|
||||
rm $(PROG) $(OBJS)
|
@ -0,0 +1,15 @@
|
||||
|
||||
#ifndef WPROGRAM_H
|
||||
#define WPROGRAM_H
|
||||
|
||||
class Stream {
|
||||
public:
|
||||
void write(uint8_t val);
|
||||
int available(void);
|
||||
int read(void);
|
||||
};
|
||||
|
||||
|
||||
extern unsigned int millis(void);
|
||||
|
||||
#endif
|
@ -0,0 +1,110 @@
|
||||
// -*- Mode: C++; c-basic-offset: 8; indent-tabs-mode: nil -*-
|
||||
|
||||
// test harness for the APM_BinComm bits
|
||||
|
||||
#include <stdint.h>
|
||||
#include <err.h>
|
||||
#include <unistd.h>
|
||||
#include <fcntl.h>
|
||||
#include <termios.h>
|
||||
#include <stdio.h>
|
||||
|
||||
#include "WProgram.h"
|
||||
|
||||
#include "../APM_BinComm.h"
|
||||
|
||||
static void handler(void *arg, uint8_t messageId, uint8_t messageVersion, void *messageData);
|
||||
|
||||
BinComm::MessageHandler handlers[] = {
|
||||
{BinComm::MSG_ANY, handler, NULL},
|
||||
{BinComm::MSG_NULL, 0, 0}
|
||||
};
|
||||
|
||||
Stream port;
|
||||
BinComm comm(handlers, &port);
|
||||
|
||||
int port_fd;
|
||||
|
||||
unsigned int
|
||||
millis(void)
|
||||
{
|
||||
return 0;
|
||||
}
|
||||
|
||||
void
|
||||
Stream::write(uint8_t val)
|
||||
{
|
||||
::write(port_fd, &val, 1);
|
||||
}
|
||||
|
||||
int
|
||||
Stream::available(void)
|
||||
{
|
||||
return(1);
|
||||
}
|
||||
|
||||
int
|
||||
Stream::read(void)
|
||||
{
|
||||
int ret;
|
||||
uint8_t c;
|
||||
|
||||
switch(::read(port_fd, &c, 1)) {
|
||||
case 1:
|
||||
return c;
|
||||
case 0:
|
||||
errx(1, "device disappeared");
|
||||
|
||||
default:
|
||||
// almost certainly EWOULDBLOCK
|
||||
return -1;
|
||||
}
|
||||
}
|
||||
|
||||
void
|
||||
handler(void *arg, uint8_t messageId, uint8_t messageVersion, void *messageData)
|
||||
{
|
||||
|
||||
if (messageId == BinComm::MSG_HEARTBEAT) {
|
||||
struct BinComm::msg_heartbeat *m = (struct BinComm::msg_heartbeat *)messageData;
|
||||
printf("Heartbeat: mode %u time %u voltage %u command %u\n",
|
||||
m->flightMode, m->timeStamp, m->batteryVoltage, m->commandIndex);
|
||||
} else
|
||||
if (messageId == BinComm::MSG_ATTITUDE) {
|
||||
struct BinComm::msg_attitude *m = (struct BinComm::msg_attitude *)messageData;
|
||||
printf("Attitude: pitch %d roll %d yaw %d\n",
|
||||
m->pitch, m->roll, m->yaw);
|
||||
} else
|
||||
if (messageId == BinComm::MSG_LOCATION) {
|
||||
struct BinComm::msg_location *m = (struct BinComm::msg_location *)messageData;
|
||||
printf("Location: lat %d long %d altitude %d groundspeed %d groundcourse %d time %u\n",
|
||||
m->latitude, m->longitude, m->altitude, m->groundSpeed, m->groundCourse, m->timeOfWeek);
|
||||
} else
|
||||
if (messageId == BinComm::MSG_STATUS_TEXT) {
|
||||
struct BinComm::msg_status_text *m = (struct BinComm::msg_status_text *)messageData;
|
||||
printf("Message %d: %-50s\n", m->severity, m->text);
|
||||
} else {
|
||||
warnx("received message %d,%d", messageId, messageVersion);
|
||||
}
|
||||
}
|
||||
|
||||
int
|
||||
main(int argc, char *argv[])
|
||||
{
|
||||
struct termios t;
|
||||
|
||||
if (2 > argc)
|
||||
errx(1, "BinCommTest <port>");
|
||||
if (0 >= (port_fd = open(argv[1], O_RDWR | O_NONBLOCK)))
|
||||
err(1, "could not open port %s", argv[1]);
|
||||
if (tcgetattr(port_fd, &t))
|
||||
err(1, "tcgetattr");
|
||||
cfsetspeed(&t, 115200);
|
||||
if (tcsetattr(port_fd, TCSANOW, &t))
|
||||
err(1, "tcsetattr");
|
||||
|
||||
// spin listening
|
||||
for (;;) {
|
||||
comm.update();
|
||||
}
|
||||
}
|
@ -0,0 +1,235 @@
|
||||
// -*- tab-width: 4; Mode: C++; c-basic-offset: 3; indent-tabs-mode: t -*-
|
||||
/*
|
||||
APM_Compass.cpp - Arduino Library for HMC5843 I2C Magnetometer
|
||||
Code by Jordi Muñoz and Jose Julio. DIYDrones.com
|
||||
|
||||
This library is free software; you can redistribute it and/or
|
||||
modify it under the terms of the GNU Lesser General Public
|
||||
License as published by the Free Software Foundation; either
|
||||
version 2.1 of the License, or (at your option) any later version.
|
||||
|
||||
Sensor is conected to I2C port
|
||||
Sensor is initialized in Continuos mode (10Hz)
|
||||
|
||||
Variables:
|
||||
Heading : Magnetic heading
|
||||
Heading_X : Magnetic heading X component
|
||||
Heading_Y : Magnetic heading Y component
|
||||
Mag_X : Raw X axis magnetometer data
|
||||
Mag_Y : Raw Y axis magnetometer data
|
||||
Mag_Z : Raw Z axis magnetometer data
|
||||
lastUpdate : the time of the last successful reading
|
||||
|
||||
Methods:
|
||||
Init() : Initialization of I2C and sensor
|
||||
Read() : Read Sensor data
|
||||
Calculate(float roll, float pitch) : Calculate tilt adjusted heading
|
||||
SetOrientation(const Matrix3f &rotationMatrix) : Set orientation of compass
|
||||
SetOffsets(int x, int y, int z) : Set adjustments for HardIron disturbances
|
||||
SetDeclination(float radians) : Set heading adjustment between true north and magnetic north
|
||||
|
||||
To do : code optimization
|
||||
Mount position : UPDATED
|
||||
Big capacitor pointing backward, connector forward
|
||||
|
||||
*/
|
||||
extern "C" {
|
||||
// AVR LibC Includes
|
||||
#include <math.h>
|
||||
#include "WConstants.h"
|
||||
}
|
||||
|
||||
#include <Wire.h>
|
||||
#include "APM_Compass.h"
|
||||
|
||||
#define CompassAddress 0x1E
|
||||
#define ConfigRegA 0x00
|
||||
#define ConfigRegB 0x01
|
||||
#define MagGain 0x20
|
||||
#define PositiveBiasConfig 0x11
|
||||
#define NegativeBiasConfig 0x12
|
||||
#define NormalOperation 0x10
|
||||
#define ModeRegister 0x02
|
||||
#define ContinuousConversion 0x00
|
||||
#define SingleConversion 0x01
|
||||
|
||||
// Constructors ////////////////////////////////////////////////////////////////
|
||||
APM_Compass_Class::APM_Compass_Class() : orientation(0), declination(0.0)
|
||||
{
|
||||
// mag x y z offset initialisation
|
||||
offset[0] = 0;
|
||||
offset[1] = 0;
|
||||
offset[2] = 0;
|
||||
|
||||
// initialise orientation matrix
|
||||
orientationMatrix = ROTATION_NONE;
|
||||
}
|
||||
|
||||
// Public Methods //////////////////////////////////////////////////////////////
|
||||
bool APM_Compass_Class::Init(void)
|
||||
{
|
||||
unsigned long currentTime = millis(); // record current time
|
||||
int numAttempts = 0;
|
||||
int success = 0;
|
||||
|
||||
Wire.begin();
|
||||
|
||||
delay(10);
|
||||
|
||||
// calibration initialisation
|
||||
calibration[0] = 1.0;
|
||||
calibration[1] = 1.0;
|
||||
calibration[2] = 1.0;
|
||||
|
||||
while( success == 0 && numAttempts < 5 )
|
||||
{
|
||||
// record number of attempts at initialisation
|
||||
numAttempts++;
|
||||
|
||||
// force positiveBias (compass should return 715 for all channels)
|
||||
Wire.beginTransmission(CompassAddress);
|
||||
Wire.send(ConfigRegA);
|
||||
Wire.send(PositiveBiasConfig);
|
||||
if (0 != Wire.endTransmission())
|
||||
continue; // compass not responding on the bus
|
||||
delay(50);
|
||||
|
||||
// set gains
|
||||
Wire.beginTransmission(CompassAddress);
|
||||
Wire.send(ConfigRegB);
|
||||
Wire.send(MagGain);
|
||||
Wire.endTransmission();
|
||||
delay(10);
|
||||
|
||||
Wire.beginTransmission(CompassAddress);
|
||||
Wire.send(ModeRegister);
|
||||
Wire.send(SingleConversion);
|
||||
Wire.endTransmission();
|
||||
delay(10);
|
||||
|
||||
// read values from the compass
|
||||
Read();
|
||||
delay(10);
|
||||
|
||||
// calibrate
|
||||
if( abs(Mag_X) > 500 && abs(Mag_X) < 1000 && abs(Mag_Y) > 500 && abs(Mag_Y) < 1000 && abs(Mag_Z) > 500 && abs(Mag_Z) < 1000)
|
||||
{
|
||||
calibration[0] = abs(715.0 / Mag_X);
|
||||
calibration[1] = abs(715.0 / Mag_Y);
|
||||
calibration[2] = abs(715.0 / Mag_Z);
|
||||
|
||||
// mark success
|
||||
success = 1;
|
||||
}
|
||||
|
||||
// leave test mode
|
||||
Wire.beginTransmission(CompassAddress);
|
||||
Wire.send(ConfigRegA);
|
||||
Wire.send(NormalOperation);
|
||||
Wire.endTransmission();
|
||||
delay(50);
|
||||
|
||||
Wire.beginTransmission(CompassAddress);
|
||||
Wire.send(ModeRegister);
|
||||
Wire.send(ContinuousConversion); // Set continuous mode (default to 10Hz)
|
||||
Wire.endTransmission(); // End transmission
|
||||
delay(50);
|
||||
}
|
||||
return(success);
|
||||
}
|
||||
|
||||
// Read Sensor data
|
||||
void APM_Compass_Class::Read()
|
||||
{
|
||||
int i = 0;
|
||||
byte buff[6];
|
||||
|
||||
Wire.beginTransmission(CompassAddress);
|
||||
Wire.send(0x03); //sends address to read from
|
||||
Wire.endTransmission(); //end transmission
|
||||
|
||||
//Wire.beginTransmission(CompassAddress);
|
||||
Wire.requestFrom(CompassAddress, 6); // request 6 bytes from device
|
||||
while(Wire.available())
|
||||
{
|
||||
buff[i] = Wire.receive(); // receive one byte
|
||||
i++;
|
||||
}
|
||||
Wire.endTransmission(); //end transmission
|
||||
|
||||
if (i==6) // All bytes received?
|
||||
{
|
||||
// MSB byte first, then LSB, X,Y,Z
|
||||
Mag_X = -((((int)buff[0]) << 8) | buff[1]) * calibration[0]; // X axis
|
||||
Mag_Y = ((((int)buff[2]) << 8) | buff[3]) * calibration[1]; // Y axis
|
||||
Mag_Z = -((((int)buff[4]) << 8) | buff[5]) * calibration[2]; // Z axis
|
||||
lastUpdate = millis(); // record time of update
|
||||
}
|
||||
}
|
||||
|
||||
void APM_Compass_Class::Calculate(float roll, float pitch)
|
||||
{
|
||||
float Head_X;
|
||||
float Head_Y;
|
||||
float cos_roll;
|
||||
float sin_roll;
|
||||
float cos_pitch;
|
||||
float sin_pitch;
|
||||
Vector3f rotMagVec;
|
||||
|
||||
cos_roll = cos(roll); // Optimizacion, se puede sacar esto de la matriz DCM?
|
||||
sin_roll = sin(roll);
|
||||
cos_pitch = cos(pitch);
|
||||
sin_pitch = sin(pitch);
|
||||
|
||||
// rotate the magnetometer values depending upon orientation
|
||||
if( orientation == 0 )
|
||||
rotMagVec = Vector3f(Mag_X+offset[0],Mag_Y+offset[1],Mag_Z+offset[2]);
|
||||
else
|
||||
rotMagVec = orientationMatrix*Vector3f(Mag_X+offset[0],Mag_Y+offset[1],Mag_Z+offset[2]);
|
||||
|
||||
// Tilt compensated Magnetic field X component:
|
||||
Head_X = rotMagVec.x*cos_pitch+rotMagVec.y*sin_roll*sin_pitch+rotMagVec.z*cos_roll*sin_pitch;
|
||||
// Tilt compensated Magnetic field Y component:
|
||||
Head_Y = rotMagVec.y*cos_roll-rotMagVec.z*sin_roll;
|
||||
// Magnetic Heading
|
||||
Heading = atan2(-Head_Y,Head_X);
|
||||
|
||||
// Declination correction (if supplied)
|
||||
if( declination != 0.0 )
|
||||
{
|
||||
Heading = Heading + declination;
|
||||
if (Heading > M_PI) // Angle normalization (-180 deg, 180 deg)
|
||||
Heading -= (2.0 * M_PI);
|
||||
else if (Heading < -M_PI)
|
||||
Heading += (2.0 * M_PI);
|
||||
}
|
||||
|
||||
// Optimization for external DCM use. Calculate normalized components
|
||||
Heading_X = cos(Heading);
|
||||
Heading_Y = sin(Heading);
|
||||
}
|
||||
|
||||
void APM_Compass_Class::SetOrientation(const Matrix3f &rotationMatrix)
|
||||
{
|
||||
orientationMatrix = rotationMatrix;
|
||||
if( orientationMatrix == ROTATION_NONE )
|
||||
orientation = 0;
|
||||
else
|
||||
orientation = 1;
|
||||
}
|
||||
|
||||
void APM_Compass_Class::SetOffsets(int x, int y, int z)
|
||||
{
|
||||
offset[0] = x;
|
||||
offset[1] = y;
|
||||
offset[2] = z;
|
||||
}
|
||||
|
||||
void APM_Compass_Class::SetDeclination(float radians)
|
||||
{
|
||||
declination = radians;
|
||||
}
|
||||
|
||||
// make one instance for the user to use
|
||||
APM_Compass_Class APM_Compass;
|
@ -0,0 +1,69 @@
|
||||
#ifndef APM_Compass_h
|
||||
#define APM_Compass_h
|
||||
|
||||
#include "../AP_Math/AP_Math.h"
|
||||
|
||||
// Rotation matrices
|
||||
#define ROTATION_NONE Matrix3f(1, 0, 0, 0, 1, 0, 0 ,0, 1)
|
||||
#define ROTATION_YAW_45 Matrix3f(0.70710678, -0.70710678, 0, 0.70710678, 0.70710678, 0, 0, 0, 1)
|
||||
#define ROTATION_YAW_90 Matrix3f(0, -1, 0, 1, 0, 0, 0, 0, 1)
|
||||
#define ROTATION_YAW_135 Matrix3f(-0.70710678, -0.70710678, 0, 0.70710678, -0.70710678, 0, 0, 0, 1)
|
||||
#define ROTATION_YAW_180 Matrix3f(-1, 0, 0, 0, -1, 0, 0, 0, 1)
|
||||
#define ROTATION_YAW_225 Matrix3f(-0.70710678, 0.70710678, 0, -0.70710678, -0.70710678, 0, 0, 0, 1)
|
||||
#define ROTATION_YAW_270 Matrix3f(0, 1, 0, -1, 0, 0, 0, 0, 1)
|
||||
#define ROTATION_YAW_315 Matrix3f(0.70710678, 0.70710678, 0, -0.70710678, 0.70710678, 0, 0, 0, 1)
|
||||
#define ROTATION_ROLL_180 Matrix3f(1, 0, 0, 0, -1, 0, 0, 0, -1)
|
||||
#define ROTATION_ROLL_180_YAW_45 Matrix3f(0.70710678, 0.70710678, 0, 0.70710678, -0.70710678, 0, 0, 0, -1)
|
||||
#define ROTATION_ROLL_180_YAW_90 Matrix3f(0, 1, 0, 1, 0, 0, 0, 0, -1)
|
||||
#define ROTATION_ROLL_180_YAW_135 Matrix3f(-0.70710678, 0.70710678, 0, 0.70710678, 0.70710678, 0, 0, 0, -1)
|
||||
#define ROTATION_PITCH_180 Matrix3f(-1, 0, 0, 0, 1, 0, 0, 0, -1)
|
||||
#define ROTATION_ROLL_180_YAW_225 Matrix3f(-0.70710678, -0.70710678, 0, -0.70710678, 0.70710678, 0, 0, 0, -1)
|
||||
#define ROTATION_ROLL_180_YAW_270 Matrix3f(0, -1, 0, -1, 0, 0, 0, 0, -1)
|
||||
#define ROTATION_ROLL_180_YAW_315 Matrix3f(0.70710678, -0.70710678, 0, -0.70710678, -0.70710678, 0, 0, 0, -1)
|
||||
|
||||
#define APM_COMPASS_COMPONENTS_UP_PINS_FORWARD ROTATION_NONE
|
||||
#define APM_COMPONENTS_UP_PINS_FORWARD_RIGHT ROTATION_YAW_45
|
||||
#define APM_COMPASS_COMPONENTS_UP_PINS_RIGHT ROTATION_YAW_90
|
||||
#define APM_COMPONENTS_UP_PINS_BACK_RIGHT ROTATION_YAW_135
|
||||
#define APM_COMPASS_COMPONENTS_UP_PINS_BACK ROTATION_YAW_180
|
||||
#define APM_COMPONENTS_UP_PINS_BACK_LEFT ROTATION_YAW_225
|
||||
#define APM_COMPASS_COMPONENTS_UP_PINS_LEFT ROTATION_YAW_270
|
||||
#define APM_COMPONENTS_UP_PINS_FORWARD_LEFT ROTATION_YAW_315
|
||||
#define APM_COMPASS_COMPONENTS_DOWN_PINS_FORWARD ROTATION_ROLL_180
|
||||
#define APM_COMPONENTS_DOWN_PINS_FORWARD_RIGHT ROTATION_ROLL_180_YAW_45
|
||||
#define APM_COMPASS_COMPONENTS_DOWN_PINS_RIGHT ROTATION_ROLL_180_YAW_90
|
||||
#define APM_COMPONENTS_DOWN_PINS_BACK_RIGHT ROTATION_ROLL_180_YAW_135
|
||||
#define APM_COMPASS_COMPONENTS_DOWN_PINS_BACK ROTATION_PITCH_180
|
||||
#define APM_COMPONENTS_DOWN_PINS_BACK_LEFT ROTATION_ROLL_180_YAW_225
|
||||
#define APM_COMPASS_COMPONENTS_DOWN_PINS_LEFT ROTATION_ROLL_180_YAW_270
|
||||
#define APM_COMPONENTS_DOWN_PINS_FORWARD_LEFT ROTATION_ROLL_180_YAW_315
|
||||
|
||||
class APM_Compass_Class
|
||||
{
|
||||
private:
|
||||
int orientation;
|
||||
Matrix3f orientationMatrix;
|
||||
float calibration[3];
|
||||
int offset[3];
|
||||
float declination;
|
||||
public:
|
||||
int Mag_X;
|
||||
int Mag_Y;
|
||||
int Mag_Z;
|
||||
float Heading;
|
||||
float Heading_X;
|
||||
float Heading_Y;
|
||||
unsigned long lastUpdate;
|
||||
|
||||
APM_Compass_Class(); // Constructor
|
||||
bool Init();
|
||||
void Read();
|
||||
void Calculate(float roll, float pitch);
|
||||
void SetOrientation(const Matrix3f &rotationMatrix);
|
||||
void SetOffsets(int x, int y, int z);
|
||||
void SetDeclination(float radians);
|
||||
};
|
||||
|
||||
extern APM_Compass_Class APM_Compass;
|
||||
|
||||
#endif
|
@ -0,0 +1,87 @@
|
||||
/*
|
||||
Example of APM_Compass library (HMC5843 sensor).
|
||||
Code by Jordi MuÒoz and Jose Julio. DIYDrones.com
|
||||
|
||||
offsets displayed are calculated from the min/max values from each axis.
|
||||
rotate the compass so as to produce the maximum and minimum values
|
||||
after the offsets stop changing, edit the code to pass these offsets into
|
||||
APM_Compass.SetOffsets.
|
||||
*/
|
||||
|
||||
#include <Wire.h>
|
||||
#include <APM_Compass.h> // Compass Library
|
||||
#include <AP_Math.h> // Math library
|
||||
|
||||
#define ToRad(x) (x*0.01745329252) // *pi/180
|
||||
#define ToDeg(x) (x*57.2957795131) // *180/pi
|
||||
|
||||
unsigned long timer;
|
||||
|
||||
void setup()
|
||||
{
|
||||
Serial.begin(38400);
|
||||
Serial.println("Compass library test (HMC5843)");
|
||||
|
||||
APM_Compass.Init(); // Initialization
|
||||
APM_Compass.SetOrientation(APM_COMPASS_COMPONENTS_UP_PINS_FORWARD); // set compass's orientation on aircraft
|
||||
APM_Compass.SetOffsets(0,0,0); // set offsets to account for surrounding interference
|
||||
APM_Compass.SetDeclination(ToRad(0.0)); // set local difference between magnetic north and true north
|
||||
|
||||
delay(1000);
|
||||
timer = millis();
|
||||
}
|
||||
|
||||
void loop()
|
||||
{
|
||||
static float min[3], max[3], offset[3];
|
||||
|
||||
if((millis()- timer) > 100)
|
||||
{
|
||||
timer = millis();
|
||||
APM_Compass.Read();
|
||||
APM_Compass.Calculate(0,0); // roll = 0, pitch = 0 for this example
|
||||
|
||||
// capture min
|
||||
if( APM_Compass.Mag_X < min[0] )
|
||||
min[0] = APM_Compass.Mag_X;
|
||||
if( APM_Compass.Mag_Y < min[1] )
|
||||
min[1] = APM_Compass.Mag_Y;
|
||||
if( APM_Compass.Mag_Z < min[2] )
|
||||
min[2] = APM_Compass.Mag_Z;
|
||||
|
||||
// capture max
|
||||
if( APM_Compass.Mag_X > max[0] )
|
||||
max[0] = APM_Compass.Mag_X;
|
||||
if( APM_Compass.Mag_Y > max[1] )
|
||||
max[1] = APM_Compass.Mag_Y;
|
||||
if( APM_Compass.Mag_Z > max[2] )
|
||||
max[2] = APM_Compass.Mag_Z;
|
||||
|
||||
// calculate offsets
|
||||
offset[0] = -(max[0]+min[0])/2;
|
||||
offset[1] = -(max[1]+min[1])/2;
|
||||
offset[2] = -(max[2]+min[2])/2;
|
||||
|
||||
// display all to user
|
||||
Serial.print("Heading:");
|
||||
Serial.print(ToDeg(APM_Compass.Heading));
|
||||
Serial.print(" (");
|
||||
Serial.print(APM_Compass.Mag_X);
|
||||
Serial.print(",");
|
||||
Serial.print(APM_Compass.Mag_Y);
|
||||
Serial.print(",");
|
||||
Serial.print(APM_Compass.Mag_Z);
|
||||
Serial.print(")");
|
||||
|
||||
// display offsets
|
||||
Serial.print("\t offsets(");
|
||||
Serial.print(offset[0]);
|
||||
Serial.print(",");
|
||||
Serial.print(offset[1]);
|
||||
Serial.print(",");
|
||||
Serial.print(offset[2]);
|
||||
Serial.print(")");
|
||||
|
||||
Serial.println();
|
||||
}
|
||||
}
|
@ -0,0 +1,19 @@
|
||||
Compass KEYWORD1
|
||||
AP_Compass KEYWORD1
|
||||
APM_Compass KEYWORD1
|
||||
Init KEYWORD2
|
||||
Read KEYWORD2
|
||||
Calculate KEYWORD2
|
||||
SetOrientation KEYWORD2
|
||||
SetOffsets KEYWORDS2
|
||||
SetDeclination KEYWORDS2
|
||||
Heading KEYWORD2
|
||||
Heading_X KEYWORD2
|
||||
Heading_Y KEYWORD2
|
||||
Mag_X KEYWORD2
|
||||
Mag_Y KEYWORD2
|
||||
Mag_Z KEYWORD2
|
||||
lastUpdate KEYWORD2
|
||||
|
||||
|
||||
|
@ -0,0 +1,150 @@
|
||||
/*
|
||||
APM_FastSerial.cpp - Fast Serial Output for Ardupilot Mega Hardware (atmega1280)
|
||||
It´s also compatible with standard Arduino boards (atmega 168 and 328)
|
||||
Interrupt driven Serial output with intermediate buffer
|
||||
Code Jose Julio and Jordi Muñoz. DIYDrones.com
|
||||
|
||||
This library is free software; you can redistribute it and/or
|
||||
modify it under the terms of the GNU Lesser General Public
|
||||
License as published by the Free Software Foundation; either
|
||||
version 2.1 of the License, or (at your option) any later version.
|
||||
|
||||
This library works as a complement of the standard Arduino Serial
|
||||
library. So user must initialize Standard Serial Arduino library first.
|
||||
This library works in Serial port 0 and Serial port3(telemetry port)[APM]
|
||||
Methods: (the same as standard arduino library, inherits from Print)
|
||||
write() for bytes or array of bytes (binary output)
|
||||
print() for chars, strings, numbers and floats
|
||||
println()
|
||||
*/
|
||||
|
||||
//#include "WProgram.h"
|
||||
#include "APM_FastSerial.h"
|
||||
extern "C" {
|
||||
// AVR LibC Includes
|
||||
#include <inttypes.h>
|
||||
#include <avr/interrupt.h>
|
||||
#include <avr/io.h>
|
||||
#include "WConstants.h"
|
||||
}
|
||||
#define TX_BUFFER_SIZE 80 // Serial output buffer size
|
||||
|
||||
// Serial0 buffer
|
||||
uint8_t tx_buffer0[TX_BUFFER_SIZE];
|
||||
volatile int tx_buffer0_head=0;
|
||||
volatile int tx_buffer0_tail=0;
|
||||
|
||||
#if defined(__AVR_ATmega1280__)
|
||||
// Serial3 buffer
|
||||
uint8_t tx_buffer3[TX_BUFFER_SIZE];
|
||||
volatile int tx_buffer3_head=0;
|
||||
volatile int tx_buffer3_tail=0;
|
||||
#endif
|
||||
|
||||
#if defined(__AVR_ATmega1280__) // For atmega1280 we use Serial port 0 and 3
|
||||
// Serial0 interrupt
|
||||
ISR(SIG_USART0_DATA)
|
||||
{
|
||||
uint8_t data;
|
||||
|
||||
if (tx_buffer0_tail == tx_buffer0_head)
|
||||
UCSR0B &= ~(_BV(UDRIE0)); // Disable interrupt
|
||||
else {
|
||||
data = tx_buffer0[tx_buffer0_tail];
|
||||
tx_buffer0_tail = (tx_buffer0_tail + 1) % TX_BUFFER_SIZE;
|
||||
UDR0 = data;
|
||||
}
|
||||
}
|
||||
|
||||
// Serial3 interrupt
|
||||
ISR(SIG_USART3_DATA)
|
||||
{
|
||||
uint8_t data;
|
||||
|
||||
if (tx_buffer3_tail == tx_buffer3_head)
|
||||
UCSR3B &= ~(_BV(UDRIE3)); // Disable interrupt
|
||||
else {
|
||||
data = tx_buffer3[tx_buffer3_tail];
|
||||
tx_buffer3_tail = (tx_buffer3_tail + 1) % TX_BUFFER_SIZE;
|
||||
UDR3 = data;
|
||||
}
|
||||
}
|
||||
#else
|
||||
|
||||
// Serial interrupt
|
||||
ISR(USART_UDRE_vect)
|
||||
{
|
||||
uint8_t data;
|
||||
|
||||
if (tx_buffer0_tail == tx_buffer0_head)
|
||||
UCSR0B &= ~(_BV(UDRIE0)); // Disable interrupt
|
||||
else {
|
||||
data = tx_buffer0[tx_buffer0_tail];
|
||||
tx_buffer0_tail = (tx_buffer0_tail + 1) % TX_BUFFER_SIZE;
|
||||
UDR0 = data;
|
||||
}
|
||||
}
|
||||
#endif
|
||||
|
||||
// Constructors ////////////////////////////////////////////////////////////////
|
||||
APM_FastSerial_Class::APM_FastSerial_Class(uint8_t SerialPort)
|
||||
{
|
||||
SerialPortNumber=SerialPort; // This could be serial port 0 or 3
|
||||
}
|
||||
|
||||
// Public Methods //////////////////////////////////////////////////////////////
|
||||
|
||||
// This is the important function (basic funtion: send a byte)
|
||||
void APM_FastSerial_Class::write(uint8_t b)
|
||||
{
|
||||
uint8_t Enable_tx_int=0;
|
||||
uint8_t new_head;
|
||||
|
||||
if (SerialPortNumber==0) // Serial Port 0
|
||||
{
|
||||
// if buffer was empty then we enable Serial TX interrupt
|
||||
if (tx_buffer0_tail == tx_buffer0_head)
|
||||
Enable_tx_int=1;
|
||||
|
||||
new_head = (tx_buffer0_head + 1) % TX_BUFFER_SIZE; // Move to next position in the ring buffer
|
||||
if (new_head==tx_buffer0_tail)
|
||||
return; // This is an Error : BUFFER OVERFLOW. We lost this character!!
|
||||
|
||||
tx_buffer0[tx_buffer0_head] = b; // Add data to the tx buffer
|
||||
tx_buffer0_head = new_head; // Update head pointer
|
||||
|
||||
if (Enable_tx_int)
|
||||
UCSR0B |= _BV(UDRIE0); // Enable Serial TX interrupt
|
||||
}
|
||||
#if defined(__AVR_ATmega1280__)
|
||||
else // Serial Port 3
|
||||
{
|
||||
// if buffer was empty then we enable Serial TX interrupt
|
||||
if (tx_buffer3_tail == tx_buffer3_head)
|
||||
Enable_tx_int=1;
|
||||
|
||||
new_head = (tx_buffer3_head + 1) % TX_BUFFER_SIZE; // Move to next position in the ring buffer
|
||||
if (new_head==tx_buffer3_tail)
|
||||
return; // This is an Error : BUFFER OVERFLOW. We lost this character!!
|
||||
|
||||
tx_buffer3[tx_buffer3_head] = b; // Add data to the tx buffer
|
||||
tx_buffer3_head = new_head; // Update head pointer
|
||||
|
||||
if (Enable_tx_int)
|
||||
UCSR3B |= _BV(UDRIE3); // Enable Serial TX interrupt
|
||||
}
|
||||
#endif
|
||||
}
|
||||
|
||||
// Send a buffer of bytes (this is util for binary protocols)
|
||||
void APM_FastSerial_Class::write(const uint8_t *buffer, int size)
|
||||
{
|
||||
while (size--)
|
||||
write(*buffer++);
|
||||
}
|
||||
|
||||
// We create this two instances
|
||||
APM_FastSerial_Class APM_FastSerial(0); // For Serial port 0
|
||||
#if defined(__AVR_ATmega1280__)
|
||||
APM_FastSerial_Class APM_FastSerial3(3); // For Serial port 3 (only Atmega1280)
|
||||
#endif
|
@ -0,0 +1,25 @@
|
||||
#ifndef APM_FastSerial_h
|
||||
#define APM_FastSerial_h
|
||||
|
||||
#include <inttypes.h>
|
||||
|
||||
#include "Print.h"
|
||||
|
||||
class APM_FastSerial_Class : public Print // Inherit from Print
|
||||
{
|
||||
private:
|
||||
uint8_t SerialPortNumber;
|
||||
|
||||
public:
|
||||
APM_FastSerial_Class(uint8_t SerialPort); // Constructor
|
||||
// we overwrite the write methods
|
||||
void write(uint8_t b); // basic funtion : send a byte
|
||||
void write(const uint8_t *buffer, int size);
|
||||
};
|
||||
|
||||
extern APM_FastSerial_Class APM_FastSerial;
|
||||
#if defined(__AVR_ATmega1280__)
|
||||
extern APM_FastSerial_Class APM_FastSerial3;
|
||||
#endif
|
||||
#endif
|
||||
|
@ -0,0 +1,91 @@
|
||||
/*
|
||||
Example of APM_FastSerial library.
|
||||
Code by Jose Julio and Jordi Muñoz. DIYDrones.com
|
||||
*/
|
||||
|
||||
// FastSerial is implemented for Serial port 0 (connection to PC) and 3 (telemetry)
|
||||
#include <APM_FastSerial.h> // ArduPilot Mega Fast Serial Output Library
|
||||
|
||||
#if defined(__AVR_ATmega1280__)
|
||||
#define LED 35
|
||||
#else
|
||||
#define LED 13
|
||||
#endif
|
||||
|
||||
unsigned long timer;
|
||||
unsigned long counter;
|
||||
|
||||
void setup()
|
||||
{
|
||||
int myint=14235; // Examples of data tytpes
|
||||
long mylong=-23456432;
|
||||
float myfloat=-26.669;
|
||||
byte mybyte=0xD1;
|
||||
byte bc_bufIn[50];
|
||||
|
||||
for (int i=0;i<10;i++)
|
||||
bc_bufIn[i]=i*10+30; // we fill the byte array
|
||||
|
||||
pinMode(LED,OUTPUT);
|
||||
|
||||
// We use the standard serial port initialization
|
||||
Serial.begin(57600);
|
||||
//Serial3.begin(57600); // if we want to use port3 also (Mega boards)...
|
||||
delay(100);
|
||||
// We can use both methods to write to serial port:
|
||||
Serial.println("ArduPilot Mega FastSerial library test"); // Standard serial output
|
||||
APM_FastSerial.println("FAST_SERIAL : ArduPilot Mega FastSerial library test"); // Fast serial output
|
||||
// We can use the same on serial port3 (telemetry)
|
||||
// APM_FastSerial3.println("Serial Port3 : ArduPilot Mega FastSerial library test");
|
||||
delay(1000);
|
||||
// Examples of data types (same result as standard arduino library)
|
||||
APM_FastSerial.println("Numbers:");
|
||||
APM_FastSerial.println(myint);
|
||||
APM_FastSerial.println(mylong);
|
||||
APM_FastSerial.println(myfloat);
|
||||
APM_FastSerial.println("Byte:");
|
||||
APM_FastSerial.write(mybyte);
|
||||
APM_FastSerial.println();
|
||||
APM_FastSerial.println("Bytes:");
|
||||
APM_FastSerial.write(bc_bufIn,10);
|
||||
APM_FastSerial.println();
|
||||
delay(4000);
|
||||
}
|
||||
|
||||
void loop()
|
||||
{
|
||||
if((millis()- timer) > 20) // 50Hz loop
|
||||
{
|
||||
timer = millis();
|
||||
if (counter < 250) // we use the Normal Serial output for 5 seconds
|
||||
{
|
||||
// Example of typical telemetry output
|
||||
digitalWrite(LED,HIGH);
|
||||
Serial.println("!ATT:14.2;-5.5;-26.8"); // 20 bytes
|
||||
digitalWrite(LED,LOW);
|
||||
if ((counter%5)==0) // GPS INFO at 5Hz
|
||||
{
|
||||
digitalWrite(LED,HIGH);
|
||||
Serial.println("!LAT:26.345324;LON:-57.372638;ALT:46.7;GC:121.3;GS:23.1"); // 54 bytes
|
||||
digitalWrite(LED,LOW);
|
||||
}
|
||||
}
|
||||
else // and Fast Serial Output for other 5 seconds
|
||||
{
|
||||
// The same info...
|
||||
digitalWrite(LED,HIGH);
|
||||
APM_FastSerial.println("#ATT:14.2;-5.5;-26.8"); // 20 bytes
|
||||
digitalWrite(LED,LOW);
|
||||
if ((counter%5)==0) // GPS INFO at 5Hz
|
||||
{
|
||||
digitalWrite(LED,HIGH);
|
||||
|
||||
APM_FastSerial.println("#LAT:26.345324;LON:-57.372638;ALT:46.7;GC:121.3;GS:23.1"); // 54 bytes
|
||||
digitalWrite(LED,LOW);
|
||||
}
|
||||
if (counter>500) // Counter reset
|
||||
counter=0;
|
||||
}
|
||||
counter++;
|
||||
}
|
||||
}
|
@ -0,0 +1,2 @@
|
||||
APM_FastSerial KEYWORD1
|
||||
APM_FastSerial3 KEYWORD1
|
@ -0,0 +1,193 @@
|
||||
/*
|
||||
APM_RC.cpp - Radio Control Library for Ardupilot Mega. Arduino
|
||||
Code by Jordi Muñoz and Jose Julio. DIYDrones.com
|
||||
|
||||
This library is free software; you can redistribute it and/or
|
||||
modify it under the terms of the GNU Lesser General Public
|
||||
License as published by the Free Software Foundation; either
|
||||
version 2.1 of the License, or (at your option) any later version.
|
||||
|
||||
RC Input : PPM signal on IC4 pin
|
||||
RC Output : 11 Servo outputs (standard 20ms frame)
|
||||
|
||||
Methods:
|
||||
Init() : Initialization of interrupts an Timers
|
||||
OutpuCh(ch,pwm) : Output value to servos (range : 900-2100us) ch=0..10
|
||||
InputCh(ch) : Read a channel input value. ch=0..7
|
||||
GetState() : Returns the state of the input. 1 => New radio frame to process
|
||||
Automatically resets when we call InputCh to read channels
|
||||
|
||||
*/
|
||||
#include "APM_RC.h"
|
||||
|
||||
#include <avr/interrupt.h>
|
||||
#include "WProgram.h"
|
||||
|
||||
#if !defined(__AVR_ATmega1280__)
|
||||
# error Please check the Tools/Board menu to ensure you have selected Arduino Mega as your target.
|
||||
#else
|
||||
|
||||
// Variable definition for Input Capture interrupt
|
||||
volatile unsigned int ICR4_old;
|
||||
volatile unsigned char PPM_Counter=0;
|
||||
volatile unsigned int PWM_RAW[8] = {2400,2400,2400,2400,2400,2400,2400,2400};
|
||||
volatile unsigned char radio_status=0;
|
||||
|
||||
/****************************************************
|
||||
Input Capture Interrupt ICP4 => PPM signal read
|
||||
****************************************************/
|
||||
ISR(TIMER4_CAPT_vect)
|
||||
{
|
||||
unsigned int Pulse;
|
||||
unsigned int Pulse_Width;
|
||||
|
||||
Pulse=ICR4;
|
||||
if (Pulse<ICR4_old) // Take care of the overflow of Timer4 (TOP=40000)
|
||||
Pulse_Width=(Pulse + 40000)-ICR4_old; //Calculating pulse
|
||||
else
|
||||
Pulse_Width=Pulse-ICR4_old; //Calculating pulse
|
||||
if (Pulse_Width>8000) // SYNC pulse?
|
||||
PPM_Counter=0;
|
||||
else
|
||||
{
|
||||
PPM_Counter &= 0x07; // For safety only (limit PPM_Counter to 7)
|
||||
PWM_RAW[PPM_Counter++]=Pulse_Width; //Saving pulse.
|
||||
if (PPM_Counter >= NUM_CHANNELS)
|
||||
radio_status = 1;
|
||||
}
|
||||
ICR4_old = Pulse;
|
||||
}
|
||||
|
||||
|
||||
// Constructors ////////////////////////////////////////////////////////////////
|
||||
|
||||
APM_RC_Class::APM_RC_Class()
|
||||
{
|
||||
}
|
||||
|
||||
// Public Methods //////////////////////////////////////////////////////////////
|
||||
void APM_RC_Class::Init(void)
|
||||
{
|
||||
// Init PWM Timer 1
|
||||
pinMode(11,OUTPUT); // (PB5/OC1A)
|
||||
pinMode(12,OUTPUT); //OUT2 (PB6/OC1B)
|
||||
pinMode(13,OUTPUT); //OUT3 (PB7/OC1C)
|
||||
|
||||
//Remember the registers not declared here remains zero by default...
|
||||
TCCR1A =((1<<WGM11)|(1<<COM1A1)|(1<<COM1B1)|(1<<COM1C1)); //Please read page 131 of DataSheet, we are changing the registers settings of WGM11,COM1B1,COM1A1 to 1 thats all...
|
||||
TCCR1B = (1<<WGM13)|(1<<WGM12)|(1<<CS11); //Prescaler set to 8, that give us a resolution of 0.5us, read page 134 of data sheet
|
||||
OCR1A = 3000; //PB5, none
|
||||
OCR1B = 3000; //PB6, OUT2
|
||||
OCR1C = 3000; //PB7 OUT3
|
||||
ICR1 = 40000; //50hz freq...Datasheet says (system_freq/prescaler)/target frequency. So (16000000hz/8)/50hz=40000,
|
||||
|
||||
// Init PWM Timer 3
|
||||
pinMode(2,OUTPUT); //OUT7 (PE4/OC3B)
|
||||
pinMode(3,OUTPUT); //OUT6 (PE5/OC3C)
|
||||
pinMode(4,OUTPUT); // (PE3/OC3A)
|
||||
TCCR3A =((1<<WGM31)|(1<<COM3A1)|(1<<COM3B1)|(1<<COM3C1));
|
||||
TCCR3B = (1<<WGM33)|(1<<WGM32)|(1<<CS31);
|
||||
OCR3A = 3000; //PE3, NONE
|
||||
OCR3B = 3000; //PE4, OUT7
|
||||
OCR3C = 3000; //PE5, OUT6
|
||||
ICR3 = 40000; //50hz freq
|
||||
|
||||
// Init PWM Timer 5
|
||||
pinMode(44,OUTPUT); //OUT1 (PL5/OC5C)
|
||||
pinMode(45,OUTPUT); //OUT0 (PL4/OC5B)
|
||||
pinMode(46,OUTPUT); // (PL3/OC5A)
|
||||
|
||||
TCCR5A =((1<<WGM51)|(1<<COM5A1)|(1<<COM5B1)|(1<<COM5C1));
|
||||
TCCR5B = (1<<WGM53)|(1<<WGM52)|(1<<CS51);
|
||||
OCR5A = 3000; //PL3,
|
||||
OCR5B = 3000; //PL4, OUT0
|
||||
OCR5C = 3000; //PL5, OUT1
|
||||
ICR5 = 40000; //50hz freq
|
||||
|
||||
// Init PPM input and PWM Timer 4
|
||||
pinMode(49, INPUT); // ICP4 pin (PL0) (PPM input)
|
||||
pinMode(7,OUTPUT); //OUT5 (PH4/OC4B)
|
||||
pinMode(8,OUTPUT); //OUT4 (PH5/OC4C)
|
||||
|
||||
TCCR4A =((1<<WGM40)|(1<<WGM41)|(1<<COM4C1)|(1<<COM4B1)|(1<<COM4A1));
|
||||
//Prescaler set to 8, that give us a resolution of 0.5us
|
||||
// Input Capture rising edge
|
||||
TCCR4B = ((1<<WGM43)|(1<<WGM42)|(1<<CS41)|(1<<ICES4));
|
||||
|
||||
OCR4A = 40000; ///50hz freq.
|
||||
OCR4B = 3000; //PH4, OUT5
|
||||
OCR4C = 3000; //PH5, OUT4
|
||||
|
||||
//TCCR4B |=(1<<ICES4); //Changing edge detector (rising edge).
|
||||
//TCCR4B &=(~(1<<ICES4)); //Changing edge detector. (falling edge)
|
||||
TIMSK4 |= (1<<ICIE4); // Enable Input Capture interrupt. Timer interrupt mask
|
||||
}
|
||||
|
||||
void APM_RC_Class::OutputCh(unsigned char ch, int pwm)
|
||||
{
|
||||
pwm=constrain(pwm,MIN_PULSEWIDTH,MAX_PULSEWIDTH);
|
||||
pwm<<=1; // pwm*2;
|
||||
|
||||
switch(ch)
|
||||
{
|
||||
case 0: OCR5B=pwm; break; //ch0
|
||||
case 1: OCR5C=pwm; break; //ch1
|
||||
case 2: OCR1B=pwm; break; //ch2
|
||||
case 3: OCR1C=pwm; break; //ch3
|
||||
case 4: OCR4C=pwm; break; //ch4
|
||||
case 5: OCR4B=pwm; break; //ch5
|
||||
case 6: OCR3C=pwm; break; //ch6
|
||||
case 7: OCR3B=pwm; break; //ch7
|
||||
case 8: OCR5A=pwm; break; //ch8, PL3
|
||||
case 9: OCR1A=pwm; break; //ch9, PB5
|
||||
case 10: OCR3A=pwm; break; //ch10, PE3
|
||||
}
|
||||
}
|
||||
|
||||
int APM_RC_Class::InputCh(unsigned char ch)
|
||||
{
|
||||
int result;
|
||||
int result2;
|
||||
|
||||
// Because servo pulse variables are 16 bits and the interrupts are running values could be corrupted.
|
||||
// We dont want to stop interrupts to read radio channels so we have to do two readings to be sure that the value is correct...
|
||||
result = PWM_RAW[ch]>>1; // Because timer runs at 0.5us we need to do value/2
|
||||
result2 = PWM_RAW[ch]>>1;
|
||||
if (result != result2)
|
||||
result = PWM_RAW[ch]>>1; // if the results are different we make a third reading (this should be fine)
|
||||
|
||||
// Limit values to a valid range
|
||||
result = constrain(result,MIN_PULSEWIDTH,MAX_PULSEWIDTH);
|
||||
radio_status=0; // Radio channel read
|
||||
return(result);
|
||||
}
|
||||
|
||||
unsigned char APM_RC_Class::GetState(void)
|
||||
{
|
||||
return(radio_status);
|
||||
}
|
||||
|
||||
// InstantPWM implementation
|
||||
// This function forces the PWM output (reset PWM) on Out0 and Out1 (Timer5). For quadcopters use
|
||||
void APM_RC_Class::Force_Out0_Out1(void)
|
||||
{
|
||||
if (TCNT5>5000) // We take care that there are not a pulse in the output
|
||||
TCNT5=39990; // This forces the PWM output to reset in 5us (10 counts of 0.5us). The counter resets at 40000
|
||||
}
|
||||
// This function forces the PWM output (reset PWM) on Out2 and Out3 (Timer1). For quadcopters use
|
||||
void APM_RC_Class::Force_Out2_Out3(void)
|
||||
{
|
||||
if (TCNT1>5000)
|
||||
TCNT1=39990;
|
||||
}
|
||||
// This function forces the PWM output (reset PWM) on Out6 and Out7 (Timer3). For quadcopters use
|
||||
void APM_RC_Class::Force_Out6_Out7(void)
|
||||
{
|
||||
if (TCNT3>5000)
|
||||
TCNT3=39990;
|
||||
}
|
||||
|
||||
// make one instance for the user to use
|
||||
APM_RC_Class APM_RC;
|
||||
|
||||
#endif // defined(ATMega1280)
|
@ -0,0 +1,24 @@
|
||||
#ifndef APM_RC_h
|
||||
#define APM_RC_h
|
||||
|
||||
#define NUM_CHANNELS 8
|
||||
#define MIN_PULSEWIDTH 900
|
||||
#define MAX_PULSEWIDTH 2100
|
||||
|
||||
class APM_RC_Class
|
||||
{
|
||||
private:
|
||||
public:
|
||||
APM_RC_Class();
|
||||
void Init();
|
||||
void OutputCh(unsigned char ch, int pwm);
|
||||
int InputCh(unsigned char ch);
|
||||
unsigned char GetState();
|
||||
void Force_Out0_Out1(void);
|
||||
void Force_Out2_Out3(void);
|
||||
void Force_Out6_Out7(void);
|
||||
};
|
||||
|
||||
extern APM_RC_Class APM_RC;
|
||||
|
||||
#endif
|
@ -0,0 +1,31 @@
|
||||
/*
|
||||
Example of APM_RC library.
|
||||
Code by Jordi Muñoz and Jose Julio. DIYDrones.com
|
||||
|
||||
Print Input values and send Output to the servos
|
||||
(Works with last PPM_encoder firmware)
|
||||
*/
|
||||
|
||||
#include <APM_RC.h> // ArduPilot Mega RC Library
|
||||
|
||||
void setup()
|
||||
{
|
||||
APM_RC.Init(); // APM Radio initialization
|
||||
Serial.begin(57600);
|
||||
Serial.println("ArduPilot Mega RC library test");
|
||||
delay(1000);
|
||||
}
|
||||
void loop()
|
||||
{
|
||||
if (APM_RC.GetState()==1) // New radio frame? (we could use also if((millis()- timer) > 20)
|
||||
{
|
||||
Serial.print("CH:");
|
||||
for(int i=0;i<8;i++)
|
||||
{
|
||||
Serial.print(APM_RC.InputCh(i)); // Print channel values
|
||||
Serial.print(",");
|
||||
APM_RC.OutputCh(i,APM_RC.InputCh(i)); // Copy input to Servos
|
||||
}
|
||||
Serial.println();
|
||||
}
|
||||
}
|
@ -0,0 +1,8 @@
|
||||
APM_RC KEYWORD1
|
||||
begin KEYWORD2
|
||||
InputCh KEYWORD2
|
||||
OutputCh KEYWORD2
|
||||
GetState KEYWORD2
|
||||
Force_Out0_Out1 KEYWORD2
|
||||
Force_Out2_Out3 KEYWORD2
|
||||
Force_Out6_Out7 KEYWORD2
|
@ -0,0 +1,169 @@
|
||||
/*
|
||||
APM_RC_QUAD.cpp - Radio Control Library for Ardupilot Mega. Arduino
|
||||
Code by Jordi Muñoz and Jose Julio. DIYDrones.com
|
||||
|
||||
This library is free software; you can redistribute it and/or
|
||||
modify it under the terms of the GNU Lesser General Public
|
||||
License as published by the Free Software Foundation; either
|
||||
version 2.1 of the License, or (at your option) any later version.
|
||||
|
||||
This is a special version of the library for use in Quadcopters
|
||||
Because we have modified servo outputs 0..3 to have a higher rate (300Hz)
|
||||
|
||||
RC Input : PPM signal on IC4 pin
|
||||
RC Output : 8 servo Outputs,
|
||||
OUT0..OUT3 : 300Hz Servo output (for Quad ESC´s)
|
||||
OUT4..OUT7 : standard 50Hz servo outputs
|
||||
|
||||
Methods:
|
||||
Init() : Initialization of interrupts an Timers
|
||||
OutpuCh(ch,pwm) : Output value to servos (range : 900-2100us) ch=0..10
|
||||
InputCh(ch) : Read a channel input value. ch=0..7
|
||||
GetState() : Returns the state of the input. 1 => New radio frame to process
|
||||
Automatically resets when we call InputCh to read channels
|
||||
|
||||
*/
|
||||
#include "APM_RC_QUAD.h"
|
||||
|
||||
#include <avr/interrupt.h>
|
||||
#include "WProgram.h"
|
||||
|
||||
// Variable definition for Input Capture interrupt
|
||||
volatile unsigned int ICR4_old;
|
||||
volatile unsigned char PPM_Counter=0;
|
||||
volatile unsigned int PWM_RAW[8] = {2400,2400,2400,2400,2400,2400,2400,2400};
|
||||
volatile unsigned char radio_status=0;
|
||||
|
||||
/****************************************************
|
||||
Input Capture Interrupt ICP4 => PPM signal read
|
||||
****************************************************/
|
||||
ISR(TIMER4_CAPT_vect)
|
||||
{
|
||||
unsigned int Pulse;
|
||||
unsigned int Pulse_Width;
|
||||
|
||||
Pulse=ICR4;
|
||||
if (Pulse<ICR4_old) // Take care of the overflow of Timer4 (TOP=40000)
|
||||
Pulse_Width=(Pulse + 40000)-ICR4_old; //Calculating pulse
|
||||
else
|
||||
Pulse_Width=Pulse-ICR4_old; //Calculating pulse
|
||||
if (Pulse_Width>8000) // SYNC pulse?
|
||||
PPM_Counter=0;
|
||||
else
|
||||
{
|
||||
PPM_Counter &= 0x07; // For safety only (limit PPM_Counter to 7)
|
||||
PWM_RAW[PPM_Counter++]=Pulse_Width; //Saving pulse.
|
||||
if (PPM_Counter >= NUM_CHANNELS)
|
||||
radio_status = 1;
|
||||
}
|
||||
ICR4_old = Pulse;
|
||||
}
|
||||
|
||||
|
||||
// Constructors ////////////////////////////////////////////////////////////////
|
||||
|
||||
APM_RC_QUAD_Class::APM_RC_QUAD_Class()
|
||||
{
|
||||
}
|
||||
|
||||
// Public Methods //////////////////////////////////////////////////////////////
|
||||
void APM_RC_QUAD_Class::Init(void)
|
||||
{
|
||||
// Init PWM Timer 1
|
||||
//pinMode(11,OUTPUT); // (PB5/OC1A)
|
||||
pinMode(12,OUTPUT); //OUT2 (PB6/OC1B)
|
||||
pinMode(13,OUTPUT); //OUT3 (PB7/OC1C)
|
||||
|
||||
//Remember the registers not declared here remains zero by default...
|
||||
TCCR1A =((1<<WGM11)|(1<<COM1B1)|(1<<COM1C1)); //Please read page 131 of DataSheet, we are changing the registers settings of WGM11,COM1B1,COM1A1 to 1 thats all...
|
||||
TCCR1B = (1<<WGM13)|(1<<WGM12)|(1<<CS11); //Prescaler set to 8, that give us a resolution of 0.5us, read page 134 of data sheet
|
||||
//OCR1A = 3000; //PB5, none
|
||||
OCR1B = 3000; //PB6, OUT2
|
||||
OCR1C = 3000; //PB7 OUT3
|
||||
ICR1 = 6600; //300hz freq...
|
||||
|
||||
// Init PWM Timer 3
|
||||
pinMode(2,OUTPUT); //OUT7 (PE4/OC3B)
|
||||
pinMode(3,OUTPUT); //OUT6 (PE5/OC3C)
|
||||
//pinMode(4,OUTPUT); // (PE3/OC3A)
|
||||
TCCR3A =((1<<WGM31)|(1<<COM3B1)|(1<<COM3C1));
|
||||
TCCR3B = (1<<WGM33)|(1<<WGM32)|(1<<CS31);
|
||||
//OCR3A = 3000; //PE3, NONE
|
||||
OCR3B = 3000; //PE4, OUT7
|
||||
OCR3C = 3000; //PE5, OUT6
|
||||
ICR3 = 40000; //50hz freq (standard servos)
|
||||
|
||||
// Init PWM Timer 5
|
||||
pinMode(44,OUTPUT); //OUT1 (PL5/OC5C)
|
||||
pinMode(45,OUTPUT); //OUT0 (PL4/OC5B)
|
||||
//pinMode(46,OUTPUT); // (PL3/OC5A)
|
||||
|
||||
TCCR5A =((1<<WGM51)|(1<<COM5B1)|(1<<COM5C1));
|
||||
TCCR5B = (1<<WGM53)|(1<<WGM52)|(1<<CS51);
|
||||
//OCR5A = 3000; //PL3,
|
||||
OCR5B = 3000; //PL4, OUT0
|
||||
OCR5C = 3000; //PL5, OUT1
|
||||
ICR5 = 6600; //300hz freq
|
||||
|
||||
// Init PPM input and PWM Timer 4
|
||||
pinMode(49, INPUT); // ICP4 pin (PL0) (PPM input)
|
||||
pinMode(7,OUTPUT); //OUT5 (PH4/OC4B)
|
||||
pinMode(8,OUTPUT); //OUT4 (PH5/OC4C)
|
||||
|
||||
TCCR4A =((1<<WGM40)|(1<<WGM41)|(1<<COM4C1)|(1<<COM4B1)|(1<<COM4A1));
|
||||
//Prescaler set to 8, that give us a resolution of 0.5us
|
||||
// Input Capture rising edge
|
||||
TCCR4B = ((1<<WGM43)|(1<<WGM42)|(1<<CS41)|(1<<ICES4));
|
||||
|
||||
OCR4A = 40000; ///50hz freq. (standard servos)
|
||||
OCR4B = 3000; //PH4, OUT5
|
||||
OCR4C = 3000; //PH5, OUT4
|
||||
|
||||
//TCCR4B |=(1<<ICES4); //Changing edge detector (rising edge).
|
||||
//TCCR4B &=(~(1<<ICES4)); //Changing edge detector. (falling edge)
|
||||
TIMSK4 |= (1<<ICIE4); // Enable Input Capture interrupt. Timer interrupt mask
|
||||
}
|
||||
|
||||
void APM_RC_QUAD_Class::OutputCh(unsigned char ch, int pwm)
|
||||
{
|
||||
pwm=constrain(pwm,MIN_PULSEWIDTH,MAX_PULSEWIDTH);
|
||||
pwm<<=1; // pwm*2;
|
||||
|
||||
switch(ch)
|
||||
{
|
||||
case 0: OCR5B=pwm; break; //ch0
|
||||
case 1: OCR5C=pwm; break; //ch1
|
||||
case 2: OCR1B=pwm; break; //ch2
|
||||
case 3: OCR1C=pwm; break; //ch3
|
||||
case 4: OCR4C=pwm; break; //ch4
|
||||
case 5: OCR4B=pwm; break; //ch5
|
||||
case 6: OCR3C=pwm; break; //ch6
|
||||
case 7: OCR3B=pwm; break; //ch7
|
||||
}
|
||||
}
|
||||
|
||||
int APM_RC_QUAD_Class::InputCh(unsigned char ch)
|
||||
{
|
||||
int result;
|
||||
int result2;
|
||||
|
||||
// Because servo pulse variables are 16 bits and the interrupts are running values could be corrupted.
|
||||
// We dont want to stop interrupts to read radio channels so we have to do two readings to be sure that the value is correct...
|
||||
result = PWM_RAW[ch]>>1; // Because timer runs at 0.5us we need to do value/2
|
||||
result2 = PWM_RAW[ch]>>1;
|
||||
if (result != result2)
|
||||
result = PWM_RAW[ch]>>1; // if the results are different we make a third reading (this should be fine)
|
||||
|
||||
// Limit values to a valid range
|
||||
result = constrain(result,MIN_PULSEWIDTH,MAX_PULSEWIDTH);
|
||||
radio_status=0; // Radio channel read
|
||||
return(result);
|
||||
}
|
||||
|
||||
unsigned char APM_RC_QUAD_Class::GetState(void)
|
||||
{
|
||||
return(radio_status);
|
||||
}
|
||||
|
||||
// make one instance for the user to use
|
||||
APM_RC_QUAD_Class APM_RC_QUAD;
|
@ -0,0 +1,21 @@
|
||||
#ifndef APM_RC_QUAD_h
|
||||
#define APM_RC_QUAD_h
|
||||
|
||||
#define NUM_CHANNELS 8
|
||||
#define MIN_PULSEWIDTH 900
|
||||
#define MAX_PULSEWIDTH 2100
|
||||
|
||||
class APM_RC_QUAD_Class
|
||||
{
|
||||
private:
|
||||
public:
|
||||
APM_RC_QUAD_Class();
|
||||
void Init();
|
||||
void OutputCh(unsigned char ch, int pwm);
|
||||
int InputCh(unsigned char ch);
|
||||
unsigned char GetState();
|
||||
};
|
||||
|
||||
extern APM_RC_QUAD_Class APM_RC_QUAD;
|
||||
|
||||
#endif
|
@ -0,0 +1,31 @@
|
||||
/*
|
||||
Example of APM_RC library.
|
||||
Code by Jordi Muñoz and Jose Julio. DIYDrones.com
|
||||
|
||||
Print Input values and send Output to the servos
|
||||
(Works with last PPM_encoder firmware)
|
||||
*/
|
||||
|
||||
#include <APM_RC.h> // ArduPilot Mega RC Library
|
||||
|
||||
void setup()
|
||||
{
|
||||
APM_RC.Init(); // APM Radio initialization
|
||||
Serial.begin(57600);
|
||||
Serial.println("ArduPilot Mega RC library test");
|
||||
delay(1000);
|
||||
}
|
||||
void loop()
|
||||
{
|
||||
if (APM_RC.GetState()==1) // New radio frame? (we could use also if((millis()- timer) > 20)
|
||||
{
|
||||
Serial.print("CH:");
|
||||
for(int i=0;i<8;i++)
|
||||
{
|
||||
Serial.print(APM_RC.InputCh(i)); // Print channel values
|
||||
Serial.print(",");
|
||||
APM_RC.OutputCh(i,APM_RC.InputCh(i)); // Copy input to Servos
|
||||
}
|
||||
Serial.println();
|
||||
}
|
||||
}
|
@ -0,0 +1,5 @@
|
||||
APM_RC_QUAD KEYWORD1
|
||||
begin KEYWORD2
|
||||
InputCh KEYWORD2
|
||||
OutputCh KEYWORD2
|
||||
GetState KEYWORD2
|
@ -0,0 +1,17 @@
|
||||
// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: t -*-
|
||||
//
|
||||
// This is free software; you can redistribute it and/or modify it under
|
||||
// the terms of the GNU Lesser General Public License as published by the
|
||||
// Free Software Foundation; either version 2.1 of the License, or (at
|
||||
// your option) any later version.
|
||||
//
|
||||
|
||||
/// @file AP_Common.cpp
|
||||
/// @brief Common utility routines for the ArduPilot libraries.
|
||||
///
|
||||
/// @note Exercise restraint adding code here; everything in this
|
||||
/// library will be linked with any sketch using it.
|
||||
///
|
||||
|
||||
#include "AP_Common.h"
|
||||
|
@ -0,0 +1,53 @@
|
||||
// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: t -*-
|
||||
//
|
||||
// This is free software; you can redistribute it and/or modify it under
|
||||
// the terms of the GNU Lesser General Public License as published by the
|
||||
// Free Software Foundation; either version 2.1 of the License, or (at
|
||||
// your option) any later version.
|
||||
//
|
||||
|
||||
///
|
||||
/// @file AP_Common.h
|
||||
/// @brief Common definitions and utility routines for the ArduPilot
|
||||
/// libraries.
|
||||
///
|
||||
|
||||
#ifndef _AP_COMMON_H
|
||||
#define _AP_COMMON_H
|
||||
|
||||
#include <stdint.h>
|
||||
|
||||
#include "include/menu.h" /// simple menu subsystem
|
||||
|
||||
////////////////////////////////////////////////////////////////////////////////
|
||||
/// @name Types
|
||||
///
|
||||
/// Data structures and types used throughout the libraries and applications.
|
||||
///
|
||||
//@{
|
||||
|
||||
struct Location {
|
||||
uint8_t id; ///< command id
|
||||
uint8_t p1; ///< param 1
|
||||
int32_t alt; ///< param 2 - Altitude in centimeters (meters * 100)
|
||||
int32_t lat; ///< param 3 - Lattitude * 10**7
|
||||
int32_t lng; ///< param 4 - Longitude * 10**7
|
||||
};
|
||||
|
||||
//@}
|
||||
|
||||
////////////////////////////////////////////////////////////////////////////////
|
||||
/// @name Conversions
|
||||
///
|
||||
/// Conversion macros and factors.
|
||||
///
|
||||
//@{
|
||||
|
||||
/// XXX this should probably be replaced with radians()/degrees(), but their
|
||||
/// inclusion in wiring.h makes doing that here difficult.
|
||||
#define ToDeg(x) (x*57.2957795131) // *180/pi
|
||||
|
||||
//@}
|
||||
|
||||
|
||||
#endif // _AP_COMMON_H
|
@ -0,0 +1,21 @@
|
||||
// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: t -*-
|
||||
|
||||
//
|
||||
// C++ runtime support not provided by Arduino
|
||||
//
|
||||
// Note: use new/delete with caution. The heap is small and
|
||||
// easily fragmented.
|
||||
//
|
||||
|
||||
#include <stdlib.h>
|
||||
|
||||
void * operator new(size_t size)
|
||||
{
|
||||
return(calloc(size, 1));
|
||||
}
|
||||
|
||||
void operator delete(void *p)
|
||||
{
|
||||
if (p)
|
||||
free(p);
|
||||
}
|
@ -0,0 +1,45 @@
|
||||
|
||||
#include <FastSerial.h>
|
||||
#include <AP_Common.h>
|
||||
|
||||
#include <avr/pgmspace.h>
|
||||
|
||||
FastSerialPort0(Serial);
|
||||
|
||||
int
|
||||
menu_test(uint8_t argc, const Menu::arg *argv)
|
||||
{
|
||||
int i;
|
||||
|
||||
Serial.printf("This is a test with %d arguments\n", argc);
|
||||
for (i = 1; i < argc; i++) {
|
||||
Serial.printf("%d: int %ld float ", i, argv[i].i);
|
||||
Serial.println(argv[i].f, 6); // gross
|
||||
}
|
||||
}
|
||||
|
||||
int
|
||||
menu_auto(uint8_t argc, const Menu::arg *argv)
|
||||
{
|
||||
Serial.println("auto text");
|
||||
}
|
||||
|
||||
const struct Menu::command top_menu_commands[] PROGMEM = {
|
||||
{"*", menu_auto},
|
||||
{"test", menu_test},
|
||||
};
|
||||
|
||||
MENU(top, "menu", top_menu_commands);
|
||||
|
||||
void
|
||||
setup(void)
|
||||
{
|
||||
Serial.begin(38400);
|
||||
top.run();
|
||||
}
|
||||
|
||||
void
|
||||
loop(void)
|
||||
{
|
||||
}
|
||||
|
@ -0,0 +1,138 @@
|
||||
// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: t -*-
|
||||
|
||||
/// @file menu.h
|
||||
/// @brief Simple commandline menu subsystem.
|
||||
/// @discussion
|
||||
/// The Menu class implements a simple CLI that accepts commands typed by
|
||||
/// the user, and passes the arguments to those commands to a function
|
||||
/// defined as handing the command.
|
||||
///
|
||||
/// Commands are defined in an array of Menu::command structures passed
|
||||
/// to the constructor. Each entry in the array defines one command.
|
||||
///
|
||||
/// Arguments passed to the handler function are pre-converted to both
|
||||
/// long and float for convenience.
|
||||
///
|
||||
|
||||
#define MENU_COMMANDLINE_MAX 32 ///< maximum input line length
|
||||
#define MENU_ARGS_MAX 4 ///< maximum number of arguments
|
||||
#define MENU_COMMAND_MAX 14 ///< maximum size of a command name
|
||||
|
||||
/// Class defining and handling one menu tree
|
||||
class Menu {
|
||||
public:
|
||||
/// argument passed to a menu function
|
||||
///
|
||||
/// Space-delimited arguments are parsed from the commandline and
|
||||
/// separated into these structures.
|
||||
///
|
||||
/// If the argument cannot be parsed as a float or a long, the value
|
||||
/// of f or i respectively is undefined. You should range-check
|
||||
/// the inputs to your function.
|
||||
///
|
||||
struct arg {
|
||||
const char *str; ///< string form of the argument
|
||||
long i; ///< integer form of the argument (if a number)
|
||||
float f; ///< floating point form of the argument (if a number)
|
||||
};
|
||||
|
||||
/// menu command function
|
||||
///
|
||||
/// Functions called by menu array entries are expected to be of this
|
||||
/// type.
|
||||
///
|
||||
/// @param argc The number of valid arguments, including the
|
||||
/// name of the command in argv[0]. Will never be
|
||||
/// more than MENU_ARGS_MAX.
|
||||
/// @param argv Pointer to an array of Menu::arg structures
|
||||
/// detailing any optional arguments given to the
|
||||
/// command. argv[0] is always the name of the
|
||||
/// command, so that the same function can be used
|
||||
/// to handle more than one command.
|
||||
///
|
||||
typedef int8_t (*func)(uint8_t argc, const struct arg *argv);
|
||||
|
||||
/// menu pre-prompt function
|
||||
///
|
||||
/// Called immediately before waiting for the user to type a command; can be
|
||||
/// used to display help text or status, for example.
|
||||
///
|
||||
/// If this function returns false, the menu exits.
|
||||
///
|
||||
typedef bool (*preprompt)(void);
|
||||
|
||||
/// menu command description
|
||||
///
|
||||
struct command {
|
||||
/// Name of the command, as typed or received.
|
||||
/// Command names are limited in size to keep this structure compact.
|
||||
///
|
||||
const char command[MENU_COMMAND_MAX];
|
||||
|
||||
/// The function to call when the command is received.
|
||||
///
|
||||
/// The argc argument will be at least 1, and no more than
|
||||
/// MENU_ARGS_MAX. The argv array will be populated with
|
||||
/// arguments typed/received up to MENU_ARGS_MAX. The command
|
||||
/// name will always be in argv[0].
|
||||
///
|
||||
/// Commands may return -2 to cause the menu itself to exit.
|
||||
/// The "?", "help" and "exit" commands are always defined, but
|
||||
/// can be overridden by explicit entries in the command array.
|
||||
///
|
||||
int8_t (*func)(uint8_t argc, const struct arg *argv); ///< callback function
|
||||
};
|
||||
|
||||
/// constructor
|
||||
///
|
||||
/// Note that you should normally not call the constructor directly. Use
|
||||
/// the MENU and MENU2 macros defined below.
|
||||
///
|
||||
/// @param prompt The prompt to be displayed with this menu.
|
||||
/// @param commands An array of ::command structures in program memory (PROGMEM).
|
||||
/// @param entries The number of entries in the menu.
|
||||
///
|
||||
Menu(const char *prompt, const struct command *commands, uint8_t entries, preprompt ppfunc = 0);
|
||||
|
||||
/// menu runner
|
||||
void run(void);
|
||||
|
||||
private:
|
||||
/// Implements the default 'help' command.
|
||||
///
|
||||
void _help(void); ///< implements the 'help' command
|
||||
|
||||
/// calls the function for the n'th menu item
|
||||
///
|
||||
/// @param n Index for the menu item to call
|
||||
/// @param argc Number of arguments prepared for the menu item
|
||||
///
|
||||
int8_t _call(uint8_t n, uint8_t argc);
|
||||
|
||||
const char *_prompt; ///< prompt to display
|
||||
const command *_commands; ///< array of commands
|
||||
const uint8_t _entries; ///< size of the menu
|
||||
const preprompt _ppfunc; ///< optional pre-prompt action
|
||||
|
||||
static char _inbuf[MENU_COMMANDLINE_MAX]; ///< input buffer
|
||||
static arg _argv[MENU_ARGS_MAX + 1]; ///< arguments
|
||||
};
|
||||
|
||||
/// Macros used to define a menu.
|
||||
///
|
||||
/// The commands argument should be an arary of Menu::command structures, one
|
||||
/// per command name. The array does not need to be terminated with any special
|
||||
/// record.
|
||||
///
|
||||
/// Use name.run() to run the menu.
|
||||
///
|
||||
/// The MENU2 macro supports the optional pre-prompt printing function.
|
||||
///
|
||||
#define MENU(name, prompt, commands) \
|
||||
static const char __menu_name__ ##name[] PROGMEM = prompt; \
|
||||
static Menu name(__menu_name__ ##name, commands, sizeof(commands) / sizeof(commands[0]))
|
||||
|
||||
#define MENU2(name, prompt, commands, preprompt) \
|
||||
static const char __menu_name__ ##name[] PROGMEM = prompt; \
|
||||
static Menu name(__menu_name__ ##name, commands, sizeof(commands) / sizeof(commands[0]), preprompt)
|
||||
|
@ -0,0 +1,4 @@
|
||||
Menu KEYWORD1
|
||||
run KEYWORD2
|
||||
Location KEYWORD2
|
||||
|
@ -0,0 +1,129 @@
|
||||
// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: t -*-
|
||||
|
||||
//
|
||||
// Simple commandline menu system.
|
||||
//
|
||||
|
||||
#include <FastSerial.h>
|
||||
|
||||
#include <ctype.h>
|
||||
#include <string.h>
|
||||
#include <avr/pgmspace.h>
|
||||
|
||||
#include "include/menu.h"
|
||||
|
||||
// statics
|
||||
char Menu::_inbuf[MENU_COMMANDLINE_MAX];
|
||||
Menu::arg Menu::_argv[MENU_ARGS_MAX + 1];
|
||||
|
||||
// constructor
|
||||
Menu::Menu(const prog_char *prompt, const Menu::command *commands, uint8_t entries, preprompt ppfunc) :
|
||||
_prompt(prompt),
|
||||
_commands(commands),
|
||||
_entries(entries),
|
||||
_ppfunc(ppfunc)
|
||||
{
|
||||
}
|
||||
|
||||
// run the menu
|
||||
void
|
||||
Menu::run(void)
|
||||
{
|
||||
uint8_t len, i, ret;
|
||||
uint8_t argc;
|
||||
int c;
|
||||
char *s;
|
||||
|
||||
// loop performing commands
|
||||
for (;;) {
|
||||
|
||||
// run the pre-prompt function, if one is defined
|
||||
if ((NULL != _ppfunc) && !_ppfunc())
|
||||
return;
|
||||
|
||||
// loop reading characters from the input
|
||||
len = 0;
|
||||
Serial.printf("%S] ", _prompt);
|
||||
for (;;) {
|
||||
c = Serial.read();
|
||||
if (-1 == c)
|
||||
continue;
|
||||
// carriage return -> process command
|
||||
if ('\r' == c) {
|
||||
_inbuf[len] = '\0';
|
||||
Serial.write('\r');
|
||||
Serial.write('\n');
|
||||
break;
|
||||
}
|
||||
// backspace
|
||||
if ('\b' == c) {
|
||||
if (len > 0) {
|
||||
len--;
|
||||
Serial.write('\b');
|
||||
Serial.write(' ');
|
||||
Serial.write('\b');
|
||||
continue;
|
||||
}
|
||||
}
|
||||
// printable character
|
||||
if (isprint(c) && (len < (MENU_COMMANDLINE_MAX - 1))) {
|
||||
_inbuf[len++] = c;
|
||||
Serial.write((char)c);
|
||||
continue;
|
||||
}
|
||||
}
|
||||
|
||||
// split the input line into tokens
|
||||
argc = 0;
|
||||
_argv[argc++].str = strtok_r(_inbuf, " ", &s);
|
||||
// XXX should an empty line by itself back out of the current menu?
|
||||
while (argc <= MENU_ARGS_MAX) {
|
||||
_argv[argc].str = strtok_r(NULL, " ", &s);
|
||||
if ('\0' == _argv[argc].str)
|
||||
break;
|
||||
_argv[argc].i = atol(_argv[argc].str);
|
||||
_argv[argc].f = atof(_argv[argc].str); // calls strtod, > 700B !
|
||||
argc++;
|
||||
}
|
||||
|
||||
// look for a command matching the first word (note that it may be empty)
|
||||
for (i = 0; i < _entries; i++) {
|
||||
if (!strcasecmp_P(_argv[0].str, _commands[i].command)) {
|
||||
ret = _call(i, argc);
|
||||
if (-2 == ret)
|
||||
return;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
// implicit commands
|
||||
if (i == _entries) {
|
||||
if (!strcmp(_argv[0].str, "?") || (!strcasecmp_P(_argv[0].str, PSTR("help")))) {
|
||||
_help();
|
||||
} else if (!strcasecmp_P(_argv[0].str, PSTR("exit"))) {
|
||||
return;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// display the list of commands in response to the 'help' command
|
||||
void
|
||||
Menu::_help(void)
|
||||
{
|
||||
int i;
|
||||
|
||||
Serial.println("Commands:");
|
||||
for (i = 0; i < _entries; i++)
|
||||
Serial.printf(" %S\n", _commands[i].command);
|
||||
}
|
||||
|
||||
// run the n'th command in the menu
|
||||
int8_t
|
||||
Menu::_call(uint8_t n, uint8_t argc)
|
||||
{
|
||||
func fn;
|
||||
|
||||
fn = (func)pgm_read_word(&_commands[n].func);
|
||||
return(fn(argc, &_argv[0]));
|
||||
}
|
@ -0,0 +1,115 @@
|
||||
/*
|
||||
AP_Compass.cpp - Arduino Library for HMC5843 I2C Magnetometer
|
||||
Code by Jordi Muñoz and Jose Julio. DIYDrones.com
|
||||
|
||||
This library is free software; you can redistribute it and / or
|
||||
modify it under the terms of the GNU Lesser General Public
|
||||
License as published by the Free Software Foundation; either
|
||||
version 2.1 of the License, or (at your option) any later version.
|
||||
|
||||
Sensor is conected to I2C port
|
||||
Sensor is initialized in Continuos mode (10Hz)
|
||||
|
||||
Variables:
|
||||
heading : Magnetic heading
|
||||
heading_X : Magnetic heading X component
|
||||
heading_Y : Magnetic heading Y component
|
||||
mag_X : Raw X axis magnetometer data
|
||||
mag_Y : Raw Y axis magnetometer data
|
||||
mag_Z : Raw Z axis magnetometer data
|
||||
|
||||
Methods:
|
||||
init() : initialization of I2C and sensor
|
||||
update() : update Sensor data
|
||||
|
||||
To do : Calibration of the sensor, code optimization
|
||||
Mount position : UPDATED
|
||||
Big capacitor pointing backward, connector forward
|
||||
|
||||
*/
|
||||
|
||||
|
||||
extern "C" {
|
||||
// AVR LibC Includes
|
||||
#include <math.h>
|
||||
#include "WConstants.h"
|
||||
}
|
||||
|
||||
#include <Wire.h>
|
||||
#include "AP_Compass.h"
|
||||
|
||||
#define COMPASS_ADDRESS 0x1E
|
||||
|
||||
// Constructors ////////////////////////////////////////////////////////////////
|
||||
AP_Compass::AP_Compass()
|
||||
{
|
||||
}
|
||||
|
||||
// Public Methods //////////////////////////////////////////////////////////////
|
||||
void
|
||||
AP_Compass::init(void)
|
||||
{
|
||||
Wire.begin();
|
||||
Wire.beginTransmission(COMPASS_ADDRESS);
|
||||
Wire.send(0x02);
|
||||
Wire.send(0x00); // Set continouos mode (default to 10Hz)
|
||||
Wire.endTransmission(); // end transmission
|
||||
}
|
||||
|
||||
// update Sensor data
|
||||
void
|
||||
AP_Compass::update()
|
||||
{
|
||||
int i = 0;
|
||||
byte buff[6];
|
||||
|
||||
Wire.beginTransmission(COMPASS_ADDRESS);
|
||||
Wire.send(0x03); // sends address to read from
|
||||
Wire.endTransmission(); // end transmission
|
||||
|
||||
//Wire.beginTransmission(COMPASS_ADDRESS);
|
||||
Wire.requestFrom(COMPASS_ADDRESS, 6); // request 6 bytes from device
|
||||
while(Wire.available()){
|
||||
buff[i] = Wire.receive(); // receive one byte
|
||||
i++;
|
||||
}
|
||||
Wire.endTransmission(); // end transmission
|
||||
|
||||
// All bytes received?
|
||||
if (i == 6) {
|
||||
// MSB byte first, then LSB, X,Y,Z
|
||||
mag_X = -((((int)buff[0]) << 8) | buff[1]); // X axis
|
||||
mag_Y = ((((int)buff[2]) << 8) | buff[3]); // Y axis
|
||||
mag_Z = -((((int)buff[4]) << 8) | buff[5]); // Z axis
|
||||
}
|
||||
}
|
||||
|
||||
void
|
||||
AP_Compass::calculate(float roll, float pitch)
|
||||
{
|
||||
float head_X;
|
||||
float head_Y;
|
||||
float cos_roll;
|
||||
float sin_roll;
|
||||
float cos_pitch;
|
||||
float sin_pitch;
|
||||
|
||||
cos_roll = cos(roll); // Optimization, you can get this out of the matrix DCM?
|
||||
sin_roll = sin(roll);
|
||||
cos_pitch = cos(pitch);
|
||||
sin_pitch = sin(pitch);
|
||||
|
||||
// Tilt compensated Magnetic field X component:
|
||||
head_X = mag_X * cos_pitch + mag_Y * sin_roll * sin_pitch + mag_Z * cos_roll * sin_pitch;
|
||||
|
||||
// Tilt compensated Magnetic field Y component:
|
||||
head_Y = mag_Y * cos_roll - mag_Z * sin_roll;
|
||||
|
||||
// Magnetic heading
|
||||
heading = atan2(-head_Y, head_X);
|
||||
ground_course = (degrees(heading) + 180) * 100;
|
||||
|
||||
// Optimization for external DCM use. calculate normalized components
|
||||
heading_X = cos(heading);
|
||||
heading_Y = sin(heading);
|
||||
}
|
@ -0,0 +1,17 @@
|
||||
#ifndef AP_Compass_h
|
||||
#define AP_Compass_h
|
||||
|
||||
#include <Compass.h>
|
||||
|
||||
class AP_Compass : public Compass
|
||||
{
|
||||
public:
|
||||
AP_Compass(); // Constructor
|
||||
void init();
|
||||
void update();
|
||||
void calculate(float roll, float pitch);
|
||||
|
||||
private:
|
||||
};
|
||||
|
||||
#endif
|
@ -0,0 +1,23 @@
|
||||
#ifndef Compass_h
|
||||
#define Compass_h
|
||||
|
||||
#include <inttypes.h>
|
||||
|
||||
class Compass
|
||||
{
|
||||
public:
|
||||
virtual void init();
|
||||
virtual void update();
|
||||
virtual void calculate(float roll, float pitch);
|
||||
|
||||
int16_t mag_X;
|
||||
int16_t mag_Y;
|
||||
int16_t mag_Z;
|
||||
int32_t ground_course;
|
||||
float heading;
|
||||
float heading_X;
|
||||
float heading_Y;
|
||||
|
||||
};
|
||||
|
||||
#endif
|
@ -0,0 +1,40 @@
|
||||
/*
|
||||
Example of APM_Compass library (HMC5843 sensor).
|
||||
Code by Jordi MuÒoz and Jose Julio. DIYDrones.com
|
||||
*/
|
||||
|
||||
#include <Wire.h>
|
||||
#include <AP_Compass.h> // Compass Library
|
||||
|
||||
AP_Compass compass;
|
||||
|
||||
unsigned long timer;
|
||||
|
||||
void setup()
|
||||
{
|
||||
compass.init(); // Initialization
|
||||
Serial.begin(38400);
|
||||
Serial.println("AP Compass library test (HMC5843)");
|
||||
delay(1000);
|
||||
timer = millis();
|
||||
}
|
||||
|
||||
void loop()
|
||||
{
|
||||
float tmp;
|
||||
|
||||
if((millis()- timer) > 100){
|
||||
timer = millis();
|
||||
compass.update();
|
||||
compass.calculate(0, 0); // roll = 0, pitch = 0 for this example
|
||||
Serial.print("Heading:");
|
||||
Serial.print(compass.ground_course,DEC);
|
||||
Serial.print(" (");
|
||||
Serial.print(compass.mag_X);
|
||||
Serial.print(",");
|
||||
Serial.print(compass.mag_Y);
|
||||
Serial.print(",");
|
||||
Serial.print(compass.mag_Z);
|
||||
Serial.println(" )");
|
||||
}
|
||||
}
|
@ -0,0 +1,15 @@
|
||||
Compass KEYWORD1
|
||||
AP_Compass KEYWORD1
|
||||
APM_Compass KEYWORD1
|
||||
init KEYWORD2
|
||||
update KEYWORD2
|
||||
calculate KEYWORD2
|
||||
heading KEYWORD2
|
||||
heading_X KEYWORD2
|
||||
heading_Y KEYWORD2
|
||||
mag_X KEYWORD2
|
||||
mag_Y KEYWORD2
|
||||
mag_Z KEYWORD2
|
||||
|
||||
|
||||
|
@ -0,0 +1,13 @@
|
||||
// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: t -*-
|
||||
|
||||
/// @file AP_GPS.h
|
||||
/// @brief Catch-all header that defines all supported GPS classes.
|
||||
|
||||
#include "AP_GPS_NMEA.h"
|
||||
#include "AP_GPS_SIRF.h"
|
||||
#include "AP_GPS_406.h"
|
||||
#include "AP_GPS_UBLOX.h"
|
||||
#include "AP_GPS_MTK.h"
|
||||
#include "AP_GPS_IMU.h"
|
||||
#include "AP_GPS_None.h"
|
||||
#include "AP_GPS_Auto.h"
|
@ -0,0 +1,81 @@
|
||||
// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: t -*-
|
||||
//
|
||||
// GPS_406.cpp - 406 GPS library for Arduino
|
||||
// Code by Michael Smith, Jason Short, Jordi Muñoz and Jose Julio. DIYDrones.com
|
||||
// This code works with boards based on ATMega168/328 ATMega1280 (Serial port 1)
|
||||
//
|
||||
// This library is free software; you can redistribute it and / or
|
||||
// modify it under the terms of the GNU Lesser General Public
|
||||
// License as published by the Free Software Foundation; either
|
||||
// version 2.1 of the License, or (at your option) any later version.
|
||||
|
||||
#include "../FastSerial/FastSerial.h" // because we need to change baud rates... ugh.
|
||||
#include "AP_GPS_406.h"
|
||||
#include "WProgram.h"
|
||||
|
||||
static const char init_str[] = "$PSRF100,0,57600,8,1,0*37";
|
||||
|
||||
// Constructors ////////////////////////////////////////////////////////////////
|
||||
AP_GPS_406::AP_GPS_406(Stream *s) : AP_GPS_SIRF(s)
|
||||
{
|
||||
}
|
||||
|
||||
// Public Methods ////////////////////////////////////////////////////////////////////
|
||||
void AP_GPS_406::init(void)
|
||||
{
|
||||
_change_to_sirf_protocol(); // Changes to SIRF protocol and sets baud rate
|
||||
_configure_gps(); // Function to configure GPS, to output only the desired msg's
|
||||
|
||||
AP_GPS_SIRF::init(); // let the superclass do anything it might need here
|
||||
}
|
||||
|
||||
// Private Methods //////////////////////////////////////////////////////////////
|
||||
|
||||
void
|
||||
AP_GPS_406::_configure_gps(void)
|
||||
{
|
||||
const uint8_t gps_header[] = {0xA0, 0xA2, 0x00, 0x08, 0xA6, 0x00};
|
||||
const uint8_t gps_payload[] = {0x02, 0x04, 0x07, 0x09, 0x1B};
|
||||
const uint8_t gps_checksum[] = {0xA8, 0xAA, 0xAD, 0xAF, 0xC1};
|
||||
const uint8_t gps_ender[] = {0xB0, 0xB3};
|
||||
|
||||
for(int z = 0; z < 2; z++){
|
||||
for(int x = 0; x < 5; x++){
|
||||
_port->write(gps_header, sizeof(gps_header)); // Prints the msg header, is the same header for all msg..
|
||||
_port->write(gps_payload[x]); // Prints the payload, is not the same for every msg
|
||||
for(int y = 0; y < 6; y++) // Prints 6 zeros
|
||||
_port->write((uint8_t)0);
|
||||
_port->write(gps_checksum[x]); // Print the Checksum
|
||||
_port->write(gps_ender[0]); // Print the Ender of the string, is same on all msg's.
|
||||
_port->write(gps_ender[1]); // ender
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// The EM406 defalts to NMEA at 4800bps. We want to switch it to SiRF binary
|
||||
// mode at a higher rate.
|
||||
//
|
||||
// The change is sticky, but only for as long as the internal supercap holds
|
||||
// settings (usually less than a week).
|
||||
//
|
||||
void
|
||||
AP_GPS_406::_change_to_sirf_protocol(void)
|
||||
{
|
||||
FastSerial *fs = (FastSerial *)_port; // this is a bit grody...
|
||||
|
||||
fs->begin(4800);
|
||||
delay(300);
|
||||
_port->print(init_str);
|
||||
delay(300);
|
||||
|
||||
fs->begin(9600);
|
||||
delay(300);
|
||||
_port->print(init_str);
|
||||
delay(300);
|
||||
|
||||
fs->begin(GPS_406_BITRATE);
|
||||
delay(300);
|
||||
_port->print(init_str);
|
||||
delay(300);
|
||||
}
|
||||
|
@ -0,0 +1,32 @@
|
||||
// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: t -*-
|
||||
//
|
||||
// EM406 GPS driver for ArduPilot and ArduPilotMega.
|
||||
// Code by Michael Smith, Jason Short, Jordi Muñoz and Jose Julio. DIYDrones.com
|
||||
//
|
||||
// This library is free software; you can redistribute it and / or
|
||||
// modify it under the terms of the GNU Lesser General Public
|
||||
// License as published by the Free Software Foundation; either
|
||||
// version 2.1 of the License, or (at your option) any later version.
|
||||
//
|
||||
|
||||
#ifndef AP_GPS_406_h
|
||||
#define AP_GPS_406_h
|
||||
|
||||
#include "AP_GPS_SIRF.h"
|
||||
|
||||
#define GPS_406_BITRATE 57600
|
||||
|
||||
class AP_GPS_406 : public AP_GPS_SIRF
|
||||
{
|
||||
public:
|
||||
// Methods
|
||||
AP_GPS_406(Stream *port);
|
||||
void init(void);
|
||||
|
||||
private:
|
||||
void _change_to_sirf_protocol(void);
|
||||
void _configure_gps(void);
|
||||
};
|
||||
|
||||
#endif
|
||||
|
@ -0,0 +1,162 @@
|
||||
// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: t -*-
|
||||
//
|
||||
// Auto-detecting pseudo-GPS driver
|
||||
//
|
||||
|
||||
#include "AP_GPS.h"
|
||||
#include <stdlib.h>
|
||||
#include <stdio.h>
|
||||
#include <wiring.h>
|
||||
|
||||
static unsigned int baudrates[] = {38400U, 57600U, 9600U, 4800U};
|
||||
|
||||
void
|
||||
AP_GPS_Auto::init(void)
|
||||
{
|
||||
}
|
||||
|
||||
//
|
||||
// Called the first time that a client tries to kick the GPS to update.
|
||||
//
|
||||
// We detect the real GPS, then update the pointer we have been called through
|
||||
// and return.
|
||||
void
|
||||
AP_GPS_Auto::update(void)
|
||||
{
|
||||
GPS *gps;
|
||||
int i;
|
||||
|
||||
// loop trying to find a GPS
|
||||
for (;;) {
|
||||
// loop through possible baudrates
|
||||
for (i = 0; i < (sizeof(baudrates) / sizeof(baudrates[0])); i++) {
|
||||
printf("GPS autodetect at %d:%u\n", i, baudrates[i]);
|
||||
_port->begin(baudrates[i]);
|
||||
if (NULL != (gps = _detect())) {
|
||||
// make the detected GPS the default
|
||||
*_gps = gps;
|
||||
|
||||
// configure the detected GPS and run one update
|
||||
gps->print_errors = true; // XXX
|
||||
gps->init();
|
||||
gps->update();
|
||||
|
||||
// drop back to our caller - subsequent calls through
|
||||
// the global will not come here
|
||||
return;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
//
|
||||
// Perform one iteration of the auto-detection process.
|
||||
//
|
||||
GPS *
|
||||
AP_GPS_Auto::_detect(void)
|
||||
{
|
||||
unsigned long then;
|
||||
int fingerprint[4];
|
||||
int tries;
|
||||
GPS *gps;
|
||||
|
||||
//
|
||||
// Loop attempting to detect a recognised GPS
|
||||
//
|
||||
gps = NULL;
|
||||
for (tries = 0; tries < 2; tries++) {
|
||||
|
||||
//
|
||||
// Empty the serial buffer and wait for 50ms of quiet.
|
||||
//
|
||||
// XXX We can detect babble by counting incoming characters, but
|
||||
// what would we do about it?
|
||||
//
|
||||
_port->flush();
|
||||
then = millis();
|
||||
do {
|
||||
if (_port->available()) {
|
||||
then = millis();
|
||||
_port->read();
|
||||
}
|
||||
} while ((millis() - then) < 50);
|
||||
|
||||
//
|
||||
// Collect four characters to fingerprint a device
|
||||
//
|
||||
fingerprint[0] = _getc();
|
||||
fingerprint[1] = _getc();
|
||||
fingerprint[2] = _getc();
|
||||
fingerprint[3] = _getc();
|
||||
printf("fingerprints 0x%02x 0x%02x 0x%02x 0x%02x\n",
|
||||
fingerprint[0],
|
||||
fingerprint[1],
|
||||
fingerprint[2],
|
||||
fingerprint[3]);
|
||||
|
||||
//
|
||||
// u-blox or MTK in DIYD binary mode (whose smart idea was
|
||||
// it to make the MTK look sort-of like it was talking UBX?)
|
||||
//
|
||||
if ((0xb5 == fingerprint[0]) &&
|
||||
(0x62 == fingerprint[1]) &&
|
||||
(0x01 == fingerprint[2])) {
|
||||
|
||||
// message 5 is MTK pretending to talk UBX
|
||||
if (0x05 == fingerprint[3]) {
|
||||
printf("detected MTK in binary mode\n");
|
||||
gps = new AP_GPS_MTK(_port);
|
||||
break;
|
||||
}
|
||||
|
||||
// any other message is u-blox
|
||||
printf("detected u-blox in binary mode\n");
|
||||
gps = new AP_GPS_UBLOX(_port);
|
||||
break;
|
||||
}
|
||||
|
||||
//
|
||||
// SIRF in binary mode
|
||||
//
|
||||
if ((0xa0 == fingerprint[0]) &&
|
||||
(0xa2 == fingerprint[1])) {
|
||||
printf("detected SIRF in binary mode\n");
|
||||
gps = new AP_GPS_SIRF(_port);
|
||||
break;
|
||||
}
|
||||
|
||||
//
|
||||
// If we haven't spammed the various init strings, send them now
|
||||
// and retry to avoid a false-positive on the NMEA detector.
|
||||
//
|
||||
if (0 == tries) {
|
||||
printf("sending setup strings and trying again\n");
|
||||
_port->println(MTK_SET_BINARY);
|
||||
_port->println(UBLOX_SET_BINARY);
|
||||
_port->println(SIRF_SET_BINARY);
|
||||
continue;
|
||||
}
|
||||
|
||||
//
|
||||
// Something talking NMEA
|
||||
//
|
||||
if (('$' == fingerprint[0]) &&
|
||||
(('G' == fingerprint[1]) || ('P' == fingerprint[1]))) {
|
||||
|
||||
// XXX this may be a bit presumptive, might want to give the GPS a couple of
|
||||
// iterations around the loop to react to init strings?
|
||||
printf("detected NMEA\n");
|
||||
gps = new AP_GPS_NMEA(_port);
|
||||
break;
|
||||
}
|
||||
}
|
||||
return(gps);
|
||||
}
|
||||
|
||||
int
|
||||
AP_GPS_Auto::_getc(void)
|
||||
{
|
||||
while (0 == _port->available())
|
||||
;
|
||||
return(_port->read());
|
||||
}
|
@ -0,0 +1,52 @@
|
||||
// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: t -*-
|
||||
//
|
||||
// Auto-detecting pseudo-GPS driver
|
||||
//
|
||||
|
||||
#ifndef AP_GPS_Auto_h
|
||||
#define AP_GPS_Auto_h
|
||||
|
||||
#include <GPS.h>
|
||||
#include <FastSerial.h>
|
||||
|
||||
class AP_GPS_Auto : public GPS
|
||||
{
|
||||
public:
|
||||
/// Constructor
|
||||
///
|
||||
/// @note The stream is expected to be set up and configured for the
|
||||
/// correct bitrate before ::init is called.
|
||||
///
|
||||
/// @param port Stream connected to the GPS module.
|
||||
/// @param ptr Pointer to a GPS * that will be fixed up by ::init
|
||||
/// when the GPS type has been detected.
|
||||
///
|
||||
AP_GPS_Auto(FastSerial *port, GPS **gps) :
|
||||
_port(port),
|
||||
_gps(gps)
|
||||
{};
|
||||
|
||||
void init(void);
|
||||
|
||||
/// Detect and initialise the attached GPS unit. Returns a
|
||||
/// pointer to the allocated & initialised GPS driver.
|
||||
///
|
||||
void update(void);
|
||||
|
||||
private:
|
||||
/// Serial port connected to the GPS.
|
||||
FastSerial *_port;
|
||||
|
||||
/// global GPS driver pointer, updated by auto-detection
|
||||
///
|
||||
GPS **_gps;
|
||||
|
||||
/// low-level auto-detect routine
|
||||
///
|
||||
GPS *_detect(void);
|
||||
|
||||
/// fetch a character from the port
|
||||
///
|
||||
int _getc(void);
|
||||
};
|
||||
#endif
|
@ -0,0 +1,224 @@
|
||||
// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: t -*-
|
||||
/*
|
||||
GPS_MTK.cpp - Ublox GPS library for Arduino
|
||||
Code by Jordi Muñoz and Jose Julio. DIYDrones.com
|
||||
This code works with boards based on ATMega168/328 and ATMega1280 (Serial port 1)
|
||||
|
||||
This library is free software; you can redistribute it and/or
|
||||
modify it under the terms of the GNU Lesser General Public
|
||||
License as published by the Free Software Foundation; either
|
||||
version 2.1 of the License, or (at your option) any later version.
|
||||
|
||||
GPS configuration : Costum protocol
|
||||
Baud rate : 38400
|
||||
|
||||
Methods:
|
||||
init() : GPS initialization
|
||||
update() : Call this funcion as often as you want to ensure you read the incomming gps data
|
||||
|
||||
Properties:
|
||||
lattitude : lattitude * 10000000 (long value)
|
||||
longitude : longitude * 10000000 (long value)
|
||||
altitude : altitude * 100 (meters) (long value)
|
||||
ground_speed : Speed (m/s) * 100 (long value)
|
||||
ground_course : Course (degrees) * 100 (long value)
|
||||
new_data : 1 when a new data is received.
|
||||
You need to write a 0 to new_data when you read the data
|
||||
fix : 1: GPS NO fix, 2: 2D fix, 3: 3D fix.
|
||||
|
||||
*/
|
||||
#include "AP_GPS_IMU.h"
|
||||
#include "WProgram.h"
|
||||
|
||||
|
||||
// Constructors ////////////////////////////////////////////////////////////////
|
||||
AP_GPS_IMU::AP_GPS_IMU(Stream *s) : GPS(s)
|
||||
{
|
||||
}
|
||||
|
||||
|
||||
// Public Methods //////////////////////////////////////////////////////////////
|
||||
void AP_GPS_IMU::init(void)
|
||||
{
|
||||
// we expect the stream to already be open at the corret bitrate
|
||||
}
|
||||
|
||||
// optimization : This code doesn't wait for data. It only proccess the data available.
|
||||
// We can call this function on the main loop (50Hz loop)
|
||||
// If we get a complete packet this function calls parse_IMU_gps() to parse and update the GPS info.
|
||||
void AP_GPS_IMU::update(void)
|
||||
{
|
||||
byte data;
|
||||
int numc = 0;
|
||||
static byte message_num = 0;
|
||||
|
||||
numc = _port->available();
|
||||
|
||||
if (numc > 0){
|
||||
for (int i=0;i<numc;i++){ // Process bytes received
|
||||
|
||||
data = _port->read();
|
||||
|
||||
switch(step){ //Normally we start from zero. This is a state machine
|
||||
case 0:
|
||||
if(data == 0x44) // IMU sync char 1
|
||||
step++; //OH first data packet is correct, so jump to the next step
|
||||
break;
|
||||
|
||||
case 1:
|
||||
if(data == 0x49) // IMU sync char 2
|
||||
step++; //ooh! The second data packet is correct, jump to the step 2
|
||||
else
|
||||
step=0; //Nop, is not correct so restart to step zero and try again.
|
||||
break;
|
||||
|
||||
case 2:
|
||||
if(data == 0x59) // IMU sync char 3
|
||||
step++; //ooh! The second data packet is correct, jump to the step 2
|
||||
else
|
||||
step=0; //Nop, is not correct so restart to step zero and try again.
|
||||
break;
|
||||
|
||||
case 3:
|
||||
if(data == 0x64) // IMU sync char 4
|
||||
step++; //ooh! The second data packet is correct, jump to the step 2
|
||||
else
|
||||
step=0; //Nop, is not correct so restart to step zero and try again.
|
||||
break;
|
||||
|
||||
case 4:
|
||||
payload_length = data;
|
||||
checksum(payload_length);
|
||||
step++;
|
||||
if (payload_length > 28){
|
||||
step = 0; //Bad data, so restart to step zero and try again.
|
||||
payload_counter = 0;
|
||||
ck_a = 0;
|
||||
ck_b = 0;
|
||||
//payload_error_count++;
|
||||
}
|
||||
break;
|
||||
|
||||
case 5:
|
||||
message_num = data;
|
||||
checksum(data);
|
||||
step++;
|
||||
break;
|
||||
|
||||
case 6: // Payload data read...
|
||||
// We stay in this state until we reach the payload_length
|
||||
buffer[payload_counter] = data;
|
||||
checksum(data);
|
||||
payload_counter++;
|
||||
if (payload_counter >= payload_length) {
|
||||
step++;
|
||||
}
|
||||
break;
|
||||
|
||||
case 7:
|
||||
GPS_ck_a = data; // First checksum byte
|
||||
step++;
|
||||
break;
|
||||
|
||||
case 8:
|
||||
GPS_ck_b = data; // Second checksum byte
|
||||
|
||||
// We end the IMU/GPS read...
|
||||
// Verify the received checksum with the generated checksum..
|
||||
if((ck_a == GPS_ck_a) && (ck_b == GPS_ck_b)) {
|
||||
if (message_num == 0x02) {
|
||||
join_data();
|
||||
} else if (message_num == 0x03) {
|
||||
GPS_join_data();
|
||||
} else if (message_num == 0x04) {
|
||||
join_data_xplane();
|
||||
} else if (message_num == 0x0a) {
|
||||
//PERF_join_data();
|
||||
} else {
|
||||
_error("Invalid message number = %d\n", (int)message_num);
|
||||
}
|
||||
} else {
|
||||
_error("XXX Checksum error\n"); //bad checksum
|
||||
//imu_checksum_error_count++;
|
||||
}
|
||||
// Variable initialization
|
||||
step = 0;
|
||||
payload_counter = 0;
|
||||
ck_a = 0;
|
||||
ck_b = 0;
|
||||
break;
|
||||
}
|
||||
}// End for...
|
||||
}
|
||||
}
|
||||
|
||||
/****************************************************************
|
||||
*
|
||||
****************************************************************/
|
||||
|
||||
void AP_GPS_IMU::join_data(void)
|
||||
{
|
||||
//Verifing if we are in class 1, you can change this "IF" for a "Switch" in case you want to use other IMU classes..
|
||||
//In this case all the message im using are in class 1, to know more about classes check PAGE 60 of DataSheet.
|
||||
|
||||
//Storing IMU roll
|
||||
roll_sensor = *(int *)&buffer[0];
|
||||
|
||||
//Storing IMU pitch
|
||||
pitch_sensor = *(int *)&buffer[2];
|
||||
|
||||
//Storing IMU heading (yaw)
|
||||
ground_course = *(int *)&buffer[4];
|
||||
imu_ok = true;
|
||||
}
|
||||
|
||||
void AP_GPS_IMU::join_data_xplane()
|
||||
{
|
||||
//Storing IMU roll
|
||||
roll_sensor = *(int *)&buffer[0];
|
||||
|
||||
|
||||
//Storing IMU pitch
|
||||
pitch_sensor = *(int *)&buffer[2];
|
||||
|
||||
//Storing IMU heading (yaw)
|
||||
ground_course = *(unsigned int *)&buffer[4];
|
||||
|
||||
//Storing airspeed
|
||||
airspeed = *(int *)&buffer[6];
|
||||
|
||||
imu_ok = true;
|
||||
|
||||
}
|
||||
|
||||
void AP_GPS_IMU::GPS_join_data(void)
|
||||
{
|
||||
longitude = *(long *)&buffer[0]; // degrees * 10e7
|
||||
latitude = *(long *)&buffer[4];
|
||||
|
||||
//Storing GPS Height above the sea level
|
||||
altitude = (long)*(int *)&buffer[8] * 10;
|
||||
|
||||
//Storing Speed
|
||||
speed_3d = ground_speed = (float)*(int *)&buffer[10];
|
||||
|
||||
//We skip the gps ground course because we use yaw value from the IMU for ground course
|
||||
time = *(long *)&buffer[14];
|
||||
|
||||
imu_health = buffer[15];
|
||||
|
||||
new_data = true;
|
||||
fix = true;
|
||||
}
|
||||
|
||||
|
||||
|
||||
/****************************************************************
|
||||
*
|
||||
****************************************************************/
|
||||
// checksum algorithm
|
||||
void AP_GPS_IMU::checksum(byte data)
|
||||
{
|
||||
ck_a += data;
|
||||
ck_b += ck_a;
|
||||
}
|
@ -0,0 +1,44 @@
|
||||
// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: t -*-
|
||||
#ifndef AP_GPS_IMU_h
|
||||
#define AP_GPS_IMU_h
|
||||
|
||||
#include <GPS.h>
|
||||
#define MAXPAYLOAD 32
|
||||
|
||||
class AP_GPS_IMU : public GPS
|
||||
{
|
||||
public:
|
||||
|
||||
// Methods
|
||||
AP_GPS_IMU(Stream *s);
|
||||
void init();
|
||||
void update();
|
||||
|
||||
// Properties
|
||||
long roll_sensor; // how much we're turning in deg * 100
|
||||
long pitch_sensor; // our angle of attack in deg * 100
|
||||
int airspeed;
|
||||
float imu_health;
|
||||
uint8_t imu_ok;
|
||||
|
||||
private:
|
||||
// Packet checksums
|
||||
uint8_t ck_a;
|
||||
uint8_t ck_b;
|
||||
uint8_t GPS_ck_a;
|
||||
uint8_t GPS_ck_b;
|
||||
|
||||
uint8_t step;
|
||||
uint8_t msg_class;
|
||||
uint8_t message_num;
|
||||
uint8_t payload_length;
|
||||
uint8_t payload_counter;
|
||||
uint8_t buffer[MAXPAYLOAD];
|
||||
|
||||
void join_data();
|
||||
void join_data_xplane();
|
||||
void GPS_join_data();
|
||||
void checksum(unsigned char data);
|
||||
};
|
||||
|
||||
#endif
|
@ -0,0 +1,146 @@
|
||||
// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: t -*-
|
||||
//
|
||||
// DIYDrones Custom Mediatek GPS driver for ArduPilot and ArduPilotMega.
|
||||
// Code by Michael Smith, Jordi Munoz and Jose Julio, DIYDrones.com
|
||||
//
|
||||
// This library is free software; you can redistribute it and / or
|
||||
// modify it under the terms of the GNU Lesser General Public
|
||||
// License as published by the Free Software Foundation; either
|
||||
// version 2.1 of the License, or (at your option) any later version.
|
||||
//
|
||||
// GPS configuration : Custom protocol per "DIYDrones Custom Binary Sentence Specification V1.1"
|
||||
//
|
||||
|
||||
#include "AP_GPS_MTK.h"
|
||||
#include "WProgram.h"
|
||||
|
||||
// Constructors ////////////////////////////////////////////////////////////////
|
||||
AP_GPS_MTK::AP_GPS_MTK(Stream *s) : GPS(s)
|
||||
{
|
||||
}
|
||||
|
||||
// Public Methods //////////////////////////////////////////////////////////////
|
||||
void AP_GPS_MTK::init(void)
|
||||
{
|
||||
_port->flush();
|
||||
// initialize serial port for binary protocol use
|
||||
// XXX should assume binary, let GPS_AUTO handle dynamic config?
|
||||
_port->print(MTK_SET_BINARY);
|
||||
|
||||
// set 4Hz update rate
|
||||
_port->print(MTK_OUTPUT_4HZ);
|
||||
}
|
||||
|
||||
// Process bytes available from the stream
|
||||
//
|
||||
// The stream is assumed to contain only our custom message. If it
|
||||
// contains other messages, and those messages contain the preamble bytes,
|
||||
// it is possible for this code to become de-synchronised. Without
|
||||
// buffering the entire message and re-processing it from the top,
|
||||
// this is unavoidable.
|
||||
//
|
||||
// The lack of a standard header length field makes it impossible to skip
|
||||
// unrecognised messages.
|
||||
//
|
||||
void AP_GPS_MTK::update(void)
|
||||
{
|
||||
byte data;
|
||||
int numc;
|
||||
|
||||
numc = _port->available();
|
||||
for (int i = 0; i < numc; i++){ // Process bytes received
|
||||
|
||||
// read the next byte
|
||||
data = _port->read();
|
||||
|
||||
restart:
|
||||
switch(_step){
|
||||
|
||||
// Message preamble, class, ID detection
|
||||
//
|
||||
// If we fail to match any of the expected bytes, we
|
||||
// reset the state machine and re-consider the failed
|
||||
// byte as the first byte of the preamble. This
|
||||
// improves our chances of recovering from a mismatch
|
||||
// and makes it less likely that we will be fooled by
|
||||
// the preamble appearing as data in some other message.
|
||||
//
|
||||
case 0:
|
||||
if(PREAMBLE1 == data)
|
||||
_step++;
|
||||
break;
|
||||
case 1:
|
||||
if (PREAMBLE2 == data) {
|
||||
_step++;
|
||||
break;
|
||||
}
|
||||
_step = 0;
|
||||
goto restart;
|
||||
case 2:
|
||||
if (MESSAGE_CLASS == data) {
|
||||
_step++;
|
||||
_ck_b = _ck_a = data; // reset the checksum accumulators
|
||||
} else {
|
||||
_step = 0; // reset and wait for a message of the right class
|
||||
goto restart;
|
||||
}
|
||||
break;
|
||||
case 3:
|
||||
if (MESSAGE_ID == data) {
|
||||
_step++;
|
||||
_ck_b += (_ck_a += data);
|
||||
_payload_length = sizeof(diyd_mtk_msg); // prepare to receive our message
|
||||
_payload_counter = 0;
|
||||
} else {
|
||||
_step = 0;
|
||||
goto restart;
|
||||
}
|
||||
break;
|
||||
|
||||
// Receive message data
|
||||
//
|
||||
case 4:
|
||||
_buffer.bytes[_payload_counter++] = data;
|
||||
_ck_b += (_ck_a += data);
|
||||
if (_payload_counter == _payload_length)
|
||||
_step++;
|
||||
break;
|
||||
|
||||
// Checksum and message processing
|
||||
//
|
||||
case 5:
|
||||
_step++;
|
||||
if (_ck_a != data) {
|
||||
_error("GPS_MTK: checksum error\n");
|
||||
_step = 0;
|
||||
}
|
||||
break;
|
||||
case 6:
|
||||
_step = 0;
|
||||
if (_ck_b != data) {
|
||||
_error("GPS_MTK: checksum error\n");
|
||||
break;
|
||||
}
|
||||
_parse_gps(); // Parse the new GPS packet
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// Private Methods
|
||||
void
|
||||
AP_GPS_MTK::_parse_gps(void)
|
||||
{
|
||||
fix = (_buffer.msg.fix_type == FIX_3D);
|
||||
latitude = _swapl(&_buffer.msg.latitude) * 10;
|
||||
longitude = _swapl(&_buffer.msg.longitude) * 10;
|
||||
altitude = _swapl(&_buffer.msg.altitude);
|
||||
ground_speed = _swapl(&_buffer.msg.ground_speed);
|
||||
ground_course = _swapl(&_buffer.msg.ground_course) / 10000;
|
||||
num_sats = _buffer.msg.satellites;
|
||||
|
||||
// XXX docs say this is UTC, but our clients expect msToW
|
||||
time = _swapl(&_buffer.msg.utc_time);
|
||||
_setTime();
|
||||
valid_read = true;
|
||||
new_data = true;
|
||||
}
|
@ -0,0 +1,87 @@
|
||||
// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: t -*-
|
||||
//
|
||||
// DIYDrones Custom Mediatek GPS driver for ArduPilot and ArduPilotMega.
|
||||
// Code by Michael Smith, Jordi Munoz and Jose Julio, DIYDrones.com
|
||||
//
|
||||
// This library is free software; you can redistribute it and / or
|
||||
// modify it under the terms of the GNU Lesser General Public
|
||||
// License as published by the Free Software Foundation; either
|
||||
// version 2.1 of the License, or (at your option) any later version.
|
||||
//
|
||||
// GPS configuration : Custom protocol per "DIYDrones Custom Binary Sentence Specification V1.1"
|
||||
//
|
||||
#ifndef AP_GPS_MTK_h
|
||||
#define AP_GPS_MTK_h
|
||||
|
||||
#include <GPS.h>
|
||||
#define MAXPAYLOAD 32
|
||||
|
||||
#define MTK_SET_BINARY "$PGCMD,16,0,0,0,0,0*6A\r\n"
|
||||
#define MTK_SET_NMEA "$PGCMD,16,1,1,1,1,1*6B\r\n"
|
||||
|
||||
#define MTK_OUTPUT_1HZ "$PMTK220,1000*1F\r\n"
|
||||
#define MTK_OUTPUT_2HZ "$PMTK220,500*2B\r\n"
|
||||
#define MTK_OUTPUT_4HZ "$PMTK220,250*29\r\n"
|
||||
#define MTK_OTUPUT_5HZ "$PMTK220,200*2C\r\n"
|
||||
#define MTK_OUTPUT_10HZ "$PMTK220,100*2F\r\n"
|
||||
|
||||
#define MTK_BAUD_RATE_38400 "$PMTK251,38400*27\r\n"
|
||||
|
||||
#define SBAS_ON "$PMTK313,1*2E\r\n"
|
||||
#define SBAS_OFF "$PMTK313,0*2F\r\n"
|
||||
|
||||
#define WAAS_ON "$PSRF151,1*3F\r\n"
|
||||
#define WAAS_OFF "$PSRF151,0*3E\r\n"
|
||||
|
||||
class AP_GPS_MTK : public GPS {
|
||||
public:
|
||||
AP_GPS_MTK(Stream *s);
|
||||
virtual void init(void);
|
||||
virtual void update(void);
|
||||
|
||||
private:
|
||||
#pragma pack(1)
|
||||
struct diyd_mtk_msg {
|
||||
int32_t latitude;
|
||||
int32_t longitude;
|
||||
int32_t altitude;
|
||||
int32_t ground_speed;
|
||||
int32_t ground_course;
|
||||
uint8_t satellites;
|
||||
uint8_t fix_type;
|
||||
uint32_t utc_time;
|
||||
};
|
||||
#pragma pack(pop)
|
||||
enum diyd_mtk_fix_type {
|
||||
FIX_NONE = 1,
|
||||
FIX_2D = 2,
|
||||
FIX_3D = 3
|
||||
};
|
||||
|
||||
enum diyd_mtk_protocol_bytes {
|
||||
PREAMBLE1 = 0xb5,
|
||||
PREAMBLE2 = 0x62,
|
||||
MESSAGE_CLASS = 1,
|
||||
MESSAGE_ID = 5
|
||||
};
|
||||
|
||||
// Packet checksum accumulators
|
||||
uint8_t _ck_a;
|
||||
uint8_t _ck_b;
|
||||
|
||||
// State machine state
|
||||
uint8_t _step;
|
||||
uint8_t _payload_length;
|
||||
uint8_t _payload_counter;
|
||||
|
||||
// Receive buffer
|
||||
union {
|
||||
diyd_mtk_msg msg;
|
||||
uint8_t bytes[];
|
||||
} _buffer;
|
||||
|
||||
// Buffer parse & GPS state update
|
||||
void _parse_gps();
|
||||
};
|
||||
|
||||
#endif // AP_GPS_MTK_H
|
@ -0,0 +1,246 @@
|
||||
// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: t -*-
|
||||
/*
|
||||
GPS_NMEA.cpp - Generic NMEA GPS library for Arduino
|
||||
Code by Jordi Muñoz and Jose Julio. DIYDrones.com
|
||||
Edits by HappyKillmore
|
||||
This code works with boards based on ATMega168 / 328 and ATMega1280 (Serial port 1)
|
||||
|
||||
This library is free software; you can redistribute it and / or
|
||||
modify it under the terms of the GNU Lesser General Public
|
||||
License as published by the Free Software Foundation; either
|
||||
version 2.1 of the License, or (at your option) any later version.
|
||||
|
||||
GPS configuration : NMEA protocol
|
||||
Baud rate : 38400
|
||||
NMEA Sentences :
|
||||
$GPGGA : Global Positioning System fix Data
|
||||
$GPVTG : Ttack and Ground Speed
|
||||
|
||||
Methods:
|
||||
init() : GPS Initialization
|
||||
update() : Call this funcion as often as you want to ensure you read the incomming gps data
|
||||
|
||||
Properties:
|
||||
latitude : latitude * 10000000 (long value)
|
||||
longitude : longitude * 10000000 (long value)
|
||||
altitude : altitude * 1000 (milimeters) (long value)
|
||||
ground_speed : Speed (m / s) * 100 (long value)
|
||||
ground_course : Course (degrees) * 100 (long value)
|
||||
Type : 2 (This indicate that we are using the Generic NMEA library)
|
||||
new_data : 1 when a new data is received.
|
||||
You need to write a 0 to new_data when you read the data
|
||||
fix : > = 1: GPS FIX, 0: No fix (normal logic)
|
||||
quality : 0 = No fix
|
||||
1 = Bad (Num sats < 5)
|
||||
2 = Poor
|
||||
3 = Medium
|
||||
4 = Good
|
||||
|
||||
NOTE : This code has been tested on a Locosys 20031 GPS receiver (MTK chipset)
|
||||
*/
|
||||
#include "AP_GPS_NMEA.h"
|
||||
#include "WProgram.h"
|
||||
|
||||
// Constructors ////////////////////////////////////////////////////////////////
|
||||
AP_GPS_NMEA::AP_GPS_NMEA(Stream *s) : GPS(s)
|
||||
{
|
||||
}
|
||||
|
||||
// Public Methods //////////////////////////////////////////////////////////////
|
||||
void
|
||||
AP_GPS_NMEA::init(void)
|
||||
{
|
||||
//Type = 2;
|
||||
// initialize serial port for binary protocol use
|
||||
_port->print(NMEA_OUTPUT_SENTENCES);
|
||||
_port->print(NMEA_OUTPUT_4HZ);
|
||||
_port->print(SBAS_ON);
|
||||
_port->print(DGPS_SBAS);
|
||||
_port->print(DATUM_GOOGLE);
|
||||
}
|
||||
|
||||
// This code don´t wait for data, only proccess the data available on serial port
|
||||
// We can call this function on the main loop (50Hz loop)
|
||||
// If we get a complete packet this function call parse_nmea_gps() to parse and update the GPS info.
|
||||
void
|
||||
AP_GPS_NMEA::update(void)
|
||||
{
|
||||
char c;
|
||||
int numc;
|
||||
int i;
|
||||
|
||||
numc = _port->available();
|
||||
|
||||
if (numc > 0){
|
||||
for (i = 0; i < numc; i++){
|
||||
c = _port->read();
|
||||
if (c == '$'){ // NMEA Start
|
||||
bufferidx = 0;
|
||||
buffer[bufferidx++] = c;
|
||||
GPS_checksum = 0;
|
||||
GPS_checksum_calc = true;
|
||||
continue;
|
||||
}
|
||||
if (c == '\r'){ // NMEA End
|
||||
buffer[bufferidx++] = 0;
|
||||
parse_nmea_gps();
|
||||
} else {
|
||||
if (bufferidx < (GPS_BUFFERSIZE - 1)){
|
||||
if (c == '*')
|
||||
GPS_checksum_calc = false; // Checksum calculation end
|
||||
buffer[bufferidx++] = c;
|
||||
if (GPS_checksum_calc){
|
||||
GPS_checksum ^= c; // XOR
|
||||
}
|
||||
} else {
|
||||
bufferidx = 0; // Buffer overflow : restart
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/****************************************************************
|
||||
*
|
||||
* * ** * ** * ** * ** * ** * ** * ** * ** * ** * ** * ** * ** * ** * ** * ** * ** * ** * ** * ** * ** * **/
|
||||
// Private Methods //////////////////////////////////////////////////////////////
|
||||
void
|
||||
AP_GPS_NMEA::parse_nmea_gps(void)
|
||||
{
|
||||
uint8_t NMEA_check;
|
||||
long aux_deg;
|
||||
long aux_min;
|
||||
char *parseptr;
|
||||
|
||||
if (strncmp(buffer,"$GPGGA",6)==0){ // Check if sentence begins with $GPGGA
|
||||
if (buffer[bufferidx-4]=='*'){ // Check for the "*" character
|
||||
NMEA_check = parseHex(buffer[bufferidx - 3]) * 16 + parseHex(buffer[bufferidx - 2]); // Read the checksums characters
|
||||
if (GPS_checksum == NMEA_check){ // Checksum validation
|
||||
//Serial.println("buffer");
|
||||
_setTime();
|
||||
valid_read = true;
|
||||
new_data = true; // New GPS Data
|
||||
parseptr = strchr(buffer, ',')+1;
|
||||
//parseptr = strchr(parseptr, ',')+1;
|
||||
time = parsenumber(parseptr, 2); // GPS UTC time hhmmss.ss
|
||||
parseptr = strchr(parseptr, ',')+1;
|
||||
aux_deg = parsedecimal(parseptr, 2); // degrees
|
||||
aux_min = parsenumber(parseptr + 2, 4); // minutes (sexagesimal) => Convert to decimal
|
||||
latitude = aux_deg * 10000000 + (aux_min * 50) / 3; // degrees + minutes / 0.6 ( * 10000000) (0.6 = 3 / 5)
|
||||
parseptr = strchr(parseptr, ',')+1;
|
||||
if ( * parseptr == 'S')
|
||||
latitude = -1 * latitude; // South latitudes are negative
|
||||
parseptr = strchr(parseptr, ',')+1;
|
||||
// W longitudes are Negative
|
||||
aux_deg = parsedecimal(parseptr, 3); // degrees
|
||||
aux_min = parsenumber(parseptr + 3, 4); // minutes (sexagesimal)
|
||||
longitude = aux_deg * 10000000 + (aux_min * 50) / 3; // degrees + minutes / 0.6 ( * 10000000)
|
||||
//longitude = -1*longitude; // This Assumes that we are in W longitudes...
|
||||
parseptr = strchr(parseptr, ',')+1;
|
||||
if ( * parseptr == 'W')
|
||||
longitude = -1 * longitude; // West longitudes are negative
|
||||
parseptr = strchr(parseptr, ',')+1;
|
||||
fix = parsedecimal(parseptr, 1);
|
||||
parseptr = strchr(parseptr, ',')+1;
|
||||
num_sats = parsedecimal(parseptr, 2);
|
||||
parseptr = strchr(parseptr, ',')+1;
|
||||
HDOP = parsenumber(parseptr, 1); // HDOP * 10
|
||||
parseptr = strchr(parseptr, ',')+1;
|
||||
altitude = parsenumber(parseptr, 1) * 100; // altitude in decimeters * 100 = milimeters
|
||||
if (fix < 1)
|
||||
quality = 0; // No FIX
|
||||
else if(num_sats < 5)
|
||||
quality = 1; // Bad (Num sats < 5)
|
||||
else if(HDOP > 30)
|
||||
quality = 2; // Poor (HDOP > 30)
|
||||
else if(HDOP > 25)
|
||||
quality = 3; // Medium (HDOP > 25)
|
||||
else
|
||||
quality = 4; // Good (HDOP < 25)
|
||||
} else {
|
||||
_error("GPSERR: Checksum error!!\n");
|
||||
}
|
||||
}
|
||||
} else if (strncmp(buffer,"$GPVTG",6)==0){ // Check if sentence begins with $GPVTG
|
||||
//Serial.println(buffer);
|
||||
if (buffer[bufferidx-4]=='*'){ // Check for the "*" character
|
||||
NMEA_check = parseHex(buffer[bufferidx - 3]) * 16 + parseHex(buffer[bufferidx - 2]); // Read the checksums characters
|
||||
if (GPS_checksum == NMEA_check){ // Checksum validation
|
||||
_setTime();
|
||||
valid_read = true;
|
||||
new_data = true; // New GPS Data
|
||||
parseptr = strchr(buffer, ',')+1;
|
||||
ground_course = parsenumber(parseptr, 2); // Ground course in degrees * 100
|
||||
parseptr = strchr(parseptr, ',')+1;
|
||||
parseptr = strchr(parseptr, ',')+1;
|
||||
parseptr = strchr(parseptr, ',')+1;
|
||||
parseptr = strchr(parseptr, ',')+1;
|
||||
parseptr = strchr(parseptr, ',')+1;
|
||||
parseptr = strchr(parseptr, ',')+1;
|
||||
ground_speed = parsenumber(parseptr, 2) * 10 / 36; // Convert Km / h to m / s ( * 100)
|
||||
//GPS_line = true;
|
||||
} else {
|
||||
_error("GPSERR: Checksum error!!\n");
|
||||
}
|
||||
}
|
||||
} else {
|
||||
bufferidx = 0;
|
||||
_error("GPSERR: Bad sentence!!\n");
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
// Parse hexadecimal numbers
|
||||
uint8_t
|
||||
AP_GPS_NMEA::parseHex(char c) {
|
||||
if (c < '0')
|
||||
return (0);
|
||||
if (c <= '9')
|
||||
return (c - '0');
|
||||
if (c < 'A')
|
||||
return (0);
|
||||
if (c <= 'F')
|
||||
return ((c - 'A')+10);
|
||||
}
|
||||
|
||||
// Decimal number parser
|
||||
long
|
||||
AP_GPS_NMEA::parsedecimal(char *str, uint8_t num_car) {
|
||||
long d = 0;
|
||||
uint8_t i;
|
||||
|
||||
i = num_car;
|
||||
while ((str[0] != 0) && (i > 0)) {
|
||||
if ((str[0] > '9') || (str[0] < '0'))
|
||||
return d;
|
||||
d *= 10;
|
||||
d += str[0] - '0';
|
||||
str++;
|
||||
i--;
|
||||
}
|
||||
return d;
|
||||
}
|
||||
|
||||
// Function to parse fixed point numbers (numdec=number of decimals)
|
||||
long
|
||||
AP_GPS_NMEA::parsenumber(char *str, uint8_t numdec) {
|
||||
long d = 0;
|
||||
uint8_t ndec = 0;
|
||||
|
||||
while (str[0] != 0) {
|
||||
if (str[0] == '.'){
|
||||
ndec = 1;
|
||||
} else {
|
||||
if ((str[0] > '9') || (str[0] < '0'))
|
||||
return d;
|
||||
d *= 10;
|
||||
d += str[0] - '0';
|
||||
if (ndec > 0)
|
||||
ndec++;
|
||||
if (ndec > numdec) // we reach the number of decimals...
|
||||
return d;
|
||||
}
|
||||
str++;
|
||||
}
|
||||
return d;
|
||||
}
|
@ -0,0 +1,60 @@
|
||||
// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: t -*-
|
||||
#ifndef AP_GPS_NMEA_h
|
||||
#define AP_GPS_NMEA_h
|
||||
|
||||
#include <GPS.h>
|
||||
#define GPS_BUFFERSIZE 120
|
||||
|
||||
#define NMEA_OUTPUT_SENTENCES "$PMTK314,0,0,1,1,0,0,0,0,0,0,0,0,0,0,0,0,0*28\r\n" //Set GPGGA and GPVTG
|
||||
|
||||
#define NMEA_BAUD_RATE_4800 "$PSRF100,1,4800,8,1,0*0E\r\n"
|
||||
#define NMEA_BAUD_RATE_9600 "$PSRF100,1,9600,8,1,0*0D\r\n"
|
||||
#define NMEA_BAUD_RATE_19200 "$PSRF100,1,19200,8,1,0*38\r\n"
|
||||
#define NMEA_BAUD_RATE_38400 "$PSRF100,1,38400,8,1,0*3D\r\n"
|
||||
#define NMEA_BAUD_RATE_57600 "$PSRF100,1,57600,8,1,0*36\r\n"
|
||||
|
||||
#define NMEA_OUTPUT_1HZ "$PMTK220,1000*1F\r\n"
|
||||
#define NMEA_OUTPUT_2HZ "$PMTK220,500*2B\r\n"
|
||||
#define NMEA_OUTPUT_4HZ "$PMTK220,250*29\r\n"
|
||||
#define NMEA_OTUPUT_5HZ "$PMTK220,200*2C\r\n"
|
||||
#define NMEA_OUTPUT_10HZ "$PMTK220,100*2F\r\n"
|
||||
|
||||
#define SBAS_ON "$PMTK313,1*2E\r\n"
|
||||
#define SBAS_OFF "$PMTK313,0*2F\r\n"
|
||||
|
||||
#define WAAS_ON "$PSRF151,1*3F\r\n"
|
||||
#define WAAS_OFF "$PSRF151,0*3E\r\n"
|
||||
|
||||
#define DGPS_OFF "$PMTK301,0*2C\r\n"
|
||||
#define DGPS_RTCM "$PMTK301,1*2D\r\n"
|
||||
#define DGPS_SBAS "$PMTK301,2*2E\r\n"
|
||||
|
||||
#define DATUM_GOOGLE "$PMTK330,0*2E\r\n"
|
||||
|
||||
class AP_GPS_NMEA : public GPS
|
||||
{
|
||||
public:
|
||||
// Methods
|
||||
AP_GPS_NMEA(Stream *s);
|
||||
void init();
|
||||
void update();
|
||||
|
||||
// Properties
|
||||
uint8_t quality; // GPS Signal quality
|
||||
int HDOP; // HDOP
|
||||
|
||||
private:
|
||||
// Internal variables
|
||||
uint8_t GPS_checksum;
|
||||
uint8_t GPS_checksum_calc;
|
||||
char buffer[GPS_BUFFERSIZE];
|
||||
int bufferidx;
|
||||
|
||||
void parse_nmea_gps(void);
|
||||
uint8_t parseHex(char c);
|
||||
long parsedecimal(char *str,uint8_t num_car);
|
||||
long parsenumber(char *str,uint8_t numdec);
|
||||
|
||||
};
|
||||
|
||||
#endif
|
@ -0,0 +1,13 @@
|
||||
// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: t -*-
|
||||
|
||||
#ifndef AP_GPS_None_h
|
||||
#define AP_GPS_None_h
|
||||
|
||||
#include <GPS.h>
|
||||
|
||||
class AP_GPS_None : public GPS
|
||||
{
|
||||
virtual void init(void) {};
|
||||
virtual void update(void) {};
|
||||
};
|
||||
#endif
|
@ -0,0 +1,193 @@
|
||||
// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: t -*-
|
||||
//
|
||||
// SiRF Binary GPS driver for ArduPilot and ArduPilotMega.
|
||||
// Code by Michael Smith.
|
||||
//
|
||||
// This library is free software; you can redistribute it and / or
|
||||
// modify it under the terms of the GNU Lesser General Public
|
||||
// License as published by the Free Software Foundation; either
|
||||
// version 2.1 of the License, or (at your option) any later version.
|
||||
//
|
||||
|
||||
#include "AP_GPS_SIRF.h"
|
||||
#include "WProgram.h"
|
||||
|
||||
// Initialisation messages
|
||||
//
|
||||
// Turn off all messages except for 0x29.
|
||||
//
|
||||
// XXX the bytes show up on the wire, but at least my test unit (EM-411) seems to ignore them.
|
||||
//
|
||||
static uint8_t init_messages[] = {
|
||||
0xa0, 0xa2, 0x00, 0x08, 0xa6, 0x02, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xa8, 0xb0, 0xb3,
|
||||
0xa0, 0xa2, 0x00, 0x08, 0xa6, 0x00, 0x29, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0xd0, 0xb0, 0xb3
|
||||
};
|
||||
|
||||
// Constructors ////////////////////////////////////////////////////////////////
|
||||
AP_GPS_SIRF::AP_GPS_SIRF(Stream *s) : GPS(s)
|
||||
{
|
||||
}
|
||||
|
||||
// Public Methods //////////////////////////////////////////////////////////////
|
||||
void AP_GPS_SIRF::init(void)
|
||||
{
|
||||
_port->flush();
|
||||
|
||||
// For modules that default to something other than SiRF binary,
|
||||
// the module-specific subclass should take care of switching to binary mode
|
||||
// before calling us.
|
||||
|
||||
// send SiRF binary setup messages
|
||||
_port->write(init_messages, sizeof(init_messages));
|
||||
}
|
||||
|
||||
// Process bytes available from the stream
|
||||
//
|
||||
// The stream is assumed to contain only messages we recognise. If it
|
||||
// contains other messages, and those messages contain the preamble
|
||||
// bytes, it is possible for this code to fail to synchronise to the
|
||||
// stream immediately. Without buffering the entire message and
|
||||
// re-processing it from the top, this is unavoidable. The parser
|
||||
// attempts to avoid this when possible.
|
||||
//
|
||||
void AP_GPS_SIRF::update(void)
|
||||
{
|
||||
byte data;
|
||||
int numc;
|
||||
|
||||
numc = _port->available();
|
||||
while(numc--) {
|
||||
|
||||
// read the next byte
|
||||
data = _port->read();
|
||||
|
||||
switch(_step){
|
||||
|
||||
// Message preamble detection
|
||||
//
|
||||
// If we fail to match any of the expected bytes, we reset
|
||||
// the state machine and re-consider the failed byte as
|
||||
// the first byte of the preamble. This improves our
|
||||
// chances of recovering from a mismatch and makes it less
|
||||
// likely that we will be fooled by the preamble appearing
|
||||
// as data in some other message.
|
||||
//
|
||||
case 1:
|
||||
if (PREAMBLE2 == data) {
|
||||
_step++;
|
||||
break;
|
||||
}
|
||||
_step = 0;
|
||||
// FALLTHROUGH
|
||||
case 0:
|
||||
if(PREAMBLE1 == data)
|
||||
_step++;
|
||||
break;
|
||||
|
||||
// Message length
|
||||
//
|
||||
// We always collect the length so that we can avoid being
|
||||
// fooled by preamble bytes in messages.
|
||||
//
|
||||
case 2:
|
||||
_step++;
|
||||
_payload_length = (uint16_t)data << 8;
|
||||
break;
|
||||
case 3:
|
||||
_step++;
|
||||
_payload_length |= data;
|
||||
_payload_counter = 0;
|
||||
_checksum = 0;
|
||||
break;
|
||||
|
||||
// Message header processing
|
||||
//
|
||||
// We sniff the message ID to determine whether we are going
|
||||
// to gather the message bytes or just discard them.
|
||||
//
|
||||
case 4:
|
||||
_step++;
|
||||
_accumulate(data);
|
||||
_payload_length--;
|
||||
_gather = false;
|
||||
switch(data) {
|
||||
case MSG_GEONAV:
|
||||
if (_payload_length == sizeof(sirf_geonav)) {
|
||||
_gather = true;
|
||||
_msg_id = data;
|
||||
}
|
||||
break;
|
||||
}
|
||||
break;
|
||||
|
||||
// Receive message data
|
||||
//
|
||||
// Note that we are effectively guaranteed by the protocol
|
||||
// that the checksum and postamble cannot be mistaken for
|
||||
// the preamble, so if we are discarding bytes in this
|
||||
// message when the payload is done we return directly
|
||||
// to the preamble detector rather than bothering with
|
||||
// the checksum logic.
|
||||
//
|
||||
case 5:
|
||||
if (_gather) { // gather data if requested
|
||||
_accumulate(data);
|
||||
_buffer.bytes[_payload_counter] = data;
|
||||
if (++_payload_counter == _payload_length)
|
||||
_step++;
|
||||
} else {
|
||||
if (++_payload_counter == _payload_length)
|
||||
_step = 0;
|
||||
}
|
||||
break;
|
||||
|
||||
// Checksum and message processing
|
||||
//
|
||||
case 6:
|
||||
_step++;
|
||||
if ((_checksum >> 8) != data) {
|
||||
_error("GPS_SIRF: checksum error\n");
|
||||
_step = 0;
|
||||
}
|
||||
break;
|
||||
case 7:
|
||||
_step = 0;
|
||||
if ((_checksum & 0xff) != data) {
|
||||
_error("GPS_SIRF: checksum error\n");
|
||||
break;
|
||||
}
|
||||
if (_gather) {
|
||||
_parse_gps(); // Parse the new GPS packet
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void
|
||||
AP_GPS_SIRF::_parse_gps(void)
|
||||
{
|
||||
switch(_msg_id) {
|
||||
case MSG_GEONAV:
|
||||
time = _swapl(&_buffer.nav.time);
|
||||
//fix = (0 == _buffer.nav.fix_invalid) && (FIX_3D == (_buffer.nav.fix_type & FIX_MASK));
|
||||
fix = (0 == _buffer.nav.fix_invalid);
|
||||
latitude = _swapl(&_buffer.nav.latitude);
|
||||
longitude = _swapl(&_buffer.nav.longitude);
|
||||
altitude = _swapl(&_buffer.nav.altitude_msl);
|
||||
ground_speed = _swapi(&_buffer.nav.ground_speed);
|
||||
// at low speeds, ground course wanders wildly; suppress changes if we are not moving
|
||||
if (ground_speed > 50)
|
||||
ground_course = _swapi(&_buffer.nav.ground_course);
|
||||
num_sats = _buffer.nav.satellites;
|
||||
_setTime();
|
||||
valid_read = 1;
|
||||
break;
|
||||
}
|
||||
new_data = true;
|
||||
}
|
||||
|
||||
void
|
||||
AP_GPS_SIRF::_accumulate(uint8_t val)
|
||||
{
|
||||
_checksum = (_checksum + val) & 0x7fff;
|
||||
}
|
@ -0,0 +1,96 @@
|
||||
// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: t -*-
|
||||
//
|
||||
// SiRF Binary GPS driver for ArduPilot and ArduPilotMega.
|
||||
// Code by Michael Smith.
|
||||
//
|
||||
// This library is free software; you can redistribute it and / or
|
||||
// modify it under the terms of the GNU Lesser General Public
|
||||
// License as published by the Free Software Foundation; either
|
||||
// version 2.1 of the License, or (at your option) any later version.
|
||||
//
|
||||
#ifndef AP_GPS_SIRF_h
|
||||
#define AP_GPS_SIRF_h
|
||||
|
||||
#include <GPS.h>
|
||||
|
||||
#define SIRF_SET_BINARY "$PSRF100,0,38400,8,1,0*3C"
|
||||
|
||||
class AP_GPS_SIRF : public GPS {
|
||||
public:
|
||||
AP_GPS_SIRF(Stream *s);
|
||||
|
||||
virtual void init();
|
||||
virtual void update();
|
||||
|
||||
private:
|
||||
#pragma pack(1)
|
||||
struct sirf_geonav {
|
||||
uint16_t fix_invalid;
|
||||
uint16_t fix_type;
|
||||
uint16_t week;
|
||||
uint32_t time;
|
||||
uint16_t year;
|
||||
uint8_t month;
|
||||
uint8_t day;
|
||||
uint8_t hour;
|
||||
uint8_t minute;
|
||||
uint16_t second;
|
||||
uint32_t satellites_used;
|
||||
int32_t latitude;
|
||||
int32_t longitude;
|
||||
int32_t altitude_ellipsoid;
|
||||
int32_t altitude_msl;
|
||||
int8_t map_datum;
|
||||
int16_t ground_speed;
|
||||
int16_t ground_course;
|
||||
int16_t res1;
|
||||
int16_t climb_rate;
|
||||
uint16_t heading_rate;
|
||||
uint32_t horizontal_position_error;
|
||||
uint32_t vertical_position_error;
|
||||
uint32_t time_error;
|
||||
int16_t horizontal_velocity_error;
|
||||
int32_t clock_bias;
|
||||
uint32_t clock_bias_error;
|
||||
int32_t clock_drift;
|
||||
uint32_t clock_drift_error;
|
||||
uint32_t distance;
|
||||
uint16_t distance_error;
|
||||
uint16_t heading_error;
|
||||
uint8_t satellites;
|
||||
uint8_t hdop;
|
||||
uint8_t mode_info;
|
||||
};
|
||||
#pragma pack(pop)
|
||||
enum sirf_protocol_bytes {
|
||||
PREAMBLE1 = 0xa0,
|
||||
PREAMBLE2 = 0xa2,
|
||||
POSTAMBLE1 = 0xb0,
|
||||
POSTAMBLE2 = 0xb3,
|
||||
MSG_GEONAV = 0x29
|
||||
};
|
||||
enum sirf_fix_type {
|
||||
FIX_3D = 0x6,
|
||||
FIX_MASK = 0x7
|
||||
};
|
||||
|
||||
|
||||
// State machine state
|
||||
uint8_t _step;
|
||||
uint16_t _checksum;
|
||||
bool _gather;
|
||||
uint16_t _payload_length;
|
||||
uint16_t _payload_counter;
|
||||
uint8_t _msg_id;
|
||||
|
||||
// Message buffer
|
||||
union {
|
||||
sirf_geonav nav;
|
||||
uint8_t bytes[];
|
||||
} _buffer;
|
||||
|
||||
void _parse_gps(void);
|
||||
void _accumulate(uint8_t val);
|
||||
};
|
||||
|
||||
#endif // AP_GPS_SIRF_h
|
@ -0,0 +1,192 @@
|
||||
// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: t -*-
|
||||
//
|
||||
// u-blox UBX GPS driver for ArduPilot and ArduPilotMega.
|
||||
// Code by Michael Smith, Jordi Munoz and Jose Julio, DIYDrones.com
|
||||
//
|
||||
// This library is free software; you can redistribute it and / or
|
||||
// modify it under the terms of the GNU Lesser General Public
|
||||
// License as published by the Free Software Foundation; either
|
||||
// version 2.1 of the License, or (at your option) any later version.
|
||||
//
|
||||
|
||||
#include "AP_GPS_UBLOX.h"
|
||||
#include "WProgram.h"
|
||||
|
||||
// Constructors ////////////////////////////////////////////////////////////////
|
||||
|
||||
AP_GPS_UBLOX::AP_GPS_UBLOX(Stream *s) : GPS(s)
|
||||
{
|
||||
}
|
||||
|
||||
// Public Methods //////////////////////////////////////////////////////////////
|
||||
|
||||
void AP_GPS_UBLOX::init(void)
|
||||
{
|
||||
// XXX it might make sense to send some CFG_MSG,CFG_NMEA messages to get the
|
||||
// right reporting configuration.
|
||||
|
||||
_port->flush();
|
||||
}
|
||||
|
||||
// Process bytes available from the stream
|
||||
//
|
||||
// The stream is assumed to contain only messages we recognise. If it
|
||||
// contains other messages, and those messages contain the preamble
|
||||
// bytes, it is possible for this code to fail to synchronise to the
|
||||
// stream immediately. Without buffering the entire message and
|
||||
// re-processing it from the top, this is unavoidable. The parser
|
||||
// attempts to avoid this when possible.
|
||||
//
|
||||
void AP_GPS_UBLOX::update(void)
|
||||
{
|
||||
byte data;
|
||||
int numc;
|
||||
|
||||
numc = _port->available();
|
||||
for (int i = 0; i < numc; i++){ // Process bytes received
|
||||
|
||||
// read the next byte
|
||||
data = _port->read();
|
||||
|
||||
switch(_step){
|
||||
|
||||
// Message preamble detection
|
||||
//
|
||||
// If we fail to match any of the expected bytes, we reset
|
||||
// the state machine and re-consider the failed byte as
|
||||
// the first byte of the preamble. This improves our
|
||||
// chances of recovering from a mismatch and makes it less
|
||||
// likely that we will be fooled by the preamble appearing
|
||||
// as data in some other message.
|
||||
//
|
||||
case 1:
|
||||
if (PREAMBLE2 == data) {
|
||||
_step++;
|
||||
break;
|
||||
}
|
||||
_step = 0;
|
||||
// FALLTHROUGH
|
||||
case 0:
|
||||
if(PREAMBLE1 == data)
|
||||
_step++;
|
||||
break;
|
||||
|
||||
// Message header processing
|
||||
//
|
||||
// We sniff the class and message ID to decide whether we
|
||||
// are going to gather the message bytes or just discard
|
||||
// them.
|
||||
//
|
||||
// We always collect the length so that we can avoid being
|
||||
// fooled by preamble bytes in messages.
|
||||
//
|
||||
case 2:
|
||||
_step++;
|
||||
if (CLASS_NAV == data) {
|
||||
_gather = true; // class is interesting, maybe gather
|
||||
_ck_b = _ck_a = data; // reset the checksum accumulators
|
||||
} else {
|
||||
_error("ignoring class 0x%x\n", (int)data);
|
||||
_gather = false; // class is not interesting, discard
|
||||
}
|
||||
break;
|
||||
case 3:
|
||||
_step++;
|
||||
_ck_b += (_ck_a += data); // checksum byte
|
||||
_msg_id = data;
|
||||
if (_gather) { // if class was interesting
|
||||
switch(data) {
|
||||
case MSG_POSLLH: // message is interesting
|
||||
_expect = sizeof(ubx_nav_posllh);
|
||||
break;
|
||||
case MSG_STATUS:
|
||||
_expect = sizeof(ubx_nav_status);
|
||||
break;
|
||||
case MSG_SOL:
|
||||
_expect = sizeof(ubx_nav_solution);
|
||||
break;
|
||||
case MSG_VELNED:
|
||||
_expect = sizeof(ubx_nav_velned);
|
||||
break;
|
||||
default:
|
||||
_error("ignoring message 0x%x\n", (int)data);
|
||||
_gather = false; // message is not interesting
|
||||
}
|
||||
}
|
||||
break;
|
||||
case 4:
|
||||
_step++;
|
||||
_ck_b += (_ck_a += data); // checksum byte
|
||||
_payload_length = data; // payload length low byte
|
||||
break;
|
||||
case 5:
|
||||
_step++;
|
||||
_ck_b += (_ck_a += data); // checksum byte
|
||||
_payload_length += (uint16_t)data; // payload length high byte
|
||||
_payload_counter = 0; // prepare to receive payload
|
||||
if (_payload_length != _expect) {
|
||||
_error("payload %d expected %d\n", _payload_length, _expect);
|
||||
_gather = false;
|
||||
}
|
||||
break;
|
||||
|
||||
// Receive message data
|
||||
//
|
||||
case 6:
|
||||
_ck_b += (_ck_a += data); // checksum byte
|
||||
if (_gather) // gather data if requested
|
||||
_buffer.bytes[_payload_counter] = data;
|
||||
if (++_payload_counter == _payload_length)
|
||||
_step++;
|
||||
break;
|
||||
|
||||
// Checksum and message processing
|
||||
//
|
||||
case 7:
|
||||
_step++;
|
||||
if (_ck_a != data) {
|
||||
_error("GPS_UBLOX: checksum error\n");
|
||||
_step = 0;
|
||||
}
|
||||
break;
|
||||
case 8:
|
||||
_step = 0;
|
||||
if (_ck_b != data) {
|
||||
_error("GPS_UBLOX: checksum error\n");
|
||||
break;
|
||||
}
|
||||
if (_gather)
|
||||
_parse_gps(); // Parse the new GPS packet
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// Private Methods /////////////////////////////////////////////////////////////
|
||||
|
||||
void
|
||||
AP_GPS_UBLOX::_parse_gps(void)
|
||||
{
|
||||
switch (_msg_id) {
|
||||
case MSG_POSLLH:
|
||||
time = _buffer.posllh.time;
|
||||
longitude = _buffer.posllh.longitude;
|
||||
latitude = _buffer.posllh.latitude;
|
||||
altitude = _buffer.posllh.altitude_msl / 10;
|
||||
break;
|
||||
case MSG_STATUS:
|
||||
fix = (_buffer.status.fix_status & NAV_STATUS_FIX_VALID) && (_buffer.status.fix_type == FIX_3D);
|
||||
break;
|
||||
case MSG_SOL:
|
||||
fix = (_buffer.solution.fix_status & NAV_STATUS_FIX_VALID) && (_buffer.solution.fix_type == FIX_3D);
|
||||
num_sats = _buffer.solution.satellites;
|
||||
break;
|
||||
case MSG_VELNED:
|
||||
speed_3d = _buffer.velned.speed_3d; // cm/s
|
||||
ground_speed = _buffer.velned.speed_2d; // cm/s
|
||||
ground_course = _buffer.velned.heading_2d / 1000; // Heading 2D deg * 100000 rescaled to deg * 100
|
||||
break;
|
||||
}
|
||||
_setTime();
|
||||
valid_read = 1;
|
||||
new_data = 1;
|
||||
}
|
@ -0,0 +1,124 @@
|
||||
// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: t -*-
|
||||
//
|
||||
// u-blox UBX GPS driver for ArduPilot and ArduPilotMega.
|
||||
// Code by Michael Smith, Jordi Munoz and Jose Julio, DIYDrones.com
|
||||
//
|
||||
// This library is free software; you can redistribute it and / or
|
||||
// modify it under the terms of the GNU Lesser General Public
|
||||
// License as published by the Free Software Foundation; either
|
||||
// version 2.1 of the License, or (at your option) any later version.
|
||||
//
|
||||
#ifndef AP_GPS_UBLOX_h
|
||||
#define AP_GPS_UBLOX_h
|
||||
|
||||
#include <GPS.h>
|
||||
|
||||
#define UBLOX_SET_BINARY "$PUBX,41,1,0003,0001,38400,0*26"
|
||||
|
||||
class AP_GPS_UBLOX : public GPS
|
||||
{
|
||||
public:
|
||||
// Methods
|
||||
AP_GPS_UBLOX(Stream *s = NULL);
|
||||
void init(void);
|
||||
void update();
|
||||
|
||||
private:
|
||||
// u-blox UBX protocol essentials
|
||||
#pragma pack(1)
|
||||
struct ubx_nav_posllh {
|
||||
uint32_t time; // GPS msToW
|
||||
int32_t longitude;
|
||||
int32_t latitude;
|
||||
int32_t altitude_ellipsoid;
|
||||
int32_t altitude_msl;
|
||||
uint32_t horizontal_accuracy;
|
||||
uint32_t vertical_accuracy;
|
||||
};
|
||||
struct ubx_nav_status {
|
||||
uint32_t time; // GPS msToW
|
||||
uint8_t fix_type;
|
||||
uint8_t fix_status;
|
||||
uint8_t differential_status;
|
||||
uint8_t res;
|
||||
uint32_t time_to_first_fix;
|
||||
uint32_t uptime; // milliseconds
|
||||
};
|
||||
struct ubx_nav_solution {
|
||||
uint32_t time;
|
||||
int32_t time_nsec;
|
||||
int16_t week;
|
||||
uint8_t fix_type;
|
||||
uint8_t fix_status;
|
||||
int32_t ecef_x;
|
||||
int32_t ecef_y;
|
||||
int32_t ecef_z;
|
||||
uint32_t position_accuracy_3d;
|
||||
int32_t ecef_x_velocity;
|
||||
int32_t ecef_y_velocity;
|
||||
int32_t ecef_z_velocity;
|
||||
uint32_t speed_accuracy;
|
||||
uint16_t position_DOP;
|
||||
uint8_t res;
|
||||
uint8_t satellites;
|
||||
uint32_t res2;
|
||||
};
|
||||
struct ubx_nav_velned {
|
||||
uint32_t time; // GPS msToW
|
||||
int32_t ned_north;
|
||||
int32_t ned_east;
|
||||
int32_t ned_down;
|
||||
uint32_t speed_3d;
|
||||
uint32_t speed_2d;
|
||||
int32_t heading_2d;
|
||||
uint32_t speed_accuracy;
|
||||
uint32_t heading_accuracy;
|
||||
};
|
||||
#pragma pack(pop)
|
||||
enum ubs_protocol_bytes {
|
||||
PREAMBLE1 = 0xb5,
|
||||
PREAMBLE2 = 0x62,
|
||||
CLASS_NAV = 0x1,
|
||||
MSG_POSLLH = 0x2,
|
||||
MSG_STATUS = 0x3,
|
||||
MSG_SOL = 0x6,
|
||||
MSG_VELNED = 0x12
|
||||
};
|
||||
enum ubs_nav_fix_type {
|
||||
FIX_NONE = 0,
|
||||
FIX_DEAD_RECKONING = 1,
|
||||
FIX_2D = 2,
|
||||
FIX_3D = 3,
|
||||
FIX_GPS_DEAD_RECKONING = 4,
|
||||
FIX_TIME = 5
|
||||
};
|
||||
enum ubx_nav_status_bits {
|
||||
NAV_STATUS_FIX_VALID = 1
|
||||
};
|
||||
|
||||
// Packet checksum accumulators
|
||||
uint8_t _ck_a;
|
||||
uint8_t _ck_b;
|
||||
|
||||
// State machine state
|
||||
uint8_t _step;
|
||||
uint8_t _msg_id;
|
||||
bool _gather;
|
||||
uint16_t _expect;
|
||||
uint16_t _payload_length;
|
||||
uint16_t _payload_counter;
|
||||
|
||||
// Receive buffer
|
||||
union {
|
||||
ubx_nav_posllh posllh;
|
||||
ubx_nav_status status;
|
||||
ubx_nav_solution solution;
|
||||
ubx_nav_velned velned;
|
||||
uint8_t bytes[];
|
||||
} _buffer;
|
||||
|
||||
// Buffer parse & GPS state update
|
||||
void _parse_gps();
|
||||
};
|
||||
|
||||
#endif
|
@ -0,0 +1,35 @@
|
||||
// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: t -*-
|
||||
|
||||
#include "GPS.h"
|
||||
#include "WProgram.h"
|
||||
#include <stdio.h>
|
||||
|
||||
int
|
||||
GPS::status(void)
|
||||
{
|
||||
if (millis() - _lastTime >= 500){
|
||||
return 0;
|
||||
} else if (fix == 0) {
|
||||
return 1;
|
||||
} else {
|
||||
return 2;
|
||||
}
|
||||
}
|
||||
|
||||
void
|
||||
GPS::_setTime(void)
|
||||
{
|
||||
_lastTime = millis();
|
||||
}
|
||||
|
||||
void
|
||||
GPS::_error(const char *fmt, ...)
|
||||
{
|
||||
va_list ap;
|
||||
|
||||
if (print_errors && stderr) {
|
||||
va_start(ap, fmt);
|
||||
vfprintf(stderr, fmt, ap);
|
||||
va_end(ap);
|
||||
}
|
||||
}
|
@ -0,0 +1,150 @@
|
||||
// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: t -*-
|
||||
|
||||
/// @file GPS.h
|
||||
/// @brief Interface definition for the various GPS drivers.
|
||||
|
||||
#ifndef GPS_h
|
||||
#define GPS_h
|
||||
|
||||
#include <inttypes.h>
|
||||
#include <Stream.h>
|
||||
|
||||
/// @class GPS
|
||||
/// @brief Abstract base class for GPS receiver drivers.
|
||||
class GPS
|
||||
{
|
||||
public:
|
||||
|
||||
/// Constructor
|
||||
///
|
||||
/// @note The stream is expected to be set up and configured for the
|
||||
/// correct bitrate before ::init is called.
|
||||
///
|
||||
/// @param s Stream connected to the GPS module. If NULL, assumed
|
||||
/// to be set up at ::init time. Support for setting
|
||||
/// the port in the ctor for backwards compatibility.
|
||||
///
|
||||
GPS(Stream *s = NULL) : _port(s) {};
|
||||
|
||||
/// Startup initialisation.
|
||||
///
|
||||
/// This routine performs any one-off initialisation required to set the
|
||||
/// GPS up for use.
|
||||
///
|
||||
/// Must be implemented by the GPS driver.
|
||||
///
|
||||
/// @param s Stream connected to the GPS module. If NULL, assumed to
|
||||
/// have been set up at constructor time.
|
||||
///
|
||||
virtual void init(void) = 0;
|
||||
|
||||
/// Update GPS state based on possible bytes received from the module.
|
||||
///
|
||||
/// This routine must be called periodically to process incoming data.
|
||||
///
|
||||
/// Must be implemented by the GPS driver.
|
||||
///
|
||||
virtual void update(void) = 0;
|
||||
|
||||
/// Query GPS status
|
||||
///
|
||||
/// The 'valid message' status indicates that a recognised message was
|
||||
/// received from the GPS within the last 500ms.
|
||||
///
|
||||
/// @todo should probably return an enumeration here.
|
||||
///
|
||||
/// @return 0 No GPS connected/detected
|
||||
/// @return 1 Receiving valid GPS messages but no lock
|
||||
/// @return 2 Receiving valid messages and locked
|
||||
///
|
||||
int status(void);
|
||||
|
||||
// Properties
|
||||
long time; ///< GPS time in milliseconds from the start of the week
|
||||
long latitude; ///< latitude in degrees * 10,000,000
|
||||
long longitude; ///< longitude in degrees * 10,000,000
|
||||
long altitude; ///< altitude in cm
|
||||
long ground_speed; ///< ground speed in cm/sec
|
||||
long ground_course; ///< ground course in 100ths of a degree
|
||||
long speed_3d; ///< 3D speed in cm/sec (not always available)
|
||||
uint8_t num_sats; ///< Number of visible satelites
|
||||
|
||||
/// Set to true when new data arrives. A client may set this
|
||||
/// to false in order to avoid processing data they have
|
||||
/// already seen.
|
||||
bool new_data;
|
||||
|
||||
// Deprecated properties
|
||||
bool fix; ///< true if we have a position fix (use ::status instead)
|
||||
bool valid_read; ///< true if we have seen data from the GPS (use ::status instead)
|
||||
|
||||
// Debug support
|
||||
bool print_errors; ///< if true, errors will be printed to stderr
|
||||
|
||||
protected:
|
||||
Stream *_port; ///< stream port the GPS is attached to
|
||||
unsigned long _lastTime; ///< Timer for lost connection
|
||||
|
||||
/// reset the last-message-received timer used by ::status
|
||||
///
|
||||
void _setTime(void);
|
||||
|
||||
/// perform an endian swap on a long
|
||||
///
|
||||
/// @param bytes pointer to a buffer containing bytes representing a
|
||||
/// long in the wrong byte order
|
||||
/// @returns endian-swapped value
|
||||
///
|
||||
long _swapl(const void *bytes);
|
||||
|
||||
/// perform an endian swap on an int
|
||||
///
|
||||
/// @param bytes pointer to a buffer containing bytes representing an
|
||||
/// int in the wrong byte order
|
||||
/// @returns endian-swapped value
|
||||
int _swapi(const void *bytes);
|
||||
|
||||
/// emit an error message
|
||||
///
|
||||
/// based on the value of print_errors, emits the printf-formatted message
|
||||
/// in msg,... to stderr
|
||||
///
|
||||
/// @param fmt printf-like format string
|
||||
///
|
||||
void _error(const char *msg, ...);
|
||||
|
||||
};
|
||||
|
||||
inline long
|
||||
GPS::_swapl(const void *bytes)
|
||||
{
|
||||
const uint8_t *b = (const uint8_t *)bytes;
|
||||
union {
|
||||
long v;
|
||||
uint8_t b[4];
|
||||
} u;
|
||||
|
||||
u.b[0] = b[3];
|
||||
u.b[1] = b[2];
|
||||
u.b[2] = b[1];
|
||||
u.b[3] = b[0];
|
||||
|
||||
return(u.v);
|
||||
}
|
||||
|
||||
inline int16_t
|
||||
GPS::_swapi(const void *bytes)
|
||||
{
|
||||
const uint8_t *b = (const uint8_t *)bytes;
|
||||
union {
|
||||
int16_t v;
|
||||
uint8_t b[2];
|
||||
} u;
|
||||
|
||||
u.b[0] = b[1];
|
||||
u.b[1] = b[0];
|
||||
|
||||
return(u.v);
|
||||
}
|
||||
|
||||
#endif
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue
Block a user