|
|
#include "SWM221.h"
|
|
|
#include "IOI2C_SLV.h"
|
|
|
|
|
|
#define SLV_ADDR 0x6C
|
|
|
|
|
|
char mst_txbuff[4] = {0x37, 0x55, 0xAA, 0x78};
|
|
|
char mst_rxbuff[4] = {0};
|
|
|
|
|
|
volatile uint32_t mst_txindx = 0;
|
|
|
volatile uint32_t mst_rxindx = 0;
|
|
|
|
|
|
#define MST_OP_TX 1
|
|
|
#define MST_OP_RX 2
|
|
|
volatile uint32_t mst_oper = 0;
|
|
|
|
|
|
#define MST_RES_SUCC 1
|
|
|
#define MST_RES_FAIL 2
|
|
|
volatile uint32_t mst_result = 0;
|
|
|
|
|
|
void SerialInit(void);
|
|
|
void I2C_Mst_Init(void);
|
|
|
|
|
|
int main(void)
|
|
|
{
|
|
|
uint32_t i;
|
|
|
|
|
|
SystemInit();
|
|
|
|
|
|
SerialInit();
|
|
|
|
|
|
IOI2C_SLV_Init();
|
|
|
|
|
|
I2C_Mst_Init();
|
|
|
|
|
|
NVIC_SetPriority(I2C0_IRQn, 3); // 设置从机优先级比主机高,因为若从机不能及时处理数据会响应NAK
|
|
|
|
|
|
while(1==1)
|
|
|
{
|
|
|
/*************************** Master Write ************************************/
|
|
|
mst_oper = MST_OP_TX;
|
|
|
mst_result = 0;
|
|
|
mst_txindx = 0;
|
|
|
|
|
|
I2C_Start(I2C0, (SLV_ADDR << 1) | 0, 0);
|
|
|
|
|
|
while(mst_result != MST_RES_SUCC)
|
|
|
{
|
|
|
if(mst_result == MST_RES_FAIL)
|
|
|
goto nextloop;
|
|
|
}
|
|
|
|
|
|
printf("Master Send %X %X %X %X\r\n", mst_txbuff[0], mst_txbuff[1], mst_txbuff[2], mst_txbuff[3]);
|
|
|
|
|
|
/********************************** Master Read *******************************/
|
|
|
mst_oper = MST_OP_RX;
|
|
|
mst_result = 0;
|
|
|
mst_rxindx = 0;
|
|
|
|
|
|
I2C_Start(I2C0, (SLV_ADDR << 1) | 1, 0);
|
|
|
|
|
|
while(mst_result != MST_RES_SUCC)
|
|
|
{
|
|
|
if(mst_result == MST_RES_FAIL)
|
|
|
goto nextloop;
|
|
|
}
|
|
|
|
|
|
printf("Master Read %X %X %X %X\r\n", mst_rxbuff[0], mst_rxbuff[1], mst_rxbuff[2], mst_rxbuff[3]);
|
|
|
|
|
|
if((mst_txbuff[0] == mst_rxbuff[0]) && (mst_txbuff[1] == mst_rxbuff[1]) && (mst_txbuff[2] == mst_rxbuff[2]) && (mst_txbuff[3] == mst_rxbuff[3]))
|
|
|
printf("Success\r\n");
|
|
|
else
|
|
|
printf("Fail\r\n");
|
|
|
|
|
|
nextloop:
|
|
|
I2C_Stop(I2C0, 1);
|
|
|
for(i = 0; i < SystemCoreClock/3; i++) __NOP();
|
|
|
}
|
|
|
}
|
|
|
|
|
|
|
|
|
void I2C_Mst_Init(void)
|
|
|
{
|
|
|
I2C_InitStructure I2C_initStruct;
|
|
|
|
|
|
PORT_Init(PORTA, PIN6, PORTA_PIN6_I2C0_SCL, 1); //GPIOA.6配置为I2C0 SCL引脚
|
|
|
PORTA->OPEND |= (1 << PIN6); //开漏输出
|
|
|
PORTA->PULLU |= (1 << PIN6); //使能上拉
|
|
|
PORT_Init(PORTA, PIN7, PORTA_PIN7_I2C0_SDA, 1); //GPIOA.7配置为I2C0 SDA引脚
|
|
|
PORTA->OPEND |= (1 << PIN7); //开漏输出
|
|
|
PORTA->PULLU |= (1 << PIN7); //使能上拉
|
|
|
|
|
|
I2C_initStruct.Master = 1;
|
|
|
I2C_initStruct.MstClk = 100000;
|
|
|
I2C_initStruct.Addr10b = 0;
|
|
|
I2C_initStruct.TXEmptyIEn = 0;
|
|
|
I2C_initStruct.RXNotEmptyIEn = 0;
|
|
|
I2C_Init(I2C0, &I2C_initStruct);
|
|
|
|
|
|
I2C_INTEn(I2C0, I2C_IT_TX_DONE);
|
|
|
/* 注意:此例程中不能使用 RX_NOT_EMPTY 中断替代 RX_DONE 中断,
|
|
|
因为当 RX_NOT_EMPTY 发生时,接收还未完成,I2C0->MCR.RD 还没清零,
|
|
|
这时候进入中断执行 I2C_Read() 设置 I2C0->MCR.RD 位没有作用,导致无法接收下一个数据
|
|
|
*/
|
|
|
I2C_INTEn(I2C0, I2C_IT_RX_DONE);
|
|
|
NVIC_EnableIRQ(I2C0_IRQn);
|
|
|
|
|
|
I2C_Open(I2C0);
|
|
|
}
|
|
|
|
|
|
void I2C0_Handler(void)
|
|
|
{
|
|
|
if(I2C0->IF & I2C_IF_TXDONE_Msk) //发送完成
|
|
|
{
|
|
|
I2C0->IF = I2C_IF_TXDONE_Msk;
|
|
|
|
|
|
if(mst_oper == MST_OP_TX)
|
|
|
{
|
|
|
if(I2C_IsAck(I2C0))
|
|
|
{
|
|
|
if(mst_txindx < 4)
|
|
|
{
|
|
|
I2C_Write(I2C0, mst_txbuff[mst_txindx], 0);
|
|
|
|
|
|
mst_txindx++;
|
|
|
}
|
|
|
else
|
|
|
{
|
|
|
I2C_Stop(I2C0, 0);
|
|
|
|
|
|
mst_result = MST_RES_SUCC;
|
|
|
}
|
|
|
}
|
|
|
else
|
|
|
{
|
|
|
if(mst_txindx == 0)
|
|
|
printf("Slave send NACK for address\r\n");
|
|
|
else
|
|
|
printf("Slave send NACK for data\r\n");
|
|
|
|
|
|
mst_result = MST_RES_FAIL;
|
|
|
}
|
|
|
}
|
|
|
else // mst_oper == MST_OP_RX
|
|
|
{
|
|
|
if(I2C_IsAck(I2C0))
|
|
|
{
|
|
|
I2C_Read(I2C0, 1, 0);
|
|
|
}
|
|
|
else
|
|
|
{
|
|
|
printf("Slave send NACK for address\r\n");
|
|
|
|
|
|
mst_result = MST_RES_FAIL;
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
else if(I2C0->IF & I2C_IF_RXDONE_Msk) //接收完成
|
|
|
{
|
|
|
I2C0->IF = I2C_IF_RXDONE_Msk;
|
|
|
|
|
|
if(mst_rxindx < 4)
|
|
|
{
|
|
|
mst_rxbuff[mst_rxindx] = I2C0->RXDATA;
|
|
|
|
|
|
mst_rxindx++;
|
|
|
|
|
|
if(mst_rxindx < 3)
|
|
|
{
|
|
|
I2C_Read(I2C0, 1, 0);
|
|
|
}
|
|
|
else if(mst_rxindx < 4)
|
|
|
{
|
|
|
I2C_Read(I2C0, 0, 0);
|
|
|
}
|
|
|
else
|
|
|
{
|
|
|
I2C_Stop(I2C0, 0);
|
|
|
|
|
|
mst_result = MST_RES_SUCC;
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
|
|
|
|
|
|
uint32_t IOI2C_SLV_EmptyCallback(char *TxBuff)
|
|
|
{
|
|
|
uint32_t i = 0;
|
|
|
|
|
|
while(IOI2C_SLV_RxNotEmpty())
|
|
|
TxBuff[i++] = IOI2C_SLV_FetchByte();
|
|
|
|
|
|
return i;
|
|
|
}
|
|
|
|
|
|
|
|
|
void SerialInit(void)
|
|
|
{
|
|
|
UART_InitStructure UART_initStruct;
|
|
|
|
|
|
PORT_Init(PORTA, PIN0, PORTA_PIN0_UART0_RX, 1); //GPIOA.0配置为UART0 RXD
|
|
|
PORT_Init(PORTA, PIN1, PORTA_PIN1_UART0_TX, 0); //GPIOA.1配置为UART0 TXD
|
|
|
|
|
|
UART_initStruct.Baudrate = 57600;
|
|
|
UART_initStruct.DataBits = UART_DATA_8BIT;
|
|
|
UART_initStruct.Parity = UART_PARITY_NONE;
|
|
|
UART_initStruct.StopBits = UART_STOP_1BIT;
|
|
|
UART_initStruct.RXThreshold = 3;
|
|
|
UART_initStruct.RXThresholdIEn = 0;
|
|
|
UART_initStruct.TXThreshold = 3;
|
|
|
UART_initStruct.TXThresholdIEn = 0;
|
|
|
UART_initStruct.TimeoutTime = 10;
|
|
|
UART_initStruct.TimeoutIEn = 0;
|
|
|
UART_Init(UART0, &UART_initStruct);
|
|
|
UART_Open(UART0);
|
|
|
}
|
|
|
|
|
|
/******************************************************************************************************************************************
|
|
|
* 函数名称: fputc()
|
|
|
* 功能说明: printf()使用此函数完成实际的串口打印动作
|
|
|
* 输 入: int ch 要打印的字符
|
|
|
* FILE *f 文件句柄
|
|
|
* 输 出: 无
|
|
|
* 注意事项: 无
|
|
|
******************************************************************************************************************************************/
|
|
|
int fputc(int ch, FILE *f)
|
|
|
{
|
|
|
UART_WriteByte(UART0, ch);
|
|
|
|
|
|
while(UART_IsTXBusy(UART0));
|
|
|
|
|
|
return ch;
|
|
|
}
|