2022-12-22 14:03:01 +01:00

271 lines
7.3 KiB
C++

#pragma once
//
// FILE: AS5600.h
// AUTHOR: Rob Tillaart
// VERSION: 0.3.4
// PURPOSE: Arduino library for AS5600 magnetic rotation meter
// DATE: 2022-05-28
// URL: https://github.com/RobTillaart/AS5600
#include "Arduino.h"
#include "Wire.h"
#define AS5600_LIB_VERSION (F("0.3.4"))
// default addresses
const uint8_t AS5600_DEFAULT_ADDRESS = 0x36;
const uint8_t AS5600L_DEFAULT_ADDRESS = 0x40;
// setDirection
const uint8_t AS5600_CLOCK_WISE = 0; // LOW
const uint8_t AS5600_COUNTERCLOCK_WISE = 1; // HIGH
// 0.087890625;
const float AS5600_RAW_TO_DEGREES = 360.0 / 4096;
// 0.00153398078788564122971808758949;
const float AS5600_RAW_TO_RADIANS = PI * 2.0 / 4096;
// 4.06901041666666e-6
const float AS5600_RAW_TO_RPM = 60.0 / 4096;
// getAngularSpeed
const uint8_t AS5600_MODE_DEGREES = 0;
const uint8_t AS5600_MODE_RADIANS = 1;
const uint8_t AS5600_MODE_RPM = 2;
// CONFIGURE CONSTANTS
// check datasheet for details
// setOutputMode
const uint8_t AS5600_OUTMODE_ANALOG_100 = 0;
const uint8_t AS5600_OUTMODE_ANALOG_90 = 1;
const uint8_t AS5600_OUTMODE_PWM = 2;
// setPowerMode
const uint8_t AS5600_POWERMODE_NOMINAL = 0;
const uint8_t AS5600_POWERMODE_LOW1 = 1;
const uint8_t AS5600_POWERMODE_LOW2 = 2;
const uint8_t AS5600_POWERMODE_LOW3 = 3;
// setPWMFrequency
const uint8_t AS5600_PWM_115 = 0;
const uint8_t AS5600_PWM_230 = 1;
const uint8_t AS5600_PWM_460 = 2;
const uint8_t AS5600_PWM_920 = 3;
// setHysteresis
const uint8_t AS5600_HYST_OFF = 0;
const uint8_t AS5600_HYST_LSB1 = 1;
const uint8_t AS5600_HYST_LSB2 = 2;
const uint8_t AS5600_HYST_LSB3 = 3;
// setSlowFilter
const uint8_t AS5600_SLOW_FILT_16X = 0;
const uint8_t AS5600_SLOW_FILT_8X = 1;
const uint8_t AS5600_SLOW_FILT_4X = 2;
const uint8_t AS5600_SLOW_FILT_2X = 3;
// setFastFilter
const uint8_t AS5600_FAST_FILT_NONE = 0;
const uint8_t AS5600_FAST_FILT_LSB6 = 1;
const uint8_t AS5600_FAST_FILT_LSB7 = 2;
const uint8_t AS5600_FAST_FILT_LSB9 = 3;
const uint8_t AS5600_FAST_FILT_LSB18 = 4;
const uint8_t AS5600_FAST_FILT_LSB21 = 5;
const uint8_t AS5600_FAST_FILT_LSB24 = 6;
const uint8_t AS5600_FAST_FILT_LSB10 = 7;
// setWatchDog
const uint8_t AS5600_WATCHDOG_OFF = 0;
const uint8_t AS5600_WATCHDOG_ON = 1;
class AS5600
{
public:
AS5600(TwoWire *wire = &Wire);
#if defined (ESP8266) || defined(ESP32)
// 255 is software controlled direction pin
bool begin(int dataPin, int clockPin, uint8_t directionPin = 255);
#endif
// 255 is software controlled direction pin
bool begin(uint8_t directionPin = 255); // MAGIC NR => const int?
bool isConnected();
uint8_t getAddress(); // 0x36 for AS5600, 0x40 for AS5600L
// SET CONFIGURE REGISTERS
// read datasheet first
// 0 = AS5600_CLOCK_WISE
// 1 = AS5600_COUNTERCLOCK_WISE
// all other = AS5600_COUNTERCLOCK_WISE
void setDirection(uint8_t direction = AS5600_CLOCK_WISE);
uint8_t getDirection();
uint8_t getZMCO();
// 0 .. 4095
// returns false if parameter out of range
bool setZPosition(uint16_t value);
uint16_t getZPosition();
// 0 .. 4095
// returns false if parameter out of range
bool setMPosition(uint16_t value);
uint16_t getMPosition();
// 0 .. 4095
// returns false if parameter out of range
bool setMaxAngle(uint16_t value);
uint16_t getMaxAngle();
// access the whole configuration register
// check datasheet for bit fields
// returns false if parameter out of range
bool setConfigure(uint16_t value);
uint16_t getConfigure();
// access details of the configuration register
// 0 = Normal
// 1,2,3 are low power mode - check datasheet
// returns false if parameter out of range
bool setPowerMode(uint8_t powerMode);
uint8_t getPowerMode();
// 0 = off 1 = lsb1 2 = lsb2 3 = lsb3
// returns false if parameter out of range
// suppresses noise when the magnet is not moving.
bool setHysteresis(uint8_t hysteresis);
uint8_t getHysteresis();
// 0 = analog 0-100%
// 1 = analog 10-90%
// 2 = PWM
// returns false if parameter out of range
bool setOutputMode(uint8_t outputMode);
uint8_t getOutputMode();
// 0 = 115 1 = 230 2 = 460 3 = 920 (Hz)
// returns false if parameter out of range
bool setPWMFrequency(uint8_t pwmFreq);
uint8_t getPWMFrequency();
// 0 = 16x 1 = 8x 2 = 4x 3 = 2x
// returns false if parameter out of range
bool setSlowFilter(uint8_t mask);
uint8_t getSlowFilter();
// 0 = none 1 = LSB6 2 = LSB7 3 = LSB9
// 4 = LSB18 5 = LSB21 6 = LSB24 7 = LSB10
// returns false if parameter out of range
bool setFastFilter(uint8_t mask);
uint8_t getFastFilter();
// 0 = OFF
// 1 = ON (auto low power mode)
// returns false if parameter out of range
bool setWatchDog(uint8_t mask);
uint8_t getWatchDog();
// READ OUTPUT REGISTERS
uint16_t rawAngle();
uint16_t readAngle();
// software based offset.
// degrees = -359.99 .. 359.99 (preferred)
// returns false if abs(parameter) > 36000
// => expect loss of precision
bool setOffset(float degrees);
float getOffset();
// READ STATUS REGISTERS
uint8_t readStatus();
uint8_t readAGC();
uint16_t readMagnitude();
// access detail status register
bool detectMagnet();
bool magnetTooStrong();
bool magnetTooWeak();
// BURN COMMANDS
// DO NOT UNCOMMENT - USE AT OWN RISK - READ DATASHEET
// void burnAngle();
// void burnSetting();
// Experimental 0.1.2 - to be tested.
// approximation of the angular speed in rotations per second.
// mode == 1: radians /second
// mode == 0: degrees /second (default)
float getAngularSpeed(uint8_t mode = AS5600_MODE_DEGREES);
// EXPERIMENTAL CUMULATIVE POSITION
// reads sensor and updates cumulative position
int32_t getCumulativePosition();
// converts last position to whole revolutions.
int32_t getRevolutions();
// resets position, returns last position.
int32_t resetPosition();
protected:
uint8_t readReg(uint8_t reg);
uint16_t readReg2(uint8_t reg);
uint8_t writeReg(uint8_t reg, uint8_t value);
uint8_t writeReg2(uint8_t reg, uint16_t value);
uint8_t _address = AS5600_DEFAULT_ADDRESS;
uint8_t _directionPin = 255;
uint8_t _direction = AS5600_CLOCK_WISE;
uint8_t _error = 0;
TwoWire* _wire;
// for getAngularSpeed()
uint32_t _lastMeasurement = 0;
int16_t _lastAngle = 0;
// for readAngle() and rawAngle()
uint16_t _offset = 0;
// EXPERIMENTAL
// cumulative position counter
// works only if the sensor is read often enough.
int32_t _position = 0;
int16_t _lastPosition = 0;
};
/////////////////////////////////////////////////////////////////////////////
//
// AS5600L
//
class AS5600L : public AS5600
{
public:
AS5600L(uint8_t address = AS5600L_DEFAULT_ADDRESS, TwoWire *wire = &Wire);
bool setAddress(uint8_t address);
// UPDT = UPDATE
// are these two needed?
bool setI2CUPDT(uint8_t value);
uint8_t getI2CUPDT();
};
// -- END OF FILE --