relocate.c 2.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104
  1. /*
  2. * Copyright (C) 2013-2014 Synopsys, Inc. All rights reserved.
  3. *
  4. * SPDX-License-Identifier: GPL-2.0+
  5. */
  6. #include <common.h>
  7. #include <elf.h>
  8. #include <asm-generic/sections.h>
  9. extern ulong __image_copy_start;
  10. extern ulong __ivt_end;
  11. DECLARE_GLOBAL_DATA_PTR;
  12. int copy_uboot_to_ram(void)
  13. {
  14. size_t len = (size_t)&__image_copy_end - (size_t)&__image_copy_start;
  15. if (gd->flags & GD_FLG_SKIP_RELOC)
  16. return 0;
  17. memcpy((void *)gd->relocaddr, (void *)&__image_copy_start, len);
  18. return 0;
  19. }
  20. int clear_bss(void)
  21. {
  22. ulong dst_addr = (ulong)&__bss_start + gd->reloc_off;
  23. size_t len = (size_t)&__bss_end - (size_t)&__bss_start;
  24. memset((void *)dst_addr, 0x00, len);
  25. return 0;
  26. }
  27. /*
  28. * Base functionality is taken from x86 version with added ARC-specifics
  29. */
  30. int do_elf_reloc_fixups(void)
  31. {
  32. Elf32_Rela *re_src = (Elf32_Rela *)(&__rel_dyn_start);
  33. Elf32_Rela *re_end = (Elf32_Rela *)(&__rel_dyn_end);
  34. if (gd->flags & GD_FLG_SKIP_RELOC)
  35. return 0;
  36. debug("Section .rela.dyn is located at %08x-%08x\n",
  37. (unsigned int)re_src, (unsigned int)re_end);
  38. Elf32_Addr *offset_ptr_rom, *last_offset = NULL;
  39. Elf32_Addr *offset_ptr_ram;
  40. do {
  41. /* Get the location from the relocation entry */
  42. offset_ptr_rom = (Elf32_Addr *)re_src->r_offset;
  43. /* Check that the location of the relocation is in .text */
  44. if (offset_ptr_rom >= (Elf32_Addr *)&__image_copy_start &&
  45. offset_ptr_rom > last_offset) {
  46. unsigned int val;
  47. /* Switch to the in-RAM version */
  48. offset_ptr_ram = (Elf32_Addr *)((ulong)offset_ptr_rom +
  49. gd->reloc_off);
  50. debug("Patching value @ %08x (relocated to %08x)\n",
  51. (unsigned int)offset_ptr_rom,
  52. (unsigned int)offset_ptr_ram);
  53. /*
  54. * Use "memcpy" because target location might be
  55. * 16-bit aligned on ARC so we may need to read
  56. * byte-by-byte. On attempt to read entire word by
  57. * CPU throws an exception
  58. */
  59. memcpy(&val, offset_ptr_ram, sizeof(int));
  60. #ifdef __LITTLE_ENDIAN__
  61. /* If location in ".text" section swap value */
  62. if ((unsigned int)offset_ptr_rom <
  63. (unsigned int)&__ivt_end)
  64. val = (val << 16) | (val >> 16);
  65. #endif
  66. /* Check that the target points into executable */
  67. if (val >= (unsigned int)&__image_copy_start && val <=
  68. (unsigned int)&__image_copy_end) {
  69. val += gd->reloc_off;
  70. #ifdef __LITTLE_ENDIAN__
  71. /* If location in ".text" section swap value */
  72. if ((unsigned int)offset_ptr_rom <
  73. (unsigned int)&__ivt_end)
  74. val = (val << 16) | (val >> 16);
  75. #endif
  76. memcpy(offset_ptr_ram, &val, sizeof(int));
  77. }
  78. }
  79. last_offset = offset_ptr_rom;
  80. } while (++re_src < re_end);
  81. return 0;
  82. }