riscv-stub.c 2.5 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697
  1. #include <dragonstub/dragonstub.h>
  2. #include <dragonstub/linux/unaligned.h>
  3. #include "efilib.h"
  4. #include <libfdt.h>
  5. #include <asm/csr.h>
  6. /// @brief 当前的hartid
  7. static unsigned long hartid;
  8. typedef void __noreturn (*jump_kernel_func)(unsigned long, unsigned long);
  9. static efi_status_t get_boot_hartid_from_fdt(void)
  10. {
  11. const void *fdt;
  12. int chosen_node, len;
  13. const void *prop;
  14. // efi_guid_t device_tree_guid = *(efi_guid_t *)&tmp;
  15. fdt = get_efi_config_table(DEVICE_TREE_GUID);
  16. if (!fdt) {
  17. efi_err("Failed to get FDT from EFI config table\n");
  18. return EFI_INVALID_PARAMETER;
  19. }
  20. chosen_node = fdt_path_offset(fdt, "/chosen");
  21. if (chosen_node < 0) {
  22. efi_err("Failed to find /chosen node in FDT\n");
  23. return EFI_INVALID_PARAMETER;
  24. }
  25. prop = fdt_getprop((void *)fdt, chosen_node, "boot-hartid", &len);
  26. if (!prop) {
  27. efi_err("Failed to find boot-hartid property in FDT\n");
  28. return EFI_INVALID_PARAMETER;
  29. }
  30. if (len == sizeof(u32))
  31. hartid = (unsigned long)fdt32_to_cpu(*(fdt32_t *)prop);
  32. else if (len == sizeof(u64))
  33. hartid = (unsigned long)fdt64_to_cpu(
  34. __get_unaligned_t(fdt64_t, prop));
  35. else {
  36. efi_err("Invalid boot-hartid property in FDT\n");
  37. return EFI_INVALID_PARAMETER;
  38. }
  39. return 0;
  40. }
  41. static efi_status_t get_boot_hartid_from_efi(void)
  42. {
  43. efi_guid_t boot_protocol_guid = RISCV_EFI_BOOT_PROTOCOL_GUID;
  44. struct riscv_efi_boot_protocol *boot_protocol;
  45. efi_status_t status;
  46. status = efi_bs_call(LocateProtocol, &boot_protocol_guid, NULL,
  47. (void **)&boot_protocol);
  48. if (status != EFI_SUCCESS)
  49. return status;
  50. return efi_call_proto(boot_protocol, get_boot_hartid, &hartid);
  51. }
  52. efi_status_t check_platform_features(void)
  53. {
  54. efi_info("Checking platform features...\n");
  55. efi_status_t status = -1;
  56. int ret;
  57. efi_info("Try to get boot hartid from EFI\n");
  58. status = get_boot_hartid_from_efi();
  59. if (status != EFI_SUCCESS) {
  60. efi_info("Try to get boot hartid from FDT\n");
  61. ret = get_boot_hartid_from_fdt();
  62. if (ret) {
  63. efi_err("Failed to get boot hartid!\n");
  64. return EFI_UNSUPPORTED;
  65. }
  66. }
  67. efi_info("Boot hartid: %ld\n", hartid);
  68. return EFI_SUCCESS;
  69. }
  70. void __noreturn efi_enter_kernel(struct payload_info *payload_info,
  71. unsigned long fdt, unsigned long fdt_size)
  72. {
  73. unsigned long kernel_entry = payload_info->kernel_entry;
  74. jump_kernel_func jump_kernel = (jump_kernel_func)kernel_entry;
  75. /*
  76. * Jump to real kernel here with following constraints.
  77. * 1. MMU should be disabled.
  78. * 2. a0 should contain hartid
  79. * 3. a1 should DT address
  80. */
  81. csr_write(CSR_SATP, 0);
  82. jump_kernel(hartid, fdt);
  83. }