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_fsmc.cpp 7.4KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237
  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 "../../../inc/MarlinConfig.h"
  23. #if HAS_FSMC_TFT
  24. #include "tft_fsmc.h"
  25. #include <libmaple/fsmc.h>
  26. #include <libmaple/gpio.h>
  27. #include <libmaple/dma.h>
  28. LCD_CONTROLLER_TypeDef *TFT_FSMC::LCD;
  29. /**
  30. * FSMC LCD IO
  31. */
  32. #define __ASM __asm
  33. #define __STATIC_INLINE static inline
  34. __attribute__((always_inline)) __STATIC_INLINE void __DSB() {
  35. __ASM volatile ("dsb 0xF":::"memory");
  36. }
  37. #define FSMC_CS_NE1 PD7
  38. #if ENABLED(STM32_XL_DENSITY)
  39. #define FSMC_CS_NE2 PG9
  40. #define FSMC_CS_NE3 PG10
  41. #define FSMC_CS_NE4 PG12
  42. #define FSMC_RS_A0 PF0
  43. #define FSMC_RS_A1 PF1
  44. #define FSMC_RS_A2 PF2
  45. #define FSMC_RS_A3 PF3
  46. #define FSMC_RS_A4 PF4
  47. #define FSMC_RS_A5 PF5
  48. #define FSMC_RS_A6 PF12
  49. #define FSMC_RS_A7 PF13
  50. #define FSMC_RS_A8 PF14
  51. #define FSMC_RS_A9 PF15
  52. #define FSMC_RS_A10 PG0
  53. #define FSMC_RS_A11 PG1
  54. #define FSMC_RS_A12 PG2
  55. #define FSMC_RS_A13 PG3
  56. #define FSMC_RS_A14 PG4
  57. #define FSMC_RS_A15 PG5
  58. #endif
  59. #define FSMC_RS_A16 PD11
  60. #define FSMC_RS_A17 PD12
  61. #define FSMC_RS_A18 PD13
  62. #define FSMC_RS_A19 PE3
  63. #define FSMC_RS_A20 PE4
  64. #define FSMC_RS_A21 PE5
  65. #define FSMC_RS_A22 PE6
  66. #define FSMC_RS_A23 PE2
  67. #if ENABLED(STM32_XL_DENSITY)
  68. #define FSMC_RS_A24 PG13
  69. #define FSMC_RS_A25 PG14
  70. #endif
  71. /* Timing configuration */
  72. #define FSMC_ADDRESS_SETUP_TIME 15 // AddressSetupTime
  73. #define FSMC_DATA_SETUP_TIME 15 // DataSetupTime
  74. static uint8_t fsmcInit = 0;
  75. void TFT_FSMC::Init() {
  76. uint8_t cs = FSMC_CS_PIN, rs = FSMC_RS_PIN;
  77. uint32_t controllerAddress;
  78. #if ENABLED(LCD_USE_DMA_FSMC)
  79. dma_init(FSMC_DMA_DEV);
  80. dma_disable(FSMC_DMA_DEV, FSMC_DMA_CHANNEL);
  81. dma_set_priority(FSMC_DMA_DEV, FSMC_DMA_CHANNEL, DMA_PRIORITY_MEDIUM);
  82. #endif
  83. struct fsmc_nor_psram_reg_map* fsmcPsramRegion;
  84. if (fsmcInit) return;
  85. fsmcInit = 1;
  86. switch (cs) {
  87. case FSMC_CS_NE1: controllerAddress = (uint32_t)FSMC_NOR_PSRAM_REGION1; fsmcPsramRegion = FSMC_NOR_PSRAM1_BASE; break;
  88. #if ENABLED(STM32_XL_DENSITY)
  89. case FSMC_CS_NE2: controllerAddress = (uint32_t)FSMC_NOR_PSRAM_REGION2; fsmcPsramRegion = FSMC_NOR_PSRAM2_BASE; break;
  90. case FSMC_CS_NE3: controllerAddress = (uint32_t)FSMC_NOR_PSRAM_REGION3; fsmcPsramRegion = FSMC_NOR_PSRAM3_BASE; break;
  91. case FSMC_CS_NE4: controllerAddress = (uint32_t)FSMC_NOR_PSRAM_REGION4; fsmcPsramRegion = FSMC_NOR_PSRAM4_BASE; break;
  92. #endif
  93. default: return;
  94. }
  95. #define _ORADDR(N) controllerAddress |= (_BV32(N) - 2)
  96. switch (rs) {
  97. #if ENABLED(STM32_XL_DENSITY)
  98. case FSMC_RS_A0: _ORADDR( 1); break;
  99. case FSMC_RS_A1: _ORADDR( 2); break;
  100. case FSMC_RS_A2: _ORADDR( 3); break;
  101. case FSMC_RS_A3: _ORADDR( 4); break;
  102. case FSMC_RS_A4: _ORADDR( 5); break;
  103. case FSMC_RS_A5: _ORADDR( 6); break;
  104. case FSMC_RS_A6: _ORADDR( 7); break;
  105. case FSMC_RS_A7: _ORADDR( 8); break;
  106. case FSMC_RS_A8: _ORADDR( 9); break;
  107. case FSMC_RS_A9: _ORADDR(10); break;
  108. case FSMC_RS_A10: _ORADDR(11); break;
  109. case FSMC_RS_A11: _ORADDR(12); break;
  110. case FSMC_RS_A12: _ORADDR(13); break;
  111. case FSMC_RS_A13: _ORADDR(14); break;
  112. case FSMC_RS_A14: _ORADDR(15); break;
  113. case FSMC_RS_A15: _ORADDR(16); break;
  114. #endif
  115. case FSMC_RS_A16: _ORADDR(17); break;
  116. case FSMC_RS_A17: _ORADDR(18); break;
  117. case FSMC_RS_A18: _ORADDR(19); break;
  118. case FSMC_RS_A19: _ORADDR(20); break;
  119. case FSMC_RS_A20: _ORADDR(21); break;
  120. case FSMC_RS_A21: _ORADDR(22); break;
  121. case FSMC_RS_A22: _ORADDR(23); break;
  122. case FSMC_RS_A23: _ORADDR(24); break;
  123. #if ENABLED(STM32_XL_DENSITY)
  124. case FSMC_RS_A24: _ORADDR(25); break;
  125. case FSMC_RS_A25: _ORADDR(26); break;
  126. #endif
  127. default: return;
  128. }
  129. rcc_clk_enable(RCC_FSMC);
  130. gpio_set_mode(GPIOD, 14, GPIO_AF_OUTPUT_PP); // FSMC_D00
  131. gpio_set_mode(GPIOD, 15, GPIO_AF_OUTPUT_PP); // FSMC_D01
  132. gpio_set_mode(GPIOD, 0, GPIO_AF_OUTPUT_PP); // FSMC_D02
  133. gpio_set_mode(GPIOD, 1, GPIO_AF_OUTPUT_PP); // FSMC_D03
  134. gpio_set_mode(GPIOE, 7, GPIO_AF_OUTPUT_PP); // FSMC_D04
  135. gpio_set_mode(GPIOE, 8, GPIO_AF_OUTPUT_PP); // FSMC_D05
  136. gpio_set_mode(GPIOE, 9, GPIO_AF_OUTPUT_PP); // FSMC_D06
  137. gpio_set_mode(GPIOE, 10, GPIO_AF_OUTPUT_PP); // FSMC_D07
  138. gpio_set_mode(GPIOE, 11, GPIO_AF_OUTPUT_PP); // FSMC_D08
  139. gpio_set_mode(GPIOE, 12, GPIO_AF_OUTPUT_PP); // FSMC_D09
  140. gpio_set_mode(GPIOE, 13, GPIO_AF_OUTPUT_PP); // FSMC_D10
  141. gpio_set_mode(GPIOE, 14, GPIO_AF_OUTPUT_PP); // FSMC_D11
  142. gpio_set_mode(GPIOE, 15, GPIO_AF_OUTPUT_PP); // FSMC_D12
  143. gpio_set_mode(GPIOD, 8, GPIO_AF_OUTPUT_PP); // FSMC_D13
  144. gpio_set_mode(GPIOD, 9, GPIO_AF_OUTPUT_PP); // FSMC_D14
  145. gpio_set_mode(GPIOD, 10, GPIO_AF_OUTPUT_PP); // FSMC_D15
  146. gpio_set_mode(GPIOD, 4, GPIO_AF_OUTPUT_PP); // FSMC_NOE
  147. gpio_set_mode(GPIOD, 5, GPIO_AF_OUTPUT_PP); // FSMC_NWE
  148. gpio_set_mode(PIN_MAP[cs].gpio_device, PIN_MAP[cs].gpio_bit, GPIO_AF_OUTPUT_PP); //FSMC_CS_NEx
  149. gpio_set_mode(PIN_MAP[rs].gpio_device, PIN_MAP[rs].gpio_bit, GPIO_AF_OUTPUT_PP); //FSMC_RS_Ax
  150. fsmcPsramRegion->BCR = FSMC_BCR_WREN | FSMC_BCR_MTYP_SRAM | FSMC_BCR_MWID_16BITS | FSMC_BCR_MBKEN;
  151. fsmcPsramRegion->BTR = (FSMC_DATA_SETUP_TIME << 8) | FSMC_ADDRESS_SETUP_TIME;
  152. afio_remap(AFIO_REMAP_FSMC_NADV);
  153. LCD = (LCD_CONTROLLER_TypeDef*)controllerAddress;
  154. }
  155. void TFT_FSMC::Transmit(uint16_t Data) {
  156. LCD->RAM = Data;
  157. __DSB();
  158. }
  159. void TFT_FSMC::WriteReg(uint16_t Reg) {
  160. LCD->REG = Reg;
  161. __DSB();
  162. }
  163. uint32_t TFT_FSMC::GetID() {
  164. uint32_t id;
  165. WriteReg(0x0000);
  166. id = LCD->RAM;
  167. if (id == 0)
  168. id = ReadID(LCD_READ_ID);
  169. if ((id & 0xFFFF) == 0 || (id & 0xFFFF) == 0xFFFF)
  170. id = ReadID(LCD_READ_ID4);
  171. if ((id & 0xFF00) == 0 && (id & 0xFF) != 0)
  172. id = ReadID(LCD_READ_ID4);
  173. return id;
  174. }
  175. uint32_t TFT_FSMC::ReadID(uint16_t Reg) {
  176. uint32_t id;
  177. WriteReg(Reg);
  178. id = LCD->RAM; // dummy read
  179. id = Reg << 24;
  180. id |= (LCD->RAM & 0x00FF) << 16;
  181. id |= (LCD->RAM & 0x00FF) << 8;
  182. id |= LCD->RAM & 0x00FF;
  183. return id;
  184. }
  185. bool TFT_FSMC::isBusy() {
  186. return false;
  187. }
  188. void TFT_FSMC::Abort() {
  189. }
  190. void TFT_FSMC::TransmitDMA(uint32_t MemoryIncrease, uint16_t *Data, uint16_t Count) {
  191. #if defined(FSMC_DMA_DEV) && defined(FSMC_DMA_CHANNEL)
  192. dma_setup_transfer(FSMC_DMA_DEV, FSMC_DMA_CHANNEL, Data, DMA_SIZE_16BITS, &LCD->RAM, DMA_SIZE_16BITS, DMA_MEM_2_MEM | MemoryIncrease);
  193. dma_set_num_transfers(FSMC_DMA_DEV, FSMC_DMA_CHANNEL, Count);
  194. dma_clear_isr_bits(FSMC_DMA_DEV, FSMC_DMA_CHANNEL);
  195. dma_enable(FSMC_DMA_DEV, FSMC_DMA_CHANNEL);
  196. while ((dma_get_isr_bits(FSMC_DMA_DEV, FSMC_DMA_CHANNEL) & 0x0A) == 0) {};
  197. dma_disable(FSMC_DMA_DEV, FSMC_DMA_CHANNEL);
  198. #endif
  199. }
  200. #endif // HAS_FSMC_TFT