reloc_x86_64_efi.c 1.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566
  1. /*
  2. * reloc_x86_64.c - position independent x86_64 ELF shared object relocator
  3. * Copyright (C) 1999 Hewlett-Packard Co.
  4. * Contributed by David Mosberger <davidm@hpl.hp.com>.
  5. * Copyright (C) 2005 Intel Co.
  6. * Contributed by Fenghua Yu <fenghua.yu@intel.com>.
  7. *
  8. * All rights reserved.
  9. *
  10. * SPDX-License-Identifier: BSD-3-Clause
  11. */
  12. #include <common.h>
  13. #include <efi.h>
  14. #include <elf.h>
  15. #include <asm/elf.h>
  16. efi_status_t _relocate(long ldbase, Elf64_Dyn *dyn, efi_handle_t image,
  17. struct efi_system_table *systab)
  18. {
  19. long relsz = 0, relent = 0;
  20. Elf64_Rel *rel = 0;
  21. unsigned long *addr;
  22. int i;
  23. for (i = 0; dyn[i].d_tag != DT_NULL; ++i) {
  24. switch (dyn[i].d_tag) {
  25. case DT_RELA:
  26. rel = (Elf64_Rel *)
  27. ((unsigned long)dyn[i].d_un.d_ptr + ldbase);
  28. break;
  29. case DT_RELASZ:
  30. relsz = dyn[i].d_un.d_val;
  31. break;
  32. case DT_RELAENT:
  33. relent = dyn[i].d_un.d_val;
  34. break;
  35. default:
  36. break;
  37. }
  38. }
  39. if (!rel && relent == 0)
  40. return EFI_SUCCESS;
  41. if (!rel || relent == 0)
  42. return EFI_LOAD_ERROR;
  43. while (relsz > 0) {
  44. /* apply the relocs */
  45. switch (ELF64_R_TYPE(rel->r_info)) {
  46. case R_X86_64_NONE:
  47. break;
  48. case R_X86_64_RELATIVE:
  49. addr = (unsigned long *)(ldbase + rel->r_offset);
  50. *addr += ldbase;
  51. break;
  52. default:
  53. break;
  54. }
  55. rel = (Elf64_Rel *)((char *)rel + relent);
  56. relsz -= relent;
  57. }
  58. return EFI_SUCCESS;
  59. }