reloc_x86_64_efi.c 1.3 KB

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