reloc.c 2.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109
  1. #include "mm_aspace.h"
  2. #include <rtthread.h>
  3. #include <stdint.h>
  4. #include <string.h>
  5. #include <elf.h>
  6. #ifdef ARCH_MM_MMU
  7. #include <mmu.h>
  8. #include <page.h>
  9. #endif
  10. typedef struct
  11. {
  12. Elf64_Word st_name;
  13. Elf64_Addr st_value;
  14. Elf64_Word st_size;
  15. unsigned char st_info;
  16. unsigned char st_other;
  17. Elf64_Half st_shndx;
  18. } Elf64_sym;
  19. #ifdef ARCH_MM_MMU
  20. void arch_elf_reloc(rt_aspace_t aspace, void *text_start, void *rel_dyn_start, size_t rel_dyn_size, void *got_start, size_t got_size, Elf64_sym *dynsym)
  21. {
  22. size_t rel_off;
  23. void* addr;
  24. if (rel_dyn_size && !dynsym)
  25. {
  26. return;
  27. }
  28. for (rel_off = 0; rel_off < rel_dyn_size; rel_off += 8)
  29. {
  30. uint32_t v1, v2;
  31. addr = rt_hw_mmu_v2p(aspace, (void *)(((rt_size_t)rel_dyn_start) + rel_off));
  32. memcpy(&v1, addr, 4);
  33. addr = rt_hw_mmu_v2p(aspace, (void *)(((rt_size_t)rel_dyn_start) + rel_off + 4));
  34. memcpy(&v2, addr, 4);
  35. addr = rt_hw_mmu_v2p(aspace, (void *)((rt_size_t)text_start + v1));
  36. if ((v2 & 0xff) == R_ARM_RELATIVE)
  37. {
  38. *(rt_size_t*)addr += (rt_size_t)text_start;
  39. }
  40. else if ((v2 & 0xff) == R_ARM_ABS32)
  41. {
  42. uint32_t t;
  43. t = (v2 >> 8);
  44. if (t) /* 0 is UDF */
  45. {
  46. *(rt_size_t*)addr = (((rt_size_t)text_start) + dynsym[t].st_value);
  47. }
  48. }
  49. }
  50. /* modify got */
  51. if (got_size)
  52. {
  53. uint32_t *got_item = (uint32_t*)got_start;
  54. for (rel_off = 0; rel_off < got_size; rel_off += 4, got_item++)
  55. {
  56. addr = rt_hw_mmu_v2p(aspace, got_item);
  57. *(rt_size_t *)addr += (rt_size_t)text_start;
  58. }
  59. }
  60. }
  61. #else
  62. void arch_elf_reloc(void *text_start, void *rel_dyn_start, size_t rel_dyn_size, void *got_start, size_t got_size, Elf64_sym *dynsym)
  63. {
  64. size_t rel_off;
  65. if (rel_dyn_size && !dynsym)
  66. {
  67. return;
  68. }
  69. for (rel_off = 0; rel_off < rel_dyn_size; rel_off += 8)
  70. {
  71. uint32_t v1, v2;
  72. memcpy(&v1, ((rt_uint8_t *)rel_dyn_start) + rel_off, 4);
  73. memcpy(&v2, ((rt_uint8_t *)rel_dyn_start) + rel_off + 4, 4);
  74. if ((v2 & 0xff) == R_ARM_RELATIVE)
  75. {
  76. *(uint32_t*)(((rt_size_t)text_start) + v1) += (uint32_t)text_start;
  77. }
  78. else if ((v2 & 0xff) == R_ARM_ABS32)
  79. {
  80. uint32_t t;
  81. t = (v2 >> 8);
  82. if (t) /* 0 is UDF */
  83. {
  84. *(uint32_t*)(((rt_size_t)text_start) + v1) = (uint32_t)(((rt_size_t)text_start) + dynsym[t].st_value);
  85. }
  86. }
  87. }
  88. /* modify got */
  89. if (got_size)
  90. {
  91. uint32_t *got_item = (uint32_t*)got_start;
  92. for (rel_off = 0; rel_off < got_size; rel_off += 4, got_item++)
  93. {
  94. *got_item += (uint32_t)text_start;
  95. }
  96. }
  97. }
  98. #endif