/**
* @file ch423s.c
* @brief
* @author lch (lch_work@foxmail.com)
* @version 1.0
* @date 20251224
*
* @copyright Copyright (c) 2025 by OLE, All Rights Reserved.
*
* @par 修改日志:
*
* | Date | Version | Author | Description
* |
|---|
| 20251224 | 2.0 | test | 内容
* |
*/
#include
#include
#include
#include
#include
#include "head.h"
#include "ch423s.h"
#if defined IO_CHIP_CH423S
/* 设置系统参数命令 */
#define CH423_SYS_CMD 0x48 // 设置系统参数命令,默认方式
#define BIT_SLEEP 0x30 // 低功耗睡眠控制 0正常工作 1睡眠模式
#define BIT_INTENS 0x20 // 动态显示驱动亮度控制
#define BIT_OD_OE 0x10 // 输出引脚 OC15~OC0 开漏输出使能 0推挽输出低电平和高电平 1开漏输出低电平和不输出
#define BIT_X_INT 0x08 // 使能输入电平变化中断 0禁止输入电平变化中断 为1并且DEC_H为0允许从OC15引脚输出电平变化中断
#define BIT_DEC_H 0x04 // 控制开漏输出引脚高8位的片选译码 0通用输出 1分时显示扫描计数器译码后
#define BIT_DEC_L 0x02 // 控制开漏输出引脚低8位的片选译码 0通用输出 1分时显示扫描计数器译码后
#define BIT_IO_OE 0x01 // 控制双向输入输出引脚的三态输出 0禁止输出 1允许输出
/* 设置低8位开漏输出命令 */
#define CH423_OC_L_CMD 0x44 // 设置低8位开漏输出命令,默认方式
#define BIT_OC0_L_DAT 0x01 // OC0为0则使引脚输出低电平,为1则引脚不输出
#define BIT_OC1_L_DAT 0x02 // OC1为0则使引脚输出低电平,为1则引脚不输出
#define BIT_OC2_L_DAT 0x04 // OC2为0则使引脚输出低电平,为1则引脚不输出
#define BIT_OC3_L_DAT 0x08 // OC3为0则使引脚输出低电平,为1则引脚不输出
#define BIT_OC4_L_DAT 0x10 // OC4为0则使引脚输出低电平,为1则引脚不输出
#define BIT_OC5_L_DAT 0x20 // OC5为0则使引脚输出低电平,为1则引脚不输出
#define BIT_OC6_L_DAT 0x40 // OC6为0则使引脚输出低电平,为1则引脚不输出
#define BIT_OC7_L_DAT 0x80 // OC7为0则使引脚输出低电平,为1则引脚不输出
/* 设置高8位开漏输出命令 */
#define CH423_OC_H_CMD 0x46 // 设置低8位开漏输出命令,默认方式
#define BIT_OC8_L_DAT 0x01 // OC8为0则使引脚输出低电平,为1则引脚不输出
#define BIT_OC9_L_DAT 0x02 // OC9为0则使引脚输出低电平,为1则引脚不输出
#define BIT_OC10_L_DAT 0x04 // OC10为0则使引脚输出低电平,为1则引脚不输出
#define BIT_OC11_L_DAT 0x08 // OC11为0则使引脚输出低电平,为1则引脚不输出
#define BIT_OC12_L_DAT 0x10 // OC12为0则使引脚输出低电平,为1则引脚不输出
#define BIT_OC13_L_DAT 0x20 // OC13为0则使引脚输出低电平,为1则引脚不输出
#define BIT_OC14_L_DAT 0x40 // OC14为0则使引脚输出低电平,为1则引脚不输出
#define BIT_OC15_L_DAT 0x80 // OC15为0则使引脚输出低电平,为1则引脚不输出
/* 设置双向输入输出命令 */
#define CH423_SET_IO_CMD 0x60 // 设置双向输入输出命令,默认方式
#define BIT_IO0_DAT 0x01 // 写入双向输入输出引脚的输出寄存器,当IO_OE=1,IO0为0输出低电平,为1输出高电平
#define BIT_IO1_DAT 0x02 // 写入双向输入输出引脚的输出寄存器,当IO_OE=1,IO1为0输出低电平,为1输出高电平
#define BIT_IO2_DAT 0x04 // 写入双向输入输出引脚的输出寄存器,当IO_OE=1,IO2为0输出低电平,为1输出高电平
#define BIT_IO3_DAT 0x08 // 写入双向输入输出引脚的输出寄存器,当IO_OE=1,IO3为0输出低电平,为1输出高电平
#define BIT_IO4_DAT 0x10 // 写入双向输入输出引脚的输出寄存器,当IO_OE=1,IO4为0输出低电平,为1输出高电平
#define BIT_IO5_DAT 0x20 // 写入双向输入输出引脚的输出寄存器,当IO_OE=1,IO5为0输出低电平,为1输出高电平
#define BIT_IO6_DAT 0x40 // 写入双向输入输出引脚的输出寄存器,当IO_OE=1,IO6为0输出低电平,为1输出高电平
#define BIT_IO7_DAT 0x80 // 写入双向输入输出引脚的输出寄存器,当IO_OE=1,IO7为0输出低电平,为1输出高电平
/**< led配置结构体 */
struct t_config_led
{
uint8_t uc_port;
uint8_t uc_pin;
uint8_t uc_en; /**< 使能标志 */
uint8_t uc_mode; /**< 模式 0灭 1亮 2闪烁 */
uint8_t uc_state; /**< 目前状态 */
uint32_t ul_time; /**< 翻转时间 */
uint32_t ul_count; /**< 翻转计时 */
};
static struct t_config_led mst_led_cfg[LED_MAX_NUM] =
{
{CH423_OC_L_CMD, 0, 0, 0, 0, 0},
{CH423_OC_L_CMD, 1, 0, 0, 0, 0},
{CH423_OC_L_CMD, 2, 0, 0, 0, 0},
{CH423_OC_L_CMD, 3, 0, 0, 0, 0},
{CH423_OC_L_CMD, 4, 0, 0, 0, 0},
{CH423_OC_L_CMD, 5, 0, 0, 0, 0},
{CH423_OC_L_CMD, 6, 0, 0, 0, 0},
{CH423_OC_L_CMD, 7, 0, 0, 0, 0},
{CH423_OC_H_CMD, 8, 0, 0, 0, 0},
{CH423_OC_H_CMD, 9, 0, 0, 0, 0},
{CH423_OC_H_CMD, 10, 0, 0, 0, 0},
{CH423_OC_H_CMD, 11, 0, 0, 0, 0},
{CH423_OC_H_CMD, 12, 0, 0, 0, 0},
{CH423_OC_H_CMD, 13, 0, 0, 0, 0},
{CH423_OC_H_CMD, 14, 0, 0, 0, 0},
{CH423_OC_H_CMD, 15, 0, 0, 0, 0},
{CH423_SET_IO_CMD, 16, 0, 0, 0, 0},
{CH423_SET_IO_CMD, 17, 0, 0, 0, 0},
{CH423_SET_IO_CMD, 18, 0, 0, 0, 0},
{CH423_SET_IO_CMD, 19, 0, 0, 0, 0},
{CH423_SET_IO_CMD, 20, 0, 0, 0, 0},
{CH423_SET_IO_CMD, 21, 0, 0, 0, 0},
{CH423_SET_IO_CMD, 22, 0, 0, 0, 0},
{CH423_SET_IO_CMD, 23, 0, 0, 0, 0},
};
#define I2C_DEV "/dev/i2c-5"
static int sl_ch423s_fd = -1, sl_ch423s_init = -1;
static uint32_t sul_io_status = 0xffffff;
static uint8_t msuc_test_flag = 0;
/**
* @brief
* @author lch (lch_work@foxmail.com)
* @version 1.0
* @date 20251224
* @param[in/out] {uint8_t} _uc_cmd 命令
* @param[in/out] {uint8_t} _uc_state 状态
* @return * int
* @retval none
*
* @warning none
* @note 注意这个原件没有固定的iic地址 抓数据发现 命令 + 数据 就可以操作
*/
static int ch423_write_data(uint8_t _uc_cmd, uint8_t _uc_state)
{
/* 使用寄存器地址来寻址 */
if (ioctl(sl_ch423s_fd, I2C_SLAVE, _uc_cmd >> 1) < 0)
{
dp_err_n_c("Failed to set I2C slave address: %s\n", strerror(errno));
return -1;
}
if (write(sl_ch423s_fd, &_uc_state, 1) != 1)
{
dp_err_n_c("Failed to write data: %s", strerror(errno));
return -2;
}
return 0;
}
/**
* @brief
* @author lch (lch_work@foxmail.com)
* @version 1.0
* @date 20251224
* @param[in/out] {uint8_t} _uc_idx
* @param[in/out] {uint8_t} _uc_state
* @return * void
* @retval none
*
* @warning none
* @note none
*/
static void ch423s_updata_port_status(uint8_t _uc_idx, uint8_t _uc_state)
{
uint32_t ul_data_1 = 0, ul_data_2 = 0;
ul_data_1 = sul_io_status;
if ((sul_io_status >> _uc_idx) & 0x01)
{
sul_io_status ^= (1 << _uc_idx);
}
ul_data_2 = sul_io_status;
/* 重新设置 */
sul_io_status |= _uc_state ? (0 << _uc_idx) : (1 << _uc_idx);
// dp_err_n_c_rt("ul_data_1 = 0x%06x, ul_data_2 = 0x%06x, sul_io_status = 0x%06x\r\n", ul_data_1, ul_data_1, sul_io_status);
}
/**
* @brief 定时更新gpio状态
* @author lch (lch_work@foxmail.com)
* @version 1.0
* @date 20251224
* @return * int
* @retval none
*
* @warning none
* @note none
*/
int ch423sio_updata_status_ontime(void)
{
uint8_t uca_state[3] = {0};
int l_ret = 0;
uca_state[0] = (uint8_t)(sul_io_status & 0xff);
(0 == ch423_write_data(CH423_OC_L_CMD, uca_state[0])) ? l_ret : l_ret--;
uca_state[1] = (uint8_t)((sul_io_status >> 8) & 0xff);
(0 == ch423_write_data(CH423_OC_H_CMD, uca_state[1])) ? l_ret : l_ret--;
uca_state[2] = (uint8_t)((sul_io_status >> 16) & 0xff);
(0 == ch423_write_data(CH423_SET_IO_CMD, uca_state[2])) ? l_ret : l_ret--;
return l_ret;
}
/**
* @brief 设置io状态
* @author lch (lch_work@foxmail.com)
* @version 1.0
* @date 20251224
* @param[in/out] {uint8_t} _uc_idx 下标
* @param[in/out] {uint8_t} _uc_state 电平状态
* @return * int
* @retval none
*
* @warning none
* @note none
*/
int ch423sio_set_io(uint8_t _uc_idx, uint8_t _uc_state)
{
if (_uc_idx >= LED_MAX_NUM || sl_ch423s_init < 0 || _uc_state > 1)
{
return -1;
}
switch (mst_led_cfg[_uc_idx].uc_port)
{
case CH423_OC_L_CMD:
ch423s_updata_port_status(mst_led_cfg[_uc_idx].uc_pin, _uc_state);
break;
case CH423_OC_H_CMD:
ch423s_updata_port_status(mst_led_cfg[_uc_idx].uc_pin, _uc_state);
break;
case CH423_SET_IO_CMD:
ch423s_updata_port_status(mst_led_cfg[_uc_idx].uc_pin, _uc_state);
break;
default:
return -2;
break;
}
return 0;
}
/**
* @brief 设置io状态
* @author lch (lch_work@foxmail.com)
* @version 1.0
* @date 20251224
* @param[in/out] {uint8_t} _uc_idx 下标
* @param[in/out] {uint8_t} _uc_state 电平状态
* @return * int
* @retval none
*
* @warning none
* @note none
*/
int ch423sio_set_ioupdata(uint8_t _uc_idx, uint8_t _uc_state)
{
int ret = -1;
uint8_t io_val[3] = {0};
if (_uc_idx >= LED_MAX_NUM || sl_ch423s_init < 0 || _uc_state > 1)
{
dp_err_n_c("_uc_idx = %d, sl_ch423s_init = %d, _uc_state = %d", _uc_idx, sl_ch423s_init, _uc_state);
return -1;
}
switch (mst_led_cfg[_uc_idx].uc_port)
{
case CH423_OC_L_CMD:
ch423s_updata_port_status(mst_led_cfg[_uc_idx].uc_pin, _uc_state);
io_val[0] = (uint8_t)(sul_io_status & 0xff);
ret = ch423_write_data(CH423_OC_L_CMD, io_val[0]);
break;
case CH423_OC_H_CMD:
ch423s_updata_port_status(mst_led_cfg[_uc_idx].uc_pin, _uc_state);
io_val[1] = (uint8_t)((sul_io_status & 0xff00) >> 8);
ret = ch423_write_data(CH423_OC_H_CMD, io_val[1]);
break;
case CH423_SET_IO_CMD:
ch423s_updata_port_status(mst_led_cfg[_uc_idx].uc_pin, _uc_state);
io_val[2] = (uint8_t)((sul_io_status & 0xff0000) >> 16);
ret = ch423_write_data(CH423_SET_IO_CMD, io_val[2]);
break;
default:
return -2;
break;
}
return ret;
}
/**
* @brief io测试
* @author lch (lch_work@foxmail.com)
* @version 1.0
* @date 20251230
* @return * int
* @retval none
*
* @warning none
* @note none
*/
static int ch423sio_run_test(void)
{
int i = 0;
for (i = 0; i < LED_MAX_NUM; i++)
{
ch423sio_set_ioupdata(i, 1);
usleep(1000 * 50);
}
usleep(1000 * 100);
for (i = (LED_MAX_NUM - 1); i >= 0; i--)
{
ch423sio_set_ioupdata(i, 0);
usleep(1000 * 50);
}
usleep(1000 * 100);
for (i = 0; i < LED_MAX_NUM; i++)
{
ch423sio_set_ioupdata(i, 1);
}
usleep(1000 * 100);
for (i = 0; i < LED_MAX_NUM; i++)
{
ch423sio_set_ioupdata(i, 0);
}
usleep(1000 * 100);
return 0;
}
/**
* @brief 初始化设备
* @author lch (lch_work@foxmail.com)
* @version 1.0
* @date 20251224
* @return * int
* @retval none
*
* @warning none
* @note none
*/
int ch423sio_init_dev(void)
{
// 打开I2C设备
sl_ch423s_fd = open(I2C_DEV, O_RDWR);
if (sl_ch423s_fd < 0)
{
dp_err_n_c_rt("Failed to open I2C device %s: %s", I2C_DEV, strerror(errno));
return -1;
}
ch423_write_data(CH423_SYS_CMD, BIT_IO_OE); /* 将IO7~IO0设置为输出 其它参数使用默认值 */
ch423_write_data(CH423_OC_L_CMD, 0xff);
ch423_write_data(CH423_OC_H_CMD, 0xff);
ch423_write_data(CH423_SET_IO_CMD, 0xff);
sl_ch423s_init = 0;
ch423sio_run_test();
return 0;
}
/**
* @brief 关闭句柄
* @author lch (lch_work@foxmail.com)
* @version 1.0
* @date 20251224
* @return * int
* @retval none
*
* @warning none
* @note none
*/
int ch423sio_exit_dev(void)
{
if (sl_ch423s_fd >= 0)
{
close(sl_ch423s_fd);
sl_ch423s_fd = -1;
}
return 0;
}
/**
* @brief 控制led驱动
*
* @param[in] {uint8_t} _uc_idx 硬件地址
* @param[in] {uint8_t} _uc_set 模式 0灭 1亮
*/
static void ch423_control_drive(uint8_t _uc_idx, uint8_t _uc_set)
{
if (_uc_idx >= LED_MAX_NUM || _uc_set >= e_io_mode_max)
return;
switch (_uc_set)
{
case e_io_mode_off:
ch423sio_set_io(_uc_idx, e_io_mode_off);
mst_led_cfg[_uc_idx].uc_state = e_io_mode_off;
break;
case e_io_mode_on:
ch423sio_set_io(_uc_idx, e_io_mode_on);
mst_led_cfg[_uc_idx].uc_state = e_io_mode_on;
break;
default:
return;
break;
}
}
/**
* @brief 切换led模式
*
* @details 闪烁模式一直在计时,uc_en使能标志不会清零
*/
void ch423sio_handle_mode(void)
{
uint8_t uc_updata = 0;
#if (0) /* 测试打印代码 */
static uint64_t sull_printf_led = 0;
if (sull_printf_led++ >= (10 * 10))
{
sull_printf_led = 0;
dp_info_n_c("dTCounter = %d", dTCounter);
for (uint8_t j = 0; j < LED_MAX_NUM; j++)
{
dp_info_n_c("[%02d] uc_en = %d, mode = %d, state = %d, ul_time = %d, ul_count = %d",
j,
mst_led_cfg[j].uc_en,
mst_led_cfg[j].uc_mode,
mst_led_cfg[j].uc_state,
mst_led_cfg[j].ul_time,
mst_led_cfg[j].ul_count);
}
}
#endif
for (uint8_t i = 0; i < LED_MAX_NUM; i++)
{
if (mst_led_cfg[i].uc_en == 0)
continue;
mst_led_cfg[i].ul_count += 100;
if (mst_led_cfg[i].ul_count > mst_led_cfg[i].ul_time)
{
mst_led_cfg[i].ul_count = 0;
uc_updata = 1;
if (mst_led_cfg[i].uc_mode != e_io_mode_shine)
{
mst_led_cfg[i].uc_en = 0;
ch423_control_drive(i, mst_led_cfg[i].uc_mode);
}
else
{
if (e_io_mode_on == mst_led_cfg[i].uc_state)
{
ch423_control_drive(i, e_io_mode_off);
}
else
{
ch423_control_drive(i, e_io_mode_on);
}
}
}
}
if (1 == uc_updata)
{
ch423sio_updata_status_ontime();
}
}
/**
* @brief
*
* @param[in] {uint8_t} _uc_idx led映射
* @param[in] {enum e_mode_ch423} _e_mode 模式 0灭 1亮 2闪烁
* @param[in] {uint32_t} _ul_time 闪烁时间
* @return int
* @retval 0设置模式成功
* @details none
*/
int ch423sio_control_state(uint8_t _uc_idx, enum e_mode_ch423 _e_mode, uint32_t _ul_time)
{
if (_uc_idx >= LED_MAX_NUM || _e_mode >= e_io_mode_max)
return -1;
mst_led_cfg[_uc_idx].uc_en = 1;
mst_led_cfg[_uc_idx].uc_mode = _e_mode;
if (_e_mode == e_io_mode_shine)
mst_led_cfg[_uc_idx].ul_time = _ul_time;
return 0;
}
#endif /* TMP_CHIP_AHT20 */