217 lines
7.4 KiB
Markdown
Raw Normal View History

2022-12-01 17:28:33 +01:00
[![Arduino CI](https://github.com/RobTillaart/AD56X8/workflows/Arduino%20CI/badge.svg)](https://github.com/marketplace/actions/arduino_ci)
[![Arduino-lint](https://github.com/RobTillaart/AD56X8/actions/workflows/arduino-lint.yml/badge.svg)](https://github.com/RobTillaart/AD56X8/actions/workflows/arduino-lint.yml)
[![JSON check](https://github.com/RobTillaart/AD56X8/actions/workflows/jsoncheck.yml/badge.svg)](https://github.com/RobTillaart/AD56X8/actions/workflows/jsoncheck.yml)
[![License: MIT](https://img.shields.io/badge/license-MIT-green.svg)](https://github.com/RobTillaart/AD56X8/blob/master/LICENSE)
[![GitHub release](https://img.shields.io/github/release/RobTillaart/AD56X8.svg?maxAge=3600)](https://github.com/RobTillaart/AD56X8/releases)
# AD56X8
Experimental Library for the AD56X8 series digital analog convertor.
NOTE: not yet tested, TODO buy hardware.
Feedback, issues, improvements are welcome.
Please file an issue on GitHub.
## Description
The AD56X8 is an 8 channel DAC convertor, that has 12, 14 or 16 bit accuracy.
Furthermore it is available in 2.5 V and 5.0 V version, see table below.
The device allows to set the outputs directly, or prepare them and update them simultaneously (Latch DAC).
| type | output | resolution | power up |
|:---------|:--------:|:----------:|:---------:|
| AD5668-1 | 2.5 V | 16 bit | 0.0 V |
| AD5648-1 | 2.5 V | 14 bit | 0.0 V |
| AD5628-1 | 2.5 V | 12 bit | 0.0 V |
| AD5668-3 | 5.0 V | 16 bit | **2.5 V** |
| AD5668-2 | 5.0 V | 16 bit | 0.0 V |
| AD5648-2 | 5.0 V | 14 bit | 0.0 V |
| AD5628-2 | 5.0 V | 12 bit | 0.0 V |
The library is usable but not functional complete yet.
At least it lacks support for:
- RESET pin,
- LDAC pin,
- VREF pin.
- other points mentioned in future section below.
## Links
This library is partly inspired by https://github.com/bobhart/AD5668-Library, kudo's to Bob!
Discussed here - https://forum.arduino.cc/t/new-library-for-the-ad5668-dac/340393
Furthermore it has the SPI part from https://github.com/RobTillaart/MCP_DAC a.o.
Some differences between this library and Bob Harts. This library
- caches the values of all channels, so they can be read back.
- has derived classes for every specific type DAC.
This allows value range checking in the future. Not Implemented Yet.
- caches the LDAC mask, so it can be read back and updated.
- has extended (ESP32) SPI interface (similar and tested with MCP_DAC)
- allows to set SPI-speed.
- has faster software SPI transfer, on ESP32 this rivals HW SPI.
- MIT license instead of GNU v3
## Interface
```cpp
#include "AD56X8.h"
```
#### Base class
Should not be used to instantiate a device as the derived types have correct number of bits.
- **AD56X8(uint8_t slaveSelect)** constructor base class, sets HW SPI.
Uses default a 16 bit interface.
Sets internal values to zero.
- **AD56X8(uint8_t spiData, uint8_t spiClock, uint8_t slaveSelect)** constructor, sets SW SPI.
Uses default a 16 bit interface.
Sets internal values to zero.
- **begin()** initializes the SPI and sets internal state.
#### Set DAC
- **bool setValue(uint8_t channel, uint16_t value)** set value to the output immediately, effectively a prepare + update in one call.
Returns false if channel out of range.
- **uint16_t getValue(uint8_t channel)** returns set OR prepared value.
At power up the DAC's will be reset to 0 V except the AD5668-3 (2.5V).
2023-01-10 19:34:29 +01:00
- **bool setPercentage(uint8_t channel, float percentage)** idem.
- **float getPercentage(uint8_t channel)** idem.
2022-12-01 17:28:33 +01:00
- **bool prepareChannel(uint8_t channel, uint16_t value)** prepares the value for a channel.
Returns false if channel out of range.
- **bool updateChannel(uint8_t channel)** writes the prepared value to ADC.
Returns false if channel out of range.
- **void updateAllChannels()** writes all prepared channels to their ADC simultaneously by applying a SW latch pulse (LDAC).
NOte: the valid range of **value** is not checked by the library.
#### LDAC
The AD56X8 has an LDAC register with one bit per channel.
This is to configure which channels are updated simultaneously.
Read datasheet for details.
- **void setLDACmask(uint8_t mask = 0x00)** sets 8 channels with one call by using a bit mask. Default value 0x00 clears all channels.
- **uint8_t getLDACmask()** return the current (cached) LDAC bit mask, default = 0x00.
- **bool inLDACmask(uint8_t channel)** returns true if a channel is in the current LDAC bit mask.
Returns also false if channel is out of range.
#### Powermode
- **bool setPowerMode(uint8_t powerDownMode, uint8_t mask = 0x00)** powerDownMode = 0..3.
Default is setting the channels to **PWR_NORMAL**.
Returns false if powerDownMode is out of range.
| mode | define |
|:------:|:-------------------|
| 0x00 | AD56X8_PWR_NORMAL |
| 0x01 | AD56X8_PWR_1K |
| 0x02 | AD56X8_PWR_100K |
| 0x03 | AD56X8_PWR_TRI |
#### Misc
- **void reset()** software reset.
- **bool setClearCode(uint8_t CCmode)** Set the startup value.
CCmode = 0..3, see table below.
Returns false if mode out of range.
| mode | define | notes |
|:------:|:-------------------|:--------|
| 0x00 | AD56X8_CC_0000 |
| 0x01 | AD56X8_CC_8000 |
| 0x02 | AD56X8_CC_FFFF |
| 0x03 | AD56X8_CC_NOP | do not use => Read datasheet.
#### SPI
- **void setSPIspeed(uint32_t speed)** sets SPI clock in **Hz**, please read datasheet
of the ADC first to get optimal speed.
- **uint32_t getSPIspeed()** gets current speed in **Hz**.
- **bool usesHWSPI()** returns true if HW SPI is used.
#### SPI ESP32 specific
("inherited" from MPC_DAC library)
- **void selectHSPI()** in case hardware SPI, the ESP32 has two options HSPI and VSPI.
- **void selectVSPI()** see above.
- **bool usesHSPI()** returns true if HSPI is used.
- **bool usesVSPI()** returns true if VSPI is used.
The **selectVSPI()** or the **selectHSPI()** needs to be called
BEFORE the **begin()** function.
(experimental)
- **void setGPIOpins(uint8_t clk, uint8_t miso, uint8_t mosi, uint8_t select)**
overrule GPIO pins of ESP32 for hardware SPI. needs to be called AFTER the **begin()** function.
Note: earlier experiments shows that on a ESP32 SW-SPI is equally fast as HW-SPI and in fact a bit more stable.
The SW pulses are a bit slower than the HW pulses and therefore more square. The HW-SPI has some overhead SW-SPI hasn't.
### Derived classes (preferred use)
The parameters for the specific constructors are identical
to the base class.
- **AD5668_3(..)** constructor, 16 bit.
Starts up at **midscale = 32768**.
- **AD5668(..)** constructor, 16 bit.
- **AD5648(..)** constructor, 14 bit.
- **AD5628(..)** constructor, 12 bit.
## Operation
The examples (should) show the basic working of the functions.
Note that the library is not tested with hardware yet.
(based upon datasheet)
## Future
#### Must
2023-01-10 19:34:29 +01:00
2022-12-01 17:28:33 +01:00
- update and improve documentation
- get test hardware
- test the library
- write unit test
- check TODO's in code
#### Should
2023-01-10 19:34:29 +01:00
2022-12-01 17:28:33 +01:00
- write examples
- LDAC
- power mode
- support for RESET pin
- support for LDAC pin
- support for EXTERNAL VREF
- investigate value range checking for AD5648 and AD5628
- now setValue() returns false if value > max,
- should value be clipped instead?
- **bool loadLDAC()** TODO?
- investigate different type for AD5668_3 (as it does midscale)
#### Could
2023-01-10 19:34:29 +01:00
2022-12-01 17:28:33 +01:00
- CCmode + reset implies start value for getValue(ch)
- is this implementable? costs?