473 lines
13 KiB
C
Raw Normal View History

/*
* Copyright : (C) 2022 Phytium Information Technology, Inc.
* All Rights Reserved.
*
* This program is OPEN SOURCE software: you can redistribute it and/or modify it
* under the terms of the Phytium Public License as published by the Phytium Technology Co.,Ltd,
* either version 1.0 of the License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,but WITHOUT ANY WARRANTY;
* without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
* See the Phytium Public License for more details.
*
*
* FilePath: fi2c_hw.c
* Date: 2021-11-01 14:53:42
* LastEditTime: 2022-02-18 08:36:22
* Description:  This file is for I2C register read/write operation
*
* Modify History:
* Ver   Who        Date         Changes
* ----- ------     --------    --------------------------------------
* 1.0 zhugengyu 2021/11/1 first commit
* 1.1 liushengming 2022/02/18 support slave mode
*/
/***************************** Include Files *********************************/
#include <string.h>
#include "fsleep.h"
#include "fdebug.h"
#include "ferror_code.h"
#include "fassert.h"
#include "fi2c_hw.h"
#include "fi2c.h"
/************************** Constant Definitions *****************************/
/**************************** Type Definitions *******************************/
typedef struct
{
u32 speed_mode;
u32 scl_lcnt;
u32 scl_hcnt;
u32 sda_hold;
} FI2cSpeedCfg; /* speed related configs */
typedef struct
{
u32 speed;
u32 min_scl_hightime_ns;
u32 min_scl_lowtime_ns;
u32 def_risetime_ns;
u32 def_falltime_ns;
} FI2cSpeedModeInfo; /* speed calculation related configs */
/************************** Variable Definitions *****************************/
static const FI2cSpeedModeInfo I2C_SPEED_CFG[FI2C_SPEED_MODE_MAX] =
{
[FI2C_STANDARD_SPEED] = {
FI2C_SPEED_STANDARD_RATE,
4000,
4700,
1000,
300,
},
[FI2C_FAST_SPEED] = {
FI2C_SPEED_FAST_RATE,
1000,
1300,
300,
300,
}
};
/***************** Macros (Inline Functions) Definitions *********************/
#define FI2C_DEBUG_TAG "I2C_HW"
#define FI2C_ERROR(format, ...) FT_DEBUG_PRINT_E(FI2C_DEBUG_TAG, format, ##__VA_ARGS__)
#define FI2C_INFO(format, ...) FT_DEBUG_PRINT_I(FI2C_DEBUG_TAG, format, ##__VA_ARGS__)
#define FI2C_DEBUG(format, ...) FT_DEBUG_PRINT_D(FI2C_DEBUG_TAG, format, ##__VA_ARGS__)
#define FI2C_TIMEOUT 500
/************************** Function Prototypes ******************************/
/**
* @name: FI2cSetEnable
* @msg: I2C控制器的使能状态
* @return {*}
* @param {uintptr} addr I2c控制器基地址
* @param {boolean} enable TRUE: 使FALSE: 使
*/
FError FI2cSetEnable(uintptr addr, boolean enable)
{
u32 status = enable ? FI2C_IC_ENABLE : FI2C_IC_DISABLE;
u32 timeout = FI2C_TIMEOUT;
do
{
FI2C_WRITE_REG32(addr, FI2C_ENABLE_OFFSET, status);
if (((FI2C_READ_REG32(addr, FI2C_ENABLE_STATUS_OFFSET)) & FI2C_IC_ENABLE_MASK) == status)
{
return FI2C_SUCCESS;
}
}
while (0 != timeout--);
FI2C_ERROR("Timeout in %sabling I2C ctrl.", enable ? "en" : "dis");
return FI2C_ERR_TIMEOUT;
}
/**
* @name: FI2cCalcTiming
* @msg: I2C的上升沿下降沿配置
* @return {*}
* @param {u32} bus_clk_hz I2C总线时钟速度 Hz48MHz
* @param {u32} spk_cnt, spk数目
* @param {FI2cSpeedCfg} *speed_cfg_p
*/
static FError FI2cCalcTiming(u32 bus_clk_hz, u32 spk_cnt, FI2cSpeedCfg *speed_cfg_p)
{
FASSERT(speed_cfg_p);
u32 speed_mode = speed_cfg_p->speed_mode;
const FI2cSpeedModeInfo *info_p = &I2C_SPEED_CFG[speed_mode];
int fall_cnt, rise_cnt, min_t_low_cnt, min_t_high_cnt;
int hcnt, lcnt, period_cnt, diff, tot;
int sda_hold_time_ns, scl_rise_time_ns, scl_fall_time_ns;
period_cnt = bus_clk_hz / info_p->speed;
scl_rise_time_ns = info_p->def_risetime_ns;
scl_fall_time_ns = info_p->def_falltime_ns;
/* convert a period to a number of IC clk cycles */
rise_cnt = DIV_ROUND_UP(bus_clk_hz / 1000 * scl_rise_time_ns, NANO_TO_KILO);
fall_cnt = DIV_ROUND_UP(bus_clk_hz / 1000 * scl_fall_time_ns, NANO_TO_KILO);
min_t_low_cnt = DIV_ROUND_UP(bus_clk_hz / 1000 * info_p->min_scl_lowtime_ns, NANO_TO_KILO);
min_t_high_cnt = DIV_ROUND_UP(bus_clk_hz / 1000 * info_p->min_scl_hightime_ns, NANO_TO_KILO);
FI2C_INFO("i2c: mode %d, bus_clk %d, speed %d, period %d rise %d fall %d tlow %d thigh %d spk %d\n",
speed_mode, bus_clk_hz, info_p->speed, period_cnt, rise_cnt, fall_cnt,
min_t_low_cnt, min_t_high_cnt, spk_cnt);
/*
* Back-solve for hcnt and lcnt according to the following equations:
* SCL_High_time = [(HCNT + IC_*_SPKLEN + 7) * icClk] + SCL_Fall_time
* SCL_Low_time = [(LCNT + 1) * icClk] - SCL_Fall_time + SCL_Rise_time
*/
hcnt = min_t_high_cnt - fall_cnt - 7 - spk_cnt;
lcnt = min_t_low_cnt - rise_cnt + fall_cnt - 1;
if (hcnt < 0 || lcnt < 0)
{
FI2C_ERROR("i2c: bad counts. hcnt = %d lcnt = %d\n", hcnt, lcnt);
return FI2C_ERR_INVAL_PARM;
}
/*
* Now add things back up to ensure the period is hit. If it is off,
* split the difference and bias to lcnt for remainder
*/
tot = hcnt + lcnt + 7 + spk_cnt + rise_cnt + 1;
if (tot < period_cnt)
{
diff = (period_cnt - tot) / 2;
hcnt += diff;
lcnt += diff;
tot = hcnt + lcnt + 7 + spk_cnt + rise_cnt + 1;
lcnt += period_cnt - tot;
}
speed_cfg_p->scl_lcnt = lcnt;
speed_cfg_p->scl_hcnt = hcnt;
speed_cfg_p->sda_hold = DIV_ROUND_UP(bus_clk_hz / 1000 * 300, NANO_TO_KILO); /* Use internal default unless other value is specified */
FI2C_INFO("i2c: hcnt = %d lcnt = %d sda hold = %d\n",
speed_cfg_p->scl_hcnt,
speed_cfg_p->scl_lcnt,
speed_cfg_p->sda_hold);
return FI2C_SUCCESS;
}
/**
* @name: FI2cCalcSpeedCfg
* @msg: I2C的速度配置
* @return {*}
* @param {uintptr} addr, I2C控制器基地址
* @param {u32} speed, I2C传输速率
* @param {u32} bus_clk_hz, I2C时钟频率
* @param {FI2cSpeedCfg} *speed_cfg_p, I2C速度配置
*/
static FError FI2cCalcSpeedCfg(uintptr addr, u32 speed, u32 bus_clk_hz, FI2cSpeedCfg *speed_cfg_p)
{
FASSERT(speed_cfg_p);
u32 spk_cnt = 0;
if (FI2C_SPEED_FAST_RATE <= speed)
{
speed_cfg_p->speed_mode = FI2C_FAST_SPEED;
}
else if (FI2C_SPEED_STANDARD_RATE <= speed)
{
speed_cfg_p->speed_mode = FI2C_STANDARD_SPEED;
}
else
{
return FI2C_ERR_INVAL_PARM;
}
spk_cnt = FI2C_READ_REG32(addr, FI2C_FS_SPKLEN_OFFSET);
return FI2cCalcTiming(bus_clk_hz, spk_cnt, speed_cfg_p);
}
/**
* @name: FI2cSetSpeed
* @msg: I2C控制器的速率
* @return {*}
* @param {uintptr} addr, I2C控制器基地址
* @param {u32} speed_rate, I2C传输速率
*/
FError FI2cSetSpeed(uintptr addr, u32 speed_rate)
{
FError ret = FI2C_SUCCESS;
FI2cSpeedCfg speed_cfg;
u32 enable_status;
u32 reg_val;
memset(&speed_cfg, 0, sizeof(speed_cfg));
ret = FI2cCalcSpeedCfg(addr, speed_rate, FI2C_CLK_FREQ_HZ, &speed_cfg);
if (FI2C_SUCCESS != ret)
{
return ret;
}
/* get enable setting for restore later */
enable_status = FI2cGetEnable(addr);
/* reset speed mode bits */
reg_val = ((FI2C_READ_REG32(addr, FI2C_CON_OFFSET)) & (~FI2C_CON_SPEED_MASK));
switch (speed_cfg.speed_mode)
{
case FI2C_STANDARD_SPEED:
reg_val |= FI2C_CON_STD_SPEED;
FI2C_WRITE_REG32(addr, FI2C_SS_SCL_HCNT_OFFSET, speed_cfg.scl_hcnt);
FI2C_WRITE_REG32(addr, FI2C_SS_SCL_LCNT_OFFSET, speed_cfg.scl_lcnt);
break;
case FI2C_FAST_SPEED:
reg_val |= FI2C_CON_FAST_SPEED;
FI2C_WRITE_REG32(addr, FI2C_FS_SCL_HCNT_OFFSET, speed_cfg.scl_hcnt);
FI2C_WRITE_REG32(addr, FI2C_FS_SCL_LCNT_OFFSET, speed_cfg.scl_lcnt);
break;
default:
ret |= FI2C_ERR_INVAL_PARM;
break;
}
FI2C_WRITE_REG32(addr, FI2C_CON_OFFSET, reg_val);
/* Configure SDA Hold Time if required */
if (0 != speed_cfg.sda_hold)
{
FI2C_WRITE_REG32(addr, FI2C_SDA_HOLD_OFFSET, speed_cfg.sda_hold);
}
/* Restore back i2c now speed set */
if (FI2C_IC_ENABLE == enable_status)
{
ret |= FI2cSetEnable(addr, TRUE);
}
return ret;
}
/**
* @name: FI2cWaitStatus
* @msg: I2C状态位直到状态不存在或者超时
* @return {*}
* @param {uintptr} addr, I2C控制器基地址
* @param {u32} stat_bit, I2C状态位
*/
FError FI2cWaitStatus(uintptr addr, u32 stat_bit)
{
u32 timeout = 0;
/* wait until statbit was set or timeout */
do
{
fsleep_millisec(2); /*wait 2 ms*/
}
while (!((FI2C_READ_REG32(addr, FI2C_STATUS_OFFSET)) & stat_bit) && (FI2C_TIMEOUT > ++timeout));
if (FI2C_TIMEOUT <= timeout)
{
FI2C_ERROR("Timeout when wait status: 0x%x.", stat_bit);
return FI2C_ERR_TIMEOUT;
}
return FI2C_SUCCESS;
}
/**
* @name: FI2cWaitBusBusy
* @msg: I2C总线忙
* @return {*}
* @param {uintptr} addr, I2C控制器基地址
*/
FError FI2cWaitBusBusy(uintptr addr)
{
u32 timeout = FI2C_TIMEOUT;
u32 ret = FI2C_SUCCESS;
if (((FI2C_READ_REG32(addr, FI2C_STATUS_OFFSET)) & FI2C_STATUS_MST_ACTIVITY) &&
(FI2C_SUCCESS != FI2cWaitStatus(addr, FI2C_STATUS_TFE)))
{
ret = FI2C_ERR_TIMEOUT;
FI2C_ERROR("Timeout when wait i2c bus busy.");
}
return ret;
}
/**
* @name: FI2cSetTar
* @msg: I2C主机通信的从机地址
* @return {*}
* @param {uintptr} addr, I2C控制器基地址
* @param {u32} tar_addr, I2C从机地址
*/
FError FI2cSetTar(uintptr addr, u32 tar_addr)
{
u32 enable_status = FI2cGetEnable(addr);
u32 ret = FI2C_SUCCESS;
u32 reg_val = 0;
if (FI2C_IC_ENABLE == enable_status)
{
ret = FI2cSetEnable(addr, FALSE);
}
if (FI2C_SUCCESS == ret)
{
FI2C_WRITE_REG32(addr, FI2C_TAR_OFFSET, (tar_addr & FI2C_IC_TAR_MASK));
}
if (FI2C_IC_ENABLE == enable_status)
{
ret = FI2cSetEnable(addr, TRUE);
}
return ret;
}
/**
* @name: FI2cSetSar
* @msg: I2C地址
* @return {*}
* @param {uintptr} addr, I2C控制器基地址
* @param {u32} sar_addr,
*/
FError FI2cSetSar(uintptr addr, u32 sar_addr)
{
u32 enable_status = FI2cGetEnable(addr);
u32 ret = FI2C_SUCCESS;
u32 reg_val = 0;
if (FI2C_IC_ENABLE == enable_status)
{
ret = FI2cSetEnable(addr, FALSE);
}
if (FI2C_SUCCESS == ret)
{
FI2C_WRITE_REG32(addr, FI2C_SAR_OFFSET, (sar_addr & FI2C_IC_SAR_MASK));
}
if (FI2C_IC_ENABLE == enable_status)
{
ret = FI2cSetEnable(addr, TRUE);
}
return ret;
}
/**
* @name: FI2cFlushRxFifo
* @msg: Fifo传输完成
* @return {*}
* @param {uintptr} addr, I2C控制器基地址
*/
FError FI2cFlushRxFifo(uintptr addr)
{
u8 data;
int timeout = 0;
FError ret = FI2C_SUCCESS;
/* read data to trigger trans until fifo empty */
while (FI2C_GET_STATUS(addr) & FI2C_STATUS_RFNE)
{
data = FI2C_READ_DATA(addr);
if (FI2C_TIMEOUT < ++timeout)
{
ret = FI2C_TIMEOUT;
FI2C_ERROR("Timeout when flush fifo.");
break;
}
}
return ret;
}
/**
* @name: FI2cClearIntrBits
* @msg:
* @return {*}
* @param {uintptr} addr, I2C控制器基地址
* @param {u32} *last_err_p, Abort错误
*/
u32 FI2cClearIntrBits(uintptr addr, u32 *last_err_p)
{
FASSERT(last_err_p);
const u32 stat = FI2C_READ_INTR_STAT(addr);
/* read to clr intrrupt status bits */
if (stat & FI2C_INTR_TX_ABRT)
{
*last_err_p = FI2C_READ_REG32(addr, FI2C_TX_ABRT_SOURCE_OFFSET); /* read out abort sources */
FI2C_READ_REG32(addr, FI2C_CLR_TX_ABRT_OFFSET);
}
if (stat & FI2C_INTR_RX_UNDER)
{
FI2C_READ_REG32(addr, FI2C_CLR_RX_UNDER_OFFSET);
}
if (stat & FI2C_INTR_RX_OVER)
{
FI2C_READ_REG32(addr, FI2C_CLR_RX_OVER_OFFSET);
}
if (stat & FI2C_INTR_TX_OVER)
{
FI2C_READ_REG32(addr, FI2C_CLR_TX_OVER_OFFSET);
}
if (stat & FI2C_INTR_RX_DONE)
{
FI2C_READ_REG32(addr, FI2C_CLR_RX_DONE_OFFSET);
}
if (stat & FI2C_INTR_ACTIVITY)
{
FI2C_READ_REG32(addr, FI2C_CLR_ACTIVITY_OFFSET);
}
if (stat & FI2C_INTR_STOP_DET)
{
FI2C_READ_REG32(addr, FI2C_CLR_STOP_DET_OFFSET);
}
if (stat & FI2C_INTR_START_DET)
{
FI2C_READ_REG32(addr, FI2C_CLR_START_DET_OFFSET);
}
if (stat & FI2C_INTR_GEN_CALL)
{
FI2C_READ_REG32(addr, FI2C_CLR_GEN_CALL_OFFSET);
}
return stat;
}