157 lines
6.4 KiB
Markdown
Raw Normal View History

2021-01-29 12:31:58 +01:00
[![Arduino CI](https://github.com/RobTillaart/PCF8575/workflows/Arduino%20CI/badge.svg)](https://github.com/marketplace/actions/arduino_ci)
2021-12-01 14:50:20 +01:00
[![Arduino-lint](https://github.com/RobTillaart/PCF8575/actions/workflows/arduino-lint.yml/badge.svg)](https://github.com/RobTillaart/PCF8575/actions/workflows/arduino-lint.yml)
[![JSON check](https://github.com/RobTillaart/PCF8575/actions/workflows/jsoncheck.yml/badge.svg)](https://github.com/RobTillaart/PCF8575/actions/workflows/jsoncheck.yml)
2021-01-29 12:31:58 +01:00
[![License: MIT](https://img.shields.io/badge/license-MIT-green.svg)](https://github.com/RobTillaart/PCF8575/blob/master/LICENSE)
[![GitHub release](https://img.shields.io/github/release/RobTillaart/PCF8575.svg?maxAge=3600)](https://github.com/RobTillaart/PCF8575/releases)
# PCF8575
2021-12-01 14:50:20 +01:00
Arduino library for PCF8575 - 16 channel I2C IO expander.
2021-01-29 12:31:58 +01:00
2021-07-09 13:02:35 +02:00
2021-01-29 12:31:58 +01:00
## Description
2021-12-01 14:50:20 +01:00
Related to the PCF8574 8 channel IO expander library https://github.com/RobTillaart/PCF8574.
2021-01-29 12:31:58 +01:00
The library gives easy control over the 16 pins of the PCF8575 chips.
2021-12-01 14:50:20 +01:00
Base address = 0x20 + 0..7 depending on address pins A0..A2.
2021-01-29 12:31:58 +01:00
2021-07-09 13:02:35 +02:00
| TYPE | ADDRESS-RANGE | notes |
|:---------|:-------------:|:------------------------:|
|PCF8575 | 0x20 to 0x27 | same range as PCF8574 !! |
| | | |
2021-01-29 12:31:58 +01:00
So you can connect up to 8 PCF8575 on one I2C bus, giving access
to 8 x 16 = 128 IO lines.
To maximize IO lines combine 8 x PCF8575 + 8 x PCF8574A giving
128 + 64 = 192 IO lines.
Be sure to have a well dimensioned power supply.
The library allows to read and write both single pins or 16 pins at once.
2021-12-01 14:50:20 +01:00
Furthermore some additional functions are implemented that are playful and useful.
2021-07-09 13:02:35 +02:00
2021-01-29 12:31:58 +01:00
## I2C Clock
Testing showed that the PCF8575 still works at 600 KHz and failed at 800 KHz.
2021-07-09 13:02:35 +02:00
These values are outside the specs of the datasheet so they are not recommended.
2021-01-29 12:31:58 +01:00
However when performance is needed you can try to overclock the chip.
2022-06-19 10:49:56 +02:00
TODO table (see PCF8574)
2021-07-09 13:02:35 +02:00
2021-01-29 12:31:58 +01:00
## Interface
**PCF8575_INITIAL_VALUE** is a define that can be set compile time or before
the include of "pcf8575.h" to overrule the default value used with the
**begin()** call.
2021-07-09 13:02:35 +02:00
2021-01-29 12:31:58 +01:00
### Constructor
2021-12-23 13:53:35 +01:00
- **PCF8575(uint8_t deviceAddress = 0x20, TwoWire \*wire = &Wire)** Constructor with the optional
I2C device address, default 0x20, and the optional Wire interface as parameter.
- **bool begin(uint8_t value = PCF8575_INITIAL_VALUE)** set the initial value for the pins and masks.
2022-04-11 10:49:34 +02:00
- **bool begin(int dataPin, int clockPin, uint8_t value = PCF8575_INITIAL_VALUE)** idem,
2021-12-01 14:50:20 +01:00
for the ESP32 where one can choose the I2C pins.
- **bool isConnected()** checks if the address is visible on the I2C bus.
2021-07-09 13:02:35 +02:00
- **bool setAddress(const uint8_t deviceAddress)** sets the device address after construction.
Can be used to switch between PCF8575 modules runtime. Note this corrupts internal buffered values,
so one might need to call **read16()** and/or **write16()**. Returns true if address can be found on I2C bus.
- **uint8_t getAddress()** returns the device address.
2021-01-29 12:31:58 +01:00
### Read and Write
2021-12-01 14:50:20 +01:00
- **uint16_t read16()** reads all 16 pins at once. This one does the actual reading.
- **uint8_t read(uint8_t pin)** reads a single pin; pin = 0..15.
- **uint16_t value()** returns the last read inputs again, as this information is buffered
2021-01-29 12:31:58 +01:00
in the class this is faster than reread the pins.
2021-12-01 14:50:20 +01:00
- **void write16(uint16_t value)** writes all 16 pins at once. This one does the actual reading.
- **void write(uint8_t pin, uint8_t value)** writes a single pin; pin = 0..15; value is HIGH(1) or LOW (0).
- **uint16_t valueOut()** returns the last written data.
2021-01-29 12:31:58 +01:00
2021-07-09 13:02:35 +02:00
2021-01-29 12:31:58 +01:00
### Button
2021-12-01 14:50:20 +01:00
The **"button"** functions are to be used when you mix input and output on one IC.
It does not change / affect the pins used for output by masking these.
Typical usage is to call **setButtonMask()** once in setup as pins do not (often) change
during program execution.
- **void setButtonMask(const uint16_t mask)** sets the (bit) mask which lines are input.
- **uint16_t getButtonMask()** returns the set buttonMask.
- **uint16_t readButton16()** use the mask set by setButtonMask to select specific input pins.
- **uint16_t readButton16(uint16_t mask)** use a specific mask to select specific input pins.
Note this can be a subset of the pins set with **setButtonMask()** if one wants to process not all.
- **uint8_t readButton(uint8_t pin)** read a singe input pin.
2021-01-29 12:31:58 +01:00
2022-06-19 10:49:56 +02:00
Background - https://github.com/RobTillaart/Arduino/issues/38
2021-07-09 13:02:35 +02:00
2021-01-29 12:31:58 +01:00
### Special
2021-12-01 14:50:20 +01:00
- **void toggle(uint8_t pin)** toggles a single pin.
- **void toggleMask(uint16_t mask)** toggles a selection of pins,
if you want to invert all pins use 0xFFFF (default value).
2022-06-19 10:49:56 +02:00
- **void shiftRight(uint8_t n = 1)** shifts output channels n pins (default 1) pins right (e.g. LEDs ).
2021-01-29 12:31:58 +01:00
Fills the higher lines with zero's.
2022-06-19 10:49:56 +02:00
- **void shiftLeft(uint8_t n = 1)** shifts output channels n pins (default 1) pins left (e.g. LEDs ).
2021-01-29 12:31:58 +01:00
Fills the lower lines with zero's.
2021-12-01 14:50:20 +01:00
- **void rotateRight(uint8_t n = 1)** rotates output channels to right, moving lowest line to highest line.
- **void rotateLeft(uint8_t n = 1)** rotates output channels to left, moving highest line to lowest line.
2022-06-19 10:49:56 +02:00
- **void reverse()** reverse the "bit pattern" of the lines, swapping pin 15 with 0, 14 with 1, 13 with 2 etc..
### Select
Some convenience wrappers.
- **void select(const uint8_t pin)** sets a single pin to HIGH, all others are set to LOW.
If pin > 15 all pins are set to LOW.
Can be used to select one of n devices.
- **void selectN(const uint8_t pin)** sets pins 0..pin to HIGH, all others are set to LOW.
If pin > 15 all pins are set to LOW.
This can typical be used to implement a VU meter.
- **void selectNone()** sets all pins to LOW.
- **void selectAll()** sets all pins to HIGH.
2021-01-29 12:31:58 +01:00
2021-07-09 13:02:35 +02:00
2021-12-01 14:50:20 +01:00
### Miscellaneous
2021-01-29 12:31:58 +01:00
2021-12-01 14:50:20 +01:00
- **int lastError()** returns the last error from the lib. (see .h file).
2021-01-29 12:31:58 +01:00
2021-07-09 13:02:35 +02:00
2021-01-29 12:31:58 +01:00
## Error codes
2021-07-09 13:02:35 +02:00
| name | value | description |
|:-------------------|:-----:|:------------------------|
2022-06-19 10:49:56 +02:00
| PCF8575_OK | 0x00 | no error |
| PCF8575_PIN_ERROR | 0x81 | pin number out of range |
| PCF8575_I2C_ERROR | 0x82 | I2C communication error |
2021-07-09 13:02:35 +02:00
2021-01-29 12:31:58 +01:00
## Testing
Testing the initial library is done by Colin Mackay (thanks!).
2021-12-01 14:50:20 +01:00
Platforms used for testing include: Nano, ESP32 and Seeed Xiao.
2021-01-29 12:31:58 +01:00
## Operation
2021-12-01 14:50:20 +01:00
See examples.
2022-06-19 10:49:56 +02:00
It is advised to use pull-up or pull-down resistors so the lines have a defined state at startup.
2021-12-01 14:50:20 +01:00
## Future
- update documentation.
- keep in sync with pcf8574 (as far as meaningful)
2021-12-23 13:53:35 +01:00
- **value()** => **lastRead16()** ??