My Marlin configs for Fabrikator Mini and CTC i3 Pro B
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.

tft_spi.cpp 8.1KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270
  1. /**
  2. * Marlin 3D Printer Firmware
  3. * Copyright (c) 2020 MarlinFirmware [https://github.com/MarlinFirmware/Marlin]
  4. *
  5. * Based on Sprinter and grbl.
  6. * Copyright (c) 2011 Camiel Gubbels / Erik van der Zalm
  7. *
  8. * This program is free software: you can redistribute it and/or modify
  9. * it under the terms of the GNU General Public License as published by
  10. * the Free Software Foundation, either version 3 of the License, or
  11. * (at your option) any later version.
  12. *
  13. * This program is distributed in the hope that it will be useful,
  14. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  15. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  16. * GNU General Public License for more details.
  17. *
  18. * You should have received a copy of the GNU General Public License
  19. * along with this program. If not, see <https://www.gnu.org/licenses/>.
  20. *
  21. */
  22. #include "../../platforms.h"
  23. #ifdef HAL_STM32
  24. #include "../../../inc/MarlinConfig.h"
  25. #if HAS_SPI_TFT
  26. #include "tft_spi.h"
  27. #include "pinconfig.h"
  28. SPI_HandleTypeDef TFT_SPI::SPIx;
  29. DMA_HandleTypeDef TFT_SPI::DMAtx;
  30. void TFT_SPI::Init() {
  31. SPI_TypeDef *spiInstance;
  32. OUT_WRITE(TFT_A0_PIN, HIGH);
  33. OUT_WRITE(TFT_CS_PIN, HIGH);
  34. if ((spiInstance = (SPI_TypeDef *)pinmap_peripheral(digitalPinToPinName(TFT_SCK_PIN), PinMap_SPI_SCLK)) == NP) return;
  35. if (spiInstance != (SPI_TypeDef *)pinmap_peripheral(digitalPinToPinName(TFT_MOSI_PIN), PinMap_SPI_MOSI)) return;
  36. #if PIN_EXISTS(TFT_MISO) && TFT_MISO_PIN != TFT_MOSI_PIN
  37. if (spiInstance != (SPI_TypeDef *)pinmap_peripheral(digitalPinToPinName(TFT_MISO_PIN), PinMap_SPI_MISO)) return;
  38. #endif
  39. SPIx.Instance = spiInstance;
  40. SPIx.State = HAL_SPI_STATE_RESET;
  41. SPIx.Init.NSS = SPI_NSS_SOFT;
  42. SPIx.Init.Mode = SPI_MODE_MASTER;
  43. SPIx.Init.Direction = (TFT_MISO_PIN == TFT_MOSI_PIN) ? SPI_DIRECTION_1LINE : SPI_DIRECTION_2LINES;
  44. SPIx.Init.BaudRatePrescaler = SPI_BAUDRATEPRESCALER_2;
  45. SPIx.Init.CLKPhase = SPI_PHASE_1EDGE;
  46. SPIx.Init.CLKPolarity = SPI_POLARITY_LOW;
  47. SPIx.Init.DataSize = SPI_DATASIZE_8BIT;
  48. SPIx.Init.FirstBit = SPI_FIRSTBIT_MSB;
  49. SPIx.Init.TIMode = SPI_TIMODE_DISABLE;
  50. SPIx.Init.CRCCalculation = SPI_CRCCALCULATION_DISABLE;
  51. SPIx.Init.CRCPolynomial = 10;
  52. pinmap_pinout(digitalPinToPinName(TFT_SCK_PIN), PinMap_SPI_SCLK);
  53. pinmap_pinout(digitalPinToPinName(TFT_MOSI_PIN), PinMap_SPI_MOSI);
  54. #if PIN_EXISTS(TFT_MISO) && TFT_MISO_PIN != TFT_MOSI_PIN
  55. pinmap_pinout(digitalPinToPinName(TFT_MISO_PIN), PinMap_SPI_MISO);
  56. #endif
  57. pin_PullConfig(get_GPIO_Port(STM_PORT(digitalPinToPinName(TFT_SCK_PIN))), STM_LL_GPIO_PIN(digitalPinToPinName(TFT_SCK_PIN)), GPIO_PULLDOWN);
  58. #ifdef SPI1_BASE
  59. if (SPIx.Instance == SPI1) {
  60. __HAL_RCC_SPI1_CLK_ENABLE();
  61. #ifdef STM32F1xx
  62. __HAL_RCC_DMA1_CLK_ENABLE();
  63. DMAtx.Instance = DMA1_Channel3;
  64. #elif defined(STM32F4xx)
  65. __HAL_RCC_DMA2_CLK_ENABLE();
  66. DMAtx.Instance = DMA2_Stream3;
  67. DMAtx.Init.Channel = DMA_CHANNEL_3;
  68. #endif
  69. SPIx.Init.BaudRatePrescaler = SPI_BAUDRATEPRESCALER_4;
  70. }
  71. #endif
  72. #ifdef SPI2_BASE
  73. if (SPIx.Instance == SPI2) {
  74. __HAL_RCC_SPI2_CLK_ENABLE();
  75. #ifdef STM32F1xx
  76. __HAL_RCC_DMA1_CLK_ENABLE();
  77. DMAtx.Instance = DMA1_Channel5;
  78. #elif defined(STM32F4xx)
  79. __HAL_RCC_DMA1_CLK_ENABLE();
  80. DMAtx.Instance = DMA1_Stream4;
  81. DMAtx.Init.Channel = DMA_CHANNEL_0;
  82. #endif
  83. }
  84. #endif
  85. #ifdef SPI3_BASE
  86. if (SPIx.Instance == SPI3) {
  87. __HAL_RCC_SPI3_CLK_ENABLE();
  88. #ifdef STM32F1xx
  89. __HAL_RCC_DMA2_CLK_ENABLE();
  90. DMAtx.Instance = DMA2_Channel2;
  91. #elif defined(STM32F4xx)
  92. __HAL_RCC_DMA1_CLK_ENABLE();
  93. DMAtx.Instance = DMA1_Stream5;
  94. DMAtx.Init.Channel = DMA_CHANNEL_0;
  95. #endif
  96. }
  97. #endif
  98. HAL_SPI_Init(&SPIx);
  99. DMAtx.Init.Direction = DMA_MEMORY_TO_PERIPH;
  100. DMAtx.Init.PeriphInc = DMA_PINC_DISABLE;
  101. DMAtx.Init.PeriphDataAlignment = DMA_PDATAALIGN_HALFWORD;
  102. DMAtx.Init.MemDataAlignment = DMA_MDATAALIGN_HALFWORD;
  103. DMAtx.Init.Mode = DMA_NORMAL;
  104. DMAtx.Init.Priority = DMA_PRIORITY_LOW;
  105. #ifdef STM32F4xx
  106. DMAtx.Init.FIFOMode = DMA_FIFOMODE_DISABLE;
  107. #endif
  108. }
  109. void TFT_SPI::DataTransferBegin(uint16_t DataSize) {
  110. SPIx.Init.DataSize = DataSize == DATASIZE_8BIT ? SPI_DATASIZE_8BIT : SPI_DATASIZE_16BIT;
  111. HAL_SPI_Init(&SPIx);
  112. WRITE(TFT_CS_PIN, LOW);
  113. }
  114. #ifdef TFT_DEFAULT_DRIVER
  115. #include "../../../lcd/tft_io/tft_ids.h"
  116. #endif
  117. uint32_t TFT_SPI::GetID() {
  118. uint32_t id;
  119. id = ReadID(LCD_READ_ID);
  120. if ((id & 0xFFFF) == 0 || (id & 0xFFFF) == 0xFFFF) {
  121. id = ReadID(LCD_READ_ID4);
  122. #ifdef TFT_DEFAULT_DRIVER
  123. if ((id & 0xFFFF) == 0 || (id & 0xFFFF) == 0xFFFF)
  124. id = TFT_DEFAULT_DRIVER;
  125. #endif
  126. }
  127. return id;
  128. }
  129. uint32_t TFT_SPI::ReadID(uint16_t Reg) {
  130. uint32_t Data = 0;
  131. #if PIN_EXISTS(TFT_MISO)
  132. uint32_t BaudRatePrescaler = SPIx.Init.BaudRatePrescaler;
  133. uint32_t i;
  134. SPIx.Init.BaudRatePrescaler = SPIx.Instance == SPI1 ? SPI_BAUDRATEPRESCALER_8 : SPI_BAUDRATEPRESCALER_4;
  135. DataTransferBegin(DATASIZE_8BIT);
  136. WriteReg(Reg);
  137. if (SPIx.Init.Direction == SPI_DIRECTION_1LINE) SPI_1LINE_RX(&SPIx);
  138. __HAL_SPI_ENABLE(&SPIx);
  139. for (i = 0; i < 4; i++) {
  140. #if TFT_MISO_PIN != TFT_MOSI_PIN
  141. //if (hspi->Init.Direction == SPI_DIRECTION_2LINES) {
  142. while (!__HAL_SPI_GET_FLAG(&SPIx, SPI_FLAG_TXE)) {}
  143. SPIx.Instance->DR = 0;
  144. //}
  145. #endif
  146. while (!__HAL_SPI_GET_FLAG(&SPIx, SPI_FLAG_RXNE)) {}
  147. Data = (Data << 8) | SPIx.Instance->DR;
  148. }
  149. __HAL_SPI_DISABLE(&SPIx);
  150. DataTransferEnd();
  151. SPIx.Init.BaudRatePrescaler = BaudRatePrescaler;
  152. #endif
  153. return Data >> 7;
  154. }
  155. bool TFT_SPI::isBusy() {
  156. #ifdef STM32F1xx
  157. volatile bool dmaEnabled = (DMAtx.Instance->CCR & DMA_CCR_EN) != RESET;
  158. #elif defined(STM32F4xx)
  159. volatile bool dmaEnabled = DMAtx.Instance->CR & DMA_SxCR_EN;
  160. #endif
  161. if (dmaEnabled) {
  162. if (__HAL_DMA_GET_FLAG(&DMAtx, __HAL_DMA_GET_TC_FLAG_INDEX(&DMAtx)) != 0 || __HAL_DMA_GET_FLAG(&DMAtx, __HAL_DMA_GET_TE_FLAG_INDEX(&DMAtx)) != 0)
  163. Abort();
  164. }
  165. else
  166. Abort();
  167. return dmaEnabled;
  168. }
  169. void TFT_SPI::Abort() {
  170. // Wait for any running spi
  171. while (!__HAL_SPI_GET_FLAG(&SPIx, SPI_FLAG_TXE)) {}
  172. while ( __HAL_SPI_GET_FLAG(&SPIx, SPI_FLAG_BSY)) {}
  173. // First, abort any running dma
  174. HAL_DMA_Abort(&DMAtx);
  175. // DeInit objects
  176. HAL_DMA_DeInit(&DMAtx);
  177. HAL_SPI_DeInit(&SPIx);
  178. // Deselect CS
  179. DataTransferEnd();
  180. }
  181. void TFT_SPI::Transmit(uint16_t Data) {
  182. if (TFT_MISO_PIN == TFT_MOSI_PIN)
  183. SPI_1LINE_TX(&SPIx);
  184. __HAL_SPI_ENABLE(&SPIx);
  185. SPIx.Instance->DR = Data;
  186. while (!__HAL_SPI_GET_FLAG(&SPIx, SPI_FLAG_TXE)) {}
  187. while ( __HAL_SPI_GET_FLAG(&SPIx, SPI_FLAG_BSY)) {}
  188. if (TFT_MISO_PIN != TFT_MOSI_PIN)
  189. __HAL_SPI_CLEAR_OVRFLAG(&SPIx); // Clear overrun flag in 2 Lines communication mode because received is not read
  190. }
  191. void TFT_SPI::TransmitDMA(uint32_t MemoryIncrease, uint16_t *Data, uint16_t Count) {
  192. // Wait last dma finish, to start another
  193. while (isBusy()) { /* nada */ }
  194. DMAtx.Init.MemInc = MemoryIncrease;
  195. HAL_DMA_Init(&DMAtx);
  196. if (TFT_MISO_PIN == TFT_MOSI_PIN)
  197. SPI_1LINE_TX(&SPIx);
  198. DataTransferBegin();
  199. HAL_DMA_Start(&DMAtx, (uint32_t)Data, (uint32_t)&(SPIx.Instance->DR), Count);
  200. __HAL_SPI_ENABLE(&SPIx);
  201. SET_BIT(SPIx.Instance->CR2, SPI_CR2_TXDMAEN); // Enable Tx DMA Request
  202. HAL_DMA_PollForTransfer(&DMAtx, HAL_DMA_FULL_TRANSFER, HAL_MAX_DELAY);
  203. Abort();
  204. }
  205. #if ENABLED(USE_SPI_DMA_TC)
  206. void TFT_SPI::TransmitDMA_IT(uint32_t MemoryIncrease, uint16_t *Data, uint16_t Count) {
  207. DMAtx.Init.MemInc = MemoryIncrease;
  208. HAL_DMA_Init(&DMAtx);
  209. if (TFT_MISO_PIN == TFT_MOSI_PIN)
  210. SPI_1LINE_TX(&SPIx);
  211. DataTransferBegin();
  212. HAL_NVIC_SetPriority(DMA2_Stream3_IRQn, 5, 0);
  213. HAL_NVIC_EnableIRQ(DMA2_Stream3_IRQn);
  214. HAL_DMA_Start_IT(&DMAtx, (uint32_t)Data, (uint32_t)&(SPIx.Instance->DR), Count);
  215. __HAL_SPI_ENABLE(&SPIx);
  216. SET_BIT(SPIx.Instance->CR2, SPI_CR2_TXDMAEN); // Enable Tx DMA Request
  217. }
  218. extern "C" void DMA2_Stream3_IRQHandler(void) { HAL_DMA_IRQHandler(&TFT_SPI::DMAtx); }
  219. #endif
  220. #endif // HAS_SPI_TFT
  221. #endif // HAL_STM32