reloc.c 3.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120
  1. #include <rtthread.h>
  2. #include <stdint.h>
  3. #include <string.h>
  4. #include <lwp_elf.h>
  5. #ifdef RT_USING_USERSPACE
  6. #include <mmu.h>
  7. #include <page.h>
  8. #endif
  9. typedef struct
  10. {
  11. Elf32_Word st_name;
  12. Elf32_Addr st_value;
  13. Elf32_Word st_size;
  14. unsigned char st_info;
  15. unsigned char st_other;
  16. Elf32_Half st_shndx;
  17. } Elf32_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, Elf32_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. /*
  31. memcpy(&v1, rel_dyn_start + rel_off, 4);
  32. memcpy(&v2, rel_dyn_start + rel_off + 4, 4);
  33. */
  34. addr = rt_hw_mmu_v2p(m_info, (void*)((char*)rel_dyn_start + rel_off));
  35. addr = (void*)((char*)addr - PV_OFFSET);
  36. memcpy(&v1, addr, 4);
  37. addr = rt_hw_mmu_v2p(m_info, (void*)((char*)rel_dyn_start + rel_off + 4));
  38. addr = (void*)((char*)addr - PV_OFFSET);
  39. memcpy(&v2, addr, 4);
  40. addr = rt_hw_mmu_v2p(m_info, (void*)((char*)text_start + v1));
  41. addr = (void*)((char*)addr - PV_OFFSET);
  42. if ((v2 & 0xff) == R_ARM_RELATIVE)
  43. {
  44. // *(uint32_t*)(text_start + v1) += (uint32_t)text_start;
  45. *(uint32_t*)addr += (uint32_t)text_start;
  46. }
  47. else if ((v2 & 0xff) == R_ARM_ABS32)
  48. {
  49. uint32_t t;
  50. t = (v2 >> 8);
  51. if (t) /* 0 is UDF */
  52. {
  53. // *(uint32_t*)(text_start + v1) = (uint32_t)(text_start + dynsym[t].st_value);
  54. *(uint32_t*)addr = (uint32_t)((char*)text_start + dynsym[t].st_value);
  55. }
  56. }
  57. }
  58. /* modify got */
  59. if (got_size)
  60. {
  61. uint32_t *got_item = (uint32_t*)got_start;
  62. for (rel_off = 0; rel_off < got_size; rel_off += 4, got_item++)
  63. {
  64. //*got_item += (uint32_t)text_start;
  65. addr = rt_hw_mmu_v2p(m_info, got_item);
  66. addr = (void*)((char*)addr - PV_OFFSET);
  67. *(uint32_t *)addr += (uint32_t)text_start;
  68. }
  69. }
  70. }
  71. #else
  72. void lwp_elf_reloc(void *text_start, void *rel_dyn_start, size_t rel_dyn_size, void *got_start, size_t got_size, Elf32_sym *dynsym)
  73. {
  74. size_t rel_off;
  75. if (rel_dyn_size && !dynsym)
  76. {
  77. return;
  78. }
  79. for (rel_off = 0; rel_off < rel_dyn_size; rel_off += 8)
  80. {
  81. uint32_t v1, v2;
  82. memcpy(&v1, (void*)((char*)rel_dyn_start + rel_off), 4);
  83. memcpy(&v2, (void*)((char*)rel_dyn_start + rel_off + 4), 4);
  84. if ((v2 & 0xff) == R_ARM_RELATIVE)
  85. {
  86. *(uint32_t*)((char*)text_start + v1) += (uint32_t)text_start;
  87. }
  88. else if ((v2 & 0xff) == R_ARM_ABS32)
  89. {
  90. uint32_t t;
  91. t = (v2 >> 8);
  92. if (t) /* 0 is UDF */
  93. {
  94. *(uint32_t*)((char*)text_start + v1) = (uint32_t)((char*)text_start + dynsym[t].st_value);
  95. }
  96. }
  97. }
  98. /* modify got */
  99. if (got_size)
  100. {
  101. uint32_t *got_item = (uint32_t*)got_start;
  102. for (rel_off = 0; rel_off < got_size; rel_off += 4, got_item++)
  103. {
  104. *got_item += (uint32_t)text_start;
  105. }
  106. }
  107. }
  108. #endif