2013-08-29 02:34:34 -03:00
|
|
|
/*
|
|
|
|
This program is free software: you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU General Public License as published by
|
|
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
|
|
(at your option) any later version.
|
2010-09-11 03:13:28 -03:00
|
|
|
|
2013-08-29 02:34:34 -03:00
|
|
|
This program is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
2010-09-11 03:13:28 -03:00
|
|
|
///
|
2012-08-17 03:18:11 -03:00
|
|
|
/// @file AP_Common.h
|
2010-09-11 03:13:28 -03:00
|
|
|
/// @brief Common definitions and utility routines for the ArduPilot
|
|
|
|
/// libraries.
|
|
|
|
///
|
|
|
|
|
2015-12-07 16:24:02 -04:00
|
|
|
#pragma once
|
2011-03-21 04:25:48 -03:00
|
|
|
|
2015-12-07 16:24:02 -04:00
|
|
|
#include <AP_HAL/AP_HAL_Boards.h>
|
2010-09-24 02:50:02 -03:00
|
|
|
#include <stdint.h>
|
2013-01-01 23:43:50 -04:00
|
|
|
#include <stdlib.h>
|
2011-01-23 22:44:31 -04:00
|
|
|
|
2013-04-19 04:47:27 -03:00
|
|
|
// used to pack structures
|
|
|
|
#define PACKED __attribute__((__packed__))
|
|
|
|
|
2015-02-17 20:12:43 -04:00
|
|
|
// used to mark a function that may be unused in some builds
|
|
|
|
#define UNUSED_FUNCTION __attribute__((unused))
|
|
|
|
|
2013-10-02 03:09:13 -03:00
|
|
|
// this can be used to optimize individual functions
|
|
|
|
#define OPTIMIZE(level) __attribute__((optimize(level)))
|
|
|
|
|
2015-05-20 20:35:26 -03:00
|
|
|
// sometimes we need to prevent inlining to prevent large stack usage
|
|
|
|
#define NOINLINE __attribute__((noinline))
|
|
|
|
|
2015-11-10 13:05:19 -04:00
|
|
|
#define FMT_PRINTF(a,b) __attribute__((format(printf, a, b)))
|
2015-11-05 23:40:18 -04:00
|
|
|
#define FMT_SCANF(a,b) __attribute__((format(scanf, a, b)))
|
2015-10-22 12:06:18 -03:00
|
|
|
|
2017-08-22 14:25:54 -03:00
|
|
|
#ifdef __has_cpp_attribute
|
|
|
|
# if __has_cpp_attribute(fallthrough)
|
|
|
|
# define FALLTHROUGH [[fallthrough]]
|
|
|
|
# elif __has_cpp_attribute(gnu::fallthrough)
|
|
|
|
# define FALLTHROUGH [[gnu::fallthrough]]
|
|
|
|
# endif
|
|
|
|
#endif
|
|
|
|
#ifndef FALLTHROUGH
|
|
|
|
# define FALLTHROUGH
|
|
|
|
#endif
|
|
|
|
|
2012-12-19 02:02:13 -04:00
|
|
|
#define ToRad(x) radians(x) // *pi/180
|
|
|
|
#define ToDeg(x) degrees(x) // *180/pi
|
2013-04-23 10:05:02 -03:00
|
|
|
|
2016-06-07 14:17:07 -03:00
|
|
|
/* Declare and implement const and non-const versions of the array subscript
|
|
|
|
* operator. The object is treated as an array of type_ values. */
|
2016-06-23 19:56:17 -03:00
|
|
|
#define DEFINE_BYTE_ARRAY_METHODS \
|
|
|
|
inline uint8_t &operator[](size_t i) { return reinterpret_cast<uint8_t *>(this)[i]; } \
|
|
|
|
inline uint8_t operator[](size_t i) const { return reinterpret_cast<const uint8_t *>(this)[i]; }
|
2016-06-07 14:17:07 -03:00
|
|
|
|
2015-10-19 04:12:59 -03:00
|
|
|
#define LOCATION_ALT_MAX_M 83000 // maximum altitude (in meters) that can be fit into Location structure's alt field
|
|
|
|
|
2013-04-23 10:05:02 -03:00
|
|
|
/*
|
|
|
|
check if bit bitnumber is set in value, returned as a
|
|
|
|
bool. Bitnumber starts at 0 for the first bit
|
|
|
|
*/
|
|
|
|
#define BIT_IS_SET(value, bitnumber) (((value) & (1U<<(bitnumber))) != 0)
|
|
|
|
|
2014-06-11 04:16:09 -03:00
|
|
|
// get high or low bytes from 2 byte integer
|
|
|
|
#define LOWBYTE(i) ((uint8_t)(i))
|
|
|
|
#define HIGHBYTE(i) ((uint8_t)(((uint16_t)(i))>>8))
|
|
|
|
|
2015-09-24 14:56:19 -03:00
|
|
|
template <typename T, size_t N>
|
|
|
|
char (&_ARRAY_SIZE_HELPER(T (&_arr)[N]))[N];
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
char (&_ARRAY_SIZE_HELPER(T (&_arr)[0]))[0];
|
|
|
|
|
|
|
|
#define ARRAY_SIZE(_arr) sizeof(_ARRAY_SIZE_HELPER(_arr))
|
2015-07-02 12:11:32 -03:00
|
|
|
|
2016-06-21 16:34:45 -03:00
|
|
|
/*
|
|
|
|
* See UNUSED_RESULT. The difference is that it receives @uniq_ as the name to
|
|
|
|
* be used for its internal variable.
|
|
|
|
*
|
|
|
|
* @uniq_: a unique name to use for variable name
|
|
|
|
* @expr_: the expression to be evaluated
|
|
|
|
*/
|
|
|
|
#define _UNUSED_RESULT(uniq_, expr_) \
|
|
|
|
do { \
|
|
|
|
decltype(expr_) uniq_ __attribute__((unused)); \
|
|
|
|
uniq_ = expr_; \
|
|
|
|
} while (0)
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Allow to call a function annotated with warn_unused_result attribute
|
|
|
|
* without getting a warning, because sometimes this is what we want to do.
|
|
|
|
*
|
|
|
|
* @expr_: the expression to be evaluated
|
|
|
|
*/
|
|
|
|
#define UNUSED_RESULT(expr_) _UNUSED_RESULT(__unique_name_##__COUNTER__, expr_)
|
|
|
|
|
2010-12-28 17:30:29 -04:00
|
|
|
// @}
|
|
|
|
|
|
|
|
|
2010-09-11 03:13:28 -03:00
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
|
|
/// @name Types
|
|
|
|
///
|
2011-03-05 00:39:23 -04:00
|
|
|
/// Data structures and types used throughout the libraries and applications. 0 = default
|
2012-08-17 03:18:11 -03:00
|
|
|
/// bit 0: Altitude is stored 0: Absolute, 1: Relative
|
2017-01-05 14:07:14 -04:00
|
|
|
/// bit 1: Change Alt between WP 0: Gradually, 1: ASAP
|
2013-01-13 09:29:53 -04:00
|
|
|
/// bit 2: Direction of loiter command 0: Clockwise 1: Counter-Clockwise
|
2011-04-16 01:49:46 -03:00
|
|
|
/// bit 3: Req.to hit WP.alt to continue 0: No, 1: Yes
|
2012-08-17 03:18:11 -03:00
|
|
|
/// bit 4: Relative to Home 0: No, 1: Yes
|
2016-03-14 15:42:28 -03:00
|
|
|
/// bit 5: Loiter crosstrack reference 0: WP center 1: Tangent exit point
|
2011-03-05 00:39:23 -04:00
|
|
|
/// bit 6:
|
2012-08-17 03:18:11 -03:00
|
|
|
/// bit 7: Move to next Command 0: YES, 1: Loiter until commanded
|
2011-03-05 00:39:23 -04:00
|
|
|
|
2010-09-11 03:13:28 -03:00
|
|
|
//@{
|
|
|
|
|
2014-03-17 02:14:23 -03:00
|
|
|
struct PACKED Location_Option_Flags {
|
2017-11-27 08:08:49 -04:00
|
|
|
uint8_t relative_alt : 1; // 1 if altitude is relative to home
|
2014-03-03 01:37:34 -04:00
|
|
|
uint8_t unused1 : 1; // unused flag (defined so that loiter_ccw uses the correct bit)
|
|
|
|
uint8_t loiter_ccw : 1; // 0 if clockwise, 1 if counter clockwise
|
2014-07-24 04:53:28 -03:00
|
|
|
uint8_t terrain_alt : 1; // this altitude is above terrain
|
2015-08-10 02:25:15 -03:00
|
|
|
uint8_t origin_alt : 1; // this altitude is above ekf origin
|
2016-03-14 15:42:28 -03:00
|
|
|
uint8_t loiter_xtrack : 1; // 0 to crosstrack from center of waypoint, 1 to crosstrack from tangent exit location
|
2014-03-03 01:37:34 -04:00
|
|
|
};
|
|
|
|
|
2014-03-17 02:14:23 -03:00
|
|
|
struct PACKED Location {
|
2014-03-03 01:37:34 -04:00
|
|
|
union {
|
|
|
|
Location_Option_Flags flags; ///< options bitmask (1<<0 = relative altitude)
|
|
|
|
uint8_t options; /// allows writing all flags to eeprom as one byte
|
|
|
|
};
|
2014-03-20 02:55:38 -03:00
|
|
|
// by making alt 24 bit we can make p1 in a command 16 bit,
|
|
|
|
// allowing an accurate angle in centi-degrees. This keeps the
|
|
|
|
// storage cost per mission item at 15 bytes, and allows mission
|
|
|
|
// altitudes of up to +/- 83km
|
2017-01-05 14:07:14 -04:00
|
|
|
int32_t alt:24; ///< param 2 - Altitude in centimeters (meters * 100) see LOCATION_ALT_MAX_M
|
2016-05-12 13:53:53 -03:00
|
|
|
int32_t lat; ///< param 3 - Latitude * 10**7
|
2012-08-17 03:18:11 -03:00
|
|
|
int32_t lng; ///< param 4 - Longitude * 10**7
|
2010-09-11 03:13:28 -03:00
|
|
|
};
|
|
|
|
|
2015-02-20 19:13:17 -04:00
|
|
|
/*
|
|
|
|
home states. Used to record if user has overridden home position.
|
|
|
|
*/
|
|
|
|
enum HomeState {
|
|
|
|
HOME_UNSET, // home is unset, no GPS positions yet received
|
|
|
|
HOME_SET_NOT_LOCKED, // home is set to EKF origin or armed location (can be moved)
|
|
|
|
HOME_SET_AND_LOCKED // home has been set by user, cannot be moved except by user initiated do-set-home command
|
|
|
|
};
|
|
|
|
|
2010-09-11 03:13:28 -03:00
|
|
|
//@}
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
|
|
|
/// @name Conversions
|
|
|
|
///
|
|
|
|
/// Conversion macros and factors.
|
|
|
|
///
|
|
|
|
//@{
|
|
|
|
|
2015-11-08 17:47:45 -04:00
|
|
|
/*
|
|
|
|
Return true if value is between lower and upper bound inclusive.
|
|
|
|
False otherwise.
|
|
|
|
*/
|
|
|
|
bool is_bounded_int32(int32_t value, int32_t lower_bound, int32_t upper_bound);
|
2015-11-04 02:43:48 -04:00
|
|
|
|
2015-12-07 16:24:02 -04:00
|
|
|
#if CONFIG_HAL_BOARD == HAL_BOARD_QURT
|
|
|
|
#include <AP_HAL_QURT/replace.h>
|
|
|
|
#endif
|
|
|
|
|
2016-05-22 21:10:29 -03:00
|
|
|
/*
|
|
|
|
useful debugging macro for SITL
|
|
|
|
*/
|
|
|
|
#if CONFIG_HAL_BOARD == HAL_BOARD_SITL
|
2017-10-23 21:25:39 -03:00
|
|
|
#include <stdio.h>
|
2016-05-22 21:10:29 -03:00
|
|
|
#define SITL_printf(fmt, args ...) do { ::printf("%s(%u): " fmt, __FILE__, __LINE__, ##args); } while(0)
|
|
|
|
#else
|
|
|
|
#define SITL_printf(fmt, args ...)
|
|
|
|
#endif
|