bootm.c 4.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197
  1. /*
  2. * (C) Copyright 2008 - 2013 Tensilica Inc.
  3. * (C) Copyright 2014 Cadence Design Systems Inc.
  4. *
  5. * SPDX-License-Identifier: GPL-2.0+
  6. */
  7. #include <common.h>
  8. #include <command.h>
  9. #include <u-boot/zlib.h>
  10. #include <asm/byteorder.h>
  11. #include <asm/addrspace.h>
  12. #include <asm/bootparam.h>
  13. #include <asm/cache.h>
  14. #include <image.h>
  15. DECLARE_GLOBAL_DATA_PTR;
  16. /*
  17. * Setup boot-parameters.
  18. */
  19. static struct bp_tag *setup_first_tag(struct bp_tag *params)
  20. {
  21. params->id = BP_TAG_FIRST;
  22. params->size = sizeof(long);
  23. *(unsigned long *)&params->data = BP_VERSION;
  24. return bp_tag_next(params);
  25. }
  26. static struct bp_tag *setup_last_tag(struct bp_tag *params)
  27. {
  28. params->id = BP_TAG_LAST;
  29. params->size = 0;
  30. return bp_tag_next(params);
  31. }
  32. static struct bp_tag *setup_memory_tag(struct bp_tag *params)
  33. {
  34. struct bd_info *bd = gd->bd;
  35. struct meminfo *mem;
  36. params->id = BP_TAG_MEMORY;
  37. params->size = sizeof(struct meminfo);
  38. mem = (struct meminfo *)params->data;
  39. mem->type = MEMORY_TYPE_CONVENTIONAL;
  40. mem->start = bd->bi_memstart;
  41. mem->end = bd->bi_memstart + bd->bi_memsize;
  42. printf(" MEMORY: tag:0x%04x, type:0X%lx, start:0X%lx, end:0X%lx\n",
  43. BP_TAG_MEMORY, mem->type, mem->start, mem->end);
  44. return bp_tag_next(params);
  45. }
  46. static struct bp_tag *setup_commandline_tag(struct bp_tag *params,
  47. char *cmdline)
  48. {
  49. int len;
  50. if (!cmdline)
  51. return params;
  52. len = strlen(cmdline);
  53. params->id = BP_TAG_COMMAND_LINE;
  54. params->size = (len + 3) & -4;
  55. strcpy((char *)params->data, cmdline);
  56. printf(" COMMAND_LINE: tag:0x%04x, size:%u, data:'%s'\n",
  57. BP_TAG_COMMAND_LINE, params->size, cmdline);
  58. return bp_tag_next(params);
  59. }
  60. static struct bp_tag *setup_ramdisk_tag(struct bp_tag *params,
  61. unsigned long rd_start,
  62. unsigned long rd_end)
  63. {
  64. struct meminfo *mem;
  65. if (rd_start == rd_end)
  66. return params;
  67. /* Add a single banked memory */
  68. params->id = BP_TAG_INITRD;
  69. params->size = sizeof(struct meminfo);
  70. mem = (struct meminfo *)params->data;
  71. mem->type = MEMORY_TYPE_CONVENTIONAL;
  72. mem->start = PHYSADDR(rd_start);
  73. mem->end = PHYSADDR(rd_end);
  74. printf(" INITRD: tag:0x%x, type:0X%04lx, start:0X%lx, end:0X%lx\n",
  75. BP_TAG_INITRD, mem->type, mem->start, mem->end);
  76. return bp_tag_next(params);
  77. }
  78. static struct bp_tag *setup_serial_tag(struct bp_tag *params)
  79. {
  80. params->id = BP_TAG_SERIAL_BAUDRATE;
  81. params->size = sizeof(unsigned long);
  82. params->data[0] = gd->baudrate;
  83. printf(" SERIAL_BAUDRATE: tag:0x%04x, size:%u, baudrate:%lu\n",
  84. BP_TAG_SERIAL_BAUDRATE, params->size, params->data[0]);
  85. return bp_tag_next(params);
  86. }
  87. #ifdef CONFIG_OF_LIBFDT
  88. static struct bp_tag *setup_fdt_tag(struct bp_tag *params, void *fdt_start)
  89. {
  90. params->id = BP_TAG_FDT;
  91. params->size = sizeof(unsigned long);
  92. params->data[0] = (unsigned long)fdt_start;
  93. printf(" FDT: tag:0x%04x, size:%u, start:0x%lx\n",
  94. BP_TAG_FDT, params->size, params->data[0]);
  95. return bp_tag_next(params);
  96. }
  97. #endif
  98. /*
  99. * Boot Linux.
  100. */
  101. int do_bootm_linux(int flag, int argc, char *argv[], bootm_headers_t *images)
  102. {
  103. struct bp_tag *params, *params_start;
  104. ulong initrd_start, initrd_end;
  105. char *commandline = getenv("bootargs");
  106. if (!(flag & (BOOTM_STATE_OS_GO | BOOTM_STATE_OS_FAKE_GO)))
  107. return 0;
  108. show_boot_progress(15);
  109. if (images->rd_start) {
  110. initrd_start = images->rd_start;
  111. initrd_end = images->rd_end;
  112. } else {
  113. initrd_start = 0;
  114. initrd_end = 0;
  115. }
  116. params_start = (struct bp_tag *)gd->bd->bi_boot_params;
  117. params = params_start;
  118. params = setup_first_tag(params);
  119. params = setup_memory_tag(params);
  120. params = setup_commandline_tag(params, commandline);
  121. params = setup_serial_tag(params);
  122. if (initrd_start)
  123. params = setup_ramdisk_tag(params, initrd_start, initrd_end);
  124. #ifdef CONFIG_OF_LIBFDT
  125. if (images->ft_addr)
  126. params = setup_fdt_tag(params, images->ft_addr);
  127. #endif
  128. printf("\n");
  129. params = setup_last_tag(params);
  130. show_boot_progress(15);
  131. printf("Transferring Control to Linux @0x%08lx ...\n\n",
  132. (ulong)images->ep);
  133. flush_dcache_range((unsigned long)params_start, (unsigned long)params);
  134. if (flag & BOOTM_STATE_OS_FAKE_GO)
  135. return 0;
  136. /*
  137. * _start() in vmlinux expects boot params in register a2.
  138. * NOTE:
  139. * Disable/delete your u-boot breakpoints before stepping into linux.
  140. */
  141. asm volatile ("mov a2, %0\n\t"
  142. "jx %1\n\t"
  143. : : "a" (params_start), "a" (images->ep)
  144. : "a2");
  145. /* Does not return */
  146. return 1;
  147. }