This commit adds the public header for the GNSS API, along with the initial GNSS Kconfig file and an entry in the common linker file for registered GNSS data callbacks. A very naive implementation of the GNSS data callback is provided as well in drivers/gnss/gnss_publish.c Signed-off-by: Bjarki Arge Andreasen <bjarkix123@gmail.com>
496 lines
14 KiB
C
496 lines
14 KiB
C
/*
|
|
* Copyright (c) 2023 Trackunit Corporation
|
|
*
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*/
|
|
|
|
/**
|
|
* @file gnss.h
|
|
* @brief Public GNSS API.
|
|
*/
|
|
|
|
#ifndef ZEPHYR_INCLUDE_DRIVERS_GNSS_H_
|
|
#define ZEPHYR_INCLUDE_DRIVERS_GNSS_H_
|
|
|
|
/**
|
|
* @brief GNSS Interface
|
|
* @defgroup gnss_interface GNSS Interface
|
|
* @ingroup io_interfaces
|
|
* @{
|
|
*/
|
|
|
|
#include <zephyr/types.h>
|
|
#include <zephyr/device.h>
|
|
#include <zephyr/data/navigation.h>
|
|
#include <errno.h>
|
|
|
|
#ifdef __cplusplus
|
|
extern "C" {
|
|
#endif
|
|
|
|
/** GNSS PPS modes */
|
|
enum gnss_pps_mode {
|
|
/** PPS output disabled */
|
|
GNSS_PPS_MODE_DISABLED = 0,
|
|
/** PPS output always enabled */
|
|
GNSS_PPS_MODE_ENABLED = 1,
|
|
/** PPS output enabled from first lock */
|
|
GNSS_PPS_MODE_ENABLED_AFTER_LOCK = 2,
|
|
/** PPS output enabled while locked */
|
|
GNSS_PPS_MODE_ENABLED_WHILE_LOCKED = 3
|
|
};
|
|
|
|
/** API for setting fix rate */
|
|
typedef int (*gnss_set_fix_rate_t)(const struct device *dev, uint32_t fix_interval_ms);
|
|
|
|
/** API for getting fix rate */
|
|
typedef int (*gnss_get_fix_rate_t)(const struct device *dev, uint32_t *fix_interval_ms);
|
|
|
|
/**
|
|
* @brief GNSS periodic tracking configuration
|
|
*
|
|
* @note Setting either active_time or inactive_time to 0 will disable periodic
|
|
* function.
|
|
*/
|
|
struct gnss_periodic_config {
|
|
/** The time the GNSS will spend in the active state in ms */
|
|
uint32_t active_time_ms;
|
|
/** The time the GNSS will spend in the inactive state in ms */
|
|
uint32_t inactive_time_ms;
|
|
};
|
|
|
|
/** API for setting periodic tracking configuration */
|
|
typedef int (*gnss_set_periodic_config_t)(const struct device *dev,
|
|
const struct gnss_periodic_config *periodic_config);
|
|
|
|
/** API for setting periodic tracking configuration */
|
|
typedef int (*gnss_get_periodic_config_t)(const struct device *dev,
|
|
struct gnss_periodic_config *periodic_config);
|
|
|
|
/** GNSS navigation modes */
|
|
enum gnss_navigation_mode {
|
|
/** Dynamics have no impact on tracking */
|
|
GNSS_NAVIGATION_MODE_ZERO_DYNAMICS = 0,
|
|
/** Low dynamics have higher impact on tracking */
|
|
GNSS_NAVIGATION_MODE_LOW_DYNAMICS = 1,
|
|
/** Low and high dynamics have equal impact on tracking */
|
|
GNSS_NAVIGATION_MODE_BALANCED_DYNAMICS = 2,
|
|
/** High dynamics have higher impact on tracking */
|
|
GNSS_NAVIGATION_MODE_HIGH_DYNAMICS = 3
|
|
};
|
|
|
|
/** API for setting navigation mode */
|
|
typedef int (*gnss_set_navigation_mode_t)(const struct device *dev,
|
|
enum gnss_navigation_mode mode);
|
|
|
|
/** API for getting navigation mode */
|
|
typedef int (*gnss_get_navigation_mode_t)(const struct device *dev,
|
|
enum gnss_navigation_mode *mode);
|
|
|
|
/** Systems contained in gnss_systems_t */
|
|
enum gnss_system {
|
|
/** Global Positioning System (GPS) */
|
|
GNSS_SYSTEM_GPS = BIT(0),
|
|
/** GLObal NAvigation Satellite System (GLONASS) */
|
|
GNSS_SYSTEM_GLONASS = BIT(1),
|
|
/** Galileo */
|
|
GNSS_SYSTEM_GALILEO = BIT(2),
|
|
/** BeiDou Navigation Satellite System */
|
|
GNSS_SYSTEM_BEIDOU = BIT(3),
|
|
/** Quasi-Zenith Satellite System (QZSS) */
|
|
GNSS_SYSTEM_QZSS = BIT(4),
|
|
/** Indian Regional Navigation Satellite System (IRNSS) */
|
|
GNSS_SYSTEM_IRNSS = BIT(5),
|
|
/** Satellite-Based Augmentation System (SBAS) */
|
|
GNSS_SYSTEM_SBAS = BIT(6),
|
|
/** Indoor Messaging System (IMES) */
|
|
GNSS_SYSTEM_IMES = BIT(7),
|
|
};
|
|
|
|
/** Type storing bitmask of GNSS systems */
|
|
typedef uint32_t gnss_systems_t;
|
|
|
|
/** API for enabling systems */
|
|
typedef int (*gnss_set_enabled_systems_t)(const struct device *dev, gnss_systems_t systems);
|
|
|
|
/** API for getting enabled systems */
|
|
typedef int (*gnss_get_enabled_systems_t)(const struct device *dev, gnss_systems_t *systems);
|
|
|
|
/** API for getting enabled systems */
|
|
typedef int (*gnss_get_supported_systems_t)(const struct device *dev, gnss_systems_t *systems);
|
|
|
|
/** GNSS fix status */
|
|
enum gnss_fix_status {
|
|
/** No GNSS fix aqcuired */
|
|
GNSS_FIX_STATUS_NO_FIX = 0,
|
|
/** GNSS fix aqcuired */
|
|
GNSS_FIX_STATUS_GNSS_FIX = 1,
|
|
/** Differential GNSS fix acquired */
|
|
GNSS_FIX_STATUS_DGNSS_FIX = 2,
|
|
/** Estimated fix acquired */
|
|
GNSS_FIX_STATUS_ESTIMATED_FIX = 3,
|
|
};
|
|
|
|
/** GNSS fix quality */
|
|
enum gnss_fix_quality {
|
|
/** Invalid fix */
|
|
GNSS_FIX_QUALITY_INVALID = 0,
|
|
/** Standard positioning service */
|
|
GNSS_FIX_QUALITY_GNSS_SPS = 1,
|
|
/** Differential GNSS */
|
|
GNSS_FIX_QUALITY_DGNSS = 2,
|
|
/** Precise positioning service */
|
|
GNSS_FIX_QUALITY_GNSS_PPS = 3,
|
|
/** Real-time kinematic */
|
|
GNSS_FIX_QUALITY_RTK = 4,
|
|
/** Floating real-time kinematic */
|
|
GNSS_FIX_QUALITY_FLOAT_RTK = 5,
|
|
/** Estimated fix */
|
|
GNSS_FIX_QUALITY_ESTIMATED = 6,
|
|
};
|
|
|
|
/** GNSS info data structure */
|
|
struct gnss_info {
|
|
/** Number of satellites being tracked */
|
|
uint16_t satellites_cnt;
|
|
/** Horizontal dilution of precision in 1/1000 */
|
|
uint16_t hdop;
|
|
/** The fix status */
|
|
enum gnss_fix_status fix_status;
|
|
/** The fix quality */
|
|
enum gnss_fix_quality fix_quality;
|
|
};
|
|
|
|
/** GNSS time data structure */
|
|
struct gnss_time {
|
|
/** Hour [0, 23] */
|
|
uint8_t hour;
|
|
/** Minute [0, 59] */
|
|
uint8_t minute;
|
|
/** Millisecond [0, 59999] */
|
|
uint16_t millisecond;
|
|
/** Day of month [1, 31] */
|
|
uint8_t month_day;
|
|
/** Month [1, 12] */
|
|
uint8_t month;
|
|
/** Year [0, 99] */
|
|
uint8_t century_year;
|
|
};
|
|
|
|
/** GNSS API structure */
|
|
__subsystem struct gnss_driver_api {
|
|
gnss_set_fix_rate_t set_fix_rate;
|
|
gnss_get_fix_rate_t get_fix_rate;
|
|
gnss_set_periodic_config_t set_periodic_config;
|
|
gnss_get_periodic_config_t get_periodic_config;
|
|
gnss_set_navigation_mode_t set_navigation_mode;
|
|
gnss_get_navigation_mode_t get_navigation_mode;
|
|
gnss_set_enabled_systems_t set_enabled_systems;
|
|
gnss_get_enabled_systems_t get_enabled_systems;
|
|
gnss_get_supported_systems_t get_supported_systems;
|
|
};
|
|
|
|
/** GNSS data structure */
|
|
struct gnss_data {
|
|
/** Navigation data acquired */
|
|
struct navigation_data nav_data;
|
|
/** GNSS info when navigation data was acquired */
|
|
struct gnss_info info;
|
|
/** UTC time when data was acquired */
|
|
struct gnss_time utc;
|
|
};
|
|
|
|
/** Template for GNSS data callback */
|
|
typedef void (*gnss_data_callback_t)(const struct device *dev, const struct gnss_data *data);
|
|
|
|
/** GNSS callback structure */
|
|
struct gnss_data_callback {
|
|
/** Filter callback to GNSS data from this device if not NULL */
|
|
const struct device *dev;
|
|
/** Callback called when GNSS data is published */
|
|
gnss_data_callback_t callback;
|
|
};
|
|
|
|
/** GNSS satellite structure */
|
|
struct gnss_satellite {
|
|
/** Pseudo-random noise sequence */
|
|
uint8_t prn;
|
|
/** Signal-to-noise ratio in dB */
|
|
uint8_t snr;
|
|
/** Elevation in degrees [0, 90] */
|
|
uint8_t elevation;
|
|
/** Azimuth relative to True North in degrees [0, 359] */
|
|
uint16_t azimuth;
|
|
/** System of satellite */
|
|
enum gnss_system system;
|
|
/** True if satellite is being tracked */
|
|
uint8_t is_tracked : 1;
|
|
};
|
|
|
|
/** Template for GNSS satellites callback */
|
|
typedef void (*gnss_satellites_callback_t)(const struct device *dev,
|
|
const struct gnss_satellite *satellites,
|
|
uint16_t size);
|
|
|
|
/** GNSS callback structure */
|
|
struct gnss_satellites_callback {
|
|
/** Filter callback to GNSS data from this device if not NULL */
|
|
const struct device *dev;
|
|
/** Callback called when GNSS satellites is published */
|
|
gnss_satellites_callback_t callback;
|
|
};
|
|
|
|
/**
|
|
* @brief Set the GNSS fix rate
|
|
*
|
|
* @param dev Device instance
|
|
* @param fix_interval_ms Fix interval to set in milliseconds
|
|
*
|
|
* @return 0 if successful
|
|
* @return -errno negative errno code on failure
|
|
*/
|
|
__syscall int gnss_set_fix_rate(const struct device *dev, uint32_t fix_interval_ms);
|
|
|
|
static inline int z_impl_gnss_set_fix_rate(const struct device *dev, uint32_t fix_interval_ms)
|
|
{
|
|
const struct gnss_driver_api *api = (const struct gnss_driver_api *)dev->api;
|
|
|
|
if (api->set_fix_rate == NULL) {
|
|
return -ENOSYS;
|
|
}
|
|
|
|
return api->set_fix_rate(dev, fix_interval_ms);
|
|
}
|
|
|
|
/**
|
|
* @brief Get the GNSS fix rate
|
|
*
|
|
* @param dev Device instance
|
|
* @param fix_interval_ms Destination for fix interval in milliseconds
|
|
*
|
|
* @return 0 if successful
|
|
* @return -errno negative errno code on failure
|
|
*/
|
|
__syscall int gnss_get_fix_rate(const struct device *dev, uint32_t *fix_interval_ms);
|
|
|
|
static inline int z_impl_gnss_get_fix_rate(const struct device *dev, uint32_t *fix_interval_ms)
|
|
{
|
|
const struct gnss_driver_api *api = (const struct gnss_driver_api *)dev->api;
|
|
|
|
if (api->get_fix_rate == NULL) {
|
|
return -ENOSYS;
|
|
}
|
|
|
|
return api->get_fix_rate(dev, fix_interval_ms);
|
|
}
|
|
|
|
/**
|
|
* @brief Set the GNSS periodic tracking configuration
|
|
*
|
|
* @param dev Device instance
|
|
* @param config Periodic tracking configuration to set
|
|
*
|
|
* @return 0 if successful
|
|
* @return -errno negative errno code on failure
|
|
*/
|
|
__syscall int gnss_set_periodic_config(const struct device *dev,
|
|
const struct gnss_periodic_config *config);
|
|
|
|
static inline int z_impl_gnss_set_periodic_config(const struct device *dev,
|
|
const struct gnss_periodic_config *config)
|
|
{
|
|
const struct gnss_driver_api *api = (const struct gnss_driver_api *)dev->api;
|
|
|
|
if (api->set_periodic_config == NULL) {
|
|
return -ENOSYS;
|
|
}
|
|
|
|
return api->set_periodic_config(dev, config);
|
|
}
|
|
|
|
/**
|
|
* @brief Get the GNSS periodic tracking configuration
|
|
*
|
|
* @param dev Device instance
|
|
* @param config Destination for periodic tracking configuration
|
|
*
|
|
* @return 0 if successful
|
|
* @return -errno negative errno code on failure
|
|
*/
|
|
__syscall int gnss_get_periodic_config(const struct device *dev,
|
|
struct gnss_periodic_config *config);
|
|
|
|
static inline int z_impl_gnss_get_periodic_config(const struct device *dev,
|
|
struct gnss_periodic_config *config)
|
|
{
|
|
const struct gnss_driver_api *api = (const struct gnss_driver_api *)dev->api;
|
|
|
|
if (api->get_periodic_config == NULL) {
|
|
return -ENOSYS;
|
|
}
|
|
|
|
return api->get_periodic_config(dev, config);
|
|
}
|
|
|
|
/**
|
|
* @brief Set the GNSS navigation mode
|
|
*
|
|
* @param dev Device instance
|
|
* @param mode Navigation mode to set
|
|
*
|
|
* @return 0 if successful
|
|
* @return -errno negative errno code on failure
|
|
*/
|
|
__syscall int gnss_set_navigation_mode(const struct device *dev,
|
|
enum gnss_navigation_mode mode);
|
|
|
|
static inline int z_impl_gnss_set_navigation_mode(const struct device *dev,
|
|
enum gnss_navigation_mode mode)
|
|
{
|
|
const struct gnss_driver_api *api = (const struct gnss_driver_api *)dev->api;
|
|
|
|
if (api->set_navigation_mode == NULL) {
|
|
return -ENOSYS;
|
|
}
|
|
|
|
return api->set_navigation_mode(dev, mode);
|
|
}
|
|
|
|
/**
|
|
* @brief Get the GNSS navigation mode
|
|
*
|
|
* @param dev Device instance
|
|
* @param mode Destination for navigation mode
|
|
*
|
|
* @return 0 if successful
|
|
* @return -errno negative errno code on failure
|
|
*/
|
|
__syscall int gnss_get_navigation_mode(const struct device *dev,
|
|
enum gnss_navigation_mode *mode);
|
|
|
|
static inline int z_impl_gnss_get_navigation_mode(const struct device *dev,
|
|
enum gnss_navigation_mode *mode)
|
|
{
|
|
const struct gnss_driver_api *api = (const struct gnss_driver_api *)dev->api;
|
|
|
|
if (api->get_navigation_mode == NULL) {
|
|
return -ENOSYS;
|
|
}
|
|
|
|
return api->get_navigation_mode(dev, mode);
|
|
}
|
|
|
|
/**
|
|
* @brief Set enabled GNSS systems
|
|
*
|
|
* @param dev Device instance
|
|
* @param systems Systems to enable
|
|
*
|
|
* @return 0 if successful
|
|
* @return -errno negative errno code on failure
|
|
*/
|
|
__syscall int gnss_set_enabled_systems(const struct device *dev, gnss_systems_t systems);
|
|
|
|
static inline int z_impl_gnss_set_enabled_systems(const struct device *dev,
|
|
gnss_systems_t systems)
|
|
{
|
|
const struct gnss_driver_api *api = (const struct gnss_driver_api *)dev->api;
|
|
|
|
if (api->set_enabled_systems == NULL) {
|
|
return -ENOSYS;
|
|
}
|
|
|
|
return api->set_enabled_systems(dev, systems);
|
|
}
|
|
|
|
/**
|
|
* @brief Get enabled GNSS systems
|
|
*
|
|
* @param dev Device instance
|
|
* @param systems Destination for enabled systems
|
|
*
|
|
* @return 0 if successful
|
|
* @return -errno negative errno code on failure
|
|
*/
|
|
__syscall int gnss_get_enabled_systems(const struct device *dev, gnss_systems_t *systems);
|
|
|
|
static inline int z_impl_gnss_get_enabled_systems(const struct device *dev,
|
|
gnss_systems_t *systems)
|
|
{
|
|
const struct gnss_driver_api *api = (const struct gnss_driver_api *)dev->api;
|
|
|
|
if (api->get_enabled_systems == NULL) {
|
|
return -ENOSYS;
|
|
}
|
|
|
|
return api->get_enabled_systems(dev, systems);
|
|
}
|
|
|
|
/**
|
|
* @brief Get supported GNSS systems
|
|
*
|
|
* @param dev Device instance
|
|
* @param systems Destination for supported systems
|
|
*
|
|
* @return 0 if successful
|
|
* @return -errno negative errno code on failure
|
|
*/
|
|
__syscall int gnss_get_supported_systems(const struct device *dev, gnss_systems_t *systems);
|
|
|
|
static inline int z_impl_gnss_get_supported_systems(const struct device *dev,
|
|
gnss_systems_t *systems)
|
|
{
|
|
const struct gnss_driver_api *api = (const struct gnss_driver_api *)dev->api;
|
|
|
|
if (api->get_supported_systems == NULL) {
|
|
return -ENOSYS;
|
|
}
|
|
|
|
return api->get_supported_systems(dev, systems);
|
|
}
|
|
|
|
/**
|
|
* @brief Register a callback structure for GNSS data published
|
|
*
|
|
* @param _dev Device pointer
|
|
* @param _callback The callback function
|
|
*/
|
|
#if CONFIG_GNSS
|
|
#define GNSS_DATA_CALLBACK_DEFINE(_dev, _callback) \
|
|
static const STRUCT_SECTION_ITERABLE(gnss_data_callback, \
|
|
_gnss_data_callback__##_callback) = { \
|
|
.dev = _dev, \
|
|
.callback = _callback, \
|
|
}
|
|
#else
|
|
#define GNSS_DATA_CALLBACK_DEFINE(_dev, _callback)
|
|
#endif
|
|
|
|
/**
|
|
* @brief Register a callback structure for GNSS satellites published
|
|
*
|
|
* @param _dev Device pointer
|
|
* @param _callback The callback function
|
|
*/
|
|
#if CONFIG_GNSS_SATELLITES
|
|
#define GNSS_SATELLITES_CALLBACK_DEFINE(_dev, _callback) \
|
|
static const STRUCT_SECTION_ITERABLE(gnss_satellites_callback, \
|
|
_gnss_satellites_callback__##_callback) = { \
|
|
.dev = _dev, \
|
|
.callback = _callback, \
|
|
}
|
|
#else
|
|
#define GNSS_SATELLITES_CALLBACK_DEFINE(_dev, _callback)
|
|
#endif
|
|
|
|
/**
|
|
* @}
|
|
*/
|
|
|
|
#ifdef __cplusplus
|
|
}
|
|
#endif
|
|
|
|
#include <syscalls/gnss.h>
|
|
|
|
#endif /* ZEPHYR_INCLUDE_DRIVERS_GNSS_H_ */
|