2020-06-09 16:49:10 +02:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2020 Freie Universität Berlin
|
|
|
|
*
|
|
|
|
* This file is subject to the terms and conditions of the GNU Lesser
|
|
|
|
* General Public License v2.1. See the file LICENSE in the top level
|
|
|
|
* directory for more details.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @ingroup drivers_pca9633
|
|
|
|
* @brief Device driver for the PCA9633 I2C PWM controller
|
|
|
|
* @author Hendrik van Essen <hendrik.ve@fu-berlin.de>
|
|
|
|
* @file
|
|
|
|
* @{
|
|
|
|
*/
|
|
|
|
|
2020-10-21 15:56:59 +02:00
|
|
|
#include <assert.h>
|
2020-06-09 16:49:10 +02:00
|
|
|
#include <stdio.h>
|
|
|
|
|
|
|
|
#include "pca9633.h"
|
|
|
|
#include "pca9633_regs.h"
|
|
|
|
|
2020-10-22 11:34:31 +02:00
|
|
|
#define ENABLE_DEBUG 0
|
2020-06-09 16:49:10 +02:00
|
|
|
#include "debug.h"
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Write data to a register.
|
|
|
|
*
|
|
|
|
* @param[in] dev Device descriptor of the PCA9633
|
|
|
|
* @param[in] reg Register address to write to
|
|
|
|
* @param[in] data Data to write
|
|
|
|
*
|
|
|
|
* @return 0 on success
|
|
|
|
* @return -PCA9633_ERROR_I2C if acquiring of I2C bus fails
|
|
|
|
* @return -EIO When slave device doesn't ACK the byte
|
|
|
|
* @return -ENXIO When no devices respond on the address sent on the bus
|
|
|
|
* @return -ETIMEDOUT When timeout occurs before device's response
|
|
|
|
* @return -EINVAL When an invalid argument is given
|
|
|
|
* @return -EOPNOTSUPP When MCU driver doesn't support the flag operation
|
|
|
|
* @return -EAGAIN When a lost bus arbitration occurs
|
|
|
|
*/
|
|
|
|
static int _write_reg(pca9633_t* dev, uint8_t reg, uint8_t data);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Read data from a register.
|
|
|
|
*
|
|
|
|
* @param[in] dev Device descriptor of the PCA9633
|
|
|
|
* @param[in] reg Register address to read from
|
|
|
|
* @param[out] data Byte read from given registerAddress
|
|
|
|
*
|
|
|
|
* @return 0 on success
|
|
|
|
* @return -PCA9633_ERROR_I2C if acquiring of I2C bus fails
|
|
|
|
* @return -EIO When slave device doesn't ACK the byte
|
|
|
|
* @return -ENXIO When no devices respond on the address sent on the bus
|
|
|
|
* @return -ETIMEDOUT When timeout occurs before device's response
|
|
|
|
* @return -EINVAL When an invalid argument is given
|
|
|
|
* @return -EOPNOTSUPP When MCU driver doesn't support the flag operation
|
|
|
|
* @return -EAGAIN When a lost bus arbitration occurs
|
|
|
|
*/
|
|
|
|
static int _read_reg(pca9633_t* dev, uint8_t reg, uint8_t* data);
|
|
|
|
|
|
|
|
int pca9633_init(pca9633_t *dev, const pca9633_params_t *params)
|
|
|
|
{
|
|
|
|
assert(dev);
|
|
|
|
assert(params);
|
|
|
|
|
|
|
|
dev->params = *params;
|
|
|
|
|
|
|
|
int rc = _write_reg(dev, PCA9633_REG_MODE1, 0x0);
|
|
|
|
_write_reg(dev, PCA9633_REG_MODE2, 0x0);
|
|
|
|
|
|
|
|
if (rc != PCA9633_OK) {
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
return PCA9633_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
void pca9633_wakeup(pca9633_t* dev)
|
|
|
|
{
|
|
|
|
uint8_t reg;
|
|
|
|
_read_reg(dev, PCA9633_REG_MODE1, ®);
|
2020-10-12 19:02:58 +02:00
|
|
|
CLRBIT(reg, PCA9633_BIT_SLEEP);
|
2020-06-09 16:49:10 +02:00
|
|
|
|
|
|
|
_write_reg(dev, PCA9633_REG_MODE1, reg);
|
|
|
|
}
|
|
|
|
|
|
|
|
void pca9633_sleep(pca9633_t* dev)
|
|
|
|
{
|
|
|
|
uint8_t reg;
|
|
|
|
_read_reg(dev, PCA9633_REG_MODE1, ®);
|
2020-10-12 19:02:58 +02:00
|
|
|
SETBIT(reg, PCA9633_BIT_SLEEP);
|
2020-06-09 16:49:10 +02:00
|
|
|
|
|
|
|
_write_reg(dev, PCA9633_REG_MODE1, reg);
|
|
|
|
}
|
|
|
|
|
|
|
|
void pca9633_turn_on(pca9633_t* dev)
|
|
|
|
{
|
|
|
|
_write_reg(dev, PCA9633_REG_LEDOUT, dev->stored_reg_ledout);
|
|
|
|
}
|
|
|
|
|
|
|
|
void pca9633_turn_off(pca9633_t* dev)
|
|
|
|
{
|
|
|
|
_read_reg(dev, PCA9633_REG_LEDOUT, &dev->stored_reg_ledout);
|
|
|
|
_write_reg(dev, PCA9633_REG_LEDOUT, PCA9633_LDR_STATE_OFF);
|
|
|
|
}
|
|
|
|
|
|
|
|
void pca9633_set_pwm(pca9633_t* dev,
|
|
|
|
pca9633_pwm_channel_t pwm_channel, uint8_t pwm)
|
|
|
|
{
|
|
|
|
_write_reg(dev, pwm_channel, pwm);
|
|
|
|
}
|
|
|
|
|
|
|
|
void pca9633_set_grp_pwm(pca9633_t* dev, uint8_t pwm)
|
|
|
|
{
|
|
|
|
_write_reg(dev, PCA9633_REG_GRPPWM, pwm);
|
|
|
|
}
|
|
|
|
|
|
|
|
void pca9633_set_blinking(pca9633_t* dev, uint16_t blink_period_ms,
|
|
|
|
uint8_t on_off_ratio)
|
|
|
|
{
|
|
|
|
/* frequency of 24 Hz is used: */
|
|
|
|
uint16_t blink_period = (24 * blink_period_ms) / 1000;
|
|
|
|
if (blink_period > 255) {
|
|
|
|
blink_period = 255;
|
|
|
|
}
|
|
|
|
|
|
|
|
_write_reg(dev, PCA9633_REG_GRPFREQ, (uint8_t) blink_period);
|
|
|
|
_write_reg(dev, PCA9633_REG_GRPPWM, on_off_ratio);
|
|
|
|
}
|
|
|
|
|
|
|
|
void pca9633_set_rgb(pca9633_t* dev, uint8_t r, uint8_t g, uint8_t b)
|
|
|
|
{
|
|
|
|
pca9633_set_pwm(dev, dev->params.reg_pwm_red, r);
|
|
|
|
pca9633_set_pwm(dev, dev->params.reg_pwm_green, g);
|
|
|
|
pca9633_set_pwm(dev, dev->params.reg_pwm_blue, b);
|
|
|
|
}
|
|
|
|
|
|
|
|
void pca9633_set_rgba(pca9633_t* dev, uint8_t r, uint8_t g, uint8_t b, uint8_t a)
|
|
|
|
{
|
|
|
|
pca9633_set_rgb(dev, r, g, b);
|
|
|
|
|
|
|
|
if (dev->params.has_amber_channel) {
|
|
|
|
pca9633_set_pwm(dev, dev->params.reg_pwm_amber, a);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void pca9633_set_ldr_state(pca9633_t* dev,
|
|
|
|
pca9633_ldr_state_t state, pca9633_pwm_channel_t pwm_channel)
|
|
|
|
{
|
2020-10-12 19:02:58 +02:00
|
|
|
uint8_t ldr_bitmask;
|
2020-06-09 16:49:10 +02:00
|
|
|
switch (pwm_channel) {
|
|
|
|
case PCA9633_PWM_CHANNEL_0:
|
2020-10-12 19:02:58 +02:00
|
|
|
ldr_bitmask = PCA9633_BITMASK_LDR0;
|
2020-06-09 16:49:10 +02:00
|
|
|
break;
|
|
|
|
case PCA9633_PWM_CHANNEL_1:
|
2020-10-12 19:02:58 +02:00
|
|
|
ldr_bitmask = PCA9633_BITMASK_LDR1;
|
2020-06-09 16:49:10 +02:00
|
|
|
break;
|
|
|
|
case PCA9633_PWM_CHANNEL_2:
|
2020-10-12 19:02:58 +02:00
|
|
|
ldr_bitmask = PCA9633_BITMASK_LDR2;
|
2020-06-09 16:49:10 +02:00
|
|
|
break;
|
|
|
|
case PCA9633_PWM_CHANNEL_3:
|
2020-10-12 19:02:58 +02:00
|
|
|
ldr_bitmask = PCA9633_BITMASK_LDR3;
|
2020-06-09 16:49:10 +02:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint8_t reg;
|
|
|
|
_read_reg(dev, PCA9633_REG_LEDOUT, ®);
|
|
|
|
|
|
|
|
/* first clear both bits of ldr */
|
2020-10-12 19:02:58 +02:00
|
|
|
CLRBIT(reg, ldr_bitmask);
|
2020-06-09 16:49:10 +02:00
|
|
|
|
2020-10-12 19:02:58 +02:00
|
|
|
/* secondly set new state to specified ldr */
|
|
|
|
SETBIT(reg, (state << bitarithm_lsb(ldr_bitmask)));
|
2020-06-09 16:49:10 +02:00
|
|
|
|
|
|
|
_write_reg(dev, PCA9633_REG_LEDOUT, reg);
|
|
|
|
}
|
|
|
|
|
|
|
|
void pca9633_set_ldr_state_all(pca9633_t* dev, pca9633_ldr_state_t state)
|
|
|
|
{
|
2020-10-12 19:02:58 +02:00
|
|
|
uint8_t reg = 0;
|
|
|
|
|
|
|
|
SETBIT(reg, (state << bitarithm_lsb(PCA9633_BITMASK_LDR3)));
|
|
|
|
SETBIT(reg, (state << bitarithm_lsb(PCA9633_BITMASK_LDR2)));
|
|
|
|
SETBIT(reg, (state << bitarithm_lsb(PCA9633_BITMASK_LDR1)));
|
|
|
|
SETBIT(reg, (state << bitarithm_lsb(PCA9633_BITMASK_LDR0)));
|
2020-06-09 16:49:10 +02:00
|
|
|
|
|
|
|
_write_reg(dev, PCA9633_REG_LEDOUT, reg);
|
|
|
|
}
|
|
|
|
|
|
|
|
void pca9633_set_auto_increment(pca9633_t* dev, pca9633_auto_inc_option_t option)
|
|
|
|
{
|
2020-10-12 19:02:58 +02:00
|
|
|
uint8_t new_reg = 0;
|
2020-06-09 16:49:10 +02:00
|
|
|
|
|
|
|
switch (option) {
|
|
|
|
|
|
|
|
case PCA9633_AI_ALL:
|
2020-10-12 19:02:58 +02:00
|
|
|
SETBIT(new_reg, PCA9633_BIT_AI2);
|
|
|
|
CLRBIT(new_reg, PCA9633_BIT_AI1);
|
|
|
|
CLRBIT(new_reg, PCA9633_BIT_AI0);
|
2020-06-09 16:49:10 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case PCA9633_AI_IND:
|
2020-10-12 19:02:58 +02:00
|
|
|
SETBIT(new_reg, PCA9633_BIT_AI2);
|
|
|
|
SETBIT(new_reg, PCA9633_BIT_AI1);
|
|
|
|
CLRBIT(new_reg, PCA9633_BIT_AI0);
|
2020-06-09 16:49:10 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case PCA9633_AI_GBL:
|
2020-10-12 19:02:58 +02:00
|
|
|
SETBIT(new_reg, PCA9633_BIT_AI2);
|
|
|
|
CLRBIT(new_reg, PCA9633_BIT_AI1);
|
|
|
|
SETBIT(new_reg, PCA9633_BIT_AI0);
|
2020-06-09 16:49:10 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case PCA9633_AI_IND_GBL:
|
2020-10-12 19:02:58 +02:00
|
|
|
SETBIT(new_reg, PCA9633_BIT_AI2);
|
|
|
|
SETBIT(new_reg, PCA9633_BIT_AI1);
|
|
|
|
SETBIT(new_reg, PCA9633_BIT_AI0);
|
2020-06-09 16:49:10 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case PCA9633_AI_DISABLED:
|
|
|
|
/* fall-thru */
|
|
|
|
default:
|
2020-10-12 19:02:58 +02:00
|
|
|
CLRBIT(new_reg, PCA9633_BIT_AI2);
|
|
|
|
CLRBIT(new_reg, PCA9633_BIT_AI1);
|
|
|
|
CLRBIT(new_reg, PCA9633_BIT_AI0);
|
2020-06-09 16:49:10 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
_write_reg(dev, PCA9633_REG_MODE1, new_reg);
|
|
|
|
}
|
|
|
|
|
|
|
|
void pca9633_set_group_control_mode(pca9633_t* dev,
|
|
|
|
pca9633_group_control_mode_t mode)
|
|
|
|
{
|
|
|
|
uint8_t prev_reg;
|
|
|
|
_read_reg(dev, PCA9633_REG_MODE2, &prev_reg);
|
|
|
|
|
|
|
|
switch (mode) {
|
|
|
|
|
|
|
|
case PCA9633_GROUP_CONTROL_MODE_BLINKING:
|
2020-10-12 19:02:58 +02:00
|
|
|
SETBIT(prev_reg, PCA9633_BIT_DMBLNK);
|
2020-06-09 16:49:10 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case PCA9633_GROUP_CONTROL_MODE_DIMMING:
|
|
|
|
default:
|
2020-10-12 19:02:58 +02:00
|
|
|
CLRBIT(prev_reg, PCA9633_BIT_DMBLNK);
|
2020-06-09 16:49:10 +02:00
|
|
|
break;
|
|
|
|
}
|
2020-10-12 19:02:58 +02:00
|
|
|
|
|
|
|
_write_reg(dev, PCA9633_REG_MODE2, prev_reg);
|
2020-06-09 16:49:10 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
int _write_reg(pca9633_t* dev, uint8_t reg, uint8_t data)
|
|
|
|
{
|
|
|
|
i2c_t i2c_dev = dev->params.i2c_dev;
|
|
|
|
|
2021-11-25 13:43:06 +01:00
|
|
|
i2c_acquire(i2c_dev);
|
2020-06-09 16:49:10 +02:00
|
|
|
int rc = i2c_write_reg(i2c_dev, dev->params.i2c_addr, reg, data, 0);
|
|
|
|
i2c_release(i2c_dev);
|
|
|
|
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
int _read_reg(pca9633_t* dev, uint8_t reg, uint8_t* data)
|
|
|
|
{
|
|
|
|
i2c_t i2c_dev = dev->params.i2c_dev;
|
|
|
|
|
2021-11-25 13:43:06 +01:00
|
|
|
i2c_acquire(i2c_dev);
|
2020-06-09 16:49:10 +02:00
|
|
|
int rc = i2c_read_reg(i2c_dev, dev->params.i2c_addr, reg, data, 0);
|
|
|
|
i2c_release(i2c_dev);
|
|
|
|
|
|
|
|
return rc;
|
|
|
|
}
|