atf.c 6.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312
  1. /**
  2. * (C) Copyright 2014, Cavium Inc.
  3. *
  4. * SPDX-License-Identifier: GPL-2.0+
  5. **/
  6. #include <common.h>
  7. #include <asm/io.h>
  8. #include <asm/system.h>
  9. #include <cavium/thunderx_svc.h>
  10. #include <cavium/atf.h>
  11. #include <cavium/atf_part.h>
  12. #include <asm/psci.h>
  13. #include <malloc.h>
  14. DECLARE_GLOBAL_DATA_PTR;
  15. ssize_t atf_read_mmc(uintptr_t offset, void *buffer, size_t size)
  16. {
  17. struct pt_regs regs;
  18. regs.regs[0] = THUNDERX_MMC_READ;
  19. regs.regs[1] = offset;
  20. regs.regs[2] = size;
  21. regs.regs[3] = (uintptr_t)buffer;
  22. smc_call(&regs);
  23. return regs.regs[0];
  24. }
  25. ssize_t atf_read_nor(uintptr_t offset, void *buffer, size_t size)
  26. {
  27. struct pt_regs regs;
  28. regs.regs[0] = THUNDERX_NOR_READ;
  29. regs.regs[1] = offset;
  30. regs.regs[2] = size;
  31. regs.regs[3] = (uintptr_t)buffer;
  32. smc_call(&regs);
  33. return regs.regs[0];
  34. }
  35. ssize_t atf_get_pcount(void)
  36. {
  37. struct pt_regs regs;
  38. regs.regs[0] = THUNDERX_PART_COUNT;
  39. smc_call(&regs);
  40. return regs.regs[0];
  41. }
  42. ssize_t atf_get_part(struct storage_partition *part, unsigned int index)
  43. {
  44. struct pt_regs regs;
  45. regs.regs[0] = THUNDERX_GET_PART;
  46. regs.regs[1] = (uintptr_t)part;
  47. regs.regs[2] = index;
  48. smc_call(&regs);
  49. return regs.regs[0];
  50. }
  51. ssize_t atf_erase_nor(uintptr_t offset, size_t size)
  52. {
  53. struct pt_regs regs;
  54. regs.regs[0] = THUNDERX_NOR_ERASE;
  55. regs.regs[1] = offset;
  56. smc_call(&regs);
  57. return regs.regs[0];
  58. }
  59. ssize_t atf_write_nor(uintptr_t offset, const void *buffer, size_t size)
  60. {
  61. struct pt_regs regs;
  62. regs.regs[0] = THUNDERX_NOR_WRITE;
  63. regs.regs[1] = offset;
  64. regs.regs[2] = size;
  65. regs.regs[3] = (uintptr_t)buffer;
  66. smc_call(&regs);
  67. return regs.regs[0];
  68. }
  69. ssize_t atf_write_mmc(uintptr_t offset, const void *buffer, size_t size)
  70. {
  71. struct pt_regs regs;
  72. regs.regs[0] = THUNDERX_MMC_WRITE;
  73. regs.regs[1] = offset;
  74. regs.regs[2] = size;
  75. regs.regs[3] = (uintptr_t)buffer;
  76. smc_call(&regs);
  77. return regs.regs[0];
  78. }
  79. ssize_t atf_dram_size(unsigned int node)
  80. {
  81. struct pt_regs regs;
  82. regs.regs[0] = THUNDERX_DRAM_SIZE;
  83. regs.regs[1] = node;
  84. smc_call(&regs);
  85. return regs.regs[0];
  86. }
  87. ssize_t atf_node_count(void)
  88. {
  89. struct pt_regs regs;
  90. regs.regs[0] = THUNDERX_NODE_COUNT;
  91. smc_call(&regs);
  92. return regs.regs[0];
  93. }
  94. ssize_t atf_env_count(void)
  95. {
  96. struct pt_regs regs;
  97. regs.regs[0] = THUNDERX_ENV_COUNT;
  98. smc_call(&regs);
  99. return regs.regs[0];
  100. }
  101. ssize_t atf_env_string(size_t index, char *str)
  102. {
  103. uint64_t *buf = (void *)str;
  104. struct pt_regs regs;
  105. regs.regs[0] = THUNDERX_ENV_STRING;
  106. regs.regs[1] = index;
  107. smc_call(&regs);
  108. if (regs.regs > 0) {
  109. buf[0] = regs.regs[0];
  110. buf[1] = regs.regs[1];
  111. buf[2] = regs.regs[2];
  112. buf[3] = regs.regs[3];
  113. return 1;
  114. } else {
  115. return regs.regs[0];
  116. }
  117. }
  118. #ifdef CONFIG_CMD_ATF
  119. static void atf_print_ver(void)
  120. {
  121. struct pt_regs regs;
  122. regs.regs[0] = ARM_STD_SVC_VERSION;
  123. smc_call(&regs);
  124. printf("ARM Std FW version: %ld.%ld\n", regs.regs[0], regs.regs[1]);
  125. regs.regs[0] = THUNDERX_SVC_VERSION;
  126. smc_call(&regs);
  127. printf("ThunderX OEM ver: %ld.%ld\n", regs.regs[0], regs.regs[1]);
  128. }
  129. static void atf_print_uid(void)
  130. {
  131. }
  132. static void atf_print_part_table(void)
  133. {
  134. size_t pcount;
  135. unsigned long i;
  136. int ret;
  137. char *ptype;
  138. struct storage_partition *part = (void *)CONFIG_SYS_LOWMEM_BASE;
  139. pcount = atf_get_pcount();
  140. printf("Partition count: %lu\n\n", pcount);
  141. printf("%10s %10s %10s\n", "Type", "Size", "Offset");
  142. for (i = 0; i < pcount; i++) {
  143. ret = atf_get_part(part, i);
  144. if (ret < 0) {
  145. printf("Uknown error while reading partition: %d\n",
  146. ret);
  147. return;
  148. }
  149. switch (part->type) {
  150. case PARTITION_NBL1FW_REST:
  151. ptype = "NBL1FW";
  152. break;
  153. case PARTITION_BL2_BL31:
  154. ptype = "BL2_BL31";
  155. break;
  156. case PARTITION_UBOOT:
  157. ptype = "BOOTLDR";
  158. break;
  159. case PARTITION_KERNEL:
  160. ptype = "KERNEL";
  161. break;
  162. case PARTITION_DEVICE_TREE:
  163. ptype = "DEVTREE";
  164. break;
  165. default:
  166. ptype = "UNKNOWN";
  167. }
  168. printf("%10s %10d %10lx\n", ptype, part->size, part->offset);
  169. }
  170. }
  171. int do_atf(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[])
  172. {
  173. ssize_t ret;
  174. size_t size, offset;
  175. void *buffer = 0;
  176. unsigned int index, node;
  177. char str[4 * sizeof(uint64_t)];
  178. if ((argc == 5) && !strcmp(argv[1], "readmmc")) {
  179. buffer = (void *)simple_strtoul(argv[2], NULL, 16);
  180. offset = simple_strtoul(argv[3], NULL, 10);
  181. size = simple_strtoul(argv[4], NULL, 10);
  182. ret = atf_read_mmc(offset, buffer, size);
  183. } else if ((argc == 5) && !strcmp(argv[1], "readnor")) {
  184. buffer = (void *)simple_strtoul(argv[2], NULL, 16);
  185. offset = simple_strtoul(argv[3], NULL, 10);
  186. size = simple_strtoul(argv[4], NULL, 10);
  187. ret = atf_read_nor(offset, buffer, size);
  188. } else if ((argc == 5) && !strcmp(argv[1], "writemmc")) {
  189. buffer = (void *)simple_strtoul(argv[2], NULL, 16);
  190. offset = simple_strtoul(argv[3], NULL, 10);
  191. size = simple_strtoul(argv[4], NULL, 10);
  192. ret = atf_write_mmc(offset, buffer, size);
  193. } else if ((argc == 5) && !strcmp(argv[1], "writenor")) {
  194. buffer = (void *)simple_strtoul(argv[2], NULL, 16);
  195. offset = simple_strtoul(argv[3], NULL, 10);
  196. size = simple_strtoul(argv[4], NULL, 10);
  197. ret = atf_write_nor(offset, buffer, size);
  198. } else if ((argc == 2) && !strcmp(argv[1], "part")) {
  199. atf_print_part_table();
  200. } else if ((argc == 4) && !strcmp(argv[1], "erasenor")) {
  201. offset = simple_strtoul(argv[2], NULL, 10);
  202. size = simple_strtoul(argv[3], NULL, 10);
  203. ret = atf_erase_nor(offset, size);
  204. } else if ((argc == 2) && !strcmp(argv[1], "envcount")) {
  205. ret = atf_env_count();
  206. printf("Number of environment strings: %zd\n", ret);
  207. } else if ((argc == 3) && !strcmp(argv[1], "envstring")) {
  208. index = simple_strtoul(argv[2], NULL, 10);
  209. ret = atf_env_string(index, str);
  210. if (ret > 0)
  211. printf("Environment string %d: %s\n", index, str);
  212. else
  213. printf("Return code: %zd\n", ret);
  214. } else if ((argc == 3) && !strcmp(argv[1], "dramsize")) {
  215. node = simple_strtoul(argv[2], NULL, 10);
  216. ret = atf_dram_size(node);
  217. printf("DRAM size: %zd Mbytes\n", ret >> 20);
  218. } else if ((argc == 2) && !strcmp(argv[1], "nodes")) {
  219. ret = atf_node_count();
  220. printf("Nodes count: %zd\n", ret);
  221. } else if ((argc == 2) && !strcmp(argv[1], "ver")) {
  222. atf_print_ver();
  223. } else if ((argc == 2) && !strcmp(argv[1], "uid")) {
  224. atf_print_uid();
  225. } else {
  226. return CMD_RET_USAGE;
  227. }
  228. return 0;
  229. }
  230. U_BOOT_CMD(
  231. atf, 10, 1, do_atf,
  232. "issue calls to ATF",
  233. "\t readmmc addr offset size - read MMC card\n"
  234. "\t readnor addr offset size - read NOR flash\n"
  235. "\t writemmc addr offset size - write MMC card\n"
  236. "\t writenor addr offset size - write NOR flash\n"
  237. "\t erasenor offset size - erase NOR flash\n"
  238. "\t nodes - number of nodes\n"
  239. "\t dramsize node - size of DRAM attached to node\n"
  240. "\t envcount - number of environment strings\n"
  241. "\t envstring index - print the environment string\n"
  242. "\t part - print MMC partition table\n"
  243. "\t ver - print ATF call set versions\n"
  244. );
  245. #endif