268 lines
8.3 KiB
C
Raw Normal View History

2020-11-27 11:28:57 +01:00
#pragma once
2016-04-24 08:26:22 +02:00
//
2022-01-06 20:19:35 +01:00
// FILE: PCA9635.h
2016-04-24 08:26:22 +02:00
// AUTHOR: Rob Tillaart
2016-04-24 09:20:33 +02:00
// DATE: 23-apr-2016
2023-05-26 16:22:43 +02:00
// VERSION: 0.4.6
2022-01-06 20:19:35 +01:00
// PURPOSE: Arduino library for PCA9635 I2C LED driver, 16 channel
2020-11-27 11:28:57 +01:00
// URL: https://github.com/RobTillaart/PCA9635
2016-04-24 08:26:22 +02:00
2021-01-29 12:31:58 +01:00
2016-04-24 08:26:22 +02:00
#include "Arduino.h"
2021-01-29 12:31:58 +01:00
#include "Wire.h"
2016-04-24 08:26:22 +02:00
2023-05-26 16:22:43 +02:00
#define PCA9635_LIB_VERSION (F("0.4.6"))
2016-04-24 08:26:22 +02:00
2023-05-07 18:51:32 +02:00
// mode codes
// NEW
#define PCA963X_MODE1 0x00
#define PCA963X_MODE2 0x01
// OLD
2021-01-29 12:31:58 +01:00
#define PCA9635_MODE1 0x00
#define PCA9635_MODE2 0x01
2022-05-16 18:25:10 +02:00
2023-05-07 18:51:32 +02:00
2022-05-16 18:25:10 +02:00
// 0x80 bit ==> Auto-Increment for all registers.
// used in writeN() - see issue #9 PCA9634
2023-05-07 18:51:32 +02:00
// NEW
#define PCA963X_PWM(x) (0x82+(x))
#define PCA963X_GRPPWM 0x12
#define PCA963X_GRPFREQ 0x13
// OLD
2022-05-16 18:25:10 +02:00
#define PCA9635_PWM(x) (0x82+(x))
2021-01-29 12:31:58 +01:00
#define PCA9635_GRPPWM 0x12
#define PCA9635_GRPFREQ 0x13
2016-04-24 08:26:22 +02:00
2023-05-07 18:51:32 +02:00
2022-01-06 20:19:35 +01:00
// check datasheet for details
2023-05-07 18:51:32 +02:00
// NEW
#define PCA963X_LEDOUT_BASE 0x14 // 0x14..0x17
#define PCA963X_LEDOFF 0x00 // default @ startup
#define PCA963X_LEDON 0x01
#define PCA963X_LEDPWM 0x02
#define PCA963X_LEDGRPPWM 0x03
// OLD
2023-03-07 16:18:49 +01:00
#define PCA9635_LEDOUT_BASE 0x14 // 0x14..0x17
#define PCA9635_LEDOFF 0x00 // default @ startup
2021-01-29 12:31:58 +01:00
#define PCA9635_LEDON 0x01
#define PCA9635_LEDPWM 0x02
#define PCA9635_LEDGRPPWM 0x03
2022-01-06 20:19:35 +01:00
// Error codes
2023-05-07 18:51:32 +02:00
// NEW
#define PCA963X_OK 0x00
#define PCA963X_ERROR 0xFF
#define PCA963X_ERR_WRITE 0xFE
#define PCA963X_ERR_CHAN 0xFD
#define PCA963X_ERR_MODE 0xFC
#define PCA963X_ERR_REG 0xFB
#define PCA963X_ERR_I2C 0xFA
// OLD
2021-01-29 12:31:58 +01:00
#define PCA9635_OK 0x00
#define PCA9635_ERROR 0xFF
#define PCA9635_ERR_WRITE 0xFE
#define PCA9635_ERR_CHAN 0xFD
#define PCA9635_ERR_MODE 0xFC
#define PCA9635_ERR_REG 0xFB
#define PCA9635_ERR_I2C 0xFA
2016-04-24 08:26:22 +02:00
2022-05-16 18:25:10 +02:00
2023-03-07 16:18:49 +01:00
// Configuration bits MODE1 register
2023-05-26 16:22:43 +02:00
// NEW
#define PCA963X_MODE1_AUTOINCR2 0x80 // ReadOnly, 0 = disable 1 = enable
#define PCA963X_MODE1_AUTOINCR1 0x40 // ReadOnly, bit1
#define PCA963X_MODE1_AUTOINCR0 0x20 // ReadOnly, bit0
#define PCA963X_MODE1_SLEEP 0x10 // 0 = normal 1 = sleep
#define PCA963X_MODE1_SUB1 0x08 // 0 = disable 1 = enable
#define PCA963X_MODE1_SUB2 0x04 // 0 = disable 1 = enable
#define PCA963X_MODE1_SUB3 0x02 // 0 = disable 1 = enable
#define PCA963X_MODE1_ALLCALL 0x01 // 0 = disable 1 = enable
#define PCA963X_MODE1_NONE 0x00
// OLD
2023-03-07 16:18:49 +01:00
#define PCA9635_MODE1_AUTOINCR2 0x80 // ReadOnly, 0 = disable 1 = enable
#define PCA9635_MODE1_AUTOINCR1 0x40 // ReadOnly, bit1
#define PCA9635_MODE1_AUTOINCR0 0x20 // ReadOnly, bit0
#define PCA9635_MODE1_SLEEP 0x10 // 0 = normal 1 = sleep
#define PCA9635_MODE1_SUB1 0x08 // 0 = disable 1 = enable
#define PCA9635_MODE1_SUB2 0x04 // 0 = disable 1 = enable
#define PCA9635_MODE1_SUB3 0x02 // 0 = disable 1 = enable
#define PCA9635_MODE1_ALLCALL 0x01 // 0 = disable 1 = enable
2022-06-02 12:11:57 +02:00
#define PCA9635_MODE1_NONE 0x00
2022-05-16 18:25:10 +02:00
2023-05-07 18:51:32 +02:00
2023-03-07 16:18:49 +01:00
// Configuration bits MODE2 register
2023-05-26 16:22:43 +02:00
// NEW
#define PCA963X_MODE2_BLINK 0x20 // 0 = dim 1 = blink
#define PCA963X_MODE2_INVERT 0x10 // 0 = normal 1 = inverted
#define PCA963X_MODE2_ACK 0x08 // 0 = on STOP 1 = on ACK
#define PCA963X_MODE2_TOTEMPOLE 0x04 // 0 = open drain 1 = totem-pole
#define PCA963X_MODE2_NONE 0x00
// OLD
2023-03-07 16:18:49 +01:00
#define PCA9635_MODE2_BLINK 0x20 // 0 = dim 1 = blink
#define PCA9635_MODE2_INVERT 0x10 // 0 = normal 1 = inverted
#define PCA9635_MODE2_ACK 0x08 // 0 = on STOP 1 = on ACK
#define PCA9635_MODE2_TOTEMPOLE 0x04 // 0 = open drain 1 = totem-pole
2022-06-02 12:11:57 +02:00
#define PCA9635_MODE2_NONE 0x00
2022-05-16 18:25:10 +02:00
2023-05-07 18:51:32 +02:00
2023-05-26 16:22:43 +02:00
// NOT IMPLEMENTED YET (TODO check)
//
// Registers in which the ALLCALL and sub-addresses are stored
// NEW
#define PCA963X_SUBADR(x) (0x17 +(x)) // x = 1..3
#define PCA963X_ALLCALLADR 0x1B
// OLD
2023-03-07 16:18:49 +01:00
#define PCA9635_SUBADR(x) (0x17+(x)) // x = 1..3
2022-01-06 20:19:35 +01:00
#define PCA9635_ALLCALLADR 0x1B
2016-04-24 08:26:22 +02:00
2021-01-29 12:31:58 +01:00
2023-05-26 16:22:43 +02:00
// Standard ALLCALL and sub-addresses --> only work for write commands and NOT for read commands
// TODO
//#define PCA963X_ALLCALL 0x70 // TDS of chip says 0xE0, however,
// in this library the LSB is added during the write command
// (0xE0 --> 0b11100000, 0x70 --> 0b1110000)
//#define PCA963X_SUB1 0x71 // see line above (0xE2 --> 0x71)
//#define PCA963X_SUB2 0x72 // see line above (0xE4 --> 0x72)
//#define PCA963X_SUB3 0x74 // see line above (0xE8 --> 0x74)
2023-05-07 18:51:32 +02:00
/////////////////////////////////////////////////////
//
// CLASS
//
2016-04-24 08:26:22 +02:00
class PCA9635
{
public:
2021-01-29 12:31:58 +01:00
explicit PCA9635(const uint8_t deviceAddress, TwoWire *wire = &Wire);
2016-04-24 08:26:22 +02:00
2020-11-27 11:28:57 +01:00
#if defined (ESP8266) || defined(ESP32)
2022-11-19 16:54:49 +01:00
bool begin(int sda, int scl,
2023-05-26 16:22:43 +02:00
uint8_t mode1_mask = PCA963X_MODE1_ALLCALL,
uint8_t mode2_mask = PCA963X_MODE2_NONE);
2020-11-27 11:28:57 +01:00
#endif
2023-05-26 16:22:43 +02:00
bool begin(uint8_t mode1_mask = PCA963X_MODE1_ALLCALL,
uint8_t mode2_mask = PCA963X_MODE2_NONE);
2021-01-29 12:31:58 +01:00
bool isConnected();
2023-03-07 16:18:49 +01:00
/////////////////////////////////////////////////////
//
// CONFIGURATION
//
uint8_t configure(uint8_t mode1_mask, uint8_t mode2_mask);
uint8_t channelCount();
2022-01-06 20:19:35 +01:00
2020-11-27 11:28:57 +01:00
uint8_t setLedDriverMode(uint8_t channel, uint8_t mode);
uint8_t getLedDriverMode(uint8_t channel);
2022-11-19 16:54:49 +01:00
// reg = 1, 2 check datasheet for values
2020-11-27 11:28:57 +01:00
uint8_t writeMode(uint8_t reg, uint8_t value);
uint8_t readMode(uint8_t reg);
2022-05-16 18:25:10 +02:00
// convenience wrappers
2023-05-07 18:51:32 +02:00
uint8_t setMode1(uint8_t value);
uint8_t setMode2(uint8_t value);
uint8_t getMode1();
uint8_t getMode2();
2022-05-16 18:25:10 +02:00
2023-05-07 18:51:32 +02:00
void setGroupPWM(uint8_t value);
uint8_t getGroupPWM();
2020-11-27 11:28:57 +01:00
2023-05-07 18:51:32 +02:00
void setGroupFREQ(uint8_t value);
uint8_t getGroupFREQ();
2020-11-27 11:28:57 +01:00
2023-03-07 16:18:49 +01:00
/////////////////////////////////////////////////////
//
// WRITE
//
// single PWM setting
uint8_t write1(uint8_t channel, uint8_t value);
// RGB setting, write three consecutive PWM registers
uint8_t write3(uint8_t channel, uint8_t R, uint8_t G, uint8_t B);
// generic worker, write N consecutive PWM registers
uint8_t writeN(uint8_t channel, uint8_t* arr, uint8_t count);
2023-05-07 18:51:32 +02:00
// generic worker, write N consecutive PWM registers without Stop command
uint8_t writeN_noStop(uint8_t channel, uint8_t* arr, uint8_t count);
// write stop command to end transmission
uint8_t writeStop();
2023-03-07 16:18:49 +01:00
/////////////////////////////////////////////////////
//
// ERROR
//
// note error flag is reset after read!
2020-11-27 11:28:57 +01:00
int lastError();
2016-04-24 08:26:22 +02:00
2023-03-07 16:18:49 +01:00
2022-06-02 12:11:57 +02:00
/////////////////////////////////////////////////////
//
2023-03-07 16:18:49 +01:00
// SUB CALL - ALL CALL (since 0.2.0)
2022-06-02 12:11:57 +02:00
//
// nr = { 1, 2, 3 }
bool enableSubCall(uint8_t nr);
bool disableSubCall(uint8_t nr);
bool isEnabledSubCall(uint8_t nr);
bool setSubCallAddress(uint8_t nr, uint8_t address);
uint8_t getSubCallAddress(uint8_t nr);
bool enableAllCall();
bool disableAllCall();
bool isEnabledAllCall();
bool setAllCallAddress(uint8_t address);
uint8_t getAllCallAddress();
2023-03-07 16:18:49 +01:00
/////////////////////////////////////////////////////
//
// OE - Output Enable control
//
bool setOutputEnablePin(uint8_t pin);
bool setOutputEnable(bool on);
uint8_t getOutputEnable();
2023-05-07 18:51:32 +02:00
/////////////////////////////////////////////////////
//
// EXPERIMENTAL
//
// 0.4.4
2023-03-17 10:31:25 +01:00
int I2C_SoftwareReset(uint8_t method);
2023-05-07 18:51:32 +02:00
// 0.4.5
// writing reg 14-17 LEDOUT
// reg LEDS
// 0 0..3
// 1 4..7
// 2 8..11
// 3 12..15
uint8_t writeLedOut(uint8_t reg, uint8_t mask);
uint8_t readLedOut(uint8_t reg);
uint8_t setLedDriverMode(uint8_t mode);
2023-03-17 10:31:25 +01:00
2016-04-24 08:26:22 +02:00
private:
2023-03-07 16:18:49 +01:00
// DIRECT CONTROL
uint8_t writeReg(uint8_t reg, uint8_t value); // returns error status.
2020-11-27 11:28:57 +01:00
uint8_t readReg(uint8_t reg);
uint8_t _address;
int _error;
2022-01-06 20:19:35 +01:00
uint8_t _channelCount = 16;
2023-03-07 16:18:49 +01:00
uint8_t _OutputEnablePin;
2021-01-29 12:31:58 +01:00
TwoWire* _wire;
2016-04-24 08:26:22 +02:00
};
2022-01-06 20:19:35 +01:00
2023-03-07 16:18:49 +01:00
// -- END OF FILE --
2022-01-06 20:19:35 +01:00