mirror of
https://github.com/eclipse/upm.git
synced 2025-03-26 02:10:02 +03:00

Signed-off-by: Adelin Dobre <adelin.dobre@rinftech.com> Signed-off-by: Mihai Tudor Panu <mihai.tudor.panu@intel.com>
454 lines
14 KiB
C++
454 lines
14 KiB
C++
/*
|
|
* Author: Jon Trulson <jtrulson@ics.com>
|
|
* Copyright (c) 2016-2017 Intel Corporation.
|
|
*
|
|
* The MIT License
|
|
*
|
|
* 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 <unistd.h>
|
|
#include <iostream>
|
|
#include <stdexcept>
|
|
#include <string>
|
|
|
|
#include "upm_string_parser.hpp"
|
|
#include "bmg160.hpp"
|
|
|
|
using namespace upm;
|
|
using namespace std;
|
|
|
|
// conversion from Celsius to Fahrenheit
|
|
|
|
static float c2f(float c)
|
|
{
|
|
return (c * (9.0 / 5.0) + 32.0);
|
|
}
|
|
|
|
BMG160::BMG160(int bus, int addr, int cs) :
|
|
m_bmg160(bmg160_init(bus, addr, cs))
|
|
{
|
|
if (!m_bmg160)
|
|
throw std::runtime_error(string(__FUNCTION__)
|
|
+ ": bmg160_init() failed");
|
|
}
|
|
|
|
BMG160::BMG160(std::string initStr) : mraaIo(initStr)
|
|
{
|
|
mraa_io_descriptor* descs = mraaIo.getMraaDescriptors();
|
|
std::vector<std::string> upmTokens;
|
|
|
|
if(!mraaIo.getLeftoverStr().empty()) {
|
|
upmTokens = UpmStringParser::parse(mraaIo.getLeftoverStr());
|
|
}
|
|
|
|
m_bmg160 = (bmg160_context)malloc(sizeof(struct _bmg160_context));
|
|
if(!m_bmg160) {
|
|
throw std::runtime_error(std::string(__FUNCTION__)
|
|
+ ": bmg160_init() failed");
|
|
}
|
|
|
|
// zero out context
|
|
memset((void *)m_bmg160, 0, sizeof(struct _bmg160_context));
|
|
|
|
// make sure MRAA is initialized
|
|
int mraa_rv;
|
|
if ((mraa_rv = mraa_init()) != MRAA_SUCCESS)
|
|
{
|
|
bmg160_close(m_bmg160);
|
|
throw std::runtime_error(std::string(__FUNCTION__)
|
|
+ ": mraa_init() failed");
|
|
}
|
|
|
|
if(descs->spis) {
|
|
m_bmg160->isSPI = true;
|
|
if( !(m_bmg160->spi = descs->spis[0]) ) {
|
|
bmg160_close(m_bmg160);
|
|
throw std::runtime_error(std::string(__FUNCTION__)
|
|
+ ": mraa_spi_init() failed");
|
|
}
|
|
if(descs->gpios) {
|
|
if( !(m_bmg160->gpioCS = descs->gpios[0]) ) {
|
|
bmg160_close(m_bmg160);
|
|
throw std::runtime_error(std::string(__FUNCTION__)
|
|
+ ": mraa_gpio_init() failed");
|
|
}
|
|
mraa_gpio_dir(m_bmg160->gpioCS, MRAA_GPIO_OUT);
|
|
} else {
|
|
throw std::runtime_error(std::string(__FUNCTION__)
|
|
+ ": mraa_gpio_init() failed");
|
|
}
|
|
mraa_spi_mode(m_bmg160->spi, MRAA_SPI_MODE0);
|
|
if (mraa_spi_frequency(m_bmg160->spi, 5000000)) {
|
|
bmg160_close(m_bmg160);
|
|
throw std::runtime_error(std::string(__FUNCTION__)
|
|
+ ": mraa_spi_frequency() failed");
|
|
}
|
|
} else {
|
|
// init the i2c context
|
|
if(!descs->i2cs) {
|
|
throw std::runtime_error(std::string(__FUNCTION__)
|
|
+ ": mraa_i2c_init() failed");
|
|
} else {
|
|
if( !(m_bmg160->i2c = descs->i2cs[0]) )
|
|
{
|
|
bmg160_close(m_bmg160);
|
|
throw std::runtime_error(std::string(__FUNCTION__)
|
|
+ ": mraa_i2c_init() failed");
|
|
}
|
|
}
|
|
}
|
|
|
|
// check the chip id
|
|
uint8_t chipID = bmg160_get_chip_id(m_bmg160);
|
|
if (chipID != BMG160_CHIPID)
|
|
{
|
|
printf("%s: invalid chip id: %02x. Expected %02x\n",
|
|
__FUNCTION__, chipID, BMG160_CHIPID);
|
|
bmg160_close(m_bmg160);
|
|
throw std::runtime_error(std::string(__FUNCTION__)
|
|
+ ": bmg160_init() failed");
|
|
}
|
|
|
|
// call devinit with default options
|
|
if (bmg160_devinit(m_bmg160, BMG160_POWER_MODE_NORMAL, BMG160_RANGE_250,
|
|
BMG160_BW_400_47))
|
|
{
|
|
printf("%s: bmg160_devinit() failed.\n", __FUNCTION__);
|
|
bmg160_close(m_bmg160);
|
|
throw std::runtime_error(std::string(__FUNCTION__)
|
|
+ ": bma250e_init() failed");
|
|
|
|
}
|
|
|
|
std::string::size_type sz;
|
|
for(std::string tok:upmTokens) {
|
|
if(tok.substr(0,11) == "enableFIFO:") {
|
|
bool useFIFO = std::stoi(tok.substr(11),&sz,0);
|
|
enableFIFO(useFIFO);
|
|
}
|
|
if(tok.substr(0,9) == "writeReg:") {
|
|
uint8_t reg = std::stoi(tok.substr(9),&sz,0);
|
|
tok = tok.substr(9);
|
|
uint8_t val = std::stoi(tok.substr(sz+1),nullptr,0);
|
|
writeReg(reg, val);
|
|
}
|
|
if(tok.substr(0,9) == "setRange:") {
|
|
BMG160_RANGE_T scale = (BMG160_RANGE_T)std::stoi(tok.substr(22),nullptr,0);
|
|
setRange(scale);
|
|
}
|
|
if(tok.substr(0,13) == "setBandwidth:") {
|
|
BMG160_BW_T bw = (BMG160_BW_T)std::stoi(tok.substr(13),nullptr,0);
|
|
setBandwidth(bw);
|
|
}
|
|
if(tok.substr(0,13) == "setPowerMode:") {
|
|
BMG160_POWER_MODE_T power = (BMG160_POWER_MODE_T)std::stoi(tok.substr(13),nullptr,0);
|
|
setPowerMode(power);
|
|
}
|
|
if(tok.substr(0,17) == "fifoSetWatermark:") {
|
|
BMG160_RANGE_T wm = (BMG160_RANGE_T)std::stoi(tok.substr(17),nullptr,0);
|
|
fifoSetWatermark(wm);
|
|
}
|
|
if(tok.substr(0,11) == "fifoConfig:") {
|
|
BMG160_FIFO_MODE_T mode = (BMG160_FIFO_MODE_T)std::stoi(tok.substr(11),&sz,0);
|
|
tok = tok.substr(11);
|
|
BMG160_FIFO_DATA_SEL_T axes = (BMG160_FIFO_DATA_SEL_T)std::stoi(tok.substr(sz+1),nullptr,0);
|
|
fifoConfig(mode, axes);
|
|
}
|
|
if(tok.substr(0,20) == "setInterruptEnable0:") {
|
|
u_int8_t bits = (u_int8_t)std::stoi(tok.substr(20),nullptr,0);
|
|
setInterruptEnable0(bits);
|
|
}
|
|
if(tok.substr(0,17) == "setInterruptMap0:") {
|
|
u_int8_t bits = (u_int8_t)std::stoi(tok.substr(17),nullptr,0);
|
|
setInterruptMap0(bits);
|
|
}
|
|
if(tok.substr(0,17) == "setInterruptMap1:") {
|
|
u_int8_t bits = (u_int8_t)std::stoi(tok.substr(17),nullptr,0);
|
|
setInterruptMap1(bits);
|
|
}
|
|
if(tok.substr(0,16) == "setInterruptSrc:") {
|
|
u_int8_t bits = (u_int8_t)std::stoi(tok.substr(16),nullptr,0);
|
|
setInterruptSrc(bits);
|
|
}
|
|
if(tok.substr(0,26) == "setInterruptOutputControl:") {
|
|
u_int8_t bits = (u_int8_t)std::stoi(tok.substr(26),nullptr,0);
|
|
setInterruptOutputControl(bits);
|
|
}
|
|
if(tok.substr(0,26) == "setInterruptLatchBehavior:") {
|
|
BMG160_RST_LATCH_T latch = (BMG160_RST_LATCH_T)std::stoi(tok.substr(26),nullptr,0);
|
|
setInterruptLatchBehavior(latch);
|
|
}
|
|
if(tok.substr(0,24) == "enableRegisterShadowing:") {
|
|
bool shadow = std::stoi(tok.substr(24),nullptr,0);
|
|
enableRegisterShadowing(shadow);
|
|
}
|
|
if(tok.substr(0,22) == "enableOutputFiltering:") {
|
|
bool filter = std::stoi(tok.substr(22),nullptr,0);
|
|
enableOutputFiltering(filter);
|
|
}
|
|
}
|
|
}
|
|
|
|
BMG160::~BMG160()
|
|
{
|
|
bmg160_close(m_bmg160);
|
|
}
|
|
|
|
void BMG160::init(BMG160_POWER_MODE_T pwr, BMG160_RANGE_T range,
|
|
BMG160_BW_T bw)
|
|
{
|
|
if (bmg160_devinit(m_bmg160, pwr, range, bw))
|
|
throw std::runtime_error(string(__FUNCTION__)
|
|
+ ": bmg160_devinit() failed");
|
|
}
|
|
|
|
void BMG160::update()
|
|
{
|
|
if (bmg160_update(m_bmg160))
|
|
throw std::runtime_error(string(__FUNCTION__)
|
|
+ ": bmg160_update() failed");
|
|
}
|
|
|
|
void BMG160::enableFIFO(bool useFIFO)
|
|
{
|
|
bmg160_enable_fifo(m_bmg160, useFIFO);
|
|
}
|
|
|
|
uint8_t BMG160::readReg(uint8_t reg)
|
|
{
|
|
return bmg160_read_reg(m_bmg160, reg);
|
|
}
|
|
|
|
int BMG160::readRegs(uint8_t reg, uint8_t *buffer, int len)
|
|
{
|
|
int rv = bmg160_read_regs(m_bmg160, reg, buffer, len);
|
|
if (rv < 0)
|
|
throw std::runtime_error(string(__FUNCTION__)
|
|
+ ": bmg160_read_regs() failed");
|
|
|
|
return rv;
|
|
}
|
|
|
|
void BMG160::writeReg(uint8_t reg, uint8_t val)
|
|
{
|
|
if (bmg160_write_reg(m_bmg160, reg, val))
|
|
throw std::runtime_error(string(__FUNCTION__)
|
|
+ ": bmg160_write_reg() failed");
|
|
}
|
|
|
|
uint8_t BMG160::getChipID()
|
|
{
|
|
return bmg160_get_chip_id(m_bmg160);
|
|
}
|
|
|
|
void BMG160::getGyroscope(float *x, float *y, float *z)
|
|
{
|
|
bmg160_get_gyroscope(m_bmg160, x, y, z);
|
|
}
|
|
|
|
std::vector<float> BMG160::getGyroscope()
|
|
{
|
|
update();
|
|
float v[3];
|
|
|
|
getGyroscope(&v[0], &v[1], &v[2]);
|
|
return std::vector<float>(v, v+3);
|
|
}
|
|
|
|
float BMG160::getTemperature(bool fahrenheit)
|
|
{
|
|
float temperature = bmg160_get_temperature(m_bmg160);
|
|
if (fahrenheit)
|
|
return c2f(temperature);
|
|
else
|
|
return temperature;
|
|
}
|
|
|
|
void BMG160::reset()
|
|
{
|
|
if (bmg160_reset(m_bmg160))
|
|
throw std::runtime_error(string(__FUNCTION__)
|
|
+ ": bmg160_reset() failed");
|
|
}
|
|
|
|
void BMG160::setRange(BMG160_RANGE_T range)
|
|
{
|
|
if (bmg160_set_range(m_bmg160, range))
|
|
throw std::runtime_error(string(__FUNCTION__)
|
|
+ ": bmg160_set_range() failed");
|
|
}
|
|
|
|
void BMG160::setBandwidth(BMG160_BW_T bw)
|
|
{
|
|
if (bmg160_set_bandwidth(m_bmg160, bw))
|
|
throw std::runtime_error(string(__FUNCTION__)
|
|
+ ": bmg160_set_bandwidth() failed");
|
|
}
|
|
|
|
void BMG160::setPowerMode(BMG160_POWER_MODE_T power)
|
|
{
|
|
if (bmg160_set_power_mode(m_bmg160, power))
|
|
throw std::runtime_error(string(__FUNCTION__)
|
|
+ ": bmg160_set_power_mode() failed");
|
|
}
|
|
|
|
void BMG160::fifoSetWatermark(int wm)
|
|
{
|
|
if (bmg160_fifo_set_watermark(m_bmg160, wm))
|
|
throw std::runtime_error(string(__FUNCTION__)
|
|
+ ": bmg160_fifo_set_watermark() failed");
|
|
}
|
|
|
|
void BMG160::fifoConfig(BMG160_FIFO_MODE_T mode, BMG160_FIFO_DATA_SEL_T axes)
|
|
{
|
|
if (bmg160_fifo_config(m_bmg160, mode, axes))
|
|
throw std::runtime_error(string(__FUNCTION__)
|
|
+ ": bmg160_fifo_config() failed");
|
|
}
|
|
|
|
uint8_t BMG160::getInterruptEnable0()
|
|
{
|
|
return bmg160_get_interrupt_enable0(m_bmg160);
|
|
}
|
|
|
|
void BMG160::setInterruptEnable0(uint8_t bits)
|
|
{
|
|
if (bmg160_set_interrupt_enable0(m_bmg160, bits))
|
|
throw std::runtime_error(string(__FUNCTION__)
|
|
+ ": bmg160_set_interrupt_enable0() failed");
|
|
}
|
|
|
|
uint8_t BMG160::getInterruptMap0()
|
|
{
|
|
return bmg160_get_interrupt_map0(m_bmg160);
|
|
}
|
|
|
|
void BMG160::setInterruptMap0(uint8_t bits)
|
|
{
|
|
if (bmg160_set_interrupt_map0(m_bmg160, bits))
|
|
throw std::runtime_error(string(__FUNCTION__)
|
|
+ ": bmg160_set_interrupt_map0() failed");
|
|
}
|
|
|
|
uint8_t BMG160::getInterruptMap1()
|
|
{
|
|
return bmg160_get_interrupt_map1(m_bmg160);
|
|
}
|
|
|
|
void BMG160::setInterruptMap1(uint8_t bits)
|
|
{
|
|
if (bmg160_set_interrupt_map1(m_bmg160, bits))
|
|
throw std::runtime_error(string(__FUNCTION__)
|
|
+ ": bmg160_set_interrupt_map1() failed");
|
|
}
|
|
|
|
uint8_t BMG160::getInterruptSrc()
|
|
{
|
|
return bmg160_get_interrupt_src(m_bmg160);
|
|
}
|
|
|
|
void BMG160::setInterruptSrc(uint8_t bits)
|
|
{
|
|
if (bmg160_set_interrupt_src(m_bmg160, bits))
|
|
throw std::runtime_error(string(__FUNCTION__)
|
|
+ ": bmg160_set_interrupt_src() failed");
|
|
}
|
|
|
|
uint8_t BMG160::getInterruptOutputControl()
|
|
{
|
|
return bmg160_get_interrupt_output_control(m_bmg160);
|
|
}
|
|
|
|
void BMG160::setInterruptOutputControl(uint8_t bits)
|
|
{
|
|
if (bmg160_set_interrupt_output_control(m_bmg160, bits))
|
|
throw std::runtime_error(string(__FUNCTION__)
|
|
+ ": bmg160_set_interrupt_output_control() failed");
|
|
}
|
|
|
|
void BMG160::clearInterruptLatches()
|
|
{
|
|
if (bmg160_clear_interrupt_latches(m_bmg160))
|
|
throw std::runtime_error(string(__FUNCTION__)
|
|
+ ": bmg160_clear_interrupt_latches() failed");
|
|
}
|
|
|
|
BMG160_RST_LATCH_T BMG160::getInterruptLatchBehavior()
|
|
{
|
|
return bmg160_get_interrupt_latch_behavior(m_bmg160);
|
|
}
|
|
|
|
void BMG160::setInterruptLatchBehavior(BMG160_RST_LATCH_T latch)
|
|
{
|
|
if (bmg160_set_interrupt_latch_behavior(m_bmg160, latch))
|
|
throw std::runtime_error(string(__FUNCTION__)
|
|
+ ": bmg160_set_interrupt_latch_behavior() failed");
|
|
}
|
|
|
|
void BMG160::enableRegisterShadowing(bool shadow)
|
|
{
|
|
if (bmg160_enable_register_shadowing(m_bmg160, shadow))
|
|
throw std::runtime_error(string(__FUNCTION__)
|
|
+ ": bmg160_enable_register_shadowing() failed");
|
|
}
|
|
|
|
void BMG160::enableOutputFiltering(bool filter)
|
|
{
|
|
if (bmg160_enable_output_filtering(m_bmg160, filter))
|
|
throw std::runtime_error(string(__FUNCTION__)
|
|
+ ": bmg160_enable_output_filtering() failed");
|
|
}
|
|
|
|
uint8_t BMG160::getInterruptStatus0()
|
|
{
|
|
return bmg160_get_interrupt_status0(m_bmg160);
|
|
}
|
|
|
|
uint8_t BMG160::getInterruptStatus1()
|
|
{
|
|
return bmg160_get_interrupt_status1(m_bmg160);
|
|
}
|
|
|
|
uint8_t BMG160::getInterruptStatus2()
|
|
{
|
|
return bmg160_get_interrupt_status2(m_bmg160);
|
|
}
|
|
|
|
uint8_t BMG160::getInterruptStatus3()
|
|
{
|
|
return bmg160_get_interrupt_status3(m_bmg160);
|
|
}
|
|
|
|
void BMG160::installISR(BMG160_INTERRUPT_PINS_T intr, int gpio,
|
|
mraa::Edge level,
|
|
void (*isr)(void *), void *arg)
|
|
{
|
|
if (bmg160_install_isr(m_bmg160, intr, gpio,
|
|
(mraa_gpio_edge_t)level, isr, arg))
|
|
throw std::runtime_error(string(__FUNCTION__)
|
|
+ ": bmg160_install_isr() failed");
|
|
}
|
|
|
|
void BMG160::uninstallISR(BMG160_INTERRUPT_PINS_T intr)
|
|
{
|
|
bmg160_uninstall_isr(m_bmg160, intr);
|
|
}
|