KT24-1110_65E-HA-651B/cpu/br25/iic_soft.c

338 lines
5.9 KiB
C
Raw Permalink Normal View History

2024-11-10 10:44:17 +00:00
//#include "system/init.h"
#include "asm/iic_soft.h"
#include "generic/gpio.h"
#define IIC_SCL_DIR(scl, val) \
gpio_set_direction(scl, val)
#define IIC_SCL_SET_PU(scl, on) \
gpio_set_pull_up(scl, on)
#define IIC_SCL_SET_PD(scl, on) \
gpio_set_pull_down(scl, on)
#define IIC_SCL_SET_DIE(scl, on) \
gpio_set_die(scl, on)
#define IIC_SCL_H(scl) \
gpio_direction_output(scl, 1)
#define IIC_SCL_L(scl) \
gpio_direction_output(scl, 0)
#define IIC_SDA_DIR(sda, val) \
gpio_set_direction(sda, val)
#define IIC_SDA_SET_PU(sda, on) \
gpio_set_pull_up(sda, on)
#define IIC_SDA_SET_PD(sda, on) \
gpio_set_pull_down(sda, on)
#define IIC_SDA_SET_DIE(sda, on) \
gpio_set_die(sda, on)
#define IIC_SDA_H(sda) \
gpio_direction_output(sda, 1)
#define IIC_SDA_L(sda) \
gpio_direction_output(sda, 0)
#define IIC_SDA_READ(sda) \
gpio_read(sda)
#define iic_get_id(iic) (iic)
static inline u32 iic_get_scl(soft_iic_dev iic)
{
u8 id = iic_get_id(iic);
return soft_iic_cfg[id].scl;
}
static inline u32 iic_get_sda(soft_iic_dev iic)
{
u8 id = iic_get_id(iic);
return soft_iic_cfg[id].sda;
}
static inline u32 iic_get_delay(soft_iic_dev iic)
{
u8 id = iic_get_id(iic);
return soft_iic_cfg[id].delay;
}
static inline u32 iic_get_io_pu(soft_iic_dev iic)
{
u8 id = iic_get_id(iic);
return soft_iic_cfg[id].io_pu;
}
int soft_iic_init(soft_iic_dev iic)
{
u32 scl, sda;
scl = iic_get_scl(iic);
sda = iic_get_sda(iic);
IIC_SCL_DIR(scl, 0);
IIC_SCL_SET_PD(scl, 0);
IIC_SCL_SET_DIE(scl, 1);
IIC_SCL_L(scl);
IIC_SDA_DIR(sda, 0);
IIC_SDA_SET_PD(sda, 0);
IIC_SDA_SET_DIE(sda, 1);
IIC_SDA_H(sda);
IIC_SCL_H(scl);
if (iic_get_io_pu(iic)) {
IIC_SCL_SET_PU(scl, 1);
IIC_SDA_SET_PU(sda, 1);
} else {
IIC_SCL_SET_PU(scl, 0);
IIC_SDA_SET_PU(sda, 0);
}
return 0;
}
void soft_iic_uninit(soft_iic_dev iic)
{
u32 scl, sda;
scl = iic_get_scl(iic);
sda = iic_get_sda(iic);
IIC_SCL_DIR(scl, 1);
IIC_SCL_SET_PU(scl, 0);
IIC_SCL_SET_PD(scl, 0);
IIC_SCL_SET_DIE(scl, 0);
IIC_SDA_DIR(sda, 1);
IIC_SDA_SET_PU(sda, 0);
IIC_SDA_SET_PD(sda, 0);
IIC_SDA_SET_DIE(sda, 0);
}
void soft_iic_suspend(soft_iic_dev iic)
{
u32 scl, sda;
scl = iic_get_scl(iic);
sda = iic_get_sda(iic);
IIC_SCL_SET_DIE(scl, 0);
IIC_SDA_SET_DIE(sda, 0);
}
void soft_iic_resume(soft_iic_dev iic)
{
u32 scl, sda;
scl = iic_get_scl(iic);
sda = iic_get_sda(iic);
IIC_SCL_SET_DIE(scl, 1);
IIC_SDA_SET_DIE(sda, 1);
}
void soft_iic_start(soft_iic_dev iic)
{
u32 scl, sda, dly_t;
scl = iic_get_scl(iic);
sda = iic_get_sda(iic);
dly_t = iic_get_delay(iic);
IIC_SDA_H(sda);
delay(dly_t);
IIC_SCL_H(scl);
delay(dly_t * 2);
IIC_SDA_L(sda);
delay(dly_t);
IIC_SCL_L(scl);
delay(dly_t);
}
void soft_iic_stop(soft_iic_dev iic)
{
u32 scl, sda, dly_t;
scl = iic_get_scl(iic);
sda = iic_get_sda(iic);
dly_t = iic_get_delay(iic);
IIC_SDA_L(sda);
delay(dly_t);
IIC_SCL_H(scl);
delay(dly_t * 2);
IIC_SDA_H(sda);
delay(dly_t);
}
static u8 soft_iic_check_ack(soft_iic_dev iic)
{
u8 ack;
u32 scl, sda, dly_t;
scl = iic_get_scl(iic);
sda = iic_get_sda(iic);
dly_t = iic_get_delay(iic);
IIC_SDA_DIR(sda, 1);
IIC_SCL_L(scl);
delay(dly_t);
IIC_SCL_H(scl);
delay(dly_t);
if (IIC_SDA_READ(sda) == 0) {
ack = 1;
} else {
ack = 0;
}
delay(dly_t);
IIC_SCL_L(scl);
delay(dly_t);
IIC_SDA_DIR(sda, 0);
IIC_SDA_L(sda);
return ack;
}
static void soft_iic_rx_ack(soft_iic_dev iic)
{
u32 scl, sda, dly_t;
scl = iic_get_scl(iic);
sda = iic_get_sda(iic);
dly_t = iic_get_delay(iic);
IIC_SDA_L(sda);
delay(dly_t);
IIC_SCL_H(scl);
delay(dly_t * 2);
IIC_SCL_L(scl);
delay(dly_t);
}
static void soft_iic_rx_nack(soft_iic_dev iic)
{
u32 scl, sda, dly_t;
scl = iic_get_scl(iic);
sda = iic_get_sda(iic);
dly_t = iic_get_delay(iic);
IIC_SDA_H(sda);
delay(dly_t);
IIC_SCL_H(scl);
delay(dly_t * 2);
IIC_SCL_L(scl);
delay(dly_t);
}
u8 soft_iic_tx_byte(soft_iic_dev iic, u8 byte)
{
u8 i, ret;
u32 scl, sda, dly_t;
scl = iic_get_scl(iic);
sda = iic_get_sda(iic);
dly_t = iic_get_delay(iic);
IIC_SCL_L(scl);
for (i = 0; i < 8; i++) { //MSB FIRST
if ((byte << i) & 0x80) {
IIC_SDA_H(sda);
} else {
IIC_SDA_L(sda);
}
delay(dly_t);
IIC_SCL_H(scl);
delay(dly_t * 2);
IIC_SCL_L(scl);
delay(dly_t);
}
return soft_iic_check_ack(iic);
}
u8 soft_iic_rx_byte(soft_iic_dev iic, u8 ack)
{
u8 byte = 0, i;
u32 scl, sda, dly_t;
scl = iic_get_scl(iic);
sda = iic_get_sda(iic);
dly_t = iic_get_delay(iic);
IIC_SDA_DIR(sda, 1);
for (i = 0; i < 8; i++) {
delay(dly_t);
IIC_SCL_H(scl);
delay(dly_t);
byte = byte << 1;
if (IIC_SDA_READ(sda)) {
byte |= 1;
}
delay(dly_t);
IIC_SCL_L(scl);
delay(dly_t);
}
IIC_SDA_DIR(sda, 0);
if (ack) {
soft_iic_rx_ack(iic);
} else {
soft_iic_rx_nack(iic);
}
return byte;
}
int soft_iic_read_buf(soft_iic_dev iic, void *buf, int len)
{
int i = 0;
if (!buf || !len) {
return -1;
}
for (i = 0; i < len - 1; i++) {
((u8 *)buf)[i] = soft_iic_rx_byte(iic, 1);
}
((u8 *)buf)[len - 1] = soft_iic_rx_byte(iic, 0);
return len;
}
int soft_iic_write_buf(soft_iic_dev iic, const void *buf, int len)
{
int i;
u8 ack;
if (!buf || !len) {
return -1;
}
for (i = 0; i < len; i++) {
ack = soft_iic_tx_byte(iic, ((u8 *)buf)[i]);
if (ack == 0) {
break;
}
}
return i;
}