iot_devkit.c 4.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184
  1. // SPDX-License-Identifier: GPL-2.0+
  2. /*
  3. * Copyright (C) 2018 Synopsys, Inc. All rights reserved.
  4. */
  5. #include <common.h>
  6. #include <malloc.h>
  7. #include <dwmmc.h>
  8. #include <linux/libfdt.h>
  9. #include <fdtdec.h>
  10. #include <asm/arcregs.h>
  11. DECLARE_GLOBAL_DATA_PTR;
  12. #define SYSCON_BASE 0xf000a000
  13. #define AHBCKDIV (void *)(SYSCON_BASE + 0x04)
  14. #define APBCKDIV (void *)(SYSCON_BASE + 0x08)
  15. #define APBCKEN (void *)(SYSCON_BASE + 0x0C)
  16. #define RESET_REG (void *)(SYSCON_BASE + 0x18)
  17. #define CLKSEL (void *)(SYSCON_BASE + 0x24)
  18. #define CLKSTAT (void *)(SYSCON_BASE + 0x28)
  19. #define PLLCON (void *)(SYSCON_BASE + 0x2C)
  20. #define APBCKSEL (void *)(SYSCON_BASE + 0x30)
  21. #define AHBCKEN (void *)(SYSCON_BASE + 0x34)
  22. #define USBPHY_PLL (void *)(SYSCON_BASE + 0x78)
  23. #define USBCFG (void *)(SYSCON_BASE + 0x7c)
  24. #define PLL_MASK_0 0xffcfffff
  25. #define PLL_MASK_1 0xffcfff00
  26. #define PLL_MASK_2 0xfbcfff00
  27. #define CLKSEL_DEFAULT 0x5a690000
  28. static int set_cpu_freq(unsigned int clk)
  29. {
  30. clk /= 1000000;
  31. /* Set clk to ext Xtal (LSN value 0) */
  32. writel(CLKSEL_DEFAULT, CLKSEL);
  33. switch (clk) {
  34. case 16:
  35. /* Bypass mode */
  36. return 0;
  37. case 50:
  38. writel(readl(PLLCON) & PLL_MASK_0, PLLCON);
  39. /* pll_off=1, M=25, N=1, OD=3, PLL_OUT_CLK=50M */
  40. writel((readl(PLLCON) & PLL_MASK_1) | 0x300191, PLLCON);
  41. /* pll_off=0, M=25, N=1, OD=3, PLL_OUT_CLK=50M */
  42. writel((readl(PLLCON) & PLL_MASK_2) | 0x300191, PLLCON);
  43. break;
  44. case 72:
  45. writel(readl(PLLCON) & PLL_MASK_0, PLLCON);
  46. /* pll_off=1, M=18, N=1, OD=2, PLL_OUT_CLK=72M */
  47. writel((readl(PLLCON) & PLL_MASK_1) | 0x200121, PLLCON);
  48. /* pll_off=0, M=18, N=1, OD=2, PLL_OUT_CLK=72M */
  49. writel((readl(PLLCON) & PLL_MASK_2) | 0x200121, PLLCON);
  50. break;
  51. case 100:
  52. writel(readl(PLLCON) & PLL_MASK_0, PLLCON);
  53. /* pll_off=1,M=25, N=1, OD=2, PLL_OUT_CLK=100M */
  54. writel((readl(PLLCON) & PLL_MASK_1) | 0x200191, PLLCON);
  55. /* pll_off=0,M=25, N=1, OD=2, PLL_OUT_CLK=100M */
  56. writel((readl(PLLCON) & PLL_MASK_2) | 0x200191, PLLCON);
  57. break;
  58. case 144:
  59. writel(readl(PLLCON) & PLL_MASK_0, PLLCON);
  60. /* pll_off=1, M=18, N=1, OD=1, PLL_OUT_CLK=144M */
  61. writel((readl(PLLCON) & PLL_MASK_1) | 0x100121, PLLCON);
  62. /* pll_off=0, M=18, N=1, OD=1, PLL_OUT_CLK=144M */
  63. writel((readl(PLLCON) & PLL_MASK_2) | 0x100121, PLLCON);
  64. break;
  65. default:
  66. return -EINVAL;
  67. }
  68. while (!(readl(CLKSTAT) & 0x4))
  69. ;
  70. /* Set clk from PLL on bus (LSN = 1) */
  71. writel(CLKSEL_DEFAULT | BIT(0), CLKSEL);
  72. return 0;
  73. }
  74. extern u8 __rom_end[];
  75. extern u8 __ram_start[];
  76. extern u8 __ram_end[];
  77. /*
  78. * Use mach_cpu_init() for .data section copy as board_early_init_f() will be
  79. * too late: initf_dm() will use a value of "av_" variable from not yet
  80. * initialized (by copy) area.
  81. */
  82. int mach_cpu_init(void)
  83. {
  84. int offset, freq;
  85. /* Don't relocate U-Boot */
  86. gd->flags |= GD_FLG_SKIP_RELOC;
  87. /* Copy data from ROM to RAM */
  88. u8 *src = __rom_end;
  89. u8 *dst = __ram_start;
  90. while (dst < __ram_end)
  91. *dst++ = *src++;
  92. /* Enable debug uart */
  93. #define DEBUG_UART_BASE 0x80014000
  94. #define DEBUG_UART_DLF_OFFSET 0xc0
  95. write_aux_reg(DEBUG_UART_BASE + DEBUG_UART_DLF_OFFSET, 1);
  96. offset = fdt_path_offset(gd->fdt_blob, "/cpu_card/core_clk");
  97. if (offset < 0)
  98. return offset;
  99. freq = fdtdec_get_int(gd->fdt_blob, offset, "clock-frequency", 0);
  100. if (!freq)
  101. return -EINVAL;
  102. /* If CPU freq > 100 MHz, divide eFLASH clock by 2 */
  103. if (freq > 100000000) {
  104. u32 reg = readl(AHBCKDIV);
  105. reg &= ~(0xF << 8);
  106. reg |= 2 << 8;
  107. writel(reg, AHBCKDIV);
  108. }
  109. return set_cpu_freq(freq);
  110. }
  111. #define ARC_PERIPHERAL_BASE 0xF0000000
  112. #define SDIO_BASE (ARC_PERIPHERAL_BASE + 0xB000)
  113. int board_mmc_init(bd_t *bis)
  114. {
  115. struct dwmci_host *host = NULL;
  116. host = malloc(sizeof(struct dwmci_host));
  117. if (!host) {
  118. printf("dwmci_host malloc fail!\n");
  119. return -ENOMEM;
  120. }
  121. memset(host, 0, sizeof(struct dwmci_host));
  122. host->name = "Synopsys Mobile storage";
  123. host->ioaddr = (void *)SDIO_BASE;
  124. host->buswidth = 4;
  125. host->dev_index = 0;
  126. host->bus_hz = 50000000;
  127. add_dwmci(host, host->bus_hz / 2, 400000);
  128. return 0;
  129. }
  130. #define IOTDK_RESET_SEQ 0x55AA6699
  131. void reset_cpu(ulong addr)
  132. {
  133. writel(IOTDK_RESET_SEQ, RESET_REG);
  134. }
  135. int checkboard(void)
  136. {
  137. puts("Board: Synopsys IoT Development Kit\n");
  138. return 0;
  139. };
  140. #ifdef CONFIG_DISPLAY_CPUINFO
  141. int print_cpuinfo(void)
  142. {
  143. printf("CPU: ARC EM9D\n");
  144. return 0;
  145. }
  146. #endif /* CONFIG_DISPLAY_CPUINFO */