aristainetos-v2.c 19 KB

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