atf.c 6.7 KB

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