You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

386 lines
12KB

  1. #include "Wire.h"
  2. #if defined(__IMXRT1052__) || defined(__IMXRT1062__)
  3. #include "debug/printf.h"
  4. void TwoWire::begin(void)
  5. {
  6. // use 24 MHz clock
  7. CCM_CSCDR2 = (CCM_CSCDR2 & ~CCM_CSCDR2_LPI2C_CLK_PODF(63)) | CCM_CSCDR2_LPI2C_CLK_SEL;
  8. hardware.clock_gate_register |= hardware.clock_gate_mask;
  9. port->MCR = LPI2C_MCR_RST;
  10. setClock(100000);
  11. // Setup SDA register
  12. *(portControlRegister(hardware.sda_pins[sda_pin_index_].pin)) |= IOMUXC_PAD_PKE | IOMUXC_PAD_PUE | IOMUXC_PAD_PUS(3);
  13. *(portConfigRegister(hardware.sda_pins[sda_pin_index_].pin)) = hardware.sda_pins[sda_pin_index_].mux_val;
  14. if (hardware.sda_pins[sda_pin_index_].select_input_register) {
  15. *(hardware.sda_pins[sda_pin_index_].select_input_register) = hardware.sda_pins[sda_pin_index_].select_val;
  16. }
  17. // setup SCL register
  18. *(portControlRegister(hardware.scl_pins[scl_pin_index_].pin)) |= IOMUXC_PAD_PKE | IOMUXC_PAD_PUE | IOMUXC_PAD_PUS(3);
  19. *(portConfigRegister(hardware.scl_pins[scl_pin_index_].pin)) = hardware.scl_pins[scl_pin_index_].mux_val;
  20. if (hardware.scl_pins[scl_pin_index_].select_input_register) {
  21. *(hardware.scl_pins[scl_pin_index_].select_input_register) = hardware.scl_pins[scl_pin_index_].select_val;
  22. }
  23. }
  24. void TwoWire::begin(uint8_t address)
  25. {
  26. // TODO: slave mode
  27. }
  28. void TwoWire::end()
  29. {
  30. }
  31. void TwoWire::setSDA(uint8_t pin) {
  32. if (pin == hardware.sda_pins[sda_pin_index_].pin) return;
  33. uint32_t newindex=0;
  34. while (1) {
  35. uint32_t sda_pin = hardware.sda_pins[newindex].pin;
  36. if (sda_pin == 255) return;
  37. if (sda_pin == pin) break;
  38. if (++newindex >= sizeof(hardware.sda_pins)) return;
  39. }
  40. if ((hardware.clock_gate_register & hardware.clock_gate_mask)) {
  41. *(portConfigRegister(hardware.sda_pins[sda_pin_index_].pin)) = 5; // hard to know what to go back to?
  42. // setup new one...
  43. *(portControlRegister(hardware.sda_pins[newindex].pin)) |= IOMUXC_PAD_PKE | IOMUXC_PAD_PUE | IOMUXC_PAD_PUS(3);
  44. *(portConfigRegister(hardware.sda_pins[newindex].pin)) = hardware.sda_pins[newindex].mux_val;
  45. if (hardware.sda_pins[newindex].select_input_register) {
  46. *(hardware.sda_pins[newindex].select_input_register) = hardware.sda_pins[newindex].select_val;
  47. }
  48. }
  49. sda_pin_index_ = newindex;
  50. }
  51. void TwoWire::setSCL(uint8_t pin) {
  52. if (pin == hardware.scl_pins[scl_pin_index_].pin) return;
  53. uint32_t newindex=0;
  54. while (1) {
  55. uint32_t scl_pin = hardware.scl_pins[newindex].pin;
  56. if (scl_pin == 255) return;
  57. if (scl_pin == pin) break;
  58. if (++newindex >= sizeof(hardware.scl_pins)) return;
  59. }
  60. if ((hardware.clock_gate_register & hardware.clock_gate_mask)) {
  61. *(portConfigRegister(hardware.scl_pins[scl_pin_index_].pin)) = 5; // hard to know what to go back to?
  62. // setup new one...
  63. *(portControlRegister(hardware.scl_pins[newindex].pin)) |= IOMUXC_PAD_PKE | IOMUXC_PAD_PUE | IOMUXC_PAD_PUS(3);
  64. *(portConfigRegister(hardware.scl_pins[newindex].pin)) = hardware.scl_pins[newindex].mux_val;
  65. if (hardware.scl_pins[newindex].select_input_register) {
  66. *(hardware.scl_pins[newindex].select_input_register) = hardware.scl_pins[newindex].select_val;
  67. }
  68. }
  69. scl_pin_index_ = newindex;
  70. }
  71. size_t TwoWire::write(uint8_t data)
  72. {
  73. if (transmitting || slave_mode) {
  74. if (txBufferLength >= BUFFER_LENGTH+1) {
  75. setWriteError();
  76. return 0;
  77. }
  78. txBuffer[txBufferLength++] = data;
  79. return 1;
  80. }
  81. return 0;
  82. }
  83. size_t TwoWire::write(const uint8_t *data, size_t quantity)
  84. {
  85. if (transmitting || slave_mode) {
  86. size_t avail = BUFFER_LENGTH+1 - txBufferLength;
  87. if (quantity > avail) {
  88. quantity = avail;
  89. setWriteError();
  90. }
  91. memcpy(txBuffer + txBufferLength, data, quantity);
  92. txBufferLength += quantity;
  93. return quantity;
  94. }
  95. return 0;
  96. }
  97. uint8_t TwoWire::endTransmission(uint8_t sendStop)
  98. {
  99. //printf("\nendTransmission %d (%x %x %x) %x\n", txBufferLength,txBuffer[0], txBuffer[1], txBuffer[2], sendStop);
  100. uint32_t i=0, len, status;
  101. len = txBufferLength;
  102. if (!len) return 4; // no data to transmit
  103. // wait while bus is busy
  104. while (1) {
  105. status = port->MSR; // pg 2899 & 2892
  106. if (!(status & LPI2C_MSR_BBF)) break; // bus is available
  107. if (status & LPI2C_MSR_MBF) break; // we already have bus control
  108. // TODO: timeout...
  109. }
  110. //printf("m=%x\n", status);
  111. // Wonder if MFSR we should maybe clear it?
  112. if ((port->MFSR & 0x7) &&
  113. ((port->MSR & (LPI2C_MSR_BBF|LPI2C_MSR_MBF)) != (LPI2C_MSR_BBF|LPI2C_MSR_MBF))) {
  114. port->MCR = LPI2C_MCR_MEN | LPI2C_MCR_RTF; // clear the FIFO
  115. port->MSR = LPI2C_MSR_PLTF | LPI2C_MSR_ALF | LPI2C_MSR_NDF | LPI2C_MSR_SDF | LPI2C_MSR_FEF; // clear flags
  116. //printf("Clear TX Fifo %lx %lx\n", port->MSR, port->MFSR);
  117. }
  118. // TODO: is this correct if the prior use didn't send stop?
  119. //port->MSR = LPI2C_MSR_PLTF | LPI2C_MSR_ALF | LPI2C_MSR_NDF | LPI2C_MSR_SDF; // clear flags
  120. port->MSR = status;
  121. //printf("MSR=%lX, MFSR=%lX\n", status, port->MFSR);
  122. //elapsedMillis timeout=0;
  123. while (1) {
  124. // transmit stuff, if we haven't already
  125. if (i <= len) {
  126. uint32_t fifo_used = port->MFSR & 0x07; // pg 2914
  127. //if (fifo_used < 4) printf("t=%ld\n", fifo_used);
  128. while (fifo_used < 4) {
  129. if (i == 0) {
  130. //printf("start %x\n", txBuffer[0]);
  131. port->MTDR = LPI2C_MTDR_CMD_START | txBuffer[0];
  132. i = 1;
  133. } else if (i < len) {
  134. port->MTDR = LPI2C_MTDR_CMD_TRANSMIT | txBuffer[i++];
  135. } else {
  136. if (sendStop) port->MTDR = LPI2C_MTDR_CMD_STOP;
  137. i++;
  138. break;
  139. }
  140. fifo_used = fifo_used + 1;
  141. }
  142. }
  143. // monitor status
  144. status = port->MSR; // pg 2899 & 2892
  145. if (status & LPI2C_MSR_ALF) {
  146. //printf("arbitration lost\n");
  147. return 4; // we lost bus arbitration to another master
  148. }
  149. if (status & LPI2C_MSR_NDF) {
  150. //printf("NACK, f=%d, i=%d\n", port->MFSR & 0x07, i);
  151. // TODO: check that hardware really sends stop automatically
  152. port->MCR = LPI2C_MCR_MEN | LPI2C_MCR_RTF; // clear the FIFO
  153. // TODO: is always sending a stop the right way to recover?
  154. port->MTDR = LPI2C_MTDR_CMD_STOP;
  155. return 2; // NACK for address
  156. //return 3; // NACK for data TODO: how to discern addr from data?
  157. }
  158. //if (status & LPI2C_MSR_PLTF) {
  159. //printf("bus stuck - what to do?\n");
  160. //return 4;
  161. //}
  162. //if (timeout > 100) {
  163. //printf("status = %x\n", status);
  164. //timeout = 0;
  165. //}
  166. if (sendStop) {
  167. if (status & LPI2C_MSR_SDF) {
  168. // master automatically sends stop condition on some
  169. // types of errors, so this flag only means success
  170. // when all comments in fifo have been fully used
  171. uint32_t fifo = port->MFSR & 0x07;
  172. if (fifo == 0) return 0;
  173. }
  174. } else {
  175. uint32_t fifo_used = port->MFSR & 0x07; // pg 2914
  176. if (fifo_used == 0) {
  177. //digitalWriteFast(15, HIGH);
  178. //delayMicroseconds(2);
  179. //digitalWriteFast(15, LOW);
  180. // TODO: this returns before the last data transmits!
  181. // Should verify every byte ACKs, arbitration not lost
  182. //printf("fifo empty, msr=%x\n", status);
  183. return 0;
  184. }
  185. }
  186. }
  187. }
  188. uint8_t TwoWire::requestFrom(uint8_t address, uint8_t length, uint8_t sendStop)
  189. {
  190. uint32_t cmd=0, status, fifo;
  191. // wait while bus is busy
  192. //printf("\nrequestFrom %x %x %x\n", address, length, sendStop);
  193. while (1) {
  194. status = port->MSR; // pg 2899 & 2892
  195. if (!(status & LPI2C_MSR_BBF)) break; // bus is available
  196. if (status & LPI2C_MSR_MBF) break; // we already have bus control
  197. // TODO: timeout...
  198. }
  199. //printf("idle2, msr=%x\n", status);
  200. // TODO: is this correct if the prior use didn't send stop?
  201. port->MSR = LPI2C_MSR_PLTF | LPI2C_MSR_ALF | LPI2C_MSR_NDF | LPI2C_MSR_SDF | LPI2C_MSR_FEF; // clear flags
  202. //printf("MSR=%lX, MCR:%lx, MFSR=%lX\n", status, port->MCR, port->MFSR);
  203. // Wonder if MFSR we should maybe clear it?
  204. if ((port->MFSR & 0x7) &&
  205. ((port->MSR & (LPI2C_MSR_BBF|LPI2C_MSR_MBF)) != (LPI2C_MSR_BBF|LPI2C_MSR_MBF))) {
  206. port->MCR = LPI2C_MCR_MEN | LPI2C_MCR_RTF; // clear the FIFO
  207. port->MSR = LPI2C_MSR_PLTF | LPI2C_MSR_ALF | LPI2C_MSR_NDF | LPI2C_MSR_SDF | LPI2C_MSR_FEF; // clear flags
  208. //printf("Clear TX Fifo %lx %lx\n", port->MSR, port->MFSR);
  209. }
  210. address = (address & 0x7F) << 1;
  211. if (length < 1) length = 1;
  212. if (length > 255) length = 255;
  213. rxBufferIndex = 0;
  214. rxBufferLength = 0;
  215. //elapsedMillis timeout=0;
  216. while (1) {
  217. // transmit stuff, if we haven't already
  218. if (cmd < 3) {
  219. fifo = port->MFSR & 0x07; // pg 2914
  220. //if (fifo < 4) printf("t=%ld\n", fifo);
  221. while (fifo < 4 && cmd < 3) {
  222. if (cmd == 0) {
  223. port->MTDR = LPI2C_MTDR_CMD_START | 1 | address;
  224. } else if (cmd == 1) {
  225. // causes bus stuck... need way to recover
  226. //port->MTDR = LPI2C_MTDR_CMD_START | (length - 1);
  227. port->MTDR = LPI2C_MTDR_CMD_RECEIVE | (length - 1);
  228. } else {
  229. if (sendStop) port->MTDR = LPI2C_MTDR_CMD_STOP;
  230. }
  231. cmd++;
  232. fifo = fifo + 1;
  233. }
  234. }
  235. // receive stuff
  236. if (rxBufferLength < sizeof(rxBuffer)) {
  237. fifo = (port->MFSR >> 16) & 0x07;
  238. //if (fifo > 0) printf("r=%ld\n", fifo);
  239. while (fifo > 0 && rxBufferLength < sizeof(rxBuffer)) {
  240. rxBuffer[rxBufferLength++] = port->MRDR;
  241. fifo = fifo - 1;
  242. }
  243. }
  244. // monitor status
  245. status = port->MSR; // pg 2899 & 2892
  246. if (status & LPI2C_MSR_ALF) {
  247. //printf("arbitration lost\n");
  248. break;
  249. }
  250. if (status & LPI2C_MSR_NDF) {
  251. //printf("got NACK\n");
  252. // TODO: how to make sure stop is sent?
  253. break;
  254. }
  255. //if (timeout > 250) {
  256. //printf("Status = %x\n", status);
  257. //timeout = 0;
  258. //}
  259. if (rxBufferLength >= length && cmd >= 3) break;
  260. }
  261. //digitalWriteFast(15, HIGH);
  262. //delayMicroseconds(2);
  263. //digitalWriteFast(15, LOW);
  264. return rxBufferLength;
  265. }
  266. uint8_t TwoWire::requestFrom(uint8_t addr, uint8_t qty, uint32_t iaddr, uint8_t n, uint8_t stop)
  267. {
  268. if (n > 0) {
  269. union { uint32_t ul; uint8_t b[4]; } iaddress;
  270. iaddress.ul = iaddr;
  271. beginTransmission(addr);
  272. if (n > 3) n = 3;
  273. do {
  274. n = n - 1;
  275. write(iaddress.b[n]);
  276. } while (n > 0);
  277. endTransmission(false);
  278. }
  279. if (qty > BUFFER_LENGTH) qty = BUFFER_LENGTH;
  280. return requestFrom(addr, qty, stop);
  281. }
  282. PROGMEM
  283. constexpr TwoWire::I2C_Hardware_t TwoWire::i2c1_hardware = {
  284. CCM_CCGR2, CCM_CCGR2_LPI2C1(CCM_CCGR_ON),
  285. {{18, 3 | 0x10, &IOMUXC_LPI2C1_SDA_SELECT_INPUT, 1}, {0xff, 0xff, nullptr, 0}},
  286. {{19, 3 | 0x10, &IOMUXC_LPI2C1_SCL_SELECT_INPUT, 1}, {0xff, 0xff, nullptr, 0}},
  287. IRQ_LPI2C1
  288. };
  289. TwoWire Wire(&IMXRT_LPI2C1, TwoWire::i2c1_hardware);
  290. PROGMEM
  291. constexpr TwoWire::I2C_Hardware_t TwoWire::i2c3_hardware = {
  292. CCM_CCGR2, CCM_CCGR2_LPI2C3(CCM_CCGR_ON),
  293. {{17, 1 | 0x10, &IOMUXC_LPI2C3_SDA_SELECT_INPUT, 2}, {36, 2 | 0x10, &IOMUXC_LPI2C3_SDA_SELECT_INPUT, 1}},
  294. {{16, 1 | 0x10, &IOMUXC_LPI2C3_SCL_SELECT_INPUT, 2}, {37, 2 | 0x10, &IOMUXC_LPI2C3_SCL_SELECT_INPUT, 1}},
  295. IRQ_LPI2C3
  296. };
  297. TwoWire Wire1(&IMXRT_LPI2C3, TwoWire::i2c3_hardware);
  298. PROGMEM
  299. constexpr TwoWire::I2C_Hardware_t TwoWire::i2c4_hardware = {
  300. CCM_CCGR6, CCM_CCGR6_LPI2C4_SERIAL(CCM_CCGR_ON),
  301. {{25, 0 | 0x10, &IOMUXC_LPI2C4_SDA_SELECT_INPUT, 1}, {0xff, 0xff, nullptr, 0}},
  302. {{24, 0 | 0x10, &IOMUXC_LPI2C4_SCL_SELECT_INPUT, 1}, {0xff, 0xff, nullptr, 0}},
  303. IRQ_LPI2C4
  304. };
  305. TwoWire Wire2(&IMXRT_LPI2C4, TwoWire::i2c4_hardware);
  306. void TwoWire::setClock(uint32_t frequency)
  307. {
  308. port->MCR = 0;
  309. if (frequency < 400000) {
  310. // 100 kHz
  311. port->MCCR0 = LPI2C_MCCR0_CLKHI(55) | LPI2C_MCCR0_CLKLO(59) |
  312. LPI2C_MCCR0_DATAVD(25) | LPI2C_MCCR0_SETHOLD(40);
  313. port->MCFGR1 = LPI2C_MCFGR1_PRESCALE(1);
  314. port->MCFGR2 = LPI2C_MCFGR2_FILTSDA(5) | LPI2C_MCFGR2_FILTSCL(5) |
  315. LPI2C_MCFGR2_BUSIDLE(3900);
  316. } else if (frequency < 1000000) {
  317. // 400 kHz
  318. port->MCCR0 = LPI2C_MCCR0_CLKHI(26) | LPI2C_MCCR0_CLKLO(28) |
  319. LPI2C_MCCR0_DATAVD(12) | LPI2C_MCCR0_SETHOLD(18);
  320. port->MCFGR1 = LPI2C_MCFGR1_PRESCALE(0);
  321. port->MCFGR2 = LPI2C_MCFGR2_FILTSDA(2) | LPI2C_MCFGR2_FILTSCL(2) |
  322. LPI2C_MCFGR2_BUSIDLE(3900);
  323. } else {
  324. // 1 MHz
  325. port->MCCR0 = LPI2C_MCCR0_CLKHI(9) | LPI2C_MCCR0_CLKLO(10) |
  326. LPI2C_MCCR0_DATAVD(4) | LPI2C_MCCR0_SETHOLD(7);
  327. port->MCFGR1 = LPI2C_MCFGR1_PRESCALE(0);
  328. port->MCFGR2 = LPI2C_MCFGR2_FILTSDA(1) | LPI2C_MCFGR2_FILTSCL(1) |
  329. LPI2C_MCFGR2_BUSIDLE(3900);
  330. }
  331. port->MCCR1 = port->MCCR0;
  332. port->MCFGR0 = 0;
  333. port->MCFGR3 = LPI2C_MCFGR3_PINLOW(3900);
  334. port->MFCR = LPI2C_MFCR_RXWATER(1) | LPI2C_MFCR_TXWATER(1);
  335. port->MCR = LPI2C_MCR_MEN;
  336. }
  337. #endif