aristainetos-v2.c 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685
  1. // SPDX-License-Identifier: GPL-2.0+
  2. /*
  3. * (C) Copyright 2015
  4. * Heiko Schocher, DENX Software Engineering, hs@denx.de.
  5. *
  6. * Based on:
  7. * Copyright (C) 2012 Freescale Semiconductor, Inc.
  8. *
  9. * Author: Fabio Estevam <fabio.estevam@freescale.com>
  10. */
  11. #include <asm/arch/clock.h>
  12. #include <asm/arch/imx-regs.h>
  13. #include <asm/arch/iomux.h>
  14. #include <asm/arch/mx6-pins.h>
  15. #include <linux/errno.h>
  16. #include <asm/gpio.h>
  17. #include <asm/mach-imx/iomux-v3.h>
  18. #include <asm/mach-imx/boot_mode.h>
  19. #include <asm/mach-imx/mxc_i2c.h>
  20. #include <asm/mach-imx/video.h>
  21. #include <mmc.h>
  22. #include <fsl_esdhc.h>
  23. #include <miiphy.h>
  24. #include <netdev.h>
  25. #include <asm/arch/mxc_hdmi.h>
  26. #include <asm/arch/crm_regs.h>
  27. #include <linux/fb.h>
  28. #include <ipu_pixfmt.h>
  29. #include <asm/io.h>
  30. #include <asm/arch/sys_proto.h>
  31. #include <pwm.h>
  32. #include <micrel.h>
  33. #include <spi.h>
  34. #include <video.h>
  35. #include <../drivers/video/ipu.h>
  36. #if defined(CONFIG_VIDEO_BMP_LOGO)
  37. #include <bmp_logo.h>
  38. #endif
  39. #define USDHC2_PAD_CTRL (PAD_CTL_SPEED_LOW | \
  40. PAD_CTL_DSE_40ohm | PAD_CTL_SRE_FAST)
  41. #if (CONFIG_SYS_BOARD_VERSION == 2)
  42. /* 4.3 display controller */
  43. #define ECSPI1_CS0 IMX_GPIO_NR(4, 9)
  44. #define ECSPI4_CS0 IMX_GPIO_NR(3, 29)
  45. #elif (CONFIG_SYS_BOARD_VERSION == 3)
  46. #define ECSPI1_CS0 IMX_GPIO_NR(2, 30) /* NOR flash */
  47. /* 4.3 display controller */
  48. #define ECSPI1_CS1 IMX_GPIO_NR(4, 10)
  49. #endif
  50. #define SOFT_RESET_GPIO IMX_GPIO_NR(7, 13)
  51. #define SD2_DRIVER_ENABLE IMX_GPIO_NR(7, 8)
  52. struct i2c_pads_info i2c_pad_info3 = {
  53. .scl = {
  54. .i2c_mode = MX6_PAD_GPIO_5__I2C3_SCL | PC,
  55. .gpio_mode = MX6_PAD_GPIO_5__GPIO1_IO05 | PC,
  56. .gp = IMX_GPIO_NR(1, 5)
  57. },
  58. .sda = {
  59. .i2c_mode = MX6_PAD_GPIO_6__I2C3_SDA | PC,
  60. .gpio_mode = MX6_PAD_GPIO_6__GPIO1_IO06 | PC,
  61. .gp = IMX_GPIO_NR(1, 6)
  62. }
  63. };
  64. struct i2c_pads_info i2c_pad_info4 = {
  65. .scl = {
  66. .i2c_mode = MX6_PAD_GPIO_7__I2C4_SCL | PC,
  67. .gpio_mode = MX6_PAD_GPIO_7__GPIO1_IO07 | PC,
  68. .gp = IMX_GPIO_NR(1, 7)
  69. },
  70. .sda = {
  71. .i2c_mode = MX6_PAD_GPIO_8__I2C4_SDA | PC,
  72. .gpio_mode = MX6_PAD_GPIO_8__GPIO1_IO08 | PC,
  73. .gp = IMX_GPIO_NR(1, 8)
  74. }
  75. };
  76. iomux_v3_cfg_t const uart1_pads[] = {
  77. MX6_PAD_CSI0_DAT10__UART1_TX_DATA | MUX_PAD_CTRL(UART_PAD_CTRL),
  78. MX6_PAD_CSI0_DAT11__UART1_RX_DATA | MUX_PAD_CTRL(UART_PAD_CTRL),
  79. MX6_PAD_EIM_D19__UART1_CTS_B | MUX_PAD_CTRL(UART_PAD_CTRL),
  80. MX6_PAD_EIM_D20__UART1_RTS_B | MUX_PAD_CTRL(UART_PAD_CTRL),
  81. };
  82. iomux_v3_cfg_t const uart2_pads[] = {
  83. MX6_PAD_EIM_D26__UART2_TX_DATA | MUX_PAD_CTRL(UART_PAD_CTRL),
  84. MX6_PAD_EIM_D27__UART2_RX_DATA | MUX_PAD_CTRL(UART_PAD_CTRL),
  85. };
  86. iomux_v3_cfg_t const uart3_pads[] = {
  87. MX6_PAD_EIM_D24__UART3_TX_DATA | MUX_PAD_CTRL(UART_PAD_CTRL),
  88. MX6_PAD_EIM_D25__UART3_RX_DATA | MUX_PAD_CTRL(UART_PAD_CTRL),
  89. MX6_PAD_EIM_D31__UART3_RTS_B | MUX_PAD_CTRL(UART_PAD_CTRL),
  90. MX6_PAD_EIM_D23__UART3_CTS_B | MUX_PAD_CTRL(UART_PAD_CTRL),
  91. };
  92. iomux_v3_cfg_t const uart4_pads[] = {
  93. MX6_PAD_KEY_COL0__UART4_TX_DATA | MUX_PAD_CTRL(UART_PAD_CTRL),
  94. MX6_PAD_KEY_ROW0__UART4_RX_DATA | MUX_PAD_CTRL(UART_PAD_CTRL),
  95. };
  96. iomux_v3_cfg_t const gpio_pads[] = {
  97. /* LED enable*/
  98. MX6_PAD_ENET_CRS_DV__GPIO1_IO25 | MUX_PAD_CTRL(NO_PAD_CTRL),
  99. /* LED yellow */
  100. MX6_PAD_NANDF_CS3__GPIO6_IO16 | MUX_PAD_CTRL(NO_PAD_CTRL),
  101. /* LED red */
  102. #if (CONFIG_SYS_BOARD_VERSION == 2)
  103. MX6_PAD_EIM_EB0__GPIO2_IO28 | MUX_PAD_CTRL(NO_PAD_CTRL),
  104. #elif (CONFIG_SYS_BOARD_VERSION == 3)
  105. MX6_PAD_EIM_WAIT__GPIO5_IO00 | MUX_PAD_CTRL(NO_PAD_CTRL),
  106. #endif
  107. /* LED green */
  108. MX6_PAD_EIM_A24__GPIO5_IO04 | MUX_PAD_CTRL(NO_PAD_CTRL),
  109. /* LED blue */
  110. MX6_PAD_EIM_EB1__GPIO2_IO29 | MUX_PAD_CTRL(NO_PAD_CTRL),
  111. /* spi flash WP protect */
  112. MX6_PAD_SD4_DAT7__GPIO2_IO15 | MUX_PAD_CTRL(NO_PAD_CTRL),
  113. /* spi CS 0 */
  114. MX6_PAD_EIM_D29__GPIO3_IO29 | MUX_PAD_CTRL(NO_PAD_CTRL),
  115. /* spi bus #2 SS driver enable */
  116. MX6_PAD_EIM_A23__GPIO6_IO06 | MUX_PAD_CTRL(NO_PAD_CTRL),
  117. /* RST_LOC# PHY reset input (has pull-down!)*/
  118. MX6_PAD_GPIO_18__GPIO7_IO13 | MUX_PAD_CTRL(NO_PAD_CTRL),
  119. /* SD 2 level shifter output enable */
  120. MX6_PAD_SD3_RST__GPIO7_IO08 | MUX_PAD_CTRL(NO_PAD_CTRL),
  121. /* SD1 card detect input */
  122. MX6_PAD_ENET_RXD0__GPIO1_IO27 | MUX_PAD_CTRL(NO_PAD_CTRL),
  123. /* SD1 write protect input */
  124. MX6_PAD_DI0_PIN4__GPIO4_IO20 | MUX_PAD_CTRL(NO_PAD_CTRL),
  125. /* SD2 card detect input */
  126. MX6_PAD_GPIO_19__GPIO4_IO05 | MUX_PAD_CTRL(NO_PAD_CTRL),
  127. /* SD2 write protect input */
  128. MX6_PAD_SD4_DAT2__GPIO2_IO10 | MUX_PAD_CTRL(NO_PAD_CTRL),
  129. /* Touchscreen IRQ */
  130. MX6_PAD_SD4_DAT1__GPIO2_IO09 | MUX_PAD_CTRL(NO_PAD_CTRL),
  131. };
  132. static iomux_v3_cfg_t const misc_pads[] = {
  133. /* USB_OTG_ID = GPIO1_24*/
  134. MX6_PAD_ENET_RX_ER__USB_OTG_ID | MUX_PAD_CTRL(NO_PAD_CTRL),
  135. /* H1 Power enable = GPIO1_0*/
  136. MX6_PAD_GPIO_0__USB_H1_PWR | MUX_PAD_CTRL(NO_PAD_CTRL),
  137. /* OTG Power enable = GPIO4_15*/
  138. MX6_PAD_KEY_ROW4__USB_OTG_PWR | MUX_PAD_CTRL(NO_PAD_CTRL),
  139. };
  140. iomux_v3_cfg_t const enet_pads[] = {
  141. MX6_PAD_ENET_MDIO__ENET_MDIO | MUX_PAD_CTRL(ENET_PAD_CTRL),
  142. MX6_PAD_ENET_MDC__ENET_MDC | MUX_PAD_CTRL(ENET_PAD_CTRL),
  143. MX6_PAD_RGMII_TXC__RGMII_TXC | MUX_PAD_CTRL(ENET_PAD_CTRL),
  144. MX6_PAD_RGMII_TD0__RGMII_TD0 | MUX_PAD_CTRL(ENET_PAD_CTRL),
  145. MX6_PAD_RGMII_TD1__RGMII_TD1 | MUX_PAD_CTRL(ENET_PAD_CTRL),
  146. MX6_PAD_RGMII_TD2__RGMII_TD2 | MUX_PAD_CTRL(ENET_PAD_CTRL),
  147. MX6_PAD_RGMII_TD3__RGMII_TD3 | MUX_PAD_CTRL(ENET_PAD_CTRL),
  148. MX6_PAD_RGMII_TX_CTL__RGMII_TX_CTL | MUX_PAD_CTRL(ENET_PAD_CTRL),
  149. MX6_PAD_ENET_REF_CLK__ENET_TX_CLK | MUX_PAD_CTRL(ENET_PAD_CTRL),
  150. MX6_PAD_RGMII_RXC__RGMII_RXC | MUX_PAD_CTRL(ENET_PAD_CTRL),
  151. MX6_PAD_RGMII_RD0__RGMII_RD0 | MUX_PAD_CTRL(ENET_PAD_CTRL),
  152. MX6_PAD_RGMII_RD1__RGMII_RD1 | MUX_PAD_CTRL(ENET_PAD_CTRL),
  153. MX6_PAD_RGMII_RD2__RGMII_RD2 | MUX_PAD_CTRL(ENET_PAD_CTRL),
  154. MX6_PAD_RGMII_RD3__RGMII_RD3 | MUX_PAD_CTRL(ENET_PAD_CTRL),
  155. MX6_PAD_RGMII_RX_CTL__RGMII_RX_CTL | MUX_PAD_CTRL(ENET_PAD_CTRL),
  156. };
  157. static iomux_v3_cfg_t const backlight_pads[] = {
  158. /* backlight PWM brightness control */
  159. MX6_PAD_GPIO_9__PWM1_OUT | MUX_PAD_CTRL(NO_PAD_CTRL),
  160. /* backlight enable */
  161. MX6_PAD_EIM_BCLK__GPIO6_IO31 | MUX_PAD_CTRL(NO_PAD_CTRL),
  162. /* LCD power enable */
  163. MX6_PAD_NANDF_CS2__GPIO6_IO15 | MUX_PAD_CTRL(NO_PAD_CTRL),
  164. };
  165. static iomux_v3_cfg_t const ecspi1_pads[] = {
  166. MX6_PAD_EIM_D16__ECSPI1_SCLK | MUX_PAD_CTRL(SPI_PAD_CTRL),
  167. MX6_PAD_EIM_D17__ECSPI1_MISO | MUX_PAD_CTRL(SPI_PAD_CTRL),
  168. MX6_PAD_EIM_D18__ECSPI1_MOSI | MUX_PAD_CTRL(SPI_PAD_CTRL),
  169. #if (CONFIG_SYS_BOARD_VERSION == 2)
  170. MX6_PAD_KEY_ROW1__GPIO4_IO09 | MUX_PAD_CTRL(SPI_PAD_CTRL),
  171. #elif (CONFIG_SYS_BOARD_VERSION == 3)
  172. MX6_PAD_EIM_EB2__GPIO2_IO30 | MUX_PAD_CTRL(SPI_PAD_CTRL),
  173. MX6_PAD_KEY_COL2__GPIO4_IO10 | MUX_PAD_CTRL(NO_PAD_CTRL),
  174. #endif
  175. };
  176. static void setup_iomux_enet(void)
  177. {
  178. imx_iomux_v3_setup_multiple_pads(enet_pads, ARRAY_SIZE(enet_pads));
  179. }
  180. #if (CONFIG_SYS_BOARD_VERSION == 2)
  181. iomux_v3_cfg_t const ecspi4_pads[] = {
  182. MX6_PAD_EIM_D21__ECSPI4_SCLK | MUX_PAD_CTRL(NO_PAD_CTRL),
  183. MX6_PAD_EIM_D22__ECSPI4_MISO | MUX_PAD_CTRL(NO_PAD_CTRL),
  184. MX6_PAD_EIM_D28__ECSPI4_MOSI | MUX_PAD_CTRL(NO_PAD_CTRL),
  185. MX6_PAD_EIM_A25__GPIO5_IO02 | MUX_PAD_CTRL(NO_PAD_CTRL),
  186. MX6_PAD_EIM_D29__GPIO3_IO29 | MUX_PAD_CTRL(NO_PAD_CTRL),
  187. };
  188. #endif
  189. static iomux_v3_cfg_t const display_pads[] = {
  190. MX6_PAD_DI0_DISP_CLK__IPU1_DI0_DISP_CLK | MUX_PAD_CTRL(DISP_PAD_CTRL),
  191. MX6_PAD_DI0_PIN15__IPU1_DI0_PIN15,
  192. MX6_PAD_DI0_PIN2__IPU1_DI0_PIN02,
  193. MX6_PAD_DI0_PIN3__IPU1_DI0_PIN03,
  194. MX6_PAD_DISP0_DAT0__IPU1_DISP0_DATA00,
  195. MX6_PAD_DISP0_DAT1__IPU1_DISP0_DATA01,
  196. MX6_PAD_DISP0_DAT2__IPU1_DISP0_DATA02,
  197. MX6_PAD_DISP0_DAT3__IPU1_DISP0_DATA03,
  198. MX6_PAD_DISP0_DAT4__IPU1_DISP0_DATA04,
  199. MX6_PAD_DISP0_DAT5__IPU1_DISP0_DATA05,
  200. MX6_PAD_DISP0_DAT6__IPU1_DISP0_DATA06,
  201. MX6_PAD_DISP0_DAT7__IPU1_DISP0_DATA07,
  202. MX6_PAD_DISP0_DAT8__IPU1_DISP0_DATA08,
  203. MX6_PAD_DISP0_DAT9__IPU1_DISP0_DATA09,
  204. MX6_PAD_DISP0_DAT10__IPU1_DISP0_DATA10,
  205. MX6_PAD_DISP0_DAT11__IPU1_DISP0_DATA11,
  206. MX6_PAD_DISP0_DAT12__IPU1_DISP0_DATA12,
  207. MX6_PAD_DISP0_DAT13__IPU1_DISP0_DATA13,
  208. MX6_PAD_DISP0_DAT14__IPU1_DISP0_DATA14,
  209. MX6_PAD_DISP0_DAT15__IPU1_DISP0_DATA15,
  210. MX6_PAD_DISP0_DAT16__IPU1_DISP0_DATA16,
  211. MX6_PAD_DISP0_DAT17__IPU1_DISP0_DATA17,
  212. MX6_PAD_DISP0_DAT18__IPU1_DISP0_DATA18,
  213. MX6_PAD_DISP0_DAT19__IPU1_DISP0_DATA19,
  214. MX6_PAD_DISP0_DAT20__IPU1_DISP0_DATA20,
  215. MX6_PAD_DISP0_DAT21__IPU1_DISP0_DATA21,
  216. MX6_PAD_DISP0_DAT22__IPU1_DISP0_DATA22,
  217. MX6_PAD_DISP0_DAT23__IPU1_DISP0_DATA23,
  218. };
  219. int board_spi_cs_gpio(unsigned bus, unsigned cs)
  220. {
  221. if (bus == CONFIG_SF_DEFAULT_BUS && cs == CONFIG_SF_DEFAULT_CS)
  222. #if (CONFIG_SYS_BOARD_VERSION == 2)
  223. return IMX_GPIO_NR(5, 2);
  224. if (bus == 0 && cs == 0)
  225. return IMX_GPIO_NR(4, 9);
  226. #elif (CONFIG_SYS_BOARD_VERSION == 3)
  227. return ECSPI1_CS0;
  228. if (bus == 0 && cs == 1)
  229. return ECSPI1_CS1;
  230. #endif
  231. return -1;
  232. }
  233. static void setup_spi(void)
  234. {
  235. int i;
  236. imx_iomux_v3_setup_multiple_pads(ecspi1_pads, ARRAY_SIZE(ecspi1_pads));
  237. #if (CONFIG_SYS_BOARD_VERSION == 2)
  238. imx_iomux_v3_setup_multiple_pads(ecspi4_pads, ARRAY_SIZE(ecspi4_pads));
  239. #endif
  240. for (i = 0; i < 4; i++)
  241. enable_spi_clk(true, i);
  242. gpio_direction_output(ECSPI1_CS0, 1);
  243. #if (CONFIG_SYS_BOARD_VERSION == 2)
  244. gpio_direction_output(ECSPI4_CS1, 0);
  245. /* set cs0 to high (second device on spi bus #4) */
  246. gpio_direction_output(ECSPI4_CS0, 1);
  247. #elif (CONFIG_SYS_BOARD_VERSION == 3)
  248. gpio_direction_output(ECSPI1_CS1, 1);
  249. #endif
  250. }
  251. static void setup_iomux_uart(void)
  252. {
  253. switch (CONFIG_MXC_UART_BASE) {
  254. case UART1_BASE:
  255. imx_iomux_v3_setup_multiple_pads(uart1_pads,
  256. ARRAY_SIZE(uart1_pads));
  257. break;
  258. case UART2_BASE:
  259. imx_iomux_v3_setup_multiple_pads(uart2_pads,
  260. ARRAY_SIZE(uart2_pads));
  261. break;
  262. case UART3_BASE:
  263. imx_iomux_v3_setup_multiple_pads(uart3_pads,
  264. ARRAY_SIZE(uart3_pads));
  265. break;
  266. case UART4_BASE:
  267. imx_iomux_v3_setup_multiple_pads(uart4_pads,
  268. ARRAY_SIZE(uart4_pads));
  269. break;
  270. }
  271. }
  272. int board_phy_config(struct phy_device *phydev)
  273. {
  274. /* control data pad skew - devaddr = 0x02, register = 0x04 */
  275. ksz9031_phy_extended_write(phydev, 0x02,
  276. MII_KSZ9031_EXT_RGMII_CTRL_SIG_SKEW,
  277. MII_KSZ9031_MOD_DATA_NO_POST_INC, 0x0000);
  278. /* rx data pad skew - devaddr = 0x02, register = 0x05 */
  279. ksz9031_phy_extended_write(phydev, 0x02,
  280. MII_KSZ9031_EXT_RGMII_RX_DATA_SKEW,
  281. MII_KSZ9031_MOD_DATA_NO_POST_INC, 0x0000);
  282. /* tx data pad skew - devaddr = 0x02, register = 0x06 */
  283. ksz9031_phy_extended_write(phydev, 0x02,
  284. MII_KSZ9031_EXT_RGMII_TX_DATA_SKEW,
  285. MII_KSZ9031_MOD_DATA_NO_POST_INC, 0x0000);
  286. /* gtx and rx clock pad skew - devaddr = 0x02, register = 0x08 */
  287. ksz9031_phy_extended_write(phydev, 0x02,
  288. MII_KSZ9031_EXT_RGMII_CLOCK_SKEW,
  289. MII_KSZ9031_MOD_DATA_NO_POST_INC, 0x03FF);
  290. if (phydev->drv->config)
  291. phydev->drv->config(phydev);
  292. return 0;
  293. }
  294. int board_eth_init(bd_t *bis)
  295. {
  296. setup_iomux_enet();
  297. return cpu_eth_init(bis);
  298. }
  299. static int rotate_logo_one(unsigned char *out, unsigned char *in)
  300. {
  301. int i, j;
  302. for (i = 0; i < BMP_LOGO_WIDTH; i++)
  303. for (j = 0; j < BMP_LOGO_HEIGHT; j++)
  304. out[j * BMP_LOGO_WIDTH + BMP_LOGO_HEIGHT - 1 - i] =
  305. in[i * BMP_LOGO_WIDTH + j];
  306. return 0;
  307. }
  308. /*
  309. * Rotate the BMP_LOGO (only)
  310. * Will only work, if the logo is square, as
  311. * BMP_LOGO_HEIGHT and BMP_LOGO_WIDTH are defines, not variables
  312. */
  313. void rotate_logo(int rotations)
  314. {
  315. unsigned char out_logo[BMP_LOGO_WIDTH * BMP_LOGO_HEIGHT];
  316. unsigned char *in_logo;
  317. int i, j;
  318. if (BMP_LOGO_WIDTH != BMP_LOGO_HEIGHT)
  319. return;
  320. in_logo = bmp_logo_bitmap;
  321. /* one 90 degree rotation */
  322. if (rotations == 1 || rotations == 2 || rotations == 3)
  323. rotate_logo_one(out_logo, in_logo);
  324. /* second 90 degree rotation */
  325. if (rotations == 2 || rotations == 3)
  326. rotate_logo_one(in_logo, out_logo);
  327. /* third 90 degree rotation */
  328. if (rotations == 3)
  329. rotate_logo_one(out_logo, in_logo);
  330. /* copy result back to original array */
  331. if (rotations == 1 || rotations == 3)
  332. for (i = 0; i < BMP_LOGO_WIDTH; i++)
  333. for (j = 0; j < BMP_LOGO_HEIGHT; j++)
  334. in_logo[i * BMP_LOGO_WIDTH + j] =
  335. out_logo[i * BMP_LOGO_WIDTH + j];
  336. }
  337. static void enable_display_power(void)
  338. {
  339. imx_iomux_v3_setup_multiple_pads(backlight_pads,
  340. ARRAY_SIZE(backlight_pads));
  341. /* backlight enable */
  342. gpio_direction_output(IMX_GPIO_NR(6, 31), 1);
  343. /* LCD power enable */
  344. gpio_direction_output(IMX_GPIO_NR(6, 15), 1);
  345. /* enable backlight PWM 1 */
  346. if (pwm_init(0, 0, 0))
  347. goto error;
  348. /* duty cycle 500ns, period: 3000ns */
  349. if (pwm_config(0, 50000, 300000))
  350. goto error;
  351. if (pwm_enable(0))
  352. goto error;
  353. return;
  354. error:
  355. puts("error init pwm for backlight\n");
  356. return;
  357. }
  358. static void enable_lvds(struct display_info_t const *dev)
  359. {
  360. struct iomuxc *iomux = (struct iomuxc *)IOMUXC_BASE_ADDR;
  361. struct mxc_ccm_reg *ccm = (struct mxc_ccm_reg *)CCM_BASE_ADDR;
  362. int reg;
  363. s32 timeout = 100000;
  364. /* set PLL5 clock */
  365. reg = readl(&ccm->analog_pll_video);
  366. reg |= BM_ANADIG_PLL_VIDEO_POWERDOWN;
  367. writel(reg, &ccm->analog_pll_video);
  368. /* set PLL5 to 232720000Hz */
  369. reg &= ~BM_ANADIG_PLL_VIDEO_DIV_SELECT;
  370. reg |= BF_ANADIG_PLL_VIDEO_DIV_SELECT(0x26);
  371. reg &= ~BM_ANADIG_PLL_VIDEO_POST_DIV_SELECT;
  372. reg |= BF_ANADIG_PLL_VIDEO_POST_DIV_SELECT(0);
  373. writel(reg, &ccm->analog_pll_video);
  374. writel(BF_ANADIG_PLL_VIDEO_NUM_A(0xC0238),
  375. &ccm->analog_pll_video_num);
  376. writel(BF_ANADIG_PLL_VIDEO_DENOM_B(0xF4240),
  377. &ccm->analog_pll_video_denom);
  378. reg &= ~BM_ANADIG_PLL_VIDEO_POWERDOWN;
  379. writel(reg, &ccm->analog_pll_video);
  380. while (timeout--)
  381. if (readl(&ccm->analog_pll_video) & BM_ANADIG_PLL_VIDEO_LOCK)
  382. break;
  383. if (timeout < 0)
  384. printf("Warning: video pll lock timeout!\n");
  385. reg = readl(&ccm->analog_pll_video);
  386. reg |= BM_ANADIG_PLL_VIDEO_ENABLE;
  387. reg &= ~BM_ANADIG_PLL_VIDEO_BYPASS;
  388. writel(reg, &ccm->analog_pll_video);
  389. /* set LDB0, LDB1 clk select to 000/000 (PLL5 clock) */
  390. reg = readl(&ccm->cs2cdr);
  391. reg &= ~(MXC_CCM_CS2CDR_LDB_DI0_CLK_SEL_MASK
  392. | MXC_CCM_CS2CDR_LDB_DI1_CLK_SEL_MASK);
  393. reg |= (0 << MXC_CCM_CS2CDR_LDB_DI0_CLK_SEL_OFFSET)
  394. | (0 << MXC_CCM_CS2CDR_LDB_DI1_CLK_SEL_OFFSET);
  395. writel(reg, &ccm->cs2cdr);
  396. reg = readl(&ccm->cscmr2);
  397. reg |= MXC_CCM_CSCMR2_LDB_DI0_IPU_DIV;
  398. writel(reg, &ccm->cscmr2);
  399. reg = readl(&ccm->chsccdr);
  400. reg |= (CHSCCDR_CLK_SEL_LDB_DI0
  401. << MXC_CCM_CHSCCDR_IPU1_DI0_CLK_SEL_OFFSET);
  402. writel(reg, &ccm->chsccdr);
  403. reg = IOMUXC_GPR2_BGREF_RRMODE_EXTERNAL_RES
  404. | IOMUXC_GPR2_DI1_VS_POLARITY_ACTIVE_HIGH
  405. | IOMUXC_GPR2_DI0_VS_POLARITY_ACTIVE_HIGH
  406. | IOMUXC_GPR2_BIT_MAPPING_CH0_SPWG
  407. | IOMUXC_GPR2_DATA_WIDTH_CH0_24BIT
  408. | IOMUXC_GPR2_LVDS_CH1_MODE_DISABLED
  409. | IOMUXC_GPR2_LVDS_CH0_MODE_ENABLED_DI0;
  410. writel(reg, &iomux->gpr[2]);
  411. reg = readl(&iomux->gpr[3]);
  412. reg = (reg & ~IOMUXC_GPR3_LVDS0_MUX_CTL_MASK)
  413. | (IOMUXC_GPR3_MUX_SRC_IPU1_DI0
  414. << IOMUXC_GPR3_LVDS0_MUX_CTL_OFFSET);
  415. writel(reg, &iomux->gpr[3]);
  416. return;
  417. }
  418. static void enable_spi_display(struct display_info_t const *dev)
  419. {
  420. struct iomuxc *iomux = (struct iomuxc *)IOMUXC_BASE_ADDR;
  421. struct mxc_ccm_reg *ccm = (struct mxc_ccm_reg *)CCM_BASE_ADDR;
  422. int reg;
  423. s32 timeout = 100000;
  424. #if defined(CONFIG_VIDEO_BMP_LOGO)
  425. rotate_logo(3); /* portrait display in landscape mode */
  426. #endif
  427. /*
  428. * set ldb clock to 28341000 Hz calculated through the formula:
  429. * (XRES + LEFT_M + RIGHT_M + HSYNC_LEN) *
  430. * (YRES + UPPER_M + LOWER_M + VSYNC_LEN) * REFRESH)
  431. * see:
  432. * https://community.freescale.com/thread/308170
  433. */
  434. ipu_set_ldb_clock(28341000);
  435. reg = readl(&ccm->cs2cdr);
  436. /* select pll 5 clock */
  437. reg &= ~(MXC_CCM_CS2CDR_LDB_DI0_CLK_SEL_MASK
  438. | MXC_CCM_CS2CDR_LDB_DI1_CLK_SEL_MASK);
  439. writel(reg, &ccm->cs2cdr);
  440. /* set PLL5 to 197994996Hz */
  441. reg &= ~BM_ANADIG_PLL_VIDEO_DIV_SELECT;
  442. reg |= BF_ANADIG_PLL_VIDEO_DIV_SELECT(0x21);
  443. reg &= ~BM_ANADIG_PLL_VIDEO_POST_DIV_SELECT;
  444. reg |= BF_ANADIG_PLL_VIDEO_POST_DIV_SELECT(0);
  445. writel(reg, &ccm->analog_pll_video);
  446. writel(BF_ANADIG_PLL_VIDEO_NUM_A(0xfbf4),
  447. &ccm->analog_pll_video_num);
  448. writel(BF_ANADIG_PLL_VIDEO_DENOM_B(0xf4240),
  449. &ccm->analog_pll_video_denom);
  450. reg &= ~BM_ANADIG_PLL_VIDEO_POWERDOWN;
  451. writel(reg, &ccm->analog_pll_video);
  452. while (timeout--)
  453. if (readl(&ccm->analog_pll_video) & BM_ANADIG_PLL_VIDEO_LOCK)
  454. break;
  455. if (timeout < 0)
  456. printf("Warning: video pll lock timeout!\n");
  457. reg = readl(&ccm->analog_pll_video);
  458. reg |= BM_ANADIG_PLL_VIDEO_ENABLE;
  459. reg &= ~BM_ANADIG_PLL_VIDEO_BYPASS;
  460. writel(reg, &ccm->analog_pll_video);
  461. /* set LDB0, LDB1 clk select to 000/000 (PLL5 clock) */
  462. reg = readl(&ccm->cs2cdr);
  463. reg &= ~(MXC_CCM_CS2CDR_LDB_DI0_CLK_SEL_MASK
  464. | MXC_CCM_CS2CDR_LDB_DI1_CLK_SEL_MASK);
  465. reg |= (0 << MXC_CCM_CS2CDR_LDB_DI0_CLK_SEL_OFFSET)
  466. | (0 << MXC_CCM_CS2CDR_LDB_DI1_CLK_SEL_OFFSET);
  467. writel(reg, &ccm->cs2cdr);
  468. reg = readl(&ccm->cscmr2);
  469. reg |= MXC_CCM_CSCMR2_LDB_DI0_IPU_DIV;
  470. writel(reg, &ccm->cscmr2);
  471. reg = readl(&ccm->chsccdr);
  472. reg |= (CHSCCDR_CLK_SEL_LDB_DI0
  473. << MXC_CCM_CHSCCDR_IPU1_DI0_CLK_SEL_OFFSET);
  474. reg &= ~MXC_CCM_CHSCCDR_IPU1_DI0_PODF_MASK;
  475. reg |= (2 << MXC_CCM_CHSCCDR_IPU1_DI0_PODF_OFFSET);
  476. reg &= ~MXC_CCM_CHSCCDR_IPU1_DI0_PRE_CLK_SEL_MASK;
  477. reg |= (2 << MXC_CCM_CHSCCDR_IPU1_DI0_PRE_CLK_SEL_OFFSET);
  478. writel(reg, &ccm->chsccdr);
  479. reg = IOMUXC_GPR2_BGREF_RRMODE_EXTERNAL_RES
  480. | IOMUXC_GPR2_DI1_VS_POLARITY_ACTIVE_HIGH
  481. | IOMUXC_GPR2_DI0_VS_POLARITY_ACTIVE_HIGH
  482. | IOMUXC_GPR2_BIT_MAPPING_CH0_SPWG
  483. | IOMUXC_GPR2_DATA_WIDTH_CH0_24BIT
  484. | IOMUXC_GPR2_LVDS_CH1_MODE_DISABLED
  485. | IOMUXC_GPR2_LVDS_CH0_MODE_ENABLED_DI0;
  486. writel(reg, &iomux->gpr[2]);
  487. reg = readl(&iomux->gpr[3]);
  488. reg = (reg & ~IOMUXC_GPR3_LVDS0_MUX_CTL_MASK)
  489. | (IOMUXC_GPR3_MUX_SRC_IPU1_DI0
  490. << IOMUXC_GPR3_LVDS0_MUX_CTL_OFFSET);
  491. writel(reg, &iomux->gpr[3]);
  492. imx_iomux_v3_setup_multiple_pads(
  493. display_pads,
  494. ARRAY_SIZE(display_pads));
  495. return;
  496. }
  497. static void setup_display(void)
  498. {
  499. enable_ipu_clock();
  500. enable_display_power();
  501. }
  502. static void setup_iomux_gpio(void)
  503. {
  504. imx_iomux_v3_setup_multiple_pads(gpio_pads, ARRAY_SIZE(gpio_pads));
  505. }
  506. static void set_gpr_register(void)
  507. {
  508. struct iomuxc *iomuxc_regs = (struct iomuxc *)IOMUXC_BASE_ADDR;
  509. writel(IOMUXC_GPR1_APP_CLK_REQ_N | IOMUXC_GPR1_PCIE_RDY_L23 |
  510. IOMUXC_GPR1_EXC_MON_SLVE |
  511. (2 << IOMUXC_GPR1_ADDRS0_OFFSET) |
  512. IOMUXC_GPR1_ACT_CS0,
  513. &iomuxc_regs->gpr[1]);
  514. writel(0x0, &iomuxc_regs->gpr[8]);
  515. writel(IOMUXC_GPR12_ARMP_IPG_CLK_EN | IOMUXC_GPR12_ARMP_AHB_CLK_EN |
  516. IOMUXC_GPR12_ARMP_ATB_CLK_EN | IOMUXC_GPR12_ARMP_APB_CLK_EN,
  517. &iomuxc_regs->gpr[12]);
  518. }
  519. int board_early_init_f(void)
  520. {
  521. setup_iomux_uart();
  522. setup_iomux_gpio();
  523. gpio_direction_output(SOFT_RESET_GPIO, 1);
  524. gpio_direction_output(SD2_DRIVER_ENABLE, 1);
  525. setup_display();
  526. set_gpr_register();
  527. return 0;
  528. }
  529. static void setup_i2c4(void)
  530. {
  531. setup_i2c(3, CONFIG_SYS_I2C_SPEED, CONFIG_SYS_I2C_SLAVE,
  532. &i2c_pad_info4);
  533. }
  534. static void setup_board_gpio(void)
  535. {
  536. /* enable all LEDs */
  537. gpio_request(IMX_GPIO_NR(2, 13), "LED ena"); /* 25 */
  538. gpio_direction_output(IMX_GPIO_NR(1, 25), 0);
  539. /* switch off Status LEDs */
  540. #if (CONFIG_SYS_BOARD_VERSION == 2)
  541. gpio_request(IMX_GPIO_NR(6, 16), "LED yellow"); /* 176 */
  542. gpio_direction_output(IMX_GPIO_NR(6, 16), 1);
  543. gpio_request(IMX_GPIO_NR(2, 28), "LED red"); /* 60 */
  544. gpio_direction_output(IMX_GPIO_NR(2, 28), 1);
  545. gpio_request(IMX_GPIO_NR(5, 4), "LED green"); /* 132 */
  546. gpio_direction_output(IMX_GPIO_NR(5, 4), 1);
  547. gpio_request(IMX_GPIO_NR(2, 29), "LED blue"); /* 61 */
  548. gpio_direction_output(IMX_GPIO_NR(2, 29), 1);
  549. #elif (CONFIG_SYS_BOARD_VERSION == 3)
  550. gpio_request(IMX_GPIO_NR(6, 16), "LED yellow"); /* 176 */
  551. gpio_direction_output(IMX_GPIO_NR(6, 16), 0);
  552. gpio_request(IMX_GPIO_NR(5, 0), "LED red"); /* 128 */
  553. gpio_direction_output(IMX_GPIO_NR(5, 0), 0);
  554. gpio_request(IMX_GPIO_NR(5, 4), "LED green"); /* 132 */
  555. gpio_direction_output(IMX_GPIO_NR(5, 4), 0);
  556. gpio_request(IMX_GPIO_NR(2, 29), "LED blue"); /* 61 */
  557. gpio_direction_output(IMX_GPIO_NR(2, 29), 0);
  558. #endif
  559. }
  560. static void setup_board_spi(void)
  561. {
  562. /* enable spi bus #2 SS drivers (and spi bus #4 SS1 for rev2b) */
  563. gpio_direction_output(IMX_GPIO_NR(6, 6), 1);
  564. }
  565. int board_late_init(void)
  566. {
  567. char *my_bootdelay;
  568. char bootmode = 0;
  569. char const *panel = env_get("panel");
  570. /*
  571. * Check the boot-source. If booting from NOR Flash,
  572. * disable bootdelay
  573. */
  574. gpio_request(IMX_GPIO_NR(7, 6), "bootsel0");
  575. gpio_direction_input(IMX_GPIO_NR(7, 6));
  576. gpio_request(IMX_GPIO_NR(7, 7), "bootsel1");
  577. gpio_direction_input(IMX_GPIO_NR(7, 7));
  578. gpio_request(IMX_GPIO_NR(7, 1), "bootsel2");
  579. gpio_direction_input(IMX_GPIO_NR(7, 1));
  580. bootmode |= (gpio_get_value(IMX_GPIO_NR(7, 6)) ? 1 : 0) << 0;
  581. bootmode |= (gpio_get_value(IMX_GPIO_NR(7, 7)) ? 1 : 0) << 1;
  582. bootmode |= (gpio_get_value(IMX_GPIO_NR(7, 1)) ? 1 : 0) << 2;
  583. if (bootmode == 7) {
  584. my_bootdelay = env_get("nor_bootdelay");
  585. if (my_bootdelay != NULL)
  586. env_set("bootdelay", my_bootdelay);
  587. else
  588. env_set("bootdelay", "-2");
  589. }
  590. /* if we have the lg panel, we can initialze it now */
  591. if (panel)
  592. if (!strcmp(panel, displays[1].mode.name))
  593. lg4573_spi_startup(CONFIG_LG4573_BUS,
  594. CONFIG_LG4573_CS,
  595. 10000000, SPI_MODE_0);
  596. return 0;
  597. }