新聞中心

STM23模擬I2C讀寫(xiě)

作者: 時(shí)間:2016-11-25 來(lái)源:網(wǎng)絡(luò) 收藏
都說(shuō)STM32的I2C有問(wèn)題,不好用。我之前,在論壇上看到了Mcuplayer分享的一段代碼,拿來(lái)測(cè)試了一下,讀寫(xiě)正常,心想還挺不錯(cuò)。
但是等到真正做程序時(shí),發(fā)現(xiàn)總是在while()循環(huán)處等待。無(wú)奈,只好用軟件模擬I2C。
從網(wǎng)上找了一段程序,發(fā)現(xiàn)好多地方下載的代碼都一樣的。
代碼如下:
#i nclude "includes.h"
#i nclude "24cxx.h"
#define ADDR_24CXX0xA0
#define SCLHGPIOB->BSRR = GPIO_Pin_6
#define SCLLGPIOB->BRR = GPIO_Pin_6

#define SDAHGPIOB->BSRR = GPIO_Pin_7
#define SDALGPIOB->BRR = GPIO_Pin_7
#define SCLreadGPIOB->IDR & GPIO_Pin_6
#define SDAreadGPIOB->IDR & GPIO_Pin_7
static void I2C_GPIO_Config(void)
{
GPIO_InitTypeDef GPIO_InitStructure;

RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE);

// Configure I2C1 pins: SCL and SDA
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_6;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_OD;
GPIO_Init(GPIOB, &GPIO_InitStructure);
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_7;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_OD;
GPIO_Init(GPIOB, &GPIO_InitStructure);
}
void I2C_delay(void)
{
u8 i=50; //這里可以優(yōu)化速度,經(jīng)測(cè)試最低到5還能寫(xiě)入
while(i)
{
i--;
}
}
bool I2C_Start(void)
{
SDAH;
SCLH;
I2C_delay();
if(!SDAread)return FALSE;//SDA線為低電平則總線忙,退出
SDAL;
I2C_delay();
if(SDAread) return FALSE;//SDA線為高電平則總線出錯(cuò),退出
SDAL;
I2C_delay();
return TRUE;
}
void I2C_Stop(void)
{
SCLL;
I2C_delay();
SDAL;
I2C_delay();
SCLH;
I2C_delay();
SDAH;
I2C_delay();
}
void I2C_Ack(void)
{
SCLL;
I2C_delay();
SDAL;
I2C_delay();
SCLH;
I2C_delay();
SCLL;
I2C_delay();
}
void I2C_NoAck(void)
{
SCLL;
I2C_delay();
SDAH;
I2C_delay();
SCLH;
I2C_delay();
SCLL;
I2C_delay();
}
bool I2C_WaitAck(void) //返回為:=1有ACK,=0無(wú)ACK
{
SCLL;
I2C_delay();
SDAH;
I2C_delay();
SCLH;
I2C_delay();
if(SDAread)
{
SCLL;
return FALSE;
}
SCLL;
return TRUE;
}
void I2C_SendByte(u8 SendByte) //數(shù)據(jù)從高位到低位//
{
u8 i=8;
while(i--)
{
SCLL;
I2C_delay();
if(SendByte&0x80)
SDAH;
else
SDAL;
SendByte<<=1;
I2C_delay();
SCLH;
I2C_delay();
}
SCLL;
}
u8 I2C_ReceiveByte(void)//數(shù)據(jù)從高位到低位//
{
u8 i=8;
u8 ReceiveByte=0;
SDAH;
while(i--)
{
ReceiveByte<<=1;
SCLL;
I2C_delay();
SCLH;
I2C_delay();
if(SDAread)
{
ReceiveByte|=0x01;
}
}
SCLL;
return ReceiveByte;
}
//寫(xiě)入1字節(jié)數(shù)據(jù) 待寫(xiě)入數(shù)據(jù) 待寫(xiě)入地址 器件類(lèi)型(24c16或SD2403)
bool I2C_WriteByte(u8 SendByte, u16 WriteAddress, u8 DeviceAddress)
{
u32 j;
if(!I2C_Start())return FALSE;
//I2C_SendByte(((WriteAddress & 0x0700) >>7) | DeviceAddress & 0xFFFE);//設(shè)置高起始地址+器件地址
I2C_SendByte( DeviceAddress & 0xFE);//寫(xiě)器件地址
if(!I2C_WaitAck()){I2C_Stop(); return FALSE;}
I2C_SendByte((u8)((WriteAddress>>8) & 0xFF)); //設(shè)置高起始地址
I2C_WaitAck();
I2C_SendByte((u8)((WriteAddress) & 0xFF)); //設(shè)置低起始地址
I2C_WaitAck();
I2C_SendByte(SendByte);//寫(xiě)數(shù)據(jù)
I2C_WaitAck();
I2C_Stop();
//注意:因?yàn)檫@里要等待EEPROM寫(xiě)完,可以采用查詢或延時(shí)方式(10ms)
for(j=0;j<1500;j++)
I2C_delay();
return TRUE;
}
//讀出1字節(jié)數(shù)據(jù) 存放讀出數(shù)據(jù) 待讀出長(zhǎng)度 待讀出地址 器件類(lèi)型(24c16或SD2403)
u8 I2C_ReadByte( u16 ReadAddress, u8 DeviceAddress)
{
u8 temp;
if(!I2C_Start())return FALSE;
//I2C_SendByte(((ReadAddress & 0x0700) >>7) | DeviceAddress & 0xFFFE);//設(shè)置高起始地址+器件地址
I2C_SendByte((DeviceAddress & 0xFE));//寫(xiě)器件地址
if(!I2C_WaitAck()){I2C_Stop(); return FALSE;}
I2C_SendByte((u8)((ReadAddress>>8) & 0xFF)); //設(shè)置高起始地址
I2C_WaitAck();
I2C_SendByte((u8)((ReadAddress) & 0xFF)); //設(shè)置低起始地址
I2C_WaitAck();
I2C_Start();
I2C_SendByte((DeviceAddress & 0xFE)|0x01); //讀器件地址
I2C_WaitAck();

//*pDat = I2C_ReceiveByte();
temp = I2C_ReceiveByte();

I2C_NoAck();

I2C_Stop();
return temp;
}
void I2C_24CXX_Init(void)
{
I2C_GPIO_Config();
}
void I2C_24CXX_Write(u16 nAddr, u8* pDat, u16 nLen)
{
u16 i,j;
for(i=0;i
{
I2C_WriteByte(*(pDat+i), nAddr+i, ADDR_24CXX);
}
}
void I2C_24CXX_Read(u16 nAddr, u8* pDat, u16 nLen)
{
u16 i;
for(i=0;i
*(pDat+i)=I2C_ReadByte(nAddr+i, ADDR_24CXX);
}
可是,反復(fù)試了多次,貌似很不穩(wěn)定,有時(shí)正確,有時(shí)錯(cuò)誤。最后添加了紅字處的的延時(shí),讀寫(xiě)完全正常。用示波器觀察,延時(shí)大概7.5ms。


關(guān)鍵詞: STM23模擬I2C讀

評(píng)論


技術(shù)專(zhuān)區(qū)

關(guān)閉