667 lines
18 KiB
C
667 lines
18 KiB
C
/*************************************************************************************
|
||
* Copyright (C) 2017, Huada Semiconductor Co.,Ltd All rights reserved.
|
||
*
|
||
* This software is owned and published by:
|
||
* Huada Semiconductor Co.,Ltd ("HDSC").
|
||
*
|
||
* BY DOWNLOADING, INSTALLING OR USING THIS SOFTWARE, YOU AGREE TO BE BOUND
|
||
* BY ALL THE TERMS AND CONDITIONS OF THIS AGREEMENT.
|
||
*
|
||
* This software contains source code for use with HDSC
|
||
* components. This software is licensed by HDSC to be adapted only
|
||
* for use in systems utilizing HDSC components. HDSC shall not be
|
||
* responsible for misuse or illegal use of this software for devices not
|
||
* supported herein. HDSC is providing this software "AS IS" and will
|
||
* not be responsible for issues arising from incorrect user implementation
|
||
* of the software.
|
||
*
|
||
* Disclaimer:
|
||
* HDSC MAKES NO WARRANTY, EXPRESS OR IMPLIED, ARISING BY LAW OR OTHERWISE,
|
||
* REGARDING THE SOFTWARE (INCLUDING ANY ACOOMPANYING WRITTEN MATERIALS),
|
||
* ITS PERFORMANCE OR SUITABILITY FOR YOUR INTENDED USE, INCLUDING,
|
||
* WITHOUT LIMITATION, THE IMPLIED WARRANTY OF MERCHANTABILITY, THE IMPLIED
|
||
* WARRANTY OF FITNESS FOR A PARTICULAR PURPOSE OR USE, AND THE IMPLIED
|
||
* WARRANTY OF NONINFRINGEMENT.
|
||
* HDSC SHALL HAVE NO LIABILITY (WHETHER IN CONTRACT, WARRANTY, TORT,
|
||
* NEGLIGENCE OR OTHERWISE) FOR ANY DAMAGES WHATSOEVER (INCLUDING, WITHOUT
|
||
* LIMITATION, DAMAGES FOR LOSS OF BUSINESS PROFITS, BUSINESS INTERRUPTION,
|
||
* LOSS OF BUSINESS INFORMATION, OR OTHER PECUNIARY LOSS) ARISING FROM USE OR
|
||
* INABILITY TO USE THE SOFTWARE, INCLUDING, WITHOUT LIMITATION, ANY DIRECT,
|
||
* INDIRECT, INCIDENTAL, SPECIAL OR CONSEQUENTIAL DAMAGES OR LOSS OF DATA,
|
||
* SAVINGS OR PROFITS,
|
||
* EVEN IF Disclaimer HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH DAMAGES.
|
||
* YOU ASSUME ALL RESPONSIBILITIES FOR SELECTION OF THE SOFTWARE TO ACHIEVE YOUR
|
||
* INTENDED RESULTS, AND FOR THE INSTALLATION OF, USE OF, AND RESULTS OBTAINED
|
||
* FROM, THE SOFTWARE.
|
||
*
|
||
* This software may be replicated in part or whole for the licensed use,
|
||
* with the restriction that this Disclaimer and Copyright notice must be
|
||
* included with each copy of this software, whether used in part or whole,
|
||
* at all times.
|
||
*/
|
||
/******************************************************************************/
|
||
/** \file I2C.c
|
||
**
|
||
** WDT function driver API.
|
||
** @link SampleGroup Some description @endlink
|
||
**
|
||
** - 2018-03-13 1.0 CJ First version for Device Driver Library of Module.
|
||
**
|
||
******************************************************************************/
|
||
|
||
/******************************************************************************/
|
||
/* Include files */
|
||
/******************************************************************************/
|
||
#include "i2c.h"
|
||
|
||
/**
|
||
*******************************************************************************
|
||
** \addtogroup I2cGroup
|
||
******************************************************************************/
|
||
//@{
|
||
|
||
/******************************************************************************/
|
||
/* Local function prototypes ('static') */
|
||
/******************************************************************************/
|
||
static func_ptr_t pfnI2c0tCallback = NULL;
|
||
static func_ptr_t pfnI2c1tCallback = NULL;
|
||
/**
|
||
******************************************************************************
|
||
** \brief I2C设置波特率配置寄存器
|
||
**
|
||
** \param [in] u8Tm 波特率配置值
|
||
**
|
||
** \retval enRet 成功或失败
|
||
**
|
||
******************************************************************************/
|
||
en_result_t I2C_SetBaud(en_i2c_channel_t enCh,uint8_t u8Tm)
|
||
{
|
||
en_result_t enRet = Error;
|
||
if(I2C0 == enCh)
|
||
{
|
||
M0P_I2C0->TM = u8Tm;
|
||
}
|
||
else
|
||
{
|
||
M0P_I2C1->TM = u8Tm;
|
||
}
|
||
|
||
enRet = Ok;
|
||
return enRet;
|
||
}
|
||
/**
|
||
******************************************************************************
|
||
** \brief I2C功能设置相关函数
|
||
**
|
||
** \param [in] enFunc功能参数
|
||
**
|
||
** \retval enRet 成功或失败
|
||
**
|
||
******************************************************************************/
|
||
en_result_t I2C_SetFunc(en_i2c_channel_t enCh,en_i2c_func_t enFunc)
|
||
{
|
||
en_result_t enRet = Error;
|
||
if(I2C0 == enCh)
|
||
{
|
||
switch(enFunc)
|
||
{
|
||
case I2cMode_En:
|
||
M0P_I2C0->CR_f.ENS = 1;
|
||
break;
|
||
case I2cStart_En:
|
||
M0P_I2C0->CR_f.STA = 1;
|
||
break;
|
||
case I2cStop_En:
|
||
M0P_I2C0->CR_f.STO = 1;
|
||
break;
|
||
case I2cAck_En:
|
||
M0P_I2C0->CR_f.AA = 1;
|
||
break;
|
||
case I2cHlm_En:
|
||
M0P_I2C0->CR_f.H1M = 1;
|
||
break;
|
||
case I2cBaud_En:
|
||
M0P_I2C0->TMRUN = 0x01;
|
||
break;
|
||
default:
|
||
return ErrorInvalidParameter;
|
||
}
|
||
}
|
||
else
|
||
{
|
||
switch(enFunc)
|
||
{
|
||
case I2cMode_En:
|
||
M0P_I2C1->CR_f.ENS = 1;
|
||
break;
|
||
case I2cStart_En:
|
||
M0P_I2C1->CR_f.STA = 1;
|
||
break;
|
||
case I2cStop_En:
|
||
M0P_I2C1->CR_f.STO = 1;
|
||
break;
|
||
case I2cAck_En:
|
||
M0P_I2C1->CR_f.AA = 1;
|
||
break;
|
||
case I2cHlm_En:
|
||
M0P_I2C1->CR_f.H1M = 1;
|
||
break;
|
||
case I2cBaud_En:
|
||
M0P_I2C1->TMRUN = 0x01;
|
||
break;
|
||
default:
|
||
return ErrorInvalidParameter;
|
||
}
|
||
}
|
||
|
||
enRet = Ok;
|
||
return enRet;
|
||
}
|
||
/**
|
||
******************************************************************************
|
||
** \brief I2C功能清除相关函数
|
||
**
|
||
** \param [in] enFunc功能参数
|
||
**
|
||
** \retval enRet 成功或失败
|
||
**
|
||
******************************************************************************/
|
||
en_result_t I2C_ClearFunc(en_i2c_channel_t enCh,en_i2c_func_t enFunc)
|
||
{
|
||
en_result_t enRet = Error;
|
||
if(I2C0 == enCh)
|
||
{
|
||
switch(enFunc)
|
||
{
|
||
case I2cMode_En:
|
||
M0P_I2C0->CR_f.ENS = 0;
|
||
break;
|
||
case I2cStart_En:
|
||
M0P_I2C0->CR_f.STA = 0;
|
||
break;
|
||
case I2cStop_En:
|
||
M0P_I2C0->CR_f.STO = 0;
|
||
break;
|
||
case I2cAck_En:
|
||
M0P_I2C0->CR_f.AA = 0;
|
||
break;
|
||
case I2cHlm_En:
|
||
M0P_I2C0->CR_f.H1M = 0;
|
||
break;
|
||
case I2cBaud_En:
|
||
M0P_I2C0->TMRUN = 0x00;
|
||
break;
|
||
default:
|
||
return ErrorInvalidParameter;
|
||
}
|
||
}
|
||
else
|
||
{
|
||
switch(enFunc)
|
||
{
|
||
case I2cMode_En:
|
||
M0P_I2C1->CR_f.ENS = 0;
|
||
break;
|
||
case I2cStart_En:
|
||
M0P_I2C1->CR_f.STA = 0;
|
||
break;
|
||
case I2cStop_En:
|
||
M0P_I2C1->CR_f.STO = 0;
|
||
break;
|
||
case I2cAck_En:
|
||
M0P_I2C1->CR_f.AA = 0;
|
||
break;
|
||
case I2cHlm_En:
|
||
M0P_I2C1->CR_f.H1M = 0;
|
||
break;
|
||
case I2cBaud_En:
|
||
M0P_I2C1->TMRUN = 0x00;
|
||
break;
|
||
default:
|
||
return ErrorInvalidParameter;
|
||
}
|
||
}
|
||
enRet = Ok;
|
||
return enRet;
|
||
}
|
||
/**
|
||
******************************************************************************
|
||
** \brief I2C获取中断标记函数
|
||
**
|
||
** \param 无
|
||
**
|
||
** \retval bIrq中断标记
|
||
**
|
||
******************************************************************************/
|
||
boolean_t I2C_GetIrq(en_i2c_channel_t enCh)
|
||
{
|
||
boolean_t bIrq = FALSE;
|
||
if(I2C0 == enCh)
|
||
{
|
||
bIrq = M0P_I2C0->CR_f.SI;
|
||
}
|
||
else
|
||
{
|
||
bIrq = M0P_I2C1->CR_f.SI;
|
||
}
|
||
|
||
return bIrq;
|
||
}
|
||
/**
|
||
******************************************************************************
|
||
** \brief I2C清除中断标记函数
|
||
**
|
||
** \param 无
|
||
**
|
||
** \retval bIrq中断标记
|
||
**
|
||
******************************************************************************/
|
||
en_result_t I2C_ClearIrq(en_i2c_channel_t enCh)
|
||
{
|
||
en_result_t enRet = Error;
|
||
if(I2C0 == enCh)
|
||
{
|
||
M0P_I2C0->CR_f.SI = 0;
|
||
}
|
||
else
|
||
{
|
||
M0P_I2C1->CR_f.SI = 0;
|
||
}
|
||
enRet = Ok;
|
||
return enRet;
|
||
}
|
||
/**
|
||
******************************************************************************
|
||
** \brief I2C获取相关状态
|
||
**
|
||
** \param 无
|
||
**
|
||
** \retval I2C状态
|
||
**
|
||
******************************************************************************/
|
||
uint8_t I2C_GetState(en_i2c_channel_t enCh)
|
||
{
|
||
uint8_t u8State = 0;
|
||
if(I2C0 == enCh)
|
||
{
|
||
u8State = M0P_I2C0->STAT;
|
||
}
|
||
else
|
||
{
|
||
u8State = M0P_I2C1->STAT;
|
||
}
|
||
return u8State;
|
||
}
|
||
/**
|
||
******************************************************************************
|
||
** \brief I2C写从机地址函数
|
||
**
|
||
** \param u8SlaveAddr从机地址
|
||
**
|
||
** \retval I2C写成功与否状态
|
||
**
|
||
******************************************************************************/
|
||
en_result_t I2C_WriteSlaveAddr(en_i2c_channel_t enCh,stc_i2c_addr_t *pstcSlaveAddr)
|
||
{
|
||
en_result_t enRet = Error;
|
||
if(I2C0 == enCh)
|
||
{
|
||
M0P_I2C0->ADDR_f.ADR = pstcSlaveAddr->Addr;
|
||
M0P_I2C0->ADDR_f.GC = pstcSlaveAddr->Gc;
|
||
}
|
||
else
|
||
{
|
||
M0P_I2C1->ADDR_f.ADR = pstcSlaveAddr->Addr;
|
||
M0P_I2C1->ADDR_f.GC = pstcSlaveAddr->Gc;
|
||
}
|
||
|
||
enRet = Ok;
|
||
return enRet;
|
||
}
|
||
/**
|
||
******************************************************************************
|
||
** \brief 字节写从机函数
|
||
**
|
||
** \param u8Data写数据
|
||
**
|
||
** \retval 写数据是否成功
|
||
**
|
||
******************************************************************************/
|
||
en_result_t I2C_WriteByte(en_i2c_channel_t enCh,uint8_t u8Data)
|
||
{
|
||
en_result_t enRet = Error;
|
||
if(I2C0 == enCh)
|
||
{
|
||
M0P_I2C0->DATA = u8Data;
|
||
}
|
||
else
|
||
{
|
||
M0P_I2C1->DATA = u8Data;
|
||
}
|
||
enRet = Ok;
|
||
return enRet;
|
||
}
|
||
/**
|
||
******************************************************************************
|
||
** \brief 字节读从机函数
|
||
**
|
||
** \param 无
|
||
**
|
||
** \retval 读取数据
|
||
**
|
||
******************************************************************************/
|
||
uint8_t I2C_ReadByte(en_i2c_channel_t enCh)
|
||
{
|
||
uint8_t u8Data = 0;
|
||
if(I2C0 == enCh)
|
||
{
|
||
u8Data = M0P_I2C0->DATA;
|
||
}
|
||
else
|
||
{
|
||
u8Data = M0P_I2C1->DATA;
|
||
}
|
||
return u8Data;
|
||
}
|
||
/**
|
||
******************************************************************************
|
||
** \brief 主机发送函数
|
||
**
|
||
** \param u8Addr从机内存地址,pu8Data写数据,u32Len写数据长度
|
||
**
|
||
** \retval 写数据是否成功
|
||
**
|
||
******************************************************************************/
|
||
en_result_t I2C_MasterWriteData(en_i2c_channel_t enCh,uint8_t u8DevAddr,uint8_t u8Addr,uint8_t *pu8Data,uint32_t u32Len)
|
||
{
|
||
en_result_t enRet = Error;
|
||
uint8_t u8i=0,u8State;
|
||
|
||
I2C_SetFunc(enCh,I2cStart_En);
|
||
while(1)
|
||
{
|
||
while(0 == I2C_GetIrq(enCh))
|
||
{}
|
||
u8State = I2C_GetState(enCh);
|
||
switch(u8State)
|
||
{
|
||
case 0x08:
|
||
I2C_ClearFunc(enCh,I2cStart_En);
|
||
I2C_WriteByte(enCh,u8DevAddr);//从设备地址发送
|
||
break;
|
||
case 0x18:
|
||
I2C_WriteByte(enCh,u8Addr);//从设备内存地址发送
|
||
break;
|
||
case 0x28:
|
||
I2C_WriteByte(enCh,pu8Data[u8i++]);
|
||
break;
|
||
case 0x20:
|
||
case 0x38:
|
||
I2C_SetFunc(enCh,I2cStart_En);
|
||
break;
|
||
case 0x30:
|
||
I2C_SetFunc(enCh,I2cStop_En);
|
||
break;
|
||
default:
|
||
break;
|
||
}
|
||
if(u8i>u32Len)
|
||
{
|
||
I2C_SetFunc(enCh,I2cStop_En);//此顺序不能调换,出停止条件
|
||
I2C_ClearIrq(enCh);
|
||
break;
|
||
}
|
||
I2C_ClearIrq(enCh);
|
||
}
|
||
enRet = Ok;
|
||
return enRet;
|
||
}
|
||
/**
|
||
******************************************************************************
|
||
** \brief 从机发送函数
|
||
**
|
||
** \param pu8Data发送数据缓存,u32Len发送数据长度
|
||
**
|
||
** \retval 发送数据是否成功
|
||
**
|
||
******************************************************************************/
|
||
en_result_t I2C_SlaveWriteData(en_i2c_channel_t enCh,uint8_t *pu8Data,uint32_t *u32Len)
|
||
{
|
||
uint8_t u8i=0,u8State;
|
||
//
|
||
while(1)
|
||
{
|
||
|
||
while(0 == I2C_GetIrq(enCh))
|
||
{}
|
||
u8State = I2C_GetState(enCh);
|
||
switch(u8State)
|
||
{
|
||
case 0xA8:
|
||
case 0xB0:
|
||
I2C_WriteByte(enCh,pu8Data[u8i++]);
|
||
break;
|
||
case 0xB8:
|
||
case 0xC8:
|
||
I2C_WriteByte(enCh,pu8Data[u8i++]);
|
||
break;
|
||
case 0xF8:
|
||
*u32Len = u8i;
|
||
break;
|
||
default:
|
||
|
||
return ErrorInvalidParameter;
|
||
}
|
||
I2C_ClearIrq(enCh);
|
||
}
|
||
}
|
||
/**
|
||
******************************************************************************
|
||
** \brief 从机接收函数
|
||
**
|
||
** \param pu8Data接收数据存放缓存,u32Len接收数据指针
|
||
**
|
||
** \retval 接收数据是否成功
|
||
**
|
||
******************************************************************************/
|
||
en_result_t I2C_SlaveReadData(en_i2c_channel_t enCh,uint8_t *pu8Data,uint32_t *pu32Len)
|
||
{
|
||
uint8_t u8i=0,u8State;
|
||
while(0 == I2C_GetIrq(enCh))
|
||
{}
|
||
while(1)
|
||
{
|
||
while(0 == I2C_GetIrq(enCh))
|
||
{}
|
||
u8State = I2C_GetState(enCh);
|
||
switch(u8State)
|
||
{
|
||
case 0x60:
|
||
case 0x68:
|
||
case 0x70:
|
||
case 0x78:
|
||
break;
|
||
case 0x80:
|
||
case 0x90:
|
||
pu8Data[u8i++] = I2C_ReadByte(enCh);
|
||
break;
|
||
case 0xA0:
|
||
*pu32Len = u8i;
|
||
break;
|
||
default:
|
||
return ErrorInvalidParameter;
|
||
}
|
||
I2C_ClearIrq(enCh);
|
||
if(0xA0 == u8State)
|
||
{
|
||
return Ok;
|
||
}
|
||
}
|
||
}
|
||
|
||
/**
|
||
******************************************************************************
|
||
** \brief 主机接收函数
|
||
**
|
||
** \param u8Addr从机内存地址,pu8Data读数据存放缓存,u32Len读数据长度
|
||
**
|
||
** \retval 读数据是否成功
|
||
**
|
||
******************************************************************************/
|
||
en_result_t I2C_MasterReadData(en_i2c_channel_t enCh,uint8_t u8DevAddr,uint8_t *pu8Data,uint8_t u8Addr,uint32_t u32Len)
|
||
{
|
||
en_result_t enRet = Error;
|
||
uint8_t u8i=0,u8State;
|
||
|
||
I2C_SetFunc(enCh,I2cStart_En);
|
||
|
||
while(1)
|
||
{
|
||
while(0 == I2C_GetIrq(enCh))
|
||
{}
|
||
u8State = I2C_GetState(enCh);
|
||
switch(u8State)
|
||
{
|
||
case 0x08:
|
||
I2C_ClearFunc(enCh,I2cStart_En);
|
||
I2C_WriteByte(enCh,u8DevAddr);
|
||
break;
|
||
case 0x18:
|
||
I2C_WriteByte(enCh,u8Addr);
|
||
break;
|
||
case 0x28:
|
||
I2C_SetFunc(enCh,I2cStart_En);
|
||
break;
|
||
case 0x10:
|
||
I2C_ClearFunc(enCh,I2cStart_En);
|
||
I2C_WriteByte(enCh,u8DevAddr|0x01);//从机地址发送OK
|
||
break;
|
||
case 0x40:
|
||
if(u32Len>1)
|
||
{
|
||
I2C_SetFunc(enCh,I2cAck_En);
|
||
}
|
||
break;
|
||
case 0x50:
|
||
pu8Data[u8i++] = I2C_ReadByte(enCh);
|
||
if(u8i==u32Len-1)
|
||
{
|
||
I2C_ClearFunc(enCh,I2cAck_En);
|
||
}
|
||
break;
|
||
case 0x58:
|
||
pu8Data[u8i++] = I2C_ReadByte(enCh);
|
||
I2C_SetFunc(enCh,I2cStop_En);
|
||
break;
|
||
case 0x38:
|
||
I2C_SetFunc(enCh,I2cStart_En);
|
||
break;
|
||
case 0x48:
|
||
I2C_SetFunc(enCh,I2cStop_En);
|
||
I2C_SetFunc(enCh,I2cStart_En);
|
||
break;
|
||
default:
|
||
I2C_SetFunc(enCh,I2cStart_En);//其他错误状态,重新发送起始条件
|
||
break;
|
||
}
|
||
I2C_ClearIrq(enCh);
|
||
if(u8i==u32Len)
|
||
{
|
||
break;
|
||
}
|
||
}
|
||
enRet = Ok;
|
||
return enRet;
|
||
}
|
||
/**
|
||
******************************************************************************
|
||
** \brief I2C模块初始化
|
||
**
|
||
** \param pstcI2CCfg初始化配置结构体
|
||
**
|
||
** \retval 初始化是否成功
|
||
**
|
||
******************************************************************************/
|
||
en_result_t I2C_Init(en_i2c_channel_t enCh,stc_i2c_config_t *pstcI2CCfg)
|
||
{
|
||
en_result_t enRet = Error;
|
||
enRet = I2C_SetFunc(enCh,pstcI2CCfg->enFunc);
|
||
enRet = I2C_SetBaud(enCh,pstcI2CCfg->u8Tm);
|
||
enRet = I2C_WriteSlaveAddr(enCh,&pstcI2CCfg->stcSlaveAddr);
|
||
if(pstcI2CCfg->u8Tm<9)
|
||
{
|
||
I2C_SetFunc(enCh,I2cHlm_En);
|
||
}
|
||
if(NULL!=pstcI2CCfg->pfnI2c0Cb)
|
||
{
|
||
pfnI2c0tCallback = pstcI2CCfg->pfnI2c0Cb;
|
||
}
|
||
if(NULL!=pstcI2CCfg->pfnI2c1Cb)
|
||
{
|
||
pfnI2c1tCallback = pstcI2CCfg->pfnI2c1Cb;
|
||
}
|
||
if(TRUE == pstcI2CCfg->bTouchNvic)
|
||
{
|
||
if(I2C0 == enCh)
|
||
{
|
||
EnableNvic(I2C0_IRQn,IrqLevel3,TRUE);
|
||
}
|
||
else
|
||
{
|
||
EnableNvic(I2C1_IRQn,IrqLevel3,TRUE);
|
||
}
|
||
}
|
||
return enRet;
|
||
}
|
||
/**
|
||
******************************************************************************
|
||
** \brief I2C模块关闭初始化
|
||
**
|
||
** \param 无
|
||
**
|
||
** \retval 设置是否成功
|
||
**
|
||
******************************************************************************/
|
||
en_result_t I2C_DeInit(en_i2c_channel_t enCh)
|
||
{
|
||
en_result_t enRet = Error;
|
||
if(I2C0 == enCh)
|
||
{
|
||
M0P_I2C0->CR = 0x00;
|
||
}
|
||
else
|
||
{
|
||
M0P_I2C1->CR = 0x00;
|
||
}
|
||
enRet = Ok;
|
||
return enRet;
|
||
}
|
||
/**
|
||
******************************************************************************
|
||
** \brief I2C模块中断处理函数
|
||
**
|
||
** \param u8Param 无意义
|
||
**
|
||
** \retval 无
|
||
**
|
||
******************************************************************************/
|
||
void I2c_IRQHandler(uint8_t u8Param)
|
||
{
|
||
if(I2C0 == u8Param)
|
||
{
|
||
if(NULL != pfnI2c0tCallback)
|
||
{
|
||
pfnI2c0tCallback();
|
||
}
|
||
}
|
||
else
|
||
{
|
||
if(NULL != pfnI2c1tCallback)
|
||
{
|
||
pfnI2c1tCallback();
|
||
}
|
||
}
|
||
}
|
||
|
||
//@} // I2cGroup
|