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.

u8g_com_stm32duino_fsmc.cpp 10KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335
  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 <http://www.gnu.org/licenses/>.
  20. *
  21. */
  22. /**
  23. * u8g_com_stm32duino_fsmc.cpp
  24. *
  25. * Communication interface for FSMC
  26. */
  27. #include "../../../inc/MarlinConfig.h"
  28. #if defined(ARDUINO_ARCH_STM32F1) && PIN_EXISTS(FSMC_CS) // FSMC on 100/144 pins SoCs
  29. #if HAS_GRAPHICAL_LCD
  30. #include <U8glib.h>
  31. #include <libmaple/fsmc.h>
  32. #include <libmaple/gpio.h>
  33. #include <libmaple/dma.h>
  34. #include <boards.h>
  35. #ifndef LCD_READ_ID
  36. #define LCD_READ_ID 0x04 // Read display identification information (0xD3 on ILI9341)
  37. #endif
  38. /* Timing configuration */
  39. #define FSMC_ADDRESS_SETUP_TIME 15 // AddressSetupTime
  40. #define FSMC_DATA_SETUP_TIME 15 // DataSetupTime
  41. void LCD_IO_Init(uint8_t cs, uint8_t rs);
  42. void LCD_IO_WriteData(uint16_t RegValue);
  43. void LCD_IO_WriteReg(uint16_t Reg);
  44. uint16_t LCD_IO_ReadData(uint16_t RegValue);
  45. uint32_t LCD_IO_ReadData(uint16_t RegValue, uint8_t ReadSize);
  46. #ifdef LCD_USE_DMA_FSMC
  47. void LCD_IO_WriteMultiple(uint16_t data, uint32_t count);
  48. void LCD_IO_WriteSequence(uint16_t *data, uint16_t length);
  49. #endif
  50. static uint8_t msgInitCount = 2; // Ignore all messages until 2nd U8G_COM_MSG_INIT
  51. uint8_t u8g_com_stm32duino_fsmc_fn(u8g_t *u8g, uint8_t msg, uint8_t arg_val, void *arg_ptr) {
  52. if (msgInitCount) {
  53. if (msg == U8G_COM_MSG_INIT) msgInitCount--;
  54. if (msgInitCount) return -1;
  55. }
  56. static uint8_t isCommand;
  57. switch (msg) {
  58. case U8G_COM_MSG_STOP: break;
  59. case U8G_COM_MSG_INIT:
  60. u8g_SetPIOutput(u8g, U8G_PI_RESET);
  61. #ifdef LCD_USE_DMA_FSMC
  62. dma_init(FSMC_DMA_DEV);
  63. dma_disable(FSMC_DMA_DEV, FSMC_DMA_CHANNEL);
  64. dma_set_priority(FSMC_DMA_DEV, FSMC_DMA_CHANNEL, DMA_PRIORITY_MEDIUM);
  65. #endif
  66. LCD_IO_Init(u8g->pin_list[U8G_PI_CS], u8g->pin_list[U8G_PI_A0]);
  67. u8g_Delay(50);
  68. if (arg_ptr) {
  69. *((uint32_t *)arg_ptr) = LCD_IO_ReadData(0x0000);
  70. if (*((uint32_t *)arg_ptr) == 0)
  71. *((uint32_t *)arg_ptr) = (LCD_READ_ID << 24) | LCD_IO_ReadData(LCD_READ_ID, 3);
  72. }
  73. isCommand = 0;
  74. break;
  75. case U8G_COM_MSG_ADDRESS: // define cmd (arg_val = 0) or data mode (arg_val = 1)
  76. isCommand = arg_val == 0 ? 1 : 0;
  77. break;
  78. case U8G_COM_MSG_RESET:
  79. u8g_SetPILevel(u8g, U8G_PI_RESET, arg_val);
  80. break;
  81. case U8G_COM_MSG_WRITE_BYTE:
  82. if (isCommand)
  83. LCD_IO_WriteReg(arg_val);
  84. else
  85. LCD_IO_WriteData((uint16_t)arg_val);
  86. break;
  87. case U8G_COM_MSG_WRITE_SEQ:
  88. for (uint8_t i = 0; i < arg_val; i += 2)
  89. LCD_IO_WriteData(*(uint16_t *)(((uint32_t)arg_ptr) + i));
  90. break;
  91. }
  92. return 1;
  93. }
  94. /**
  95. * FSMC LCD IO
  96. */
  97. #define __ASM __asm
  98. #define __STATIC_INLINE static inline
  99. __attribute__((always_inline)) __STATIC_INLINE void __DSB() {
  100. __ASM volatile ("dsb 0xF":::"memory");
  101. }
  102. #define FSMC_CS_NE1 PD7
  103. #if ENABLED(STM32_XL_DENSITY)
  104. #define FSMC_CS_NE2 PG9
  105. #define FSMC_CS_NE3 PG10
  106. #define FSMC_CS_NE4 PG12
  107. #define FSMC_RS_A0 PF0
  108. #define FSMC_RS_A1 PF1
  109. #define FSMC_RS_A2 PF2
  110. #define FSMC_RS_A3 PF3
  111. #define FSMC_RS_A4 PF4
  112. #define FSMC_RS_A5 PF5
  113. #define FSMC_RS_A6 PF12
  114. #define FSMC_RS_A7 PF13
  115. #define FSMC_RS_A8 PF14
  116. #define FSMC_RS_A9 PF15
  117. #define FSMC_RS_A10 PG0
  118. #define FSMC_RS_A11 PG1
  119. #define FSMC_RS_A12 PG2
  120. #define FSMC_RS_A13 PG3
  121. #define FSMC_RS_A14 PG4
  122. #define FSMC_RS_A15 PG5
  123. #endif
  124. #define FSMC_RS_A16 PD11
  125. #define FSMC_RS_A17 PD12
  126. #define FSMC_RS_A18 PD13
  127. #define FSMC_RS_A19 PE3
  128. #define FSMC_RS_A20 PE4
  129. #define FSMC_RS_A21 PE5
  130. #define FSMC_RS_A22 PE6
  131. #define FSMC_RS_A23 PE2
  132. #if ENABLED(STM32_XL_DENSITY)
  133. #define FSMC_RS_A24 PG13
  134. #define FSMC_RS_A25 PG14
  135. #endif
  136. static uint8_t fsmcInit = 0;
  137. typedef struct {
  138. __IO uint16_t REG;
  139. __IO uint16_t RAM;
  140. } LCD_CONTROLLER_TypeDef;
  141. LCD_CONTROLLER_TypeDef *LCD;
  142. void LCD_IO_Init(uint8_t cs, uint8_t rs) {
  143. uint32_t controllerAddress;
  144. if (fsmcInit) return;
  145. fsmcInit = 1;
  146. switch (cs) {
  147. case FSMC_CS_NE1: controllerAddress = (uint32_t)FSMC_NOR_PSRAM_REGION1; break;
  148. #if ENABLED(STM32_XL_DENSITY)
  149. case FSMC_CS_NE2: controllerAddress = (uint32_t)FSMC_NOR_PSRAM_REGION2; break;
  150. case FSMC_CS_NE3: controllerAddress = (uint32_t)FSMC_NOR_PSRAM_REGION3; break;
  151. case FSMC_CS_NE4: controllerAddress = (uint32_t)FSMC_NOR_PSRAM_REGION4; break;
  152. #endif
  153. default: return;
  154. }
  155. #define _ORADDR(N) controllerAddress |= (_BV32(N) - 2)
  156. switch (rs) {
  157. #if ENABLED(STM32_XL_DENSITY)
  158. case FSMC_RS_A0: _ORADDR( 1); break;
  159. case FSMC_RS_A1: _ORADDR( 2); break;
  160. case FSMC_RS_A2: _ORADDR( 3); break;
  161. case FSMC_RS_A3: _ORADDR( 4); break;
  162. case FSMC_RS_A4: _ORADDR( 5); break;
  163. case FSMC_RS_A5: _ORADDR( 6); break;
  164. case FSMC_RS_A6: _ORADDR( 7); break;
  165. case FSMC_RS_A7: _ORADDR( 8); break;
  166. case FSMC_RS_A8: _ORADDR( 9); break;
  167. case FSMC_RS_A9: _ORADDR(10); break;
  168. case FSMC_RS_A10: _ORADDR(11); break;
  169. case FSMC_RS_A11: _ORADDR(12); break;
  170. case FSMC_RS_A12: _ORADDR(13); break;
  171. case FSMC_RS_A13: _ORADDR(14); break;
  172. case FSMC_RS_A14: _ORADDR(15); break;
  173. case FSMC_RS_A15: _ORADDR(16); break;
  174. #endif
  175. case FSMC_RS_A16: _ORADDR(17); break;
  176. case FSMC_RS_A17: _ORADDR(18); break;
  177. case FSMC_RS_A18: _ORADDR(19); break;
  178. case FSMC_RS_A19: _ORADDR(20); break;
  179. case FSMC_RS_A20: _ORADDR(21); break;
  180. case FSMC_RS_A21: _ORADDR(22); break;
  181. case FSMC_RS_A22: _ORADDR(23); break;
  182. case FSMC_RS_A23: _ORADDR(24); break;
  183. #if ENABLED(STM32_XL_DENSITY)
  184. case FSMC_RS_A24: _ORADDR(25); break;
  185. case FSMC_RS_A25: _ORADDR(26); break;
  186. #endif
  187. default: return;
  188. }
  189. rcc_clk_enable(RCC_FSMC);
  190. gpio_set_mode(GPIOD, 14, GPIO_AF_OUTPUT_PP); // FSMC_D00
  191. gpio_set_mode(GPIOD, 15, GPIO_AF_OUTPUT_PP); // FSMC_D01
  192. gpio_set_mode(GPIOD, 0, GPIO_AF_OUTPUT_PP); // FSMC_D02
  193. gpio_set_mode(GPIOD, 1, GPIO_AF_OUTPUT_PP); // FSMC_D03
  194. gpio_set_mode(GPIOE, 7, GPIO_AF_OUTPUT_PP); // FSMC_D04
  195. gpio_set_mode(GPIOE, 8, GPIO_AF_OUTPUT_PP); // FSMC_D05
  196. gpio_set_mode(GPIOE, 9, GPIO_AF_OUTPUT_PP); // FSMC_D06
  197. gpio_set_mode(GPIOE, 10, GPIO_AF_OUTPUT_PP); // FSMC_D07
  198. gpio_set_mode(GPIOE, 11, GPIO_AF_OUTPUT_PP); // FSMC_D08
  199. gpio_set_mode(GPIOE, 12, GPIO_AF_OUTPUT_PP); // FSMC_D09
  200. gpio_set_mode(GPIOE, 13, GPIO_AF_OUTPUT_PP); // FSMC_D10
  201. gpio_set_mode(GPIOE, 14, GPIO_AF_OUTPUT_PP); // FSMC_D11
  202. gpio_set_mode(GPIOE, 15, GPIO_AF_OUTPUT_PP); // FSMC_D12
  203. gpio_set_mode(GPIOD, 8, GPIO_AF_OUTPUT_PP); // FSMC_D13
  204. gpio_set_mode(GPIOD, 9, GPIO_AF_OUTPUT_PP); // FSMC_D14
  205. gpio_set_mode(GPIOD, 10, GPIO_AF_OUTPUT_PP); // FSMC_D15
  206. gpio_set_mode(GPIOD, 4, GPIO_AF_OUTPUT_PP); // FSMC_NOE
  207. gpio_set_mode(GPIOD, 5, GPIO_AF_OUTPUT_PP); // FSMC_NWE
  208. gpio_set_mode(PIN_MAP[cs].gpio_device, PIN_MAP[cs].gpio_bit, GPIO_AF_OUTPUT_PP); //FSMC_CS_NEx
  209. gpio_set_mode(PIN_MAP[rs].gpio_device, PIN_MAP[rs].gpio_bit, GPIO_AF_OUTPUT_PP); //FSMC_RS_Ax
  210. #if ENABLED(STM32_XL_DENSITY)
  211. FSMC_NOR_PSRAM4_BASE->BCR = FSMC_BCR_WREN | FSMC_BCR_MTYP_SRAM | FSMC_BCR_MWID_16BITS | FSMC_BCR_MBKEN;
  212. FSMC_NOR_PSRAM4_BASE->BTR = (FSMC_DATA_SETUP_TIME << 8) | FSMC_ADDRESS_SETUP_TIME;
  213. #else // PSRAM1 for STM32F103V (high density)
  214. FSMC_NOR_PSRAM1_BASE->BCR = FSMC_BCR_WREN | FSMC_BCR_MTYP_SRAM | FSMC_BCR_MWID_16BITS | FSMC_BCR_MBKEN;
  215. FSMC_NOR_PSRAM1_BASE->BTR = (FSMC_DATA_SETUP_TIME << 8) | FSMC_ADDRESS_SETUP_TIME;
  216. #endif
  217. afio_remap(AFIO_REMAP_FSMC_NADV);
  218. LCD = (LCD_CONTROLLER_TypeDef*)controllerAddress;
  219. }
  220. void LCD_IO_WriteData(uint16_t RegValue) {
  221. LCD->RAM = RegValue;
  222. __DSB();
  223. }
  224. void LCD_IO_WriteReg(uint16_t Reg) {
  225. LCD->REG = Reg;
  226. __DSB();
  227. }
  228. uint16_t LCD_IO_ReadData(uint16_t RegValue) {
  229. LCD->REG = RegValue;
  230. __DSB();
  231. return LCD->RAM;
  232. }
  233. uint32_t LCD_IO_ReadData(uint16_t RegValue, uint8_t ReadSize) {
  234. volatile uint32_t data;
  235. LCD->REG = RegValue;
  236. __DSB();
  237. data = LCD->RAM; // dummy read
  238. data = LCD->RAM & 0x00FF;
  239. while (--ReadSize) {
  240. data <<= 8;
  241. data |= (LCD->RAM & 0x00FF);
  242. }
  243. return uint32_t(data);
  244. }
  245. #if ENABLED(LCD_USE_DMA_FSMC)
  246. void LCD_IO_WriteMultiple(uint16_t color, uint32_t count) {
  247. while (count > 0) {
  248. dma_setup_transfer(FSMC_DMA_DEV, FSMC_DMA_CHANNEL, &color, DMA_SIZE_16BITS, &LCD->RAM, DMA_SIZE_16BITS, DMA_MEM_2_MEM);
  249. dma_set_num_transfers(FSMC_DMA_DEV, FSMC_DMA_CHANNEL, count > 65535 ? 65535 : count);
  250. dma_clear_isr_bits(FSMC_DMA_DEV, FSMC_DMA_CHANNEL);
  251. dma_enable(FSMC_DMA_DEV, FSMC_DMA_CHANNEL);
  252. while ((dma_get_isr_bits(FSMC_DMA_DEV, FSMC_DMA_CHANNEL) & 0x0A) == 0) {};
  253. dma_disable(FSMC_DMA_DEV, FSMC_DMA_CHANNEL);
  254. count = count > 65535 ? count - 65535 : 0;
  255. }
  256. }
  257. void LCD_IO_WriteSequence(uint16_t *data, uint16_t length) {
  258. dma_setup_transfer(FSMC_DMA_DEV, FSMC_DMA_CHANNEL, data, DMA_SIZE_16BITS, &LCD->RAM, DMA_SIZE_16BITS, DMA_MEM_2_MEM | DMA_PINC_MODE);
  259. dma_set_num_transfers(FSMC_DMA_DEV, FSMC_DMA_CHANNEL, length);
  260. dma_clear_isr_bits(FSMC_DMA_DEV, FSMC_DMA_CHANNEL);
  261. dma_enable(FSMC_DMA_DEV, FSMC_DMA_CHANNEL);
  262. while ((dma_get_isr_bits(FSMC_DMA_DEV, FSMC_DMA_CHANNEL) & 0x0A) == 0) {};
  263. dma_disable(FSMC_DMA_DEV, FSMC_DMA_CHANNEL);
  264. }
  265. void LCD_IO_WriteSequence_Async(uint16_t *data, uint16_t length) {
  266. dma_setup_transfer(FSMC_DMA_DEV, FSMC_DMA_CHANNEL, data, DMA_SIZE_16BITS, &LCD->RAM, DMA_SIZE_16BITS, DMA_MEM_2_MEM | DMA_PINC_MODE);
  267. dma_set_num_transfers(FSMC_DMA_DEV, FSMC_DMA_CHANNEL, length);
  268. dma_clear_isr_bits(FSMC_DMA_DEV, FSMC_DMA_CHANNEL);
  269. dma_enable(FSMC_DMA_DEV, FSMC_DMA_CHANNEL);
  270. }
  271. void LCD_IO_WaitSequence_Async() {
  272. while ((dma_get_isr_bits(FSMC_DMA_DEV, FSMC_DMA_CHANNEL) & 0x0A) == 0) {};
  273. dma_disable(FSMC_DMA_DEV, FSMC_DMA_CHANNEL);
  274. }
  275. #endif // LCD_USE_DMA_FSMC
  276. #endif // HAS_GRAPHICAL_LCD
  277. #endif // ARDUINO_ARCH_STM32F1 && FSMC_CS_PIN