Atmel Tiny88 - software I2C master

 Software I2C (master)

Microchip Tiny-88 只有一組 TWI master/slave 功能,在系統設計中,Tiny-88 一般做為電源管理,系統開關機,還有和系統主晶片組溝通。Tiny-88 和主晶片組溝通的介面會採用 TWI slave,當還需要再讀取系統溫度,系統電流,系統生產資料或者系統電池時,還需要一組 I2C master 去讀取系統的資料,只能使用 GPIO 來模擬 software I2C.   


Software I2C header

宣告程式碼中,需要用到的變數和子函式,使用宣告的好處是日後若要更換成其他組的 GPIO 容易更換,也不容易有遺漏。
  • 宣告 GPIO PORT 和接腳位置去模擬 software I2C
  • 宣告 I2C 寛度的時間設定值
  • 宣告 GPIO High / Low / Input 子函式
  1. /* Constant Variable Define */
  2. #define SW_I2C_DDR DDRD
  3. #define SW_I2C_DIN PIND
  4. #define SW_I2C_DOUT PORTD
  5. #define SW_I2C_SDA 6
  6. #define SW_I2C_SCL 7
  7. /** Bit to or with address for read start and read restart */
  8. #define I2C_READ 1
  9. /** Bit to or with address for write start and write restart */
  10. #define I2C_WRITE 0
  11. /** Delay used for software I2C */
  12. #define I2C_DELAY_USEC 10
  13. #define I2C_DELAY_CLKH 6 /* I2C Clock High Width */
  14. /* I2C Clock Low Width (by instruction run time) */
  15. #define I2C_DELAY_CLKL 0
  16. /* I2C Clock Low Width for finial bit */
  17. #define I2C_DELAY_CLKFL 4
  18. #define I2C_DELAY_DATBR 3 /* I2C Data Delay to before read */
  19. #define I2C_DELAY_DATAR 2 /* I2C Data Delay to after read */
  20. /* Declare Function */
  21. #define SET_I2C_SCL_HI() (SW_I2C_DOUT |= (1 &lt< SW_I2C_SCL))
  22. #define SET_I2C_SCL_LO() (SW_I2C_DOUT &= ~(1 &lt< SW_I2C_SCL))
  23. #define SET_I2C_SDA_HI() (SW_I2C_DOUT |= (1 &lt< SW_I2C_SDA))
  24. #define SET_I2C_SDA_LO() (SW_I2C_DOUT &= ~(1 &lt< SW_I2C_SDA))
  25. #define SET_I2C_SDA_OUT() (SW_I2C_DDR |= (1 &lt< SW_I2C_SDA))
  26. #define SET_I2C_SDA_IN() (SW_I2C_DDR &= ~(1 &lt< SW_I2C_SDA))
  27. #define GET_I2C_SDA_IN() (SW_I2C_DIN & (1 &lt< SW_I2C_SDA))

Software I2C code
I2C protocol 分為 start-bit,stop-bit,repeat start-bit,I2C-read,I2C-write 的子函式來組合成主函式。因此,在編寫程式碼前,建議先使用流程圖規劃一下,再開始編寫程式碼,先完成子函式,再組成主函式。
  • SW_I2C_start(address): 
子函式主要是用在一開始傳送 start-bit 和 slave address,也用於 read protocol 時的 repeat start-bit 只是傳遞的 8-bit 位址的 bit 0 要改為 read.
  1. /* ! \brief SW I2C issue a start condition
  2. * This function use to send start condition of SW I2C
  3. * \note
  4. * (input) slave device address
  5. * \return
  6. * 1 : the address has ACK
  7. * 0 : the address without ACK
  8. */
  9. uint8_t SW_I2C_start(uint8_t addressRW)
  10. {
  11. uint8_t result;
  12. SET_I2C_SDA_LO(); /* set data to low */
  13. _delay_us(I2C_DELAY_USEC);
  14. SET_I2C_SCL_LO(); /* set clock to low */
  15. _delay_us(I2C_DELAY_CLKFL);
  16. result = SW_I2C_write(addressRW);
  17. return result;
  18. }
  • SW_I2C_restart(address) : 
子函式用於 read protocol 的 repeat start bit ,需記得更改 slave address 為 read。  
  1. /* ! \brief SW I2C issue a re-start condition
  2. * This function use to send re-start condition of SW I2C
  3. * \note
  4. * (input) slave device address
  5. * \return
  6. * 1 : the address has ACK
  7. * 0 : the address without ACK
  8. */
  9. uint8_t SW_I2C_restart(uint8_t addressRW)
  10. {
  11. SET_I2C_SDA_HI();
  12. SET_I2C_SCL_HI();
  13. _delay_us(I2C_DELAY_USEC);
  14. return SW_I2C_start(addressRW);
  15. }
  • SW_I2C_stop(void): 
子函式 stop-bit 結束 I2C packet
  1. /* ! \brief SW I2C issue a stop condition
  2. * This function use to send stop condition of SW I2C
  3. * \note
  4. * (input) slave device address
  5. * \return
  6. * 1 : the address has ACK
  7. * 0 : the address without ACK
  8. */
  9. void SW_I2C_stop(void)
  10. {
  11. SET_I2C_SDA_LO(); /* set data to low */
  12. _delay_us(I2C_DELAY_USEC);
  13. SET_I2C_SCL_HI();
  14. _delay_us(I2C_DELAY_USEC);
  15. SET_I2C_SDA_HI();
  16. _delay_us(I2C_DELAY_USEC);
  17. }
  • SW_I2C_read(last): 
子函式讀取一個 byte 和回應 ack /nack 給 host
  1. /* ! \brief SW_I2C_read
  2. * This function use to read a byte and send ack/nack to device
  3. * \note
  4. * decide the terminate bit is ACK or NACK.
  5. * \return
  6. * data byte of I2C read
  7. */
  8. uint8_t SW_I2C_read(uint8_t last)
  9. {
  10. uint8_t b = 0;
  11. /* make sure pull-up enabled */
  12. SET_I2C_SDA_HI();
  13. SET_I2C_SDA_IN(); /* change to input mode */
  14. /* read byte */
  15. for (uint8_t i = 0; i < 8; i++) {
  16. /* do not change this loop unless you verify the change with a
  17. * scope */
  18. _delay_us(I2C_DELAY_CLKFL);
  19. b &lt<= 1;
  20. SET_I2C_SCL_HI(); /* set clock to high */
  21. _delay_us(I2C_DELAY_DATBR);
  22. if (GET_I2C_SDA_IN())
  23. b |= 1; /* read bit data */
  24. _delay_us(I2C_DELAY_DATAR);
  25. SET_I2C_SCL_LO(); /* set clock to low */
  26. }
  27. /* send Ack or Nak */
  28. SET_I2C_SDA_OUT();
  29. if (last)
  30. SET_I2C_SDA_HI();
  31. else
  32. SET_I2C_SDA_LO();
  33. SET_I2C_SCL_HI(); /* set clock to high */
  34. _delay_us(I2C_DELAY_CLKH);
  35. SET_I2C_SCL_LO(); /* set clock to low */
  36. SET_I2C_SDA_LO();
  37. return b;
  38. }
  • SW_I2C_write(data): 
子函式寫入一個 byte 並讀回 slave 的 ack/nack 
  1. /* ! \brief SW_I2C_write
  2. * This function use to write a byte to device
  3. * \note
  4. * (input) data byte
  5. * \return
  6. * 1 : slave return ACK
  7. * 0 : slave return NACK
  8. */
  9. uint8_t SW_I2C_write(uint8_t data)
  10. {
  11. uint8_t rtn;
  12. /* write byte */
  13. for (uint8_t m = 0x80; m != 0; m >>= 1) {
  14. /* don't change this loop unless you verify the change with a
  15. * scope */
  16. if (m & data)
  17. SET_I2C_SDA_HI();
  18. else
  19. SET_I2C_SDA_LO();
  20. SET_I2C_SCL_HI(); /* set clock to high */
  21. _delay_us(I2C_DELAY_CLKH);
  22. SET_I2C_SCL_LO(); /* set clock to low */
  23. _delay_us(I2C_DELAY_CLKL);
  24. }
  25. _delay_us(I2C_DELAY_CLKFL);
  26. SET_I2C_SCL_HI();
  27. _delay_us(I2C_DELAY_DATBR);
  28. /* get Ack or Nak */
  29. SET_I2C_SDA_IN(); /* change to input mode */
  30. rtn = GET_I2C_SDA_IN();
  31. _delay_us(I2C_DELAY_DATAR);
  32. SET_I2C_SCL_LO(); /* set clock to low */
  33. _delay_us(I2C_DELAY_CLKFL);
  34. SET_I2C_SDA_OUT();
  35. SET_I2C_SDA_LO(); /* set data to low */
  36. return rtn == 0;
  37. }
  • SW_I2C_buffer_read:
主函式完成 I2C read byte 功能,包含 start-bit,I2C-slave address,I2C-command,repeat start-bitaddressI2C-data buffer,stop-bit 完整的 I2C read protocol. 
  1. /* ! \brief SW I2C read protocol
  2. * This function use to SW I2C read protocol
  3. * \note
  4. * (input) address-slave device address
  5. * (input) cmd-read command
  6. * (input) buf-read protocol buffer
  7. * (input) count-read protocol byte count
  8. * \return
  9. * 1 : success of read protocol
  10. * 0 : fail of read protocol
  11. */
  12. uint8_t SW_I2C_buffer_read(uint8_t address, uint8_t cmd, uint8_t *buf,
  13. uint8_t count)
  14. {
  15. cli();
  16. /* issue a start condition, send device address and write direction
  17. * bit */
  18. if (!SW_I2C_start(address | I2C_WRITE)) {
  19. SW_I2C_stop();
  20. sei();
  21. return FALSE;
  22. }
  23. /* send the DS1307 address */
  24. if (!SW_I2C_write(cmd)) {
  25. SW_I2C_stop();
  26. sei();
  27. return FALSE;
  28. }
  29. /* issue a repeated start condition, send device address and read
  30. * direction bit */
  31. if (!SW_I2C_restart(address | I2C_READ)) {
  32. SW_I2C_stop();
  33. sei();
  34. return FALSE;
  35. }
  36. /* read data from the DS1307 */
  37. for (uint8_t i = 0; i < count; i++) {
  38. /* send Ack until last byte then send Ack */
  39. buf[i] = SW_I2C_read(i == (count - 1));
  40. }
  41. /* issue a stop condition */
  42. SW_I2C_stop();
  43. sei();
  44. return TRUE;
  45. }
  • SW_I2C_buffer_write:
主函式完成 I2C read write 功能,包含 start-bit,I2C-slave address,I2C-commandI2C-data buffer,stop-bit 完整的 I2C write protocol.
  1. /* ! \brief SW I2C write protocol
  2. * This function use to SW I2C write protocol
  3. * \note
  4. * (input) address-slave device address
  5. * (input) cmd-write command
  6. * (input) buf-write protocol buffer
  7. * (input) count-write protocol byte count
  8. * \return
  9. * 1 : success of write protocol
  10. * 0 : fail of write protocol
  11. */
  12. uint8_t SW_I2C_buffer_write(uint8_t address, uint8_t cmd, uint8_t *buf,
  13. uint8_t count)
  14. {
  15. cli();
  16. /* issue a start condition, send device address and write direction
  17. * bit */
  18. if (!SW_I2C_start(address | I2C_WRITE)) {
  19. SW_I2C_stop();
  20. sei();
  21. return FALSE;
  22. }
  23. /* send the DS1307 address */
  24. if (!SW_I2C_write(cmd)) {
  25. SW_I2C_stop();
  26. sei();
  27. return FALSE;
  28. }
  29. /* send data to the DS1307 */
  30. for (uint8_t i = 0; i < count; i++) {
  31. if (!SW_I2C_write(buf[i])) {
  32. SW_I2C_stop();
  33. sei();
  34. return FALSE;
  35. }
  36. }
  37. /* issue a stop condition */
  38. SW_I2C_stop();
  39. sei();
  40. return TRUE;
  41. }
  42.  

留言

這個網誌中的熱門文章

EC 所需知識 - SMBUS

EC 所需知識 - KBC

EC 所需知識 - LPC