stm8s开发(八) IIC的使用:IIC主机通信!

前面讲过两个常用的串口,UART和SPI,这次这次讲解一下另一个常用的串口:IIC(I2C)通信


科普IIC:一般有两根信号线,一根是双向的数据线SDA,另一根是时钟线SCL。所有接到IIC总线设备上的串行数据SDA都接到总线的SDA上,各设备的时钟线SCL接到总线的SCL上。


通信过程:
主模式时,IIC接口启动数据传输并产生时钟信号。串行数据传输总是以起始条件开始并以停止条件结束。起始条件和停止条件都是在主模式下由软件控制产生。
从模式时,IIC接口能识别它自己的地址(7位或10位)和广播呼叫地址。软件能够控制开启或禁止广播呼叫地址的识别。
数据和地址按8位/字节进行传输,高位在前。跟在起始条件后的1或2个字节是地址(7位模式为1个字节,10位模式为2个字节)。地址只在主模式发送。
在一个字节传输的8个时钟后的第9个时钟期间,接收器必须回送一个应答位(ACK)给发送器。参考下图。


详细协议可以参考:http://blog.csdn.net/subkiller/article/details/6854910


和SPI差不多,使用IIC无非就一个初始化,一个数据发送,一个数据接收,三大功能。

初始化分初始化为主机、从机,不过一般和外部芯片通信单片机都是作为主机。

  1. void IIC_Master_Init(void)
  2. {
  3. CLK_PCKENR1 |= 0x01; //使能IIC外设时钟
  4. PB_DDR &= 0xcf;
  5. PB_CR1 &= 0xcf;
  6. PB_CR2 &= 0xcf;
  7. I2C_CR1 = 0x00; //允许时钟延展,禁止广播呼叫,禁止iic
  8. I2C_FREQR = 0x01; //输入时钟频率8MHz
  9. I2C_OARH = 0x40; //七位地址模式
  10. I2C_OARL = 0xa0; //自身地址0xa0
  11. I2C_CCRL = 0xff; //
  12. I2C_CCRH = 0x00; //标准模式
  13. I2C_TRISER = 0x02;
  14. I2C_CR1 |= 0x01; //使能iic外设
  15. }

这里使用的是STM8S105片上的IIC引脚PB4、PB5。另外就是需要使能IIC的时钟。


发送和接收数据这里只给出一个最简单的例子,因为不同的外部芯片的通信方式不一样,不过一般都是:

读操作:开始  -> 发送外设地址 -> 开始  ->发送需要读取的寄存器地址 -> 读一个字节  -> (可能再读一个字节)  ->。。。  -> 结束
写操作:开始  -> 发送外设地址 -> 发送需要写入的寄存器地址 -> 写一个字节  -> (可能再写一个字节)  ->。。。  -> 结束


  1. void IIC_Write_Byte(u8 DeviceAddress, u8 Address, u8 Data)
  2. {
  3. vu8 temp = 0;
  4. while((I2C_SR3 & 0x02) != 0); //等待IIC总线空闲
  5. IIC_Start();
  6. while((I2C_SR1 & 0x01) == 0); //EV5,起始信号已经发送
  7. I2C_DR = (DeviceAddress & 0xfe); // 发送iic从器件物理地址,最低位0,写操作
  8. while((I2C_SR1 & 0x02) == 0); //地址已经被发送
  9. temp = I2C_SR1; //清除ADDR标志位
  10. temp = I2C_SR3;
  11. while((I2C_SR1 & 0x80) == 0); //等待发送寄存器为空
  12. I2C_DR = Address; //发送要写入的寄存器地址
  13. while((I2C_SR1 & 0x04) == 0); //等待发送完成
  14. while((I2C_SR1 & 0x80) == 0); //等待发送寄存器为空
  15. I2C_DR = Data; //发送要写入的数据
  16. while((I2C_SR1 & 0x04) == 0); //等待发送完成
  17. temp = I2C_SR1; //清零BTF标志位
  18. temp = I2C_DR;
  19. IIC_Stop(); //发送停止信号
  20. }
  21. unsigned char IIC_Read_Byte(u8 DeviceAddress, u8 Address)
  22. {
  23. vu8 temp = 0;
  24. short read_data = 0;
  25. while((I2C_SR3 & 0x02) != 0); //等待IIC总线空闲
  26. I2C_CR2 |= 0x04; //使能ACK
  27. IIC_Start();
  28. while((I2C_SR1 & 0x01) == 0); //EV5,起始信号已经发送
  29. I2C_DR = (DeviceAddress & 0xfe); // 发送iic从器件物理地址,最低位0,写操作
  30. while((I2C_SR1 & 0x02) == 0); //地址已经被发送
  31. temp = I2C_SR1; //清除ADDR标志位
  32. temp = I2C_SR3;
  33. while((I2C_SR1 & 0x80) == 0); //等待发送寄存器为空
  34. I2C_DR = Address; // 发送要读取的寄存器地址
  35. while((I2C_SR1 & 0x04) == 0); //等待数据发送完成
  36. IIC_Start();
  37. while((I2C_SR1 & 0x01) == 0); //EV5,起始信号已经发送
  38. I2C_DR = (DeviceAddress | 0x01); // 发送iic从器件物理地址,最低位1,读操作
  39. while((I2C_SR1 & 0x02) == 0); //地址已经被发送
  40. temp = I2C_SR1; //清除ADDR标志位
  41. temp = I2C_SR3;
  42. while((I2C_SR1 & 0x40) == 0); //等待接收数据寄存器非空
  43. read_data = I2C_DR;
  44. I2C_CR2 &= 0xfb; //读取数据下,发送stop必须禁止ack,才能释放从机
  45. temp = I2C_SR1; //清零BTF标志位
  46. temp = I2C_DR;
  47. IIC_Stop();
  48. return read_data;
  49. }

猜你喜欢

转载自blog.csdn.net/qinrenzhi/article/details/80882495