https://github.com/robtillaart/tca9555
Arduino library for I2C TCA9555 16 channel port expander
https://github.com/robtillaart/tca9555
16-bit arduino port-expander
Last synced: about 1 month ago
JSON representation
Arduino library for I2C TCA9555 16 channel port expander
- Host: GitHub
- URL: https://github.com/robtillaart/tca9555
- Owner: RobTillaart
- License: mit
- Created: 2021-06-09T15:37:55.000Z (about 4 years ago)
- Default Branch: master
- Last Pushed: 2024-07-17T06:41:49.000Z (11 months ago)
- Last Synced: 2024-08-07T18:27:17.708Z (10 months ago)
- Topics: 16-bit, arduino, port-expander
- Language: C++
- Homepage:
- Size: 5.62 MB
- Stars: 10
- Watchers: 2
- Forks: 3
- Open Issues: 0
-
Metadata Files:
- Readme: README.md
- Changelog: CHANGELOG.md
- Funding: .github/FUNDING.yml
- License: LICENSE
Awesome Lists containing this project
README
[](https://github.com/marketplace/actions/arduino_ci)
[](https://github.com/RobTillaart/TCA9555/actions/workflows/arduino-lint.yml)
[](https://github.com/RobTillaart/TCA9555/actions/workflows/jsoncheck.yml)
[](https://github.com/RobTillaart/TCA9555/issues)[](https://github.com/RobTillaart/TCA9555/blob/master/LICENSE)
[](https://github.com/RobTillaart/TCA9555/releases)
[](https://registry.platformio.org/libraries/robtillaart/TCA9555)# TCA9555
Arduino library for TCA9555 16 channel I2C port expander.
## Description
This library gives easy control over the 16 pins of a TCA9555 device.
The pins can be used for INPUT (default) and OUTPUT, and allow to set polarity.
The pins can be set per single pin, or with a mask to set either 8 or 16 pins
in one call.
Note that for the 16 bit interface settings are not perfectly simultaneous
as the 16 bit interface does in fact 2 calls to the 8 bit interface.The TCA9555 has an interrupt pin, however the library does not support this.
### TCA9535
From datasheet:
_The TCA9535 is identical to the TCA9555, except that the TCA9535 does not include the internal I/O
pull-up resistor, which requires pull-ups and pull-downs on unused I/O pins when configured as an
input and not driven. This reduces power consumption when the I/O is held low._There is a TCA9535 class which is a (convenience) wrapper around the TCA9555 class.
This allows one to create TCA9535 objects.### Compatibles
The library is expected to work for the PCA9555 / PCA9535 too for which derived
classes are made.| type | Notes |
|:---------:|:------:|
| TCA9555 | confirmed
| TCA9535 |
| PCA9555 |
| PCA9535 |
| CAT9555 | confirmed (see #24)Feedback of working types is welcome.
### Related
16 bit port expanders
- https://github.com/RobTillaart/MCP23017_RT I2C 16 IO lines.
- https://github.com/RobTillaart/MCP23S17 SPI 16 IO lines.
- https://github.com/RobTillaart/PCF8575 I2C 16 IO lines.
- https://github.com/RobTillaart/PCA9671 I2C 16 IO lines. - successor PCF8575
- https://github.com/RobTillaart/TCA9555 I2C 16 IO lines.8 bit port expanders
- https://github.com/RobTillaart/MCP23008 I2C 8 IO lines.
- https://github.com/RobTillaart/MCP23S08 SPI 8 IO lines.
- https://github.com/RobTillaart/PCF8574 I2C 8 IO lines.
- https://github.com/RobTillaart/TCA9554 I2C 8 IO lines.### 0.3.0 Breaking change
The version 0.3.0 has breaking changes in the interface.
The rationale is that the programming environment of the **Arduino ESP32 S3**
board uses a remapping by means of the include file **io_pin_remap.h**.
This file remaps the pins of several core Arduino functions.
The remapping is implemented by #define macros and these implement "hard" text
replacements without considering context.
The effect is that methods from this class (and several others) which have the same
name as those Arduino core functions will be remapped into something not working.The following library functions have been renamed:
| old name | new name | notes |
|:-----------------|:-------------|:--------|
| analogRead() | read() |
| analogWrite() | write() |
| pinMode() | pinMode1() |
| digitalRead() | read1() |
| digitalWrite() | write1() |### 0.2.0 Breaking change
Version 0.2.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()**.## Hardware
### I2C addresses
The addresses for the TCA9555 and TCA9535 are 0x20..0x27.
These are to be set with pin A0, A1, A2.Both the TCA9555 and TCA9535 support up to 400 kHz I2C.
### 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.- https://github.com/RobTillaart/TCA9548
### I2C Performance
Tested with TCA9555_performance.ino + UNO 0.4.3
Note above 800 I2C fails, up to 800 KHz seems OK.
| CLOCK | read1 | read8 | read16 | write1 | write8 | write16 |
|:-------:|:--------:|:--------:|:---------:|:--------:|:--------:|:---------:|
| 50 | 860.16 | 860.16 | 1720.32 | 860.16 | 619.64 | 1239.24 |
| 100 | 463.65 | 463.52 | 926.19 | 464.67 | 334.57 | 669.16 |
| 200 | 267.16 | 267.16 | 534.27 | 267.17 | 191.75 | 382.86 |
| 300 | 196.83 | 196.80 | 393.46 | 195.89 | 141.23 | 281.26 |
| 400 | 169.58 | 169.55 | 335.65 | 168.63 | 119.48 | 238.75 |
| 500 | 147.34 | 147.34 | 294.45 | 147.91 | 104.74 | 208.94 |
| 600 | 137.38 | 137.36 | 274.30 | 137.76 | 96.87 | 194.70 |
| 700 | 129.12 | 129.12 | 257.11 | 130.34 | 91.43 | 182.86 |
| 800 | 124.89 | 124.88 | 249.77 | 125.39 | 87.87 | 175.54 |
| 900 | 48.78 | 48.76 | 97.55 | 100.94 | 52.38 | 104.50 |### INT pin interrupts
The interrupt pin is not supported by the library.
There is an example showing how interrupts can be used.Note that the device generates interrupts on **CHANGING** pins,
so no option to select falling or rising.## Interface
Check the datasheet for details
```cpp
#include "TCA9555.h"
```### Constructor
- **TCA9555(uint8_t address, TwoWire \*wire = &Wire)** constructor, with default Wire interface.
Can be overruled with Wire0..WireN.
- **TCA9535(uint8_t address, TwoWire \*wire = &Wire)** idem.
- **PCA9555(uint8_t address, TwoWire \*wire = &Wire)** idem.
- **PCA9535(uint8_t address, TwoWire \*wire = &Wire)** idem.
- **CAT9555(uint8_t address, TwoWire \*wire = &Wire)** idem.
- **uint8_t getType()** returns 35 or 55 depending on type.
- **bool begin(uint8_t mode = INPUT, uint16_t mask = 0x0000)** initializes library.
Sets all the pins to INPUT (default) or to OUTPUT.
If mode == OUTPUT the mask sets the initial value.
If mode != OUTPUT the mode is considered INPUT.
Returns true if device can be seen on I2C bus, false otherwise.
- **bool isConnected()** returns true if device address given in the constructor can be seen
on I2C bus, returns false otherwise.
- **uint8_t getAddress()** returns set address, (debugging).### 1 pin interface
- **bool pinMode1(uint8_t pin, uint8_t mode)** idem.
If pin > 15 the function will return false.
The parameter mode must be INPUT or OUTPUT, other values will return false.
- **bool write1(uint8_t pin, uint8_t value)** pin = 0..15, value = LOW(0) HIGH (!0),
returns true if successful.
- **uint8_t read1(uint8_t pin)** pin = 0..15, returns the value of the pin HIGH or LOW.
- **bool setPolarity(uint8_t pin, uint8_t value)** inverts polarity of an INPUT pin.
- **uint8_t getPolarity(uint8_t pin)** returns 1 if a pin is inverted, zero otherwise.### 8 pin interface
port = 0..1
mask = 0..255 (0xFF)- **bool pinMode8(uint8_t port, uint8_t mask)** set the mode of eight pins in one call.
If port > 1 the function will return false.
1 bit = INPUT, 0 bit = OUTPUT.
- **bool write8(uint8_t port, uint8_t mask)** returns true if successful.
Especially useful if one needs to trigger multiple pins at the exact same time.
- **uint8_t read8(uint8_t port)** returns a bit pattern for pins 0..7 or pins 8..15.
- **bool setPolarity8(uint8_t port, uint8_t mask)** inverts polarity of the INPUT pins in one action.
- **uint8_t getPolarity(uint8_t port)** returns a mask with a 1 for every INPUT pin that is inverted.### 16 pin interface
Be aware that the 16 pins interface does two calls to the 8 pins interface.
So it is impossible to switch pins from the 2 groups of 8 at exactly the same time
without additional hardware.- **bool pinMode16(uint16_t mask)** set the mode of sixteen pins in one call.
1 bit = INPUT, 0 bit = OUTPUT.
- **bool write16(uint16_t mask)** mask = 0x0000 .. 0xFFFF, returns true if successful.
As **write16()** uses two calls to **write8()** only the pins on the same port are
set simultaneously.
- **uint16_t read16()** Returns a bit pattern for pins 0..15.
- **bool setPolarity16(uint16_t mask)** inverts polarity of the INPUT pins in one action.
Returns true upon success.
- **uint16_t getPolarity()** returns a mask of 16 bits with a 1 for every INPUT pin that
has inverted polarity.### Error codes
- **int lastError()** Above functions set an error flag that can be read with this function.
Reading it will reset the flag to **TCA9555_OK**.| Description | Value |
|:-----------------------|:-------:|
| TCA9555_OK | 0x00 |
| TCA9555_PIN_ERROR | 0x81 |
| TCA9555_I2C_ERROR | 0x82 |
| TCA9555_VALUE_ERROR | 0x83 |
| TCA9555_PORT_ERROR | 0x84 |
| TCA9555_INVALID_READ | -100 |## Future
#### Must
- update documentation
- keep TCA9554/TCA9555 in sync#### Should
- investigate optimizing the "16 pins" interface.
- read /write multiple bytes in one call, is it supported? (YES!)
- expected gain ~33%
- **setPolarity()** ==> **setPolarity1()** ? get idem.
- uniformity
- investigate map INPUT_PULLUP on INPUT (pinMode ?)
- investigate internal pull up etc.
- investigate TCA9535 differences
- pull up resistors
- elaborate derived class#### Could
- rethink class hierarchy
- TCA9535 has less functions so should be base class
- add performance example for I2C.
- buy TCA9535 / PCA9555 / PCA9535 to test#### Wont (unless)
- add TCA9535 error codes
- better reuse them? ==> TCA95X5 codes## 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,