reloc_ia32_efi.c 1.3 KB

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