Skip to content

RobTillaart/AD5144A

Repository files navigation

Arduino CI Arduino-lint JSON check GitHub issues

License: MIT GitHub release PlatformIO Registry

AD5144A

Arduino library for I2C AD5144A 4 channel digital potentiometer.

Library also provides classes for AD5123, AD5124, AD5143, AD5144, AD5144A, AD5122A, AD5142A, AD5121, AD5141. These are have different number of potentiometers, rheostats and range but are functional compatible,

Description

Experimental

The library gives easy control over the I2C AD5144A 4 channel digital potentiometer and compatibles.

From the datasheets it is expected that this library will work for the family of related AD devices. See the table below. However the library is only partially tested, therefore the label experimental.

If there are problems, please file an issue. Also interested in success stories :)

This library uses the I2C interface to communicate with the device.

The library does not work for the SPI versions of these devices. See Future section below.

Feedback as always is welcome.

Breaking change 0.4.0

Fixed the working for AD5122A and possible other 7 bit (range 0..127) devices.

As this is a bug fix, this release makes pre-0.4.0 versions obsolete.

Breaking change 0.3.0

Version 0.3.0 introduced a breaking change. You cannot set the pins in begin() any more. This reduces the dependency of processor dependent Wire implementations. The user has to call Wire.begin() and can optionally set the Wire pins before calling begin().

Types supported

device #potmeters #rheostats range tested notes
AD5123 2 2 0..127 no
AD5124 4 0 0..127 no
AD5143 2 2 0..255 no
AD5144 4 0 0..255 partial see #2
AD5144A 4 0 0..255 partial me
AD5122A 2 0 0..127 partial see #28
AD5142A 2 0 0..255 partial see #15
AD5121 1 0 0..127 no
AD5141 1 0 0..255 no

Type AD51xy decomposition

  • x = 2 => range = 0..127
  • x = 4 => range = 0..255
  • y = 1 => 1 potentiometer
  • y = 2 => 2 potentiometers
  • y = 4 => 4 potentiometers
  • y = 3 => 2 potentiometers + 2 rheostats

Related

I2C

For the selection of the address, see table 12 / 13 datasheet. Typical 0x28, 0x2A or 0x2B.

The AD5144A devices support standard 100 kHz, and fast 400 kHz, data transfer modes.

I2C multiplexing

Sometimes you need to control more devices than possible with the default address range the device provides. This is possible with an I2C multiplexer e.g. TCA9548 which creates up to eight channels (think of it as I2C subnets) which can use the complete address range of the device.

Drawback of using a multiplexer is that it takes more administration in your code e.g. which device is on which channel. This will slow down the access, which must be taken into account when deciding which devices are on which channel. Also note that switching between channels will slow down other devices too if they are behind the multiplexer.

Interface

The library has a number of functions which are all quite straightforward. Checking return values increases robustness of your code.

As the library is experimental, function signatures might change in the future.

#include "AD5144A.h"

Constructor

  • AD51XX(uint8_t address, TwoWire *wire = &Wire) base class, sets the I2C address and optional the Wire bus used. This class does not distinguish between the derived classes. It assumes a range of 0..255 and 4 potentiometers so it could work to some extent for all device types. The developer is responsible for handling the differences in range and potmeters when using the base class.

Derived classes

Same as above, now with specified range and number of potentiometers.

  • AD5123(uint8_t address, TwoWire *wire = &Wire)
  • AD5124(uint8_t address, TwoWire *wire = &Wire)
  • AD5143(uint8_t address, TwoWire *wire = &Wire)
  • AD5144(uint8_t address, TwoWire *wire = &Wire)
  • AD5144A(uint8_t address, TwoWire *wire = &Wire)
  • AD5122A(uint8_t address, TwoWire *wire = &Wire)
  • AD5142A(uint8_t address, TwoWire *wire = &Wire)
  • AD5121(uint8_t address, TwoWire *wire = &Wire)
  • AD5141(uint8_t address, TwoWire *wire = &Wire)

I2C / device initialization

  • bool begin(bool doReset = true) for UNO, if doReset == true (default) reset() is called, to load last values stored in EEPROM. Returns true if the address of the device can be found on the I2C bus. If the device cannot be found, reset() won't be called, even if doReset == true.
  • bool isConnected() returns true if the address of the device can be found on the I2C bus, false otherwise.
  • uint8_t getAddress() returns address set in the constructor.
  • uint8_t reset() calls the built in RESET command. This loads the last values stored in EEPROM in the RDAC's. Furthermore it reads back the values from EEPROM in to the cache. Factory default is midScale() check datasheet for details.

Basic IO

Used to set one channel at the time.

  • uint8_t write(uint8_t rdac, uint8_t value) set channel rdac 0..3 to value 0..255 (depending on device type less channels - 1,2 -and lower max value - 127 - should be used) The value is also written into a cache of last set values for fast retrieval later. If value > maxValue an error AD51XXA_INVALID_VALUE is returned.
  • uint8_t read(uint8_t rdac) read back set value from the cache, not from the device.

EEPROM

The value stored in the EEPROM is the value the 4 potentiometers will start at boot time. This allows to start at predefined values and makes it possibly easier to continue after a reboot.

  • uint8_t storeEEPROM(uint8_t rdac) store the current channel value in EEPROM.
  • uint8_t storeEEPROM(uint8_t rdac, uint8_t value) store a specific (new) value in EEPROM. If value > maxValue an error AD51XXA_INVALID_VALUE is returned.
  • uint8_t recallEEPROM(uint8_t rdac) get the value from EEPROM and set the channel.

Note: reset() resets all 4 channels from the last values stored in EEPROM. (Page 28 datasheet REV-C)

Asynchronous

Sets values in sequence, not at exact same time.

  • uint8_t writeAll(uint8_t value) write the same value to all channels. If value > maxValue an error AD51XXA_INVALID_VALUE is returned.
  • uint8_t zeroAll() sets all channels to 0.
  • uint8_t midScaleAll() sets all channels to their midpoint 128 / 64.
  • uint8_t maxAll() sets all channels to the max 255 / 127.
  • uint8_t zero(uint8_t rdac) sets the selected channel to 0.
  • uint8_t midScale(uint8_t rdac) sets the selected channel to its midpoint = 128 / 64.
  • uint8_t maxValue(uint8_t rdac) sets the selected channel to the max 255 / 127.

Synchronous

  • uint8_t preload(uint8_t rdac, uint8_t value) prepare a single rdac for a new value but only use it after sync() is called. If value > maxValue an error AD51XXA_INVALID_VALUE is returned.
  • uint8_t preloadAll(uint8_t value) prepare all rdacs with the same value, and wait for sync(). If value > maxValue an error AD51XXA_INVALID_VALUE is returned.
  • uint8_t sync(uint8_t mask) will transfer the preloaded values to the (4) rdacs at the very same moment. The 4-bit mask is used to select which rdacs to synchronize.

TopScale BottomScale

See page 27 datasheet REV-C - TODO testing.

WARNING - USE WITH CARE

Do not set both top scale and bottom scale as the resistance of AB will drop from 100 kΩ to 120 Ω, possibly resulting in damage. (to be verified).

  • uint8_t setTopScale(uint8_t rdac) the resistance between Terminal A and Terminal W is decreased with 1 LSB e.g. from 130 Ω to 60 Ω (RAB = 100 kΩ).
  • uint8_t clrTopScale(uint8_t rdac) go back to 130 Ω.
  • uint8_t setTopScaleAll() idem.
  • uint8_t clrTopScaleAll() idem.
  • uint8_t setBottomScale(uint8_t rdac) the resistance between Terminal B and Terminal W is decreased with 1 LSB e.g. from 130 Ω to 60 Ω (RAB = 100 kΩ).
  • uint8_t clrBottomScale(uint8_t rdac) go back to 130 Ω.
  • uint8_t setBottomScaleAll()
  • uint8_t clrBottomScaleAll()

Operational modes

See page 27-28 datasheet REV-C - TODO investigate working and testing.

  • uint8_t setLinearMode(uint8_t rdac)
  • uint8_t setPotentiometerMode(uint8_t rdac)
  • uint8_t getOperationalMode(uint8_t rdac) returns 0 = potentiometer, 1 = linear

Increment / decrement

See page 27-28 datasheet REV-C - TODO investigate working and testing.

  • uint8_t incrementLinear(uint8_t rdac)
  • uint8_t incrementLinearAll()
  • uint8_t decrementLineair(uint8_t rdac)
  • uint8_t decrementLineairAll()
  • uint8_t increment6dB(uint8_t rdac)
  • uint8_t increment6dBAll()
  • uint8_t decrement6dB(uint8_t rdac)
  • uint8_t decrement6dBAll()

ReadBack

These function read back from the internal registers of the actual device.

  • uint8_t readBackINPUT(uint8_t rdac) reads back the "preload value" in the INPUT register.
  • uint8_t readBackEEPROM(uint8_t rdac) reads the boot value for the selected rdac from EEPROM.
  • uint8_t readBackCONTROL(uint8_t rdac) read back the control register. Read the datasheet for the details of the individual bits.
  • uint8_t readBackRDAC(uint8_t rdac) reads the value of the rdac from the device.

Write control register

  • uint8_t writeControlRegister(uint8_t mask) writes to the control register. Read the datasheet for the details of the individual bits. Warning use with care!

Miscellaneous

  • uint8_t pmCount() returns the number of potentiometers / channels the device has. Useful when writing your own loops over all channels.
  • uint8_t maxValue() return maxValue of the potentiometer. Values expected are 127 or 255.
  • uint8_t shutDown() check datasheet, not tested yet, use at own risk.

Operation

The examples show the basic working of the functions.

Future

Must

  • update documentation

Should

  • more testing with (different) hardware.
  • test bottomScale and TopScale functions
    • including simultaneous. (dangerous?)
    • does library need a build in protection?
  • investigate linear / potentiometer mode
  • investigate linear and 6dB increment decrement.

Could

  • some functions can be performance optimized
    • writing a value is not needed if last value is the same?
    • should be at least written once.
    • flag cacheOn + cacheDirty or so?
  • improve unit testing CI
  • stereo, write one value to two channels.

Wont (unless requested)

  • SPI based version of the library (if requested)

Support

If you appreciate my libraries, you can support the development and maintenance. Improve the quality of the libraries by providing issues and Pull Requests, or donate through PayPal or GitHub sponsors.

Thank you,

About

Arduino library for I2C digital potentiometer AD5144A

Topics

Resources

License

Stars

Watchers

Forks

Sponsor this project

  •  

Packages

No packages published

Languages