/* * Author: Jon Trulson * 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. */ #pragma once #include #include #include #include "bma250e.h" namespace upm { /** * @brief Bosch 3-axis Accelerometer * @defgroup bma250e libupm-bma250e * @ingroup i2c spi gpio bosch accelerometer */ /** * @library bma250e * @sensor bma250e * @comname Digital Triaxial Acceleration Sensor * @type accelerometer * @man bosch * @con i2c spi gpio * @web https://www.bosch-sensortec.com/bst/products/all_products/bma250e * * @brief API for the BMA250E 10 bit Triaxial Accelerometer * * The BMA250E is a triaxial, low-g acceleration sensor with digital * output for consumer applications. It allows measurements of * acceleration in three perpendicular axes. An evaluation circuitry * (ASIC) converts the output of a micromechanical * acceleration-sensing structure (MEMS) that works according to the * differential capacitance principle. * * Not all functionality of this chip has been implemented in this * driver, however all the pieces are present to add any desired * functionality. This driver supports both I2C (default) and SPI * operation. * * This driver attempts to support verious flavors of this chip, * such as the version on the BMX050, BMI050 (chipid 0xfa) and the * version on the bmc050 (chipid 0x03). Not all functionality is * appropriate, or even present on all chips. Consult the relevant * datasheets. * * This device requires 3.3v operation. * * @snippet bma250e.cxx Interesting */ class BMA250E { public: /** * BMA250E constructor. * * This device can support both I2C and SPI. For SPI, set the addr * to -1, and specify a positive integer representing the Chip * Select (CS) pin for the cs argument. If you are using a * hardware CS pin (like edison with arduino breakout), then you * can connect the proper pin to the hardware CS pin on your MCU * and supply -1 for cs. The default operating mode is I2C. * * @param bus I2C or SPI bus to use. * @param addr The address for this device. -1 for SPI. * @param cs The gpio pin to use for the SPI Chip Select. -1 for * I2C or for SPI with a hardware controlled pin. * @throws std::runtime_error on initialization failure. */ BMA250E(int bus=BMA250E_DEFAULT_I2C_BUS, int addr=BMA250E_DEFAULT_ADDR, int cs=-1); /** * BMA250E Destructor. */ ~BMA250E(); /** * Update the internal stored values from sensor data. * * @throws std::runtime_error on failure. */ void update(); /** * Return the chip ID. * * @return The chip ID. */ uint8_t getChipID(); /** * Return accelerometer data in gravities. update() must have * been called prior to calling this method. * * @param x Pointer to a floating point value that will have the * current x component placed into it. * @param y Pointer to a floating point value that will have the * current y component placed into it. * @param z Pointer to a floating point value that will have the * current z component placed into it. */ void getAccelerometer(float *x, float *y, float *z); /** * Return accelerometer data in gravities in the form of a * floating point vector. update() must have been called * prior to calling this method. * * @return A floating point vector containing x, y, and z in * that order. */ std::vector getAccelerometer(); /** * Return the current measured temperature. Note, this is not * ambient temperature. update() must have been called prior to * calling this method. * * @param fahrenheit true to return data in Fahrenheit, false for * Celicus. Celsius is the default. * @return The temperature in degrees Celsius or Fahrenheit. */ float getTemperature(bool fahrenheit=false); /** * Initialize the device and start operation. This function is * called from the constructor so will not typically need to be * called by a user unless the device is reset. * * @param pwr One of the BMA250E_POWER_MODE_T values. The default is * BMA250E_POWER_MODE_NORMAL. * @param range One of the BMA250E_RANGE_T values. The default is * BMA250E_RANGE_2G. * @param bw One of the filtering BMA250E_BW_T values. The default is * BMA250E_BW_250. * @throws std::runtime_error on failure. */ void init(BMA250E_POWER_MODE_T pwr=BMA250E_POWER_MODE_NORMAL, BMA250E_RANGE_T range=BMA250E_RANGE_2G, BMA250E_BW_T bw=BMA250E_BW_250); /** * Reset the device as if during a power on reset. All configured * values are lost when this happens. You should call init() * afterwards, or at least perform the same initialization init() * does before continuing. * * @throws std::runtime_error on failure. */ void reset(); /** * Set the acceleration scaling range. This device supports 2, 4, * 8, and 16g ranges. * * @param range One of the BMA250E_RANGE_T values. * @throws std::runtime_error on failure. */ void setRange(BMA250E_RANGE_T range); /** * Set the output filtering bandwidth of the device. * * @param bw One of the BMA250E_BW_T values. * @throws std::runtime_error on failure. */ void setBandwidth(BMA250E_BW_T bw); /** * Set the power mode of the device. Care must be taken when * setting a low power or suspend mode. By default init() calls * setLowPowerMode2() to ensure that if any of these modes are * entered we can still talk to the device. The default low power * mode is LPM1, which requires slowing down register writes, * which we cannot support. setLowPowerMode2() enables LPM2 which * keeps the digital interface operational in low power or suspend * modes. See the datasheet for details. * * So if you reset your device and don't call init() or * setLowPowerMode2(), you could lose control of the device by * calling this function with anything other than * POWER_MODE_NORMAL. You've been warned :) * * @param power One of the BMA250E_POWER_MODE_T values. * @throws std::runtime_error on failure. */ void setPowerMode(BMA250E_POWER_MODE_T power); /** * Enable update() to read from the FIFO rather than the * acceleration axis registers directly. init() enables this mode * by default if the chip variant supports a FIFO. An advantage * to this mode that all axis data is sampled from the same * timeslice. When reading directly from the acceleration output * registers, it's possible for one axis to be updated while * another is being read, causing a temporal anomaly that even * Captain Picard can't resolve. If there is no FIFO present, * this call is ignored. * * Using the FIFO removes this problem. * * @param useFIFO True to enable update() to read from the FIFO. * When false, update will read from the acceleration output * registers directly. */ void enableFIFO(bool useFIFO); /** * Set the FIFO watermark. When the watermark is reached an * interrupt (if enabled) will be generated. If there is no FIFO * present, this call is ignored. * * @param wm The FIFO watermark to use. The maximum value is 63. * @throws std::runtime_error on failure. */ void fifoSetWatermark(int wm); /** * Set the FIFO configuration. init() uses the FIFO_MODE_BYPASS * mode with axes set to FIFO_DATA_SEL_XYZ by default. If there * is no FIFO present, this call is ignored. * * @param mode One of the BMA250E_FIFO_MODE_T values. * @param axes One of the BMA250E_FIFO_DATA_SEL_T values. * @throws std::runtime_error on failure. */ void fifoConfig(BMA250E_FIFO_MODE_T mode, BMA250E_FIFO_DATA_SEL_T axes); /** * Enable, disable, and configure the built in self test on a per * axis basis. See the datasheet for details. * * @param sign True for a positive deflection, false for negative * @param amp True for a high deflection, false for a low deflection * @param axis One of the BMA250E_SELFTTEST_AXIS_T values. * Note, only one axis at a time can be tested. Accelerometer * output for other axes should be ignored. * @throws std::runtime_error on failure. */ void setSelfTest(bool sign, bool amp, BMA250E_SELFTTEST_AXIS_T axis); /** * Return the Interrupt Enables 0 register. These registers * allow you to enable various interrupt conditions. See the * datasheet for details. * * @return A bitmask of BMA250E_INT_EN_0_BITS_T bits. */ uint8_t getInterruptEnable0(); /** * Set the Interrupt Enables 0 register. See the datasheet for * details. * * @param bits A bitmask of BMA250E_INT_EN_0_BITS_T bits. * @throws std::runtime_error on failure. */ void setInterruptEnable0(uint8_t bits); /** * Return the Interrupt Enables 1 register. See the datasheet for * details. * * @return A bitmask of BMA250E_INT_EN_1_BITS_T bits. */ uint8_t getInterruptEnable1(); /** * Set the Interrupt Enables 1 register. See the datasheet for * details. * * @param bits A bitmask of BMA250E_INT_EN_1_BITS_T bits. * @throws std::runtime_error on failure. */ void setInterruptEnable1(uint8_t bits); /** * Return the Interrupt Enables 2 register. See the datasheet for * details. * * @return A bitmask of BMA250E_INT_EN_2_BITS_T bits. */ uint8_t getInterruptEnable2(); /** * Set the Interrupt Enables 2 register. See the datasheet for * details. * * @param bits A bitmask of BMA250E_INT_EN_2_BITS_T bits. * @throws std::runtime_error on failure. */ void setInterruptEnable2(uint8_t bits); /** * Return the Interrupt Map 0 register. These registers allow you * to map specific interrupts to the interrupt 1 or interrupt 2 * pin. See the datasheet for details. * * @return A bitmask of BMA250E_INT_MAP_0_BITS_T bits. */ uint8_t getInterruptMap0(); /** * Set the Interrupt Map 0 register. These registers allow you * to map specific interrupts to the interrupt 1 or interrupt 2 * pin. See the datasheet for details. * * @param bits A bitmask of BMA250E_INT_MAP_0_BITS_T bits. * @throws std::runtime_error on failure. */ void setInterruptMap0(uint8_t bits); /** * Return the Interrupt Map 1 register. See the datasheet for * details. * * @return A bitmask of BMA250E_INT_MAP_1_BITS_T bits. */ uint8_t getInterruptMap1(); /** * Set the Interrupt Map 1 register. See the datasheet for * details. * * @param bits A bitmask of BMA250E_INT_MAP_1_BITS_T bits. * @throws std::runtime_error on failure. */ void setInterruptMap1(uint8_t bits); /** * Return the Interrupt Map 2 register. See the datasheet for * details. * * @return A bitmask of BMA250E_INT_MAP_2_BITS_T bits. */ uint8_t getInterruptMap2(); /** * Set the Interrupt Map 2 register. See the datasheet for * details. * * @param bits A bitmask of BMA250E_INT_MAP_2_BITS_T bits. * @throws std::runtime_error on failure. */ void setInterruptMap2(uint8_t bits); /** * Return the Interrupt source register. This register allows * determining where data comes from (filtered/unfiltered) for * those interrupt sources where this is selectable. See the * datasheet for details. * * @return A bitmask of BMA250E_INT_SRC_BITS_T bits. */ uint8_t getInterruptSrc(); /** * Set the Interrupt source register. This register allows * determining where data comes from (filtered/unfiltered) for * those interrupt sources where this is selectable. See the * datasheet for details. * * @param bits A bitmask of BMA250E_INT_SRC_BITS_T bits. * @throws std::runtime_error on failure. */ void setInterruptSrc(uint8_t bits); /** * Return the Interrupt output control register. This register * allows determining the electrical characteristics of the 2 * interrupt pins (open-drain/push-pull and level/edge * triggering). See the datasheet for details. * * @return A bitmask of BMA250E_INT_OUT_CTRL_BITS_T bits. */ uint8_t getInterruptOutputControl(); /** * Set the Interrupt output control register. This register * allows determining the electrical characteristics of the 2 * interrupt pins (open-drain/push-pull and level/edge * triggering). See the datasheet for details. * * @param bits A bitmask of BMA250E_INT_OUT_CTRL_BITS_T bits. * @throws std::runtime_error on failure. */ void setInterruptOutputControl(uint8_t bits); /** * Clear all latched interrupts. See the datasheet for details. * * @throws std::runtime_error on failure. */ void clearInterruptLatches(); /** * Return the current interrupt latching behavior. See the * datasheet for details. * * @return One of the BMA250E_RST_LATCH_T values. */ BMA250E_RST_LATCH_T getInterruptLatchBehavior(); /** * Set the current interrupt latching behavior. See the datasheet * for details. * * @param latch One of the BMA250E_RST_LATCH_T values. * @throws std::runtime_error on failure. */ void setInterruptLatchBehavior(BMA250E_RST_LATCH_T latch); /** * Return the interrupt status 0 register. These registers * indicate which interrupts have been triggered. See the * datasheet for details. * * @return A bitmask of BMA250E_INT_STATUS_0_BITS_T bits. */ uint8_t getInterruptStatus0(); /** * Return the interrupt status 1 register. See the datasheet for * details. * * @return A bitmask of BMA250E_INT_STATUS_1_BITS_T bits. */ uint8_t getInterruptStatus1(); /** * Return the interrupt status 2 register. See the datasheet for * details. * * @return A bitmask of BMA250E_INT_STATUS_2_BITS_T bits. */ uint8_t getInterruptStatus2(); /** * Return the interrupt status 3 register bitfields. See the * datasheet for details. The Orientation value is not returned by * this function, see getInterruptStatus3Orientation() for that * information. * * @return A bitmask of BMA250E_INT_STATUS_3_BITS_T bits ONLY. */ uint8_t getInterruptStatus3Bits(); /** * Return the interrupt status 3 register Orientation value. See the * datasheet for details. * * @return One of the BMA250E_ORIENT_T values. */ BMA250E_ORIENT_T getInterruptStatus3Orientation(); /** * Enable shadowing of the accelerometer output registers. When * enabled, a read of an axis LSB register automatically locks the * MSB register of that axis until it has been read. This is * usually a good thing to have enabled. init() enables this by * default. If disabled, then it becomes possible for part of an * axis value to change while another part is being read, causing * inconsistent data. * * @param shadow True to enable axis register shadowing, false * otherwise. * @throws std::runtime_error on failure. */ void enableRegisterShadowing(bool shadow); /** * Enable filtering of the accelerometer axis data. init() * enables this by default. If disabled, then accelerometer data * that is read will be raw and unfiltered (rated R). See the * datasheet for details. * * @param filter True to enable filtering, false to disable. * @throws std::runtime_error on failure. */ void enableOutputFiltering(bool filter); /** * Make sure low power mode config (LPM2) is set in case we later * go into the low power or suspend power modes. LPM1 mode (the * default) requires drastically slowed register writes which we * cannot handle. * * @throws std::runtime_error on failure. */ void setLowPowerMode2(); /** * install an interrupt handler. * * @param intr One of the BMA250E_INTERRUPT_PINS_T values * specifying which interrupt pin you are installing. * @param gpio GPIO pin to use as interrupt pin. * @param level The interrupt trigger level (one of mraa::Edge * values). Make sure that you have configured the interrupt pin * properly for whatever level you choose. * @param isr The interrupt handler, accepting a void * argument. * @param arg The argument to pass the the interrupt handler. * @throws std::runtime_error on failure. */ void installISR(BMA250E_INTERRUPT_PINS_T intr, int gpio, mraa::Edge level, void (*isr)(void *), void *arg); /** * uninstall a previously installed interrupt handler * * @param intr One of the BMA250E_INTERRUPT_PINS_T values * specifying which interrupt pin you are removing. */ void uninstallISR(BMA250E_INTERRUPT_PINS_T intr); /** * Read a register. * * @param reg The register to read. * @return The value of the register. */ uint8_t readReg(uint8_t reg); /** * Read contiguous registers into a buffer. * * @param buffer The buffer to store the results. * @param len The number of registers to read. * @return The number of bytes read. * @throws std::runtime_error on failure. */ int readRegs(uint8_t reg, uint8_t *buffer, int len); /** * Write to a register. * * @param reg The register to write to. * @param val The value to write. * @throws std::runtime_error on failure. */ void writeReg(uint8_t reg, uint8_t val); protected: bma250e_context m_bma250e; private: /* Disable implicit copy and assignment operators */ BMA250E(const BMA250E&) = delete; BMA250E &operator=(const BMA250E&) = delete; }; }