STM8S105S4 I2C
STM8S105S4 I2C這樣配置可以從機(jī)發(fā)數(shù)據(jù),從機(jī)接數(shù)據(jù)就不行,是什么問題 //I2C端口初始化 GPIO_Init(GPIOE,GPIO_PIN_1|GPIO_PIN_2, GPIO_MODE_OUT_OD_HIZ_SLOW); void I2C_init(void) { I2C_CCRH = 0; //標(biāo)準(zhǔn)I2C接口 I2C_CCRL = 80 ; I2C_FREQR = 16; I2C_OARL = (I2CAddr<<1); I2C_OARH= 0x40; I2C_ITR =0x06; I2C_CR1 =0x01; I2C_CR2 |=0x04; } @far @interrupt void I2C_Handler (void) { u8 state1,state2,state3; state1 = I2C_SR1; state2 = I2C_SR2; state3 = I2C_SR3; //地址匹配 if((state1&0x02)!=0){I2C_CR2 |= 0x04;} //接收到數(shù)據(jù),需要處理 if( (state1& 0x40) != 0) { I2C_CR2|= 0x04; temp1= I2C_DR; SetOutput(temp1); } //從機(jī)需要發(fā)送數(shù)據(jù) if((state1 & 0x80) != 0) { I2C_DR = num++; } if((state2 & 0x04) != 0)I2C_SR2 &= ~0x04; //應(yīng)答失敗,清除該位 if((state1 & 0x10) != 0) { I2C_CR2 = 0x02; } } 我寫成這樣子,發(fā)現(xiàn)現(xiàn)在是從機(jī)發(fā)數(shù)據(jù)是沒問題,從機(jī)接收數(shù)據(jù)不行,沒有進(jìn)入中斷。 |
成功傳輸一次正確的數(shù)據(jù)后就進(jìn)不去了。程序還正常運(yùn)行
本文引用地址:http://m.butianyuan.cn/article/201612/325048.htm#define I2CAddr 0x01
在主機(jī)
//主機(jī)寫數(shù)據(jù),從機(jī)接,這個(gè)不行
Soft_I2C_Start();
Soft_I2C_Write(0x02);
Soft_I2C_Write(0xAA);
Soft_I2C_Stop();
//主機(jī)接收數(shù)據(jù)。沒有運(yùn)行上一段程序,主機(jī)可以接收數(shù)據(jù),運(yùn)行上段,主機(jī)接收數(shù)據(jù)都是255
Soft_I2C_Start();
Soft_I2C_Write(0x03);
var0=Soft_I2C_Read(1);
Soft_I2C_Stop();
這個(gè)軟I2C在控制其它設(shè)備是沒有問題的,主機(jī)用的芯片是STM32。應(yīng)該是沒有問題的
解決問題:
這就是了,你第一次成功后了,你把應(yīng)答給關(guān)掉了,還怎么接收啊。。。以后配寄存器要多一個(gè)心眼才是。
問題就在這里I2C_CR2 = 0x02; 修改為I2C_CR2 |= 0x02;原因是修改了,第一次成功后,沒有再回復(fù) |
1.PC0,PC1初始化,GPIO_Init(GPIOC, GPIO_Pin_1 | GPIO_Pin_0, GPIO_Mode_Out_OD_HiZ_Fast);
2.必須連接硬件
3.總線不能busy,一定要空閑
初始化代碼如下:
GPIO_Init(GPIOC, GPIO_Pin_0, GPIO_Mode_Out_OD_HiZ_Slow);
GPIO_Init(GPIOC, GPIO_Pin_1, GPIO_Mode_Out_OD_HiZ_Slow);
CLK_PeripheralClockConfig(CLK_Peripheral_I2C1, ENABLE);
I2C_Init(I2C1, 50000, 0xA5, I2C_Mode_I2C, I2C_DutyCycle_2,
I2C_Ack_Enable, I2C_AcknowledgedAddress_7bit);
//注:I2C_Init()里已經(jīng)使能了I2C,所以可以不用I2C_Cmd(I2C1, ENABLE);
while(I2C_GetFlagStatus(I2C1, I2C_FLAG_BUSY));
stm8 自帶I2C終于調(diào)試成功。
1,此次調(diào)試I2C受益多多,首先證明官方的那個(gè)寫EEPROM的程序是對的。
2,這次調(diào)試TW8816開始調(diào)不通的原因是誤以為寄存器地址是16位的(芯片公司的業(yè)務(wù)也是這么說的,我相信了),才讓我看到的數(shù)據(jù)不同,但我想,有數(shù)據(jù)了,而且用WHILE等待的方法,程序跑通了,說明連上了從機(jī)。
3, I2C的GPIO不要配置,我看到有些同仁把GPIO配置了。
4,為了大家用好STM8的I2C,我把程序烤下來。
void IIC_Init(void)
{
UCHAR temp;
CLK_PCKENR1 |= 0x01;
I2C_FREQR |= 0x10; //輸入外設(shè)時(shí)鐘頻率為1MHz
I2C_CR1 = 0x00; //禁止I2C外設(shè)
I2C_CCRH&= ~0xcf;
I2C_CCRL&= ~0xff;
I2C_TRISER = 0x11;
I2C_CCRL = 0x10;
I2C_CCRH = 0x00;
I2C_CR1 |= 0x01; //開啟I2C外設(shè)
I2C_CR2 |= 0x04; //當(dāng)前接收字節(jié)返回應(yīng)答
I2C_CR2 &= 0x08;
I2C_OARL = 0x86; //自身地址
I2C_OARH = 0x40;
}
void Read_8816(UCHAR *pBuffer, UCHAR index, UCHAR NumByteToRead)
{
UCHAR temp;
while(I2C_SR3 & 0x02); //等待總線空閑
I2C_CR2 |= 0x01; //產(chǎn)生起始位
while(!(I2C_SR1 & 0x01)); //等待START發(fā)送完
I2C_DR = 0x8a; //發(fā)送8816器件地址
while(!(I2C_SR1 & 0x02)); //等特7位器件地址發(fā)送完
temp = I2C_SR1;
temp = I2C_SR3;
I2C_DR = (UCHAR)(index);
while(!(I2C_SR1 & 0x84));
I2C_CR2 |= 0x01; //產(chǎn)生重復(fù)起始位
while(!(I2C_SR1 & 0x01)); //等待START發(fā)送完
I2C_DR = 0x8b; //讀
while(!(I2C_SR1 & 0x02)); //等特7位器件地址發(fā)送完
temp = I2C_SR1;
temp = I2C_SR3;
while(NumByteToRead) //要讀幾個(gè)字節(jié)
{
if(NumByteToRead == 1)
{
I2C_CR2 &= ~0x04; //不返回應(yīng)答
I2C_CR2 |= 0x02; //產(chǎn)生停止位
}
if(I2C_SR1 & 0x40)
{
temp = I2C_SR1;
Buff[8-NumByteToRead]=I2C_DR;
*pBuffer = Buff[8-NumByteToRead];
pBuffer++;
NumByteToRead--;
}
}
I2C_CR2 |= 0x04;
I2C_CR2 &= ~0x08; //為下一次接收使能應(yīng)答
}
評論