Files
Karl Osterseher db0f9732b2 upgrade to IDF v5.1.1
custom board driver ma120x0 is broken
esp dsp is now IDF managed component
mdns is now IDF managed component

Signed-off-by: Karl Osterseher <karli_o@gmx.at>
2024-01-08 22:49:11 +01:00

199 lines
5.9 KiB
C

/*
* ESPRESSIF MIT License
*
* Copyright (c) 2021 <ESPRESSIF SYSTEMS (SHANGHAI) CO., LTD>
*
* Permission is hereby granted for use on all ESPRESSIF SYSTEMS products, in
* which case, it is free of charge, to any person obtaining a copy of this
* software and associated documentation files (the "Software"), to deal in the
* Software without restriction, including without limitation the rights to use,
* copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
* the Software, and to permit persons to whom the Software is furnished to do
* so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
* SOFTWARE.
*
*/
#include "tca9554.h"
#include <string.h>
#include "board.h"
#include "driver/gpio.h"
#include "esp_log.h"
#include "i2c_bus.h"
static char *TAG = "TCA9554";
static i2c_bus_handle_t i2c_handle;
#define SET_BITS(_m, _s, _v) ((_v) ? (_m) | ((_s)) : (_m) & ~((_s)))
#define GET_BITS(_m, _s) (((_m) & (_s)) ? true : false)
#define TCA9554_INPUT_PORT 0x00
#define TCA9554_OUTPUT_PORT 0x01
#define TCA9554_POLARITY_INVERSION_PORT 0x02
#define TCA9554_CONFIGURATION_PORT 0x03
typedef struct {
uint8_t addr;
char *name;
} tca9554_dev_t;
static tca9554_dev_t dev_list[] = {
{0x70, "TCA9554A"},
{0x40, "TCA9554"},
};
static uint8_t tca9554_addr;
static esp_err_t expander_dev_prob() {
for (size_t i = 0; i < sizeof(dev_list) / sizeof(dev_list[0]); i++) {
if (ESP_OK == i2c_bus_probe_addr(i2c_handle, dev_list[i].addr)) {
ESP_LOGI(TAG, "Detected IO expander device at 0x%02X, name is: %s",
dev_list[i].addr, dev_list[i].name);
tca9554_addr = dev_list[i].addr;
return ESP_OK;
}
}
tca9554_addr = 0;
ESP_LOGE(TAG, "IO expander device has not detected");
return ESP_ERR_NOT_FOUND;
}
static esp_err_t tca9554_write_reg(uint8_t reg_addr, uint8_t data) {
return i2c_bus_write_bytes(i2c_handle, tca9554_addr, &reg_addr,
sizeof(reg_addr), &data, sizeof(data));
}
static char tca9554_read_reg(uint8_t reg_addr) {
uint8_t data;
i2c_bus_read_bytes(i2c_handle, tca9554_addr, &reg_addr, sizeof(reg_addr),
&data, sizeof(data));
return data;
}
static int i2c_init(gpio_num_t clk, gpio_num_t sda) {
int res = 0;
i2c_config_t es_i2c_cfg = {
.mode = I2C_MODE_MASTER,
.sda_pullup_en = GPIO_PULLUP_ENABLE,
.scl_pullup_en = GPIO_PULLUP_ENABLE,
.master.clk_speed = 100000,
.scl_io_num = clk,
.sda_io_num = sda,
};
i2c_handle = i2c_bus_create(I2C_NUM_0, &es_i2c_cfg);
return res;
}
esp_tca9554_io_level_t tca9554_get_input_state(
esp_tca9554_gpio_num_t gpio_num) {
char data = 0;
if (gpio_num < TCA9554_GPIO_NUM_MAX) {
data = tca9554_read_reg(TCA9554_INPUT_PORT);
} else {
ESP_LOGE(TAG, "gpio num is error, current gpio: %d", gpio_num);
return TCA9554_LEVEL_ERROR;
}
return GET_BITS(data, gpio_num);
}
esp_tca9554_io_level_t tca9554_get_output_state(
esp_tca9554_gpio_num_t gpio_num) {
char data = 0;
if (gpio_num < TCA9554_GPIO_NUM_MAX) {
data = tca9554_read_reg(TCA9554_OUTPUT_PORT);
} else {
ESP_LOGE(TAG, "gpio num is error, current gpio: %d", gpio_num);
return TCA9554_LEVEL_ERROR;
}
return GET_BITS(data, gpio_num);
}
esp_err_t tca9554_set_output_state(esp_tca9554_gpio_num_t gpio_num,
esp_tca9554_io_level_t level) {
char data;
esp_err_t res = ESP_FAIL;
if (gpio_num < TCA9554_GPIO_NUM_MAX) {
data = tca9554_read_reg(TCA9554_OUTPUT_PORT);
res =
tca9554_write_reg(TCA9554_OUTPUT_PORT, SET_BITS(data, gpio_num, level));
} else {
ESP_LOGE(TAG, "gpio num is error, current gpio: %d", gpio_num);
}
return res;
}
esp_err_t tca9554_set_polarity_inversion(esp_tca9554_gpio_num_t gpio_num,
esp_tca9554_io_polarity_t polarity) {
char data;
esp_err_t res = ESP_FAIL;
if (gpio_num < TCA9554_GPIO_NUM_MAX) {
data = tca9554_read_reg(TCA9554_POLARITY_INVERSION_PORT);
res = tca9554_write_reg(TCA9554_POLARITY_INVERSION_PORT,
SET_BITS(data, gpio_num, polarity));
} else {
ESP_LOGE(TAG, "gpio num is error, current gpio: %d", gpio_num);
}
return res;
}
esp_tca9554_io_config_t tca9554_get_io_config(esp_tca9554_gpio_num_t gpio_num) {
char data = 0;
if (gpio_num < TCA9554_GPIO_NUM_MAX) {
data = tca9554_read_reg(TCA9554_CONFIGURATION_PORT);
} else {
ESP_LOGE(TAG, "gpio num is error, current gpio: %d", gpio_num);
return TCA9554_LEVEL_ERROR;
}
return GET_BITS(data, gpio_num);
}
esp_err_t tca9554_set_io_config(esp_tca9554_gpio_num_t gpio_num,
esp_tca9554_io_config_t io_config) {
char data;
esp_err_t res = ESP_FAIL;
if (gpio_num < TCA9554_GPIO_NUM_MAX) {
data = tca9554_read_reg(TCA9554_CONFIGURATION_PORT);
res = tca9554_write_reg(TCA9554_CONFIGURATION_PORT,
SET_BITS(data, gpio_num, io_config));
} else {
ESP_LOGE(TAG, "gpio num is error, current gpio: %d", gpio_num);
}
return res;
}
void tca9554_read_all() {
for (int i = 0; i < 0x04; i++) {
uint8_t reg = tca9554_read_reg(i);
ESP_LOGI(TAG, "REG:%02x, %02x", i, reg);
}
}
esp_err_t tca9554_init(esp_tca9554_config_t *cfg) {
esp_err_t ret = ESP_OK;
i2c_init(cfg->i2c_scl, cfg->i2c_sda);
expander_dev_prob();
return ret;
}
esp_err_t tca9554_deinit() {
i2c_bus_delete(i2c_handle);
return ESP_OK;
}