271 lines
8.9 KiB
Markdown
Raw Normal View History

2021-05-26 11:38:28 +02:00
[![Arduino CI](https://github.com/RobTillaart/MCP_DAC/workflows/Arduino%20CI/badge.svg)](https://github.com/marketplace/actions/arduino_ci)
2021-11-08 16:16:58 +01:00
[![Arduino-lint](https://github.com/RobTillaart/MCP_DAC/actions/workflows/arduino-lint.yml/badge.svg)](https://github.com/RobTillaart/MCP_DAC/actions/workflows/arduino-lint.yml)
[![JSON check](https://github.com/RobTillaart/MCP_DAC/actions/workflows/jsoncheck.yml/badge.svg)](https://github.com/RobTillaart/MCP_DAC/actions/workflows/jsoncheck.yml)
2021-05-26 11:38:28 +02:00
[![License: MIT](https://img.shields.io/badge/license-MIT-green.svg)](https://github.com/RobTillaart/MCP_DAC/blob/master/LICENSE)
[![GitHub release](https://img.shields.io/github/release/RobTillaart/MCP_DAC.svg?maxAge=3600)](https://github.com/RobTillaart/MCP_DAC/releases)
# MCP_DAC
2021-11-08 16:16:58 +01:00
Arduino library for Microchip SPI DAC's: 8, 10, 12 bits, 1, 2 channel.
2021-05-26 11:38:28 +02:00
## Description
2021-11-08 16:16:58 +01:00
The MCP_DAC is a library for DAC's from Microchip in the MCP48xx en MCP49xx series.
2021-05-26 11:38:28 +02:00
The library is experimental as it is not tested with all different devices.
Please post an issue if there are problems.
| Type | Channels | Bits | MaxValue | Voltage reference |
|:--------|:--------:|:----:|:--------:|:-----------------:|
| MCP4801 | 1 | 8 | 255 | internal 2.048 V |
| MCP4802 | 2 | 8 | 255 | internal 2.048 V |
| MCP4811 | 1 | 10 | 1023 | internal 2.048 V |
| MCP4812 | 2 | 10 | 1023 | internal 2.048 V |
| MCP4821 | 1 | 12 | 4095 | internal 2.048 V |
| MCP4822 | 2 | 12 | 4095 | internal 2.048 V |
| MCP4901 | 1 | 8 | 255 | external |
| MCP4902 | 2 | 8 | 255 | external |
| MCP4911 | 1 | 10 | 1023 | external |
| MCP4912 | 2 | 10 | 1023 | external |
| MCP4921 | 1 | 12 | 4095 | external |
| MCP4922 | 2 | 12 | 4095 | external |
2022-10-02 17:19:23 +02:00
The output voltage of the MCP_DAC depends on the voltage supplied,
2021-05-26 11:38:28 +02:00
which is in the range of 2.7V .. 5.5V. Check datasheet for the details.
## Interface
2023-03-24 11:08:34 +01:00
```cpp
#include "MCP_DAC.h"
```
2021-05-26 11:38:28 +02:00
2023-03-24 11:08:34 +01:00
#### Constructor
2021-05-26 11:38:28 +02:00
2022-10-19 11:48:25 +02:00
- **MCP_DAC(uint8_t dataOut = 255, uint8_t clock = 255, SPIClassRP2040 \*mySPI = &SPI)** Constructor base class for RP2040.
- **MCP_DAC(uint8_t dataOut = 255, uint8_t clock = 255, SPIClass \*mySPI = &SPI)** Constructor base class.
2021-11-08 16:16:58 +01:00
Other devices just use their name as class object e.g. MCP4801 with same parameters.
2022-10-19 11:48:25 +02:00
- **begin(uint8_t select)** defines the select pin.
2022-10-02 17:19:23 +02:00
The select pin is used for device selection in case of multiple SPI devices.
2023-03-24 11:08:34 +01:00
- **uint8_t channels()** returns the number of channels, 1 or 2.
(note channel numbers are 0 or 1).
2021-11-08 16:16:58 +01:00
- **uint16_t maxValue()** returns the maximum value that can be set.
This relates to the number of bits, see table above.
2021-05-26 11:38:28 +02:00
2023-03-24 11:08:34 +01:00
#### Gain
2021-05-26 11:38:28 +02:00
2021-11-08 16:16:58 +01:00
- **bool setGain(uint8_t gain = 1)** gain is 1 (default) or 2.
2021-05-26 11:38:28 +02:00
- **uint8_t getGain()** returns gain set, default 1.
2022-10-02 17:19:23 +02:00
The analog output cannot go beyond the supply voltage.
2021-05-26 11:38:28 +02:00
So if Vref is connected to 5V, gain=2 will not output 10 Volts.
2023-03-24 11:08:34 +01:00
#### Write
2021-05-26 11:38:28 +02:00
2023-03-24 11:08:34 +01:00
- **bool analogWrite(uint16_t value, uint8_t channel = 0)** writes value to channel.
Default for channel 0 as that works for the single DAC devices.
The value is limited to maxValue.
2021-11-08 16:16:58 +01:00
Returns false in case of an invalid channel.
2023-03-24 11:08:34 +01:00
- **uint16_t lastValue(uint8_t channel = 0)** returns last written value.
Default for channel 0 as that works for the single DAC devices.
- **void setPercentage(float percentage, uint8_t channel = 0)** percentage = 0..100.0%.
Wrapper around **analogWrite()**.
- **float getPercentage(uint8_t channel = 0)** returns percentage.
Reads from cache.
- **void fastWriteA(uint16_t value)** faster version to write to channel 0.
Does not check flags and does not update **lastValue()**
- **void fastWriteB(uint16_t value)** faster version to write to channel 1.
Does not check flags and does not update **lastValue()**
2021-08-01 15:52:01 +02:00
- **bool increment(uint8_t channel = 0)** returns true if channel is incremented, false otherwise.
- **bool decrement(uint8_t channel = 0)** returns true if channel is decremented, false otherwise.
2021-05-26 11:38:28 +02:00
2022-10-02 17:19:23 +02:00
For fastest speed there is an example added **MCP4921_standalone.ino**.
2021-05-26 11:38:28 +02:00
That squeezes the most performance out of it for now.
Code for the other MCP4xxx can be written in same way.
2023-03-24 11:08:34 +01:00
#### Shutdown
2021-05-26 11:38:28 +02:00
2023-03-24 11:08:34 +01:00
- **void shutDown()** shuts down the device, optional one might need to **triggerLatch()**.
2021-05-26 11:38:28 +02:00
- **bool isActive()** returns false if device is in shutdown mode.
2023-03-24 11:08:34 +01:00
Note: any **write()** operation will set active to true again.
2021-05-26 11:38:28 +02:00
2023-03-24 11:08:34 +01:00
#### Hardware SPI
2021-05-26 11:38:28 +02:00
To be used only if one needs a specific speed.
2023-03-24 11:08:34 +01:00
Check datasheet for details.
2021-05-26 11:38:28 +02:00
2023-03-24 11:08:34 +01:00
- **void setSPIspeed(uint32_t speed)** set SPI transfer rate.
- **uint32_t getSPIspeed()** returns SPI transfer rate.
2021-05-26 11:38:28 +02:00
2023-03-24 11:08:34 +01:00
#### LDAC
2021-05-26 11:38:28 +02:00
2023-03-24 11:08:34 +01:00
- **void setLatchPin(uint8_t latchPin)** defines the latchPin, this is optional.
The latchPin is used for simultaneous setting a value in both DAC registers.
It can also be used to synchronize the setting of multiple devices.
Note the latchPin must be the same for all instances that need to be triggered together.
2021-05-26 11:38:28 +02:00
- **triggerLatch()** toggles the defined latchPin, and all devices that are connected to it.
2023-03-24 11:08:34 +01:00
Note: pre 0.2.0 versions have the LDAC signal incorrectly inverted.
2021-05-26 11:38:28 +02:00
2023-03-24 11:08:34 +01:00
#### Buffered
2021-05-26 11:38:28 +02:00
2023-03-24 11:08:34 +01:00
**MCP49xxx series only**, see page 20 ==> not functional for MCP48xx series.
2021-05-26 11:38:28 +02:00
2023-03-24 11:08:34 +01:00
- **void setBufferedMode(bool mode = false)** set buffered mode on/off.
The default mode == false == unbuffered.
- **bool getBufferedMode()** returns set value.
2021-05-26 11:38:28 +02:00
2023-03-24 11:08:34 +01:00
#### Debug
2021-05-26 11:38:28 +02:00
2023-03-24 11:08:34 +01:00
- **void reset()** resets internal variables to initial value. (use with care!).
2021-05-26 11:38:28 +02:00
- **bool usesHWSPI()** returns true if HW SPI is used.
2022-10-02 17:19:23 +02:00
## ESP32 specific
2023-03-24 11:08:34 +01:00
#### SPI port selection
2021-07-31 17:44:31 +02:00
2021-08-01 15:52:01 +02:00
This functionality is new in 0.1.2 and it is expected that the interface will change
2022-10-02 17:19:23 +02:00
in the future.
2021-08-01 15:52:01 +02:00
2021-07-31 17:44:31 +02:00
- **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.
2022-10-02 17:19:23 +02:00
The **selectVSPI()** or the **selectHSPI()** needs to be called
2021-08-01 15:52:01 +02:00
BEFORE the **begin()** function.
2023-03-24 11:08:34 +01:00
#### Experimental
2021-08-01 15:52:01 +02:00
2023-03-24 11:08:34 +01:00
- **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.
2021-08-01 15:52:01 +02:00
```cpp
void setup()
{
MCP.selectVSPI();
MCP.begin(15);
MCP.setGPIOpins(CLK, MISO, MOSI, SELECT); // SELECT should match the param of begin()
}
```
2021-11-08 16:16:58 +01:00
This interface can change in the future as the **select** pin is known
in the code.
2021-08-01 15:52:01 +02:00
2023-03-24 11:08:34 +01:00
#### ESP32 connections to MCP4922 (example)
2021-08-01 15:52:01 +02:00
2021-11-08 16:16:58 +01:00
ESP32 has **four** SPI peripherals from which two can be used.
2021-08-01 15:52:01 +02:00
2023-03-24 11:08:34 +01:00
SPI0 and SPI1 are used to access flash memory.
SPI2 and SPI3 are "user" SPI controllers a.k.a. HSPI and VSPI.
2021-08-01 15:52:01 +02:00
2023-03-24 11:08:34 +01:00
| MCP4922 | HSPI = SPI2 | VSPI = SPI3 |
|:----------:|:-------------:|:-------------:|
| CS | SELECT = 15 | SELECT = 5 |
| SCK | SCLK = 14 | SCLK = 18 |
| SDI | MOSI = 13 | MOSI = 23 |
| not used | MISO = 12 | MISO = 19 |
2021-08-01 15:52:01 +02:00
2022-10-02 17:19:23 +02:00
By using different **SELECT** pins multiple DAC's can be controlled over
one SPI bus.
2021-07-31 17:44:31 +02:00
2021-05-26 11:38:28 +02:00
2022-10-02 17:19:23 +02:00
## RP2040 specific
2023-03-24 11:08:34 +01:00
#### SPI port selection
The SPI Port selections happens in the constructor with e.g. &SPI or &SPI1.
For the pin swap, you need the call the experimental feature **void setGPIOpins**.
In the constructor you need to call the parameter dataOut and clock both
with 255 (0xff) or otherwise it will use SoftSPI.
2022-10-02 17:19:23 +02:00
2023-03-24 11:08:34 +01:00
#### Experimental
2022-10-02 17:19:23 +02:00
2023-03-24 11:08:34 +01:00
- **void setGPIOpins(uint8_t clk, uint8_t miso, uint8_t mosi, uint8_t select)**
overrule GPIO pins of RP2040 for different SPI pins.
Needs to be called AFTER the **begin()** function.
Selected pins must match the RP2040 pinout!
2022-10-02 17:19:23 +02:00
2023-03-24 11:08:34 +01:00
Warning! This command changes the Pins of the bus not only of a specific device,
but all devices, that are connected on that bus!
2022-10-02 17:19:23 +02:00
```cpp
2022-10-19 11:48:25 +02:00
MCP4822 MCP(255, 255, &SPI1)
2022-10-02 17:19:23 +02:00
void setup()
{
MCP.setGPIOpins(CLK, MISO, MOSI, SELECT); // SELECT should match the param of begin()
2022-10-19 11:48:25 +02:00
MCP.begin(17);
2022-10-02 17:19:23 +02:00
}
```
2023-03-24 11:08:34 +01:00
#### Pico connections to MCP4922 (example)
2022-10-02 17:19:23 +02:00
The RP2040 has **two** SPI peripherals from which two can be used.
SPI (SPI0) and SPI1 can both be used to connect devices.
| MCP4922 | SPI / SPI0 | SPI1 |
|:--------:|:-------------:|:-------------:|
| CS | SELECT = 17 | SELECT = 13 |
| SCK | SCLK = 18 | SCLK = 14 |
| SDI | MOSI = 19 | MOSI = 15 |
| not used | MISO = 16 | MISO = 12 |
By using different **SELECT** pins multiple DAC's can be controlled over
one SPI bus.
2021-05-26 11:38:28 +02:00
## Operation
See examples
2022-10-02 17:19:23 +02:00
## Future
#### Must
- test test test and ....
- improve documentation.
2023-03-24 11:08:34 +01:00
2022-10-02 17:19:23 +02:00
#### Should
2023-03-24 11:08:34 +01:00
#### Could
- refactor the API (how).
- minimize conditional in code if possible.
- functional names for magic masks.
#### Wont
2022-10-02 17:19:23 +02:00
- **useSPI1** and **useHSPI** are functional identical indicators.
- how to refactor to a generic model? Should work for all libraries.
- int8_t HWSPIport = 0, 1, 2, 3, 4, .... (-1 == SW SPI ?).
- numbers are not self-documenting.
- **selectSPIport(int)** ?
- would reduce conditional code.
2023-03-24 11:08:34 +01:00
- does not improve library
2022-10-02 17:19:23 +02:00