fw_i2c.h 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129
  1. // Copyright 2021 IOsetting <iosetting(at)outlook.com>
  2. //
  3. // Licensed under the Apache License, Version 2.0 (the "License");
  4. // you may not use this file except in compliance with the License.
  5. // You may obtain a copy of the License at
  6. //
  7. // http://www.apache.org/licenses/LICENSE-2.0
  8. //
  9. // Unless required by applicable law or agreed to in writing, software
  10. // distributed under the License is distributed on an "AS IS" BASIS,
  11. // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  12. // See the License for the specific language governing permissions and
  13. // limitations under the License.
  14. #ifndef ___FW_I2C_H___
  15. #define ___FW_I2C_H___
  16. #include "fw_conf.h"
  17. #include "fw_types.h"
  18. typedef enum
  19. {
  20. I2C_WorkMode_Slave = 0x00,
  21. I2C_WorkMode_Master = 0x01,
  22. } I2C_WorkMode_t;
  23. typedef enum
  24. {
  25. I2C_MasterCmd_Wait = 0x00, // Wait, idle
  26. I2C_MasterCmd_Start = 0x01, // START
  27. I2C_MasterCmd_Send = 0x02, /* Send data. This command will generate 8 clocks on SCL, and
  28. send I2CTXD to SDA bit by bit from MSB */
  29. I2C_MasterCmd_RxAck = 0x03, /* Recive Ack. This command will generate 1 clock on SCL, and
  30. save the received bit to MSACKI(I2CMSST.1) */
  31. I2C_MasterCmd_Recv = 0x04, // Recive data
  32. I2C_MasterCmd_TxAck = 0x05, /* Send Ack. This command will generate 1 clock on SCL, and
  33. write the bit value of MSACKO(I2CMSST.0) to SDA */
  34. I2C_MasterCmd_Stop = 0x06, // STOP. This command will send STOP signal, and reset MSBUSY flag
  35. I2C_MasterCmd_StartSendRxAck = 0x09, // START + Send data + RxAck
  36. I2C_MasterCmd_SendRxAck = 0x0A, // Send data + RxAck
  37. I2C_MasterCmd_RecvTxAck0 = 0x0B, // Receive data + TxAck(0)
  38. I2C_MasterCmd_RecvNAck = 0x0C, // Receive data + NAck
  39. } I2C_MasterCmd_t;
  40. typedef enum
  41. {
  42. // SCL SDA
  43. I2C_AlterPort_P15_P14 = 0x00,
  44. I2C_AlterPort_P25_P24 = 0x01,
  45. I2C_AlterPort_P77_P76 = 0x02,
  46. I2C_AlterPort_P32_P33 = 0x03,
  47. } I2C_AlterPort_t;
  48. /**
  49. * I2C bus clock = FOSC / 2 / (prescaler * 2 + 4)
  50. * prescaler range is [0, 63]
  51. */
  52. /* Compute i2c prescaler given frequency. If the result exceeds 63, i2c might not work as expeted */
  53. #define I2C_PRESCALER_COMPUTE(__FREQ__) ((((__SYSCLOCK / __FREQ__) / 2U) - 4U) / 2U)
  54. /* Compute i2c frequency given prescaler */
  55. #define I2C_FREQUENCY_COMPUTE(__PRESCALER__) ((__SYSCLOCK / 2U) / (((__PRESCALER__) * 2U) + 4U))
  56. #define I2C_SetEnabled(__STATE__) SFRX_ASSIGN(I2CCFG, 7, __STATE__)
  57. #define I2C_SetWorkMode(__MODE__) SFRX_ASSIGN(I2CCFG, 6, __MODE__)
  58. /**
  59. * I2C bus clock = FOSC / 2 / (__DIV__ * 2 + 4)
  60. * __DIV__ range [0, 63]
  61. */
  62. #define I2C_SetClockPrescaler(__DIV__) do { \
  63. SFRX_ON(); \
  64. (I2CCFG) = (I2CCFG) & ~(0x3F) | ((__DIV__) & 0x3F); \
  65. SFRX_OFF(); \
  66. } while(0)
  67. #define I2C_SendMasterCmd(__CMD__) { \
  68. (I2CMSCR) = (I2CMSCR) & ~(0x0F) | ((__CMD__) & 0x0F); \
  69. while (!(I2CMSST & 0x40)); \
  70. I2CMSST &= ~0x40; \
  71. }
  72. #define I2C_MasterStart() I2C_SendMasterCmd(I2C_MasterCmd_Start)
  73. #define I2C_MasterSendData(__DATA__) do{I2CTXD = (__DATA__); I2C_SendMasterCmd(I2C_MasterCmd_Send);}while(0)
  74. #define I2C_MasterRxAck() I2C_SendMasterCmd(I2C_MasterCmd_RxAck)
  75. #define I2C_MasterAck() do{I2CMSST &= ~(0x01); I2C_SendMasterCmd(I2C_MasterCmd_TxAck);}while(0)
  76. #define I2C_MasterNAck() do{I2CMSST |= 0x01; I2C_SendMasterCmd(I2C_MasterCmd_TxAck);}while(0)
  77. #define I2C_MasterStop() I2C_SendMasterCmd(I2C_MasterCmd_Stop)
  78. /**
  79. * If enabled, `Send Data`+`RxAck` will be executed automatically after write operation on I2CTXD
  80. */
  81. #define I2C_SetMasterAutoSend(__STATE__) SFRX_ASSIGN(I2CMSAUX, 0, __STATE__)
  82. /**
  83. * Call P_SW2=0x80 before invoking this
  84. */
  85. #define I2C_IsMasterBusy() (I2CMSST & 0x80)
  86. #define I2C_ClearMasterCmdInterrupt() do {SFRX_ON(); I2CMSST &= ~(0x01 << 6); SFRX_OFF();} while(0)
  87. #define I2C_ResetSlaveMode() SFRX_SET(I2CSLCR, 0)
  88. /**
  89. * Call SFRX_ON() before invoking these
  90. */
  91. #define I2C_IsSlaveBusy() (I2CSLST & (0x01 << 7))
  92. #define I2C_IsSlaveStartInterrupt() (I2CSLST & (0x01 << 6))
  93. #define I2C_ClearSlaveStartInterrupt() (I2CSLST &= ~(0x01 << 6))
  94. #define I2C_IsSlaveRecvInterrupt() (I2CSLST & (0x01 << 5))
  95. #define I2C_ClearSlaveRecvInterrupt() (I2CSLST &= ~(0x01 << 5))
  96. #define I2C_IsSlaveSendInterrupt() (I2CSLST & (0x01 << 4))
  97. #define I2C_ClearSlaveSendInterrupt() (I2CSLST &= ~(0x01 << 4))
  98. #define I2C_IsSlaveStopInterrupt() (I2CSLST & (0x01 << 3))
  99. #define I2C_ClearSlaveStopInterrupt() (I2CSLST &= ~(0x01 << 3))
  100. #define I2C_ReadSlaveAckIn() (I2CSLST & (0x01 << 1))
  101. #define I2C_ReadSlaveAckOut() (I2CSLST & (0x01 << 0))
  102. #define I2C_ClearAllSlaveInterrupts() (I2CSLST = 0x00)
  103. #define I2C_SetSlaveAddrControl(__STATE__) SFRX_ASSIGN(I2CSLADR, 0, __STATE__)
  104. /**
  105. * Alternative ports
  106. */
  107. #define I2C_SetPort(__ALTER_PORT__) (P_SW2 = P_SW2 & ~(0x03 << 4) | ((__ALTER_PORT__) << 4))
  108. uint8_t I2C_Write(uint8_t devAddr, uint8_t memAddr, uint8_t *dat, uint16_t size);
  109. uint8_t I2C_Read(uint8_t devAddr, uint8_t memAddr, uint8_t *buf, uint16_t size);
  110. uint8_t I2C_Write16BitAddr(uint8_t devAddr, uint16_t memAddr, uint8_t *dat, uint16_t size);
  111. uint8_t I2C_Read16BitAddr(uint8_t devAddr, uint16_t memAddr, uint8_t *buf, uint16_t size);
  112. #endif