multiple fixes to sh (superh) dynamic linker relocations
authorRich Felker <dalias@aerifal.cx>
Tue, 17 Jun 2014 17:56:54 +0000 (13:56 -0400)
committerRich Felker <dalias@aerifal.cx>
Tue, 17 Jun 2014 17:56:54 +0000 (13:56 -0400)
the following issues are fixed:

- R_SH_REL32 was adding the load address of the module being relocated
  to the result. this seems to have been a mistake in the original
  port, since it does not match other dynamic linker implementations
  and since adding a difference between two addresses (the symbol
  value and the relocation address) to a load address does not make
  sense.

- R_SH_TLS_DTPMOD32 was wrongly accepting an inline addend (i.e. using
  += rather than = on *reloc_addr) which makes no sense; addition is
  not an operation that's defined on module ids.

- R_SH_TLS_DTPOFF32 and R_SH_TLS_TPOFF32 were wrongly using inline
  addends rather than the RELA-provided addends.

in addition, handling of R_SH_GLOB_DAT, R_SH_JMP_SLOT, and R_SH_DIR32
are merged to all honor the addend. the first two should not need it
for correct usage generated by toolchains, but other dynamic linkers
allow addends here, and it simplifies the code anyway.

these issues were spotted while reviewing the code for the purpose of
refactoring this part of the dynamic linker. no testing was performed.

arch/sh/reloc.h

index a1393bb3dd25a117fb10ded67070be5ee2590b72..9090ee3beed05f12f4282a4a2c703f7fcaab43c5 100644 (file)
@@ -18,30 +18,28 @@ static inline int do_single_reloc(
        switch(type) {
        case R_SH_GLOB_DAT:
        case R_SH_JMP_SLOT:
-               *reloc_addr = sym_val;
+       case R_SH_DIR32:
+               *reloc_addr = sym_val + addend;
                break;
        case R_SH_RELATIVE:
                *reloc_addr = (size_t)base_addr + addend;
                break;
-       case R_SH_DIR32:
-               *reloc_addr = sym_val + addend;
-               break;
        case R_SH_REL32:
-               *reloc_addr = sym_val + addend - (size_t)reloc_addr + (size_t)base_addr;
+               *reloc_addr = sym_val + addend - (size_t)reloc_addr;
                break;
        case R_SH_COPY:
                memcpy(reloc_addr, (void *)sym_val, sym_size);
                break;
        case R_SH_TLS_DTPMOD32:
-               *reloc_addr += def.dso ? def.dso->tls_id : self->tls_id;
+               *reloc_addr = def.dso ? def.dso->tls_id : self->tls_id;
                break;
        case R_SH_TLS_DTPOFF32:
-               *reloc_addr += def.sym->st_value;
+               *reloc_addr = def.sym->st_value + addend;
                break;
        case R_SH_TLS_TPOFF32:
-               *reloc_addr += def.sym
-                       ? def.sym->st_value + def.dso->tls_offset + 8
-                       : self->tls_offset + 8;
+               *reloc_addr = (def.sym
+                       ? def.sym->st_value + def.dso->tls_offset
+                       : self->tls_offset) + 8 + addend;
                break;
        }
        return 0;