mirror of
https://github.com/eclipse/upm.git
synced 2025-03-19 15:07:35 +03:00

These changes allow NMEA data to be read via I2C on UBLOX compliant devices that support this capability, such as the LEA-6H based GPS shield from DFRobot. It adds a new init() function to the C code, and a new constructor to the C++ code. It also adds 5 new examples for C, C++, Javascript, Python, and Java. Signed-off-by: Jon Trulson <jtrulson@ics.com>
292 lines
7.2 KiB
C
292 lines
7.2 KiB
C
/*
|
|
* Author: Jon Trulson <jtrulson@ics.com>
|
|
* Copyright (c) 2016 Intel Corporation.
|
|
*
|
|
* Permission is hereby granted, free of charge, to any person obtaining
|
|
* a copy of this software and associated documentation files (the
|
|
* "Software"), to deal in the Software without restriction, including
|
|
* without limitation the rights to use, copy, modify, merge, publish,
|
|
* distribute, sublicense, and/or sell copies of the Software, and to
|
|
* permit persons to whom the Software is furnished to do so, subject to
|
|
* the following conditions:
|
|
*
|
|
* The above copyright notice and this permission notice shall be
|
|
* included in all copies or substantial portions of the Software.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
|
* NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
|
* LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
|
* OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
|
* WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
*/
|
|
|
|
#include <string.h>
|
|
#include <assert.h>
|
|
|
|
#include "nmea_gps.h"
|
|
|
|
#include "upm_utilities.h"
|
|
|
|
// For ublox6 and compatible I2C communications (see the u-blox6
|
|
// Receiver Description Protocol Specification datasheet for details).
|
|
|
|
#define UBLOX6_I2C_BYTES_AVAIL_H 0xfd
|
|
#define UBLOX6_I2C_BYTES_AVAIL_L 0xfe
|
|
#define UBLOX6_I2C_BYTE_STREAM 0xff
|
|
|
|
#define UBLOX6_I2C_BYTE_NONE 0xff // read if no data avail
|
|
|
|
// static helpers for i2c reading
|
|
static uint8_t readReg(const nmea_gps_context dev, uint8_t reg)
|
|
{
|
|
assert(dev != NULL);
|
|
assert(dev->i2c != NULL);
|
|
|
|
int rv;
|
|
if ((rv = mraa_i2c_read_byte_data(dev->i2c, reg)) < 0)
|
|
printf("%s: mraa_i2c_read_byte_data() failed.\n", __FUNCTION__);
|
|
|
|
// This will return 0xff on errors, which is invalid NMEA data anyway.
|
|
return (uint8_t)rv;
|
|
}
|
|
|
|
static int readRegs(const nmea_gps_context dev, uint8_t reg,
|
|
uint8_t *buffer, int len)
|
|
{
|
|
assert(dev != NULL);
|
|
assert(dev->i2c != NULL);
|
|
|
|
int rv;
|
|
if ((rv = mraa_i2c_read_bytes_data(dev->i2c, reg, buffer, len)) < 0)
|
|
{
|
|
printf("%s: mraa_i2c_read_bytes_data() failed.\n", __FUNCTION__);
|
|
}
|
|
|
|
return rv;
|
|
}
|
|
|
|
// uart init
|
|
nmea_gps_context nmea_gps_init(unsigned int uart, unsigned int baudrate,
|
|
int enable_pin)
|
|
{
|
|
nmea_gps_context dev =
|
|
(nmea_gps_context)malloc(sizeof(struct _nmea_gps_context));
|
|
|
|
if (!dev)
|
|
return NULL;
|
|
|
|
// zero out context
|
|
memset((void *)dev, 0, sizeof(struct _nmea_gps_context));
|
|
|
|
dev->uart = NULL;
|
|
dev->i2c = NULL;
|
|
dev->gpio_en = NULL;
|
|
|
|
// initialize the MRAA contexts
|
|
|
|
// uart, default should be 8N1
|
|
if (!(dev->uart = mraa_uart_init(uart)))
|
|
{
|
|
printf("%s: mraa_uart_init() failed.\n", __FUNCTION__);
|
|
nmea_gps_close(dev);
|
|
return NULL;
|
|
}
|
|
|
|
if (nmea_gps_set_baudrate(dev, baudrate))
|
|
{
|
|
printf("%s: nmea_gps_set_baudrate() failed.\n", __FUNCTION__);
|
|
nmea_gps_close(dev);
|
|
return NULL;
|
|
}
|
|
|
|
mraa_uart_set_flowcontrol(dev->uart, false, false);
|
|
|
|
// now the gpio_en, if enabled
|
|
if (enable_pin >= 0)
|
|
{
|
|
if (!(dev->gpio_en = mraa_gpio_init(enable_pin)))
|
|
{
|
|
printf("%s: mraa_gpio_init() failed.\n", __FUNCTION__);
|
|
nmea_gps_close(dev);
|
|
return NULL;
|
|
}
|
|
|
|
mraa_gpio_dir(dev->gpio_en, MRAA_GPIO_OUT);
|
|
|
|
// wake up
|
|
nmea_gps_enable(dev, true);
|
|
}
|
|
|
|
return dev;
|
|
}
|
|
|
|
// i2c ublox init
|
|
nmea_gps_context nmea_gps_init_ublox_i2c(unsigned int bus, uint8_t addr)
|
|
{
|
|
nmea_gps_context dev =
|
|
(nmea_gps_context)malloc(sizeof(struct _nmea_gps_context));
|
|
|
|
if (!dev)
|
|
return NULL;
|
|
|
|
// zero out context
|
|
memset((void *)dev, 0, sizeof(struct _nmea_gps_context));
|
|
|
|
dev->uart = NULL;
|
|
dev->i2c = NULL;
|
|
dev->gpio_en = NULL;
|
|
|
|
// initialize the MRAA contexts
|
|
|
|
if (!(dev->i2c = mraa_i2c_init(bus)))
|
|
{
|
|
printf("%s: mraa_i2c_init() failed.\n", __FUNCTION__);
|
|
nmea_gps_close(dev);
|
|
return NULL;
|
|
}
|
|
|
|
// This device cannot operate at more than 100Khz, so we set that
|
|
// here and bail if it fails.
|
|
|
|
if (mraa_i2c_frequency(dev->i2c, MRAA_I2C_STD))
|
|
{
|
|
printf("%s: mraa_i2c_frequency(MRAA_I2C_STD) failed.\n", __FUNCTION__);
|
|
nmea_gps_close(dev);
|
|
return NULL;
|
|
}
|
|
|
|
if (mraa_i2c_address(dev->i2c, addr))
|
|
{
|
|
printf("%s: mraa_i2c_address() failed.\n", __FUNCTION__);
|
|
nmea_gps_close(dev);
|
|
return NULL;
|
|
}
|
|
|
|
return dev;
|
|
}
|
|
|
|
void nmea_gps_close(nmea_gps_context dev)
|
|
{
|
|
assert(dev != NULL);
|
|
|
|
// sleepy-time
|
|
nmea_gps_enable(dev, false);
|
|
|
|
if (dev->uart)
|
|
mraa_uart_stop(dev->uart);
|
|
if (dev->i2c)
|
|
mraa_i2c_stop(dev->i2c);
|
|
if (dev->gpio_en)
|
|
mraa_gpio_close(dev->gpio_en);
|
|
|
|
free(dev);
|
|
}
|
|
|
|
upm_result_t nmea_gps_enable(const nmea_gps_context dev, bool enable)
|
|
{
|
|
assert(dev != NULL);
|
|
|
|
if (!dev->gpio_en)
|
|
return UPM_ERROR_NO_RESOURCES;
|
|
|
|
if (enable)
|
|
mraa_gpio_write(dev->gpio_en, 1);
|
|
else
|
|
mraa_gpio_write(dev->gpio_en, 0);
|
|
|
|
return UPM_SUCCESS;
|
|
}
|
|
|
|
int nmea_gps_read(const nmea_gps_context dev, char *buffer, size_t len)
|
|
{
|
|
assert(dev != NULL);
|
|
|
|
// i2c ublox
|
|
if (dev->i2c)
|
|
{
|
|
int rv;
|
|
if ((rv = readRegs(dev, UBLOX6_I2C_BYTE_STREAM, buffer, len)) < 0)
|
|
return rv;
|
|
|
|
// now we need to go through the bytes returned, and stop
|
|
// counting "real" bytes when we hit any character with the high
|
|
// bit set. The documentation implies that only a 0xff will be
|
|
// sent when no new data is available, but it seems sometimes
|
|
// the return contains 0xbf and 0xc3 bytes. So we stop counting
|
|
// as soon as we see any "8 bit" character (which isn't allowed
|
|
// by NMEA anyway).
|
|
int realBytes = 0;
|
|
int i;
|
|
|
|
for (i=0; i<rv; i++)
|
|
{
|
|
if (buffer[i] & 0x80)
|
|
break;
|
|
|
|
realBytes++;
|
|
}
|
|
|
|
return realBytes;
|
|
}
|
|
|
|
// uart
|
|
return mraa_uart_read(dev->uart, buffer, len);
|
|
}
|
|
|
|
int nmea_gps_write(const nmea_gps_context dev, char *buffer, size_t len)
|
|
{
|
|
assert(dev != NULL);
|
|
|
|
if (!dev->uart)
|
|
return UPM_ERROR_NO_RESOURCES;
|
|
|
|
return mraa_uart_write(dev->uart, buffer, len);
|
|
}
|
|
|
|
bool nmea_gps_data_available(const nmea_gps_context dev, unsigned int millis)
|
|
{
|
|
assert(dev != NULL);
|
|
|
|
// i2c ublox
|
|
if (dev->i2c)
|
|
{
|
|
// here millis is ignored
|
|
uint8_t h, l;
|
|
|
|
h = readReg(dev, UBLOX6_I2C_BYTES_AVAIL_H);
|
|
l = readReg(dev, UBLOX6_I2C_BYTES_AVAIL_L);
|
|
|
|
uint16_t total = (h << 8) | l;
|
|
// 0 means no data available, 0xffff means read errors
|
|
if (total == 0x0000 || total == 0xffff)
|
|
return false;
|
|
else
|
|
return true;
|
|
}
|
|
|
|
// uart
|
|
if (mraa_uart_data_available(dev->uart, millis))
|
|
return true;
|
|
else
|
|
return false;
|
|
}
|
|
|
|
upm_result_t nmea_gps_set_baudrate(const nmea_gps_context dev,
|
|
unsigned int baudrate)
|
|
{
|
|
assert(dev != NULL);
|
|
|
|
if (!dev->uart)
|
|
return UPM_ERROR_NO_RESOURCES;
|
|
|
|
if (mraa_uart_set_baudrate(dev->uart, baudrate))
|
|
{
|
|
printf("%s: mraa_uart_set_baudrate() failed.\n", __FUNCTION__);
|
|
return UPM_ERROR_OPERATION_FAILED;
|
|
}
|
|
|
|
return UPM_SUCCESS;
|
|
}
|