reloc.c 2.8 KB

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