GY-63_MS5611/libraries/AS5600/README.md

664 lines
22 KiB
Markdown
Raw Normal View History

2022-05-29 16:07:48 -04:00
[![Arduino CI](https://github.com/RobTillaart/AS5600/workflows/Arduino%20CI/badge.svg)](https://github.com/marketplace/actions/arduino_ci)
[![Arduino-lint](https://github.com/RobTillaart/AS5600/actions/workflows/arduino-lint.yml/badge.svg)](https://github.com/RobTillaart/AS5600/actions/workflows/arduino-lint.yml)
[![JSON check](https://github.com/RobTillaart/AS5600/actions/workflows/jsoncheck.yml/badge.svg)](https://github.com/RobTillaart/AS5600/actions/workflows/jsoncheck.yml)
[![License: MIT](https://img.shields.io/badge/license-MIT-green.svg)](https://github.com/RobTillaart/AS5600/blob/master/LICENSE)
[![GitHub release](https://img.shields.io/github/release/RobTillaart/AS5600.svg?maxAge=3600)](https://github.com/RobTillaart/AS5600/releases)
# AS5600
2022-09-01 05:31:32 -04:00
Arduino library for AS5600 and AS5600L magnetic rotation meter.
2022-05-29 16:07:48 -04:00
## Description
2023-02-01 10:57:56 -05:00
#### AS5600
**AS5600** is a library for an AS5600 / AS5600L based magnetic **rotation** meter.
2023-02-21 05:41:58 -05:00
More exact, it measures the angle (rotation w.r.t. reference) and not RPM.
2023-02-01 10:57:56 -05:00
Multiple angle measurements allows one to calculate / estimate the RPM.
2022-05-29 16:07:48 -04:00
2023-02-01 10:57:56 -05:00
The AS5600 and AS5600L sensors are pin compatible (always check datasheet).
2022-05-29 16:07:48 -04:00
2022-07-07 06:53:02 -04:00
**Warning: experimental**
2022-05-29 16:07:48 -04:00
2022-06-27 05:27:48 -04:00
The sensor can measure a full rotation in 4096 steps.
2023-02-01 10:57:56 -05:00
The precision of the position is therefore limited to approx 0.1°.
2022-09-01 05:31:32 -04:00
Noise levels are unknown, but one might expect that the sensor is affected by electric
2022-06-27 05:27:48 -04:00
and or magnetic signals in the environment.
2022-07-07 06:53:02 -04:00
Also unknown is the influence of metals near the sensor or an unstable
or fluctuating power supply.
2022-06-27 05:27:48 -04:00
2022-09-01 05:31:32 -04:00
Please share your experiences.
2023-02-01 10:57:56 -05:00
#### Related libraries
2022-09-01 05:31:32 -04:00
- https://github.com/RobTillaart/Angle
2022-12-21 15:44:41 -05:00
- https://github.com/RobTillaart/AngleConvertor
2022-09-01 05:31:32 -04:00
- https://github.com/RobTillaart/AverageAngle
- https://github.com/RobTillaart/runningAngle
2022-06-27 05:27:48 -04:00
2022-10-17 03:48:27 -04:00
## Hardware connection
2022-05-29 16:07:48 -04:00
2022-10-17 03:48:27 -04:00
The I2C address of the **AS5600** is always 0x36.
2022-07-07 06:53:02 -04:00
2022-10-17 03:48:27 -04:00
The AS5600 datasheet states it supports Fast-Mode == 400 KHz
and Fast-Mode-Plus == 1000 KHz.
2023-02-01 10:57:56 -05:00
Tests with an AS5600L (UNO) failed at 400 KHz (needs investigation).
2022-05-29 16:07:48 -04:00
2022-10-17 03:48:27 -04:00
The sensor should connect the I2C lines SDA and SCL and the
VCC and GND to communicate with the processor.
2023-02-01 10:57:56 -05:00
Do not forget to add the pull up resistors to improve the I2C signals.
2022-10-17 03:48:27 -04:00
2023-02-01 10:57:56 -05:00
#### DIR pin
2022-10-17 03:48:27 -04:00
From datasheet, page 30 - Direction (clockwise vs. counter-clockwise)
_**The AS5600 allows controlling the direction of the magnet
rotation with the DIR pin. If DIR is connected to GND (DIR = 0)
a clockwise rotation viewed from the top will generate an
increment of the calculated angle. If the DIR pin is connected
to VDD (DIR = 1) an increment of the calculated angle will
happen with counter-clockwise rotation.**_
2022-09-01 05:31:32 -04:00
2022-10-17 03:48:27 -04:00
This AS5600 library offers a 3rd option for the DIR (direction) pin of the sensor:
2022-09-01 05:31:32 -04:00
2022-10-17 03:48:27 -04:00
1. Connect to **GND** ==> fixed clockwise(\*). This is the default.
1. Connect to **VCC** ==> fixed counter-clockwise.
1. Connect to an IO pin of the processor == Hardware Direction Control by library.
2022-09-01 05:31:32 -04:00
2022-10-17 03:48:27 -04:00
In the 3rd configuration the library controls the direction of counting by initializing
this pin in **begin(directionPin)**, followed by **setDirection(direction)**.
For the parameter direction the library defines two constants named:
2022-09-01 05:31:32 -04:00
2022-10-17 03:48:27 -04:00
- **AS5600_CLOCK_WISE (0)**
- **AS5600_COUNTERCLOCK_WISE (1)**
(\*) if **begin()** is called without **directionPin** or with this
parameter set to **255**, Software Direction Control is enabled.
2022-09-01 05:31:32 -04:00
2022-10-17 03:48:27 -04:00
See Software Direction Control below for more information.
2022-09-01 05:31:32 -04:00
2022-05-29 16:07:48 -04:00
2023-02-01 10:57:56 -05:00
#### OUT pin
2022-06-27 05:27:48 -04:00
The sensor has an output pin named **OUT**.
2022-09-01 05:31:32 -04:00
This pin can be used for an analogue or PWM output signal (AS5600),
and only for PWM by the AS5600L.
2022-07-07 06:53:02 -04:00
See **Analogue Pin** and **PWM Pin** below.
2022-06-27 05:27:48 -04:00
Examples are added to show how to use this pin with **setOutputMode()**.
2022-05-29 16:07:48 -04:00
2022-10-17 03:48:27 -04:00
See more in the sections Analog OUT and PWM OUT below.
2022-05-29 16:07:48 -04:00
2023-02-01 10:57:56 -05:00
#### PGO pin
2022-05-29 16:07:48 -04:00
2022-09-01 05:31:32 -04:00
Not tested. ==> Read the datasheet!
2022-05-29 16:07:48 -04:00
2022-07-07 06:53:02 -04:00
PGO stands for Programming Option, it is used to calibrate / program the sensor.
2022-06-27 05:27:48 -04:00
As the sensor can be programmed only a few times one should
use this functionality with extreme care.
See datasheet for a detailed list of steps to be done.
2022-07-07 06:53:02 -04:00
See **Make configuration persistent** below.
2022-06-27 05:27:48 -04:00
2022-10-17 03:48:27 -04:00
## I2C
2022-07-07 06:53:02 -04:00
2023-02-01 10:57:56 -05:00
#### Address
2022-07-07 06:53:02 -04:00
2022-10-17 03:48:27 -04:00
| sensor | address | changeable |
|:--------:|:---------:|:-------------|
2023-02-01 10:57:56 -05:00
| AS5600 | 0x36 | NO |
| AS5600L | 0x40 | YES, check setAddress() |
2022-07-07 06:53:02 -04:00
2022-10-17 03:48:27 -04:00
To use more than one **AS5600** on one I2C bus, see Multiplexing below.
2022-07-07 06:53:02 -04:00
2022-10-17 03:48:27 -04:00
The **AS5600L** supports the change of I2C address, optionally permanent.
Check the **setAddress()** function for non-permanent change.
2022-06-27 05:27:48 -04:00
2022-05-29 16:07:48 -04:00
2023-02-01 10:57:56 -05:00
#### Performance
| board | sensor | results | notes |
|:-------------:|:---------:|:-----------------|:--------|
| Arduino UNO | AS5600 | up to 900 KHz. | https://github.com/RobTillaart/AS5600/issues/22
| Arduino UNO | AS5600L | up to 300 KHz. |
| ESP32 | AS5600 | no data |
| ESP32 | AS5600L | up to 800 KHz |
No other boards tested yet.
#### ESP32 and I2C
When polling the AS5600 with an ESP32 to measure RPM an issue has been reported.
See https://github.com/RobTillaart/AS5600/issues/28
2022-07-04 06:10:00 -04:00
2023-02-01 10:57:56 -05:00
The problem is that the ESP32 can be blocking for up to one second if there is a
problem in the connection with the sensor. Using **setWireTimeout()** does not seem
to solve the problem (2023-01-31). In the issue the goal was to measure the turns
of a rotating device at around 3800 RPM. To do this one need roughly 1 angle measurement
per 5 milliseconds.
which
2022-05-29 16:07:48 -04:00
2022-07-07 06:53:02 -04:00
## Interface
2022-05-29 16:07:48 -04:00
2023-02-01 10:57:56 -05:00
```cpp
#include "AS5600.h"
```
2022-05-29 16:07:48 -04:00
2023-02-01 10:57:56 -05:00
#### Constants
2022-05-29 16:07:48 -04:00
2022-07-04 06:10:00 -04:00
Most important are:
2022-05-29 16:07:48 -04:00
2022-06-27 05:27:48 -04:00
```cpp
// setDirection
const uint8_t AS5600_CLOCK_WISE = 0; // LOW
const uint8_t AS5600_COUNTERCLOCK_WISE = 1; // HIGH
2022-07-04 06:10:00 -04:00
// 0.087890625;
const float AS5600_RAW_TO_DEGREES = 360.0 / 4096;
// 0.00153398078788564122971808758949;
const float AS5600_RAW_TO_RADIANS = PI * 2.0 / 4096;
2022-12-21 15:44:41 -05:00
// 4.06901041666666e-6
const float AS5600_RAW_TO_RPM = 1.0 / 4096 / 60;
2022-06-27 05:27:48 -04:00
// getAngularSpeed
const uint8_t AS5600_MODE_DEGREES = 0;
const uint8_t AS5600_MODE_RADIANS = 1;
2022-12-21 15:44:41 -05:00
const uint8_t AS5600_MODE_RPM = 2;
2022-06-27 05:27:48 -04:00
```
2022-05-29 16:07:48 -04:00
2022-07-04 06:10:00 -04:00
See AS5600.h file (and datasheet) for all constants.
Also Configuration bits below for configuration related ones.
2022-05-29 16:07:48 -04:00
2023-02-01 10:57:56 -05:00
#### Constructor + I2C
2022-05-29 16:07:48 -04:00
2022-07-07 06:53:02 -04:00
- **AS5600(TwoWire \*wire = &Wire)** Constructor with optional Wire
interface as parameter.
2023-05-11 05:48:26 -04:00
- **bool begin(uint8_t directionPin = AS5600_SW_DIRECTION_PIN)**
set the value for the directionPin.
If the pin is set to AS5600_SW_DIRECTION_PIN, the default value,
there will be software direction control instead of hardware control.
2022-07-04 06:10:00 -04:00
See below.
2023-05-11 05:48:26 -04:00
- **bool begin(int dataPin, int clockPin, uint8_t directionPin = AS5600_SW_DIRECTION_PIN)** idem,
2022-07-07 06:53:02 -04:00
for the ESP32 where one can choose the I2C pins.
2023-05-11 05:48:26 -04:00
If the pin is set to AS5600_SW_DIRECTION_PIN, the default value, there will be software
2022-07-07 06:53:02 -04:00
direction control instead of hardware control.
2022-07-04 06:10:00 -04:00
See below.
2022-09-01 05:31:32 -04:00
- **bool isConnected()** checks if the address 0x36 (AS5600) is on the I2C bus.
- **uint8_t getAddress()** returns the fixed device address 0x36 (AS5600).
2022-05-29 16:07:48 -04:00
2023-02-01 10:57:56 -05:00
#### Direction
2022-05-29 16:07:48 -04:00
To define in which way the sensor counts up.
2022-07-04 06:10:00 -04:00
- **void setDirection(uint8_t direction = AS5600_CLOCK_WISE)** idem.
- **uint8_t getDirection()** returns AS5600_CLOCK_WISE (0) or
AS5600_COUNTERCLOCK_WISE (1).
2022-05-29 16:07:48 -04:00
2022-10-17 03:48:27 -04:00
See Software Direction Control below for more information.
2022-05-29 16:07:48 -04:00
2023-02-01 10:57:56 -05:00
#### Configuration registers
2022-05-29 16:07:48 -04:00
2022-07-04 06:10:00 -04:00
Please read the datasheet for details.
2022-05-29 16:07:48 -04:00
2022-09-01 05:31:32 -04:00
- **bool setZPosition(uint16_t value)** set start position for limited range.
Value = 0..4095. Returns false if parameter is out of range.
2022-05-29 16:07:48 -04:00
- **uint16_t getZPosition()** get current start position.
2022-09-01 05:31:32 -04:00
- **bool setMPosition(uint16_t value)** set stop position for limited range.
Value = 0..4095. Returns false if parameter is out of range.
2022-05-29 16:07:48 -04:00
- **uint16_t getMPosition()** get current stop position.
2022-07-07 06:53:02 -04:00
- **bool setMaxAngle(uint16_t value)** set limited range.
2022-09-01 05:31:32 -04:00
Value = 0..4095. Returns false if parameter is out of range.
2022-05-29 16:07:48 -04:00
See datasheet **Angle Programming**
- **uint16_t getMaxAngle()** get limited range.
2022-07-04 06:10:00 -04:00
2022-05-29 16:07:48 -04:00
#### Configuration bits
Please read datasheet for details.
2022-12-21 15:44:41 -05:00
- **bool setConfigure(uint16_t value)** value == 0..0x3FFF
Access the register as bit mask.
Returns false if parameter is out of range.
- **uint16_t getConfigure()** returns the current configuration register a bit mask.
2023-02-01 10:57:56 -05:00
| Bit | short | Description | Values |
|:-------:|:--------|:----------------|:-------------------------------------------------------|
| 0-1 | PM | Power mode | 00 = NOM, 01 = LPM1, 10 = LPM2, 11 = LPM3 |
| 2-3 | HYST | Hysteresis | 00 = OFF, 01 = 1 LSB, 10 = 2 LSB, 11 = 3 LSB |
| 4-5 | OUTS | Output Stage | 00 = analog (0-100%), 01 = analog (10-90%), 10 = PWM |
| 6-7 | PWMF | PWM frequency | 00 = 115, 01 = 230, 10 = 460, 11 = 920 (Hz) |
| 8-9 | SF | Slow Filter | 00 = 16x, 01 = 8x, 10 = 4x, 11 = 2x |
| 10-12 | FTH | Fast Filter | Threshold 000 - 111 check datasheet |
| 13 | WD | Watch Dog | 0 = OFF, 1 = ON |
| 15-14 | | not used |
2022-05-29 16:07:48 -04:00
2022-07-07 06:53:02 -04:00
The library has functions to address these fields directly.
2022-12-21 15:44:41 -05:00
The setters() returns false if parameter is out of range.
2022-07-07 06:53:02 -04:00
- **bool setPowerMode(uint8_t powerMode)**
2022-12-21 15:44:41 -05:00
- **uint8_t getPowerMode()**
- **bool setHysteresis(uint8_t hysteresis)**
Suppresses "noise" on the output when the magnet is not moving.
2022-07-07 06:53:02 -04:00
In a way one is trading precision for stability.
2022-12-21 15:44:41 -05:00
- **uint8_t getHysteresis()**
- **bool setOutputMode(uint8_t outputMode)**
- **uint8_t getOutputMode()**
- **bool setPWMFrequency(uint8_t pwmFreq)**
- **uint8_t getPWMFrequency()**
- **bool setSlowFilter(uint8_t mask)**
- **uint8_t getSlowFilter()**
- **bool setFastFilter(uint8_t mask)**
- **uint8_t getFastFilter()**
- **bool setWatchDog(uint8_t mask)**
- **uint8_t getWatchDog()**
2022-07-07 06:53:02 -04:00
2022-05-29 16:07:48 -04:00
2023-02-01 10:57:56 -05:00
#### Read Angle
2022-05-29 16:07:48 -04:00
2022-12-21 15:44:41 -05:00
- **uint16_t rawAngle()** returns 0 .. 4095. (12 bits)
2022-07-04 06:10:00 -04:00
Conversion factor AS5600_RAW_TO_DEGREES = 360 / 4096 = 0.087890625
or use AS5600_RAW_TO_RADIANS if needed.
2022-12-21 15:44:41 -05:00
- **uint16_t readAngle()** returns 0 .. 4095. (12 bits)
Conversion factor AS5600_RAW_TO_DEGREES = 360 / 4096 = 0.087890625
or use AS5600_RAW_TO_RADIANS if needed.
The value of this register can be affected by the configuration bits above.
This is the one most used.
2022-07-04 06:10:00 -04:00
- **void setOffset(float degrees)** sets an offset in degrees,
e.g. to calibrate the sensor after mounting.
Typical values are -359.99 - 359.99 probably smaller.
Larger values will be mapped back to this interval.
2022-09-01 05:31:32 -04:00
Be aware that larger values will affect / decrease the precision of the
measurements as floats have only 7 significant digits.
2022-07-04 06:10:00 -04:00
Verify this for your application.
- **float getOffset()** returns offset in degrees.
2023-02-01 10:57:56 -05:00
In issue #14 there is a discussion about **setOffset()**.
2022-07-04 06:10:00 -04:00
A possible implementation is to ignore all values outside the
-359.99 - 359.99 range.
2022-09-01 05:31:32 -04:00
This would help to keep the precision high. User responsibility.
2022-05-29 16:07:48 -04:00
2023-02-01 10:57:56 -05:00
#### Angular Speed
2022-06-02 14:10:23 -04:00
2022-07-04 06:10:00 -04:00
- **float getAngularSpeed(uint8_t mode = AS5600_MODE_DEGREES)** is an experimental function that returns
2022-06-26 07:58:33 -04:00
an approximation of the angular speed in rotations per second.
2022-06-02 14:10:23 -04:00
The function needs to be called at least **four** times per rotation
2022-07-04 06:10:00 -04:00
or once per second to get a reasonably precision.
2022-06-02 14:10:23 -04:00
2022-12-21 15:44:41 -05:00
| mode | value | description | notes |
|:----------------------|:-------:|:---------------|:--------|
| AS5600_MODE_RADIANS | 1 | radians /sec | |
| AS5600_MODE_DEGREES | 0 | degrees /sec | default |
| other | - | degrees /sec | |
2022-06-26 07:58:33 -04:00
2022-12-21 15:44:41 -05:00
Negative return values indicate reverse rotation.
What that exactly means depends on the setup of your project.
2022-06-02 14:10:23 -04:00
Note: the first call will return an erroneous value as it has no
2022-07-04 06:10:00 -04:00
reference angle or time.
Also if one stops calling this function
2022-06-02 14:10:23 -04:00
for some time the first call after such delays will be incorrect.
2022-07-04 06:10:00 -04:00
Note: the frequency of calling this function of the sensor depends on the application.
The faster the magnet rotates, the faster it may be called.
Also if one wants to detect minute movements, calling it more often is the way to go.
2022-09-01 05:31:32 -04:00
An alternative implementation is possible in which the angle is measured twice
with a short interval. The only limitation then is that both measurements
should be within 180° = half a rotation.
2022-06-02 14:10:23 -04:00
2023-02-01 10:57:56 -05:00
#### Cumulative position (experimental)
2022-12-22 08:03:01 -05:00
Since 0.3.3 an experimental cumulative position can be requested from the library.
The sensor does not provide interrupts to indicate a movement or revolution
2023-02-01 10:57:56 -05:00
Therefore one has to poll the sensor at a frequency at least **three** times
per revolution with **getCumulativePosition()**
2022-12-22 08:03:01 -05:00
2023-02-01 10:57:56 -05:00
The cumulative position (32 bits) consists of 3 parts
2022-12-22 08:03:01 -05:00
| bit | meaning | notes |
|:-------:|:--------------|:--------|
| 31 | sign | typical + == CW, - == CCW
| 30-12 | revolutions |
| 11-00 | raw angle | call getCumulativePosition()
2022-12-21 15:44:41 -05:00
2022-12-22 08:03:01 -05:00
Functions are:
2022-12-21 15:44:41 -05:00
2022-12-22 08:03:01 -05:00
- **int32_t getCumulativePosition()** reads sensor and updates cumulative position.
- **int32_t getRevolutions()** converts last position to whole revolutions.
Convenience function.
2023-02-21 05:41:58 -05:00
- **int32_t resetPosition(int32_t position = 0)** resets the "revolutions" to position (default 0).
It does not reset the delta (rotation) since last call to **getCumulativePosition()**.
Returns last position (before reset).
- **int32_t resetCumulativePosition(int32_t position = 0)** completely resets the cumulative counter.
This includes the delta (rotation) since last call to **getCumulativePosition()**.
Returns last position (before reset).
As this code is experimental, names might change in the future (0.4.0)?
2022-12-22 08:03:01 -05:00
As the function are mostly about counting revolutions the current thoughts for new names are:
```cpp
int32_t updateRevolutions() replaces getCumulativePosition()
int32_t getRevolutions()
int32_t resetRevolutions() replaces resetPosition()
```
2022-12-21 15:44:41 -05:00
2023-02-01 10:57:56 -05:00
#### Status registers
2022-05-29 16:07:48 -04:00
2022-07-04 06:10:00 -04:00
- **uint8_t readStatus()** see Status bits below.
2022-05-29 16:07:48 -04:00
- **uint8_t readAGC()** returns the Automatic Gain Control.
0..255 in 5V mode, 0..128 in 3V3 mode.
- **uint16_t readMagnitude()** reads the current internal magnitude.
2022-07-04 06:10:00 -04:00
(page 9 datasheet)
Scale is unclear, can be used as relative scale.
2022-05-29 16:07:48 -04:00
- **bool detectMagnet()** returns true if device sees a magnet.
2022-09-01 05:31:32 -04:00
- **bool magnetTooStrong()** idem.
- **bool magnetTooWeak()** idem.
2022-05-29 16:07:48 -04:00
#### Status bits
Please read datasheet for details.
2022-12-21 15:44:41 -05:00
| Bit | short | Description | Values |
2022-05-29 16:07:48 -04:00
|:-----:|:------|:-------------:|:----------------------|
2022-12-21 15:44:41 -05:00
| 0-2 | | not used | |
| 3 | MH | overflow | 1 = magnet too strong |
| 4 | ML | underflow | 1 = magnet too weak |
| 5 | MD | magnet detect | 1 = magnet detected |
| 6-7 | | not used | |
2022-05-29 16:07:48 -04:00
2023-02-01 10:57:56 -05:00
## Make configuration persistent.
2022-05-29 16:07:48 -04:00
**USE AT OWN RISK**
2022-12-21 15:44:41 -05:00
Please read datasheet **twice** as these changes are not reversible.
2022-05-29 16:07:48 -04:00
2022-06-26 07:58:33 -04:00
The burn functions are used to make settings persistent.
2022-12-21 15:44:41 -05:00
These burn functions are permanent, therefore they are commented in the library.
2022-09-01 05:31:32 -04:00
Please read datasheet twice, before uncomment them.
2022-05-29 16:07:48 -04:00
2022-09-01 05:31:32 -04:00
The risk is that you make your AS5600 / AS5600L **USELESS**.
2022-05-29 16:07:48 -04:00
**USE AT OWN RISK**
2022-06-26 07:58:33 -04:00
- **uint8_t getZMCO()** reads back how many times the ZPOS and MPOS
registers are written to permanent memory.
2022-09-01 05:31:32 -04:00
You can only burn a new Angle 3 times to the AS5600, and only 2 times for the AS5600L.
2022-06-26 07:58:33 -04:00
- **void burnAngle()** writes the ZPOS and MPOS registers to permanent memory.
2022-09-01 05:31:32 -04:00
You can only burn a new Angle maximum **THREE** times to the AS5600
and **TWO** times for the AS5600L.
2022-06-26 07:58:33 -04:00
- **void burnSetting()** writes the MANG register to permanent memory.
You can write this only **ONE** time to the AS5600.
2022-05-29 16:07:48 -04:00
2023-02-01 10:57:56 -05:00
**USE AT OWN RISK**
2022-05-29 16:07:48 -04:00
2022-07-04 06:10:00 -04:00
## Software Direction Control
Experimental 0.2.0
2022-07-07 06:53:02 -04:00
Normally one controls the direction of the sensor by connecting the DIR pin
to one of the available IO pins of the processor.
This IO pin is set in the library as parameter of the **begin(directionPin)** function.
2022-07-04 06:10:00 -04:00
2022-10-17 03:48:27 -04:00
The directionPin is default set to 255, which defines a **Software Direction Control**.
To have this working one has to connect the **DIR pin of the sensor to GND**.
This puts the sensor in a hardware clockwise mode. Then it is up to the library
2022-07-07 06:53:02 -04:00
to do the additional math so the **readAngle()** and **rawAngle()** behave as
2022-10-17 03:48:27 -04:00
if the DIR pin was connected to a processor IO pin.
2022-07-04 06:10:00 -04:00
2022-10-17 03:48:27 -04:00
The user still calls **setDirection()** as before to change the direction
of the increments and decrements.
2022-07-04 06:10:00 -04:00
2022-10-17 03:48:27 -04:00
The advantage is one does not need that extra IO pin from the processor.
This makes connecting the sensor a bit easier.
2022-07-04 06:10:00 -04:00
TODO: measure performance impact.
TODO: investigate impact on functionality of other registers.
2022-09-01 05:31:32 -04:00
## Analog OUT
2022-07-07 06:53:02 -04:00
2022-09-01 05:31:32 -04:00
(details datasheet - page 25 = AS5600)
2022-07-07 06:53:02 -04:00
The OUT pin can be configured with the function:
- **bool setOutputMode(uint8_t outputMode)**
2022-09-01 05:31:32 -04:00
#### AS5600
2022-07-07 06:53:02 -04:00
When the analog OUT mode is set the OUT pin provides a voltage
which is linear with the angle.
| VDD | mode | percentage | output | 1° in V |
|:---:|:------:|:----------:|:---------:|:----------:|
| 5V0 | 0 | 0 - 100% | 0.0 - 5.0 | 0.01388889 |
| 5V0 | 1 | 10 - 90% | 0.5 - 4.5 | 0.01111111 |
| 3V3 | 0 | 0 - 100% | 0.0 - 3.3 | 0.00916667 |
| 3V3 | 1 | 10 - 90% | 0.3 - 3.0 | 0.00750000 |
To measure these angles a 10 bit ADC or higher is needed.
When analog OUT is selected **readAngle()** will still return valid values.
2022-09-01 05:31:32 -04:00
#### AS5600L
The AS5600L does **NOT** support analog OUT.
Both mode 0 and 1 will set the OUT pin to VDD (+5V0 or 3V3).
2022-07-07 06:53:02 -04:00
## PWM OUT
2022-09-01 05:31:32 -04:00
(details datasheet - page 27 = AS5600)
2022-07-07 06:53:02 -04:00
The OUT pin can be configured with the function:
- **bool setOutputMode(uint8_t outputMode)** outputMode = 2 = PWM
When the PWM OUT mode is set the OUT pin provides a duty cycle which is linear
with the angle. However they PWM has a lead in (HIGH) and a lead out (LOW).
The pulse width is 4351 units, 128 high, 4095 angle, 128 low.
| Angle | HIGH | LOW | HIGH % | LOW % | Notes |
|:-------:|:-------:|:------:|:--------:|:--------:|:--------|
| 0 | 128 | 4223 | 2,94% | 97,06% |
| 10 | 242 | 4109 | 5,56% | 94,44% |
| 20 | 356 | 3996 | 8,17% | 91,83% |
| 45 | 640 | 3711 | 14,71% | 85,29% |
| 90 | 1152 | 3199 | 26,47% | 73,53% |
| 135 | 1664 | 2687 | 38,24% | 61,76% |
| 180 | 2176 | 2176 | 50,00% | 50,00% |
| 225 | 2687 | 1664 | 61,76% | 38,24% |
| 270 | 3199 | 1152 | 73,53% | 26,47% |
| 315 | 3711 | 640 | 85,29% | 14,71% |
| 360 | 4223 | 128 | 97,06% | 2,94% | in fact 359.9 something as 360 == 0
#### Formula:
based upon the table above ```angle = map(dutyCycle, 2.94, 97.06, 0.0, 359.9);```
or in code ..
```cpp
t0 = micros(); // rising;
t1 = micros(); // falling;
t2 = micros(); // rising; new t0
// note that t2 - t0 should be a constant depending on frequency set.
// however as there might be up to 5% variation it cannot be hard coded.
float dutyCycle = (1.0 * (t1 - t0)) / (t2 - t0);
float angle = (dutyCycle - 0.0294) * (359.9 / (0.9706 - 0.0294));
```
#### PWM frequency
The AS5600 allows one to set the PWM base frequency (~5%)
- **bool setPWMFrequency(uint8_t pwmFreq)**
| mode | pwmFreq | step in us | 1° in time |
|:----:|:-------:|:----------:|:----------:|
| 0 | 115 Hz | 2.123 | 24.15 |
| 1 | 230 Hz | 1.062 | 12.77 |
| 2 | 460 Hz | 0.531 | 6.39 |
| 3 | 920 Hz | 0.216 | 3.20 |
Note that at the higher frequencies the step size becomes smaller
and smaller and it becomes harder to measure.
You need a sub-micro second hardware timer to measure the pulse width
with enough precision to get the max resolution.
When PWM OUT is selected **readAngle()** will still return valid values.
2022-06-27 05:27:48 -04:00
## Multiplexing
The I2C address of the **AS5600** is always 0x36.
To use more than one **AS5600** on one I2C bus, one needs an I2C multiplexer,
e.g. https://github.com/RobTillaart/TCA9548.
Alternative could be the use of a AND port for the I2C clock line to prevent
the sensor from listening to signals on the I2C bus.
Finally the sensor has an analogue output **OUT**.
2022-07-04 06:10:00 -04:00
This output could be used to connect multiple sensors to different analogue ports of the processor.
2022-06-27 05:27:48 -04:00
2022-07-07 06:53:02 -04:00
**Warning**: If and how well this analog option works is not verified or tested.
2022-06-27 05:27:48 -04:00
2022-12-21 15:44:41 -05:00
----
2022-06-02 14:10:23 -04:00
2022-12-21 15:44:41 -05:00
## AS5600L class
2022-09-01 05:31:32 -04:00
- **AS5600L(uint8_t address = 0x40, TwoWire \*wire = &Wire)** constructor.
As the I2C address can be changed in the AS5600L, the address is a parameter of the constructor.
This is a difference with the AS5600 constructor.
2023-02-01 10:57:56 -05:00
#### Setting I2C address
2022-09-01 05:31:32 -04:00
- **bool setAddress(uint8_t address)** Returns false if the I2C address is not in valid range (8-119).
2023-02-01 10:57:56 -05:00
#### Setting I2C UPDT
2022-09-01 05:31:32 -04:00
UPDT = update page 30 - AS5600L
- **bool setI2CUPDT(uint8_t value)**
- **uint8_t getI2CUPDT()**
2022-10-17 03:48:27 -04:00
These functions seems to have only a function in relation to **setAddress()** so possibly obsolete in the future.
If you got other insights on these functions please let me know.
2022-09-01 05:31:32 -04:00
2022-12-21 15:44:41 -05:00
----
2022-09-01 05:31:32 -04:00
2022-05-29 16:07:48 -04:00
## Operational
The base functions are:
```cpp
AS5600 as5600;
void setup()
{
Serial.begin(115200);
...
as5900.begin(4); // set the direction pin
as5600.setDirection(AS5600_CLOCK_WISE);
...
}
void loop()
{
...
Serial.println(as5600.readAngle());
delay(1000);
...
}
```
See examples.
## Future
Some ideas are kept here so they won't get lost.
2022-10-17 03:48:27 -04:00
priority is relative.
2022-05-29 16:07:48 -04:00
2022-06-26 07:58:33 -04:00
2022-12-21 15:44:41 -05:00
#### Must
2022-05-29 16:07:48 -04:00
2023-05-11 05:48:26 -04:00
- re-organize readme (0.4.0)
2023-02-01 10:57:56 -05:00
- rename revolution functions (0.4.0)
2023-05-11 05:48:26 -04:00
- to what?
2023-02-01 10:57:56 -05:00
2022-05-29 16:07:48 -04:00
2022-12-21 15:44:41 -05:00
#### Should
2022-06-26 07:58:33 -04:00
2022-07-04 06:10:00 -04:00
- investigate **readMagnitude()**
- combination of AGC and MD, ML and MH flags?
2022-09-01 05:31:32 -04:00
- investigate OUT behaviour in practice
2022-07-07 06:53:02 -04:00
- analogue
- PWM
2022-09-01 05:31:32 -04:00
- need AS5600 on breakout with support
2022-07-04 06:10:00 -04:00
- write examples:
- as5600_calibration.ino (needs HW and lots of time)
- different configuration options
2022-10-17 03:48:27 -04:00
- check / verify Power-up time
- 1 minute (need HW)
- check Timing Characteristics (datasheet)
- is there improvement possible.
2023-02-01 10:57:56 -05:00
- investigate why AS5600L failed at 400 KHz I2C
- repeatable?
2022-10-17 03:48:27 -04:00
2022-07-04 06:10:00 -04:00
2022-12-21 15:44:41 -05:00
#### Could
2022-07-04 06:10:00 -04:00
- add error handling
- investigate PGO programming pin.
2023-02-21 05:41:58 -05:00
- check for compatible devices
- AS5200 ?
2023-05-11 05:48:26 -04:00
- investigate performance
- basic performance per function
- I2C improvements
- software direction
2022-05-29 16:07:48 -04:00
2023-02-01 10:57:56 -05:00
2023-05-11 05:48:26 -04:00
#### Wont (unless)
- fix for AS5600L as it does not support analog OUT.
- type field?
- other class hierarchy?
- base class with commonalities?
- ==> just ignore for now.
- add mode parameter to offset functions.
- see getAngularSpeed()
2023-02-01 10:57:56 -05:00