binfmt/elf: Support loading fully linked executables.

The following changes make it possible for Nuttx to load binaries in ELF format which are fully linked.

The change does not include the necessary modifications to produce such binaries. In order to build an applicable binary:
 - The userspace applications linker script (`gnu-elf.ld`) needs to modified so the data and text section origin's match those setup by the address environment.
 - The makefile used, in `apps/import/Make.defs` needs to remove the `-r` LDELFFLAG.
This commit is contained in:
Stuart Ianna 2023-05-26 14:22:36 +10:00 committed by Alan Carvalho de Assis
parent c12a122663
commit a05e8fd9ff
2 changed files with 32 additions and 7 deletions

View File

@ -247,6 +247,8 @@ static int elf_loadbinary(FAR struct binary_s *binp,
/* Bind the program to the exported symbol table */ /* Bind the program to the exported symbol table */
if (loadinfo.ehdr.e_type == ET_REL)
{
ret = elf_bind(&loadinfo, exports, nexports); ret = elf_bind(&loadinfo, exports, nexports);
if (ret != 0) if (ret != 0)
{ {
@ -254,9 +256,31 @@ static int elf_loadbinary(FAR struct binary_s *binp,
goto errout_with_load; goto errout_with_load;
} }
binp->entrypt = (main_t)(loadinfo.textalloc + loadinfo.ehdr.e_entry);
}
else if (loadinfo.ehdr.e_type == ET_EXEC)
{
if (nexports > 0)
{
berr("Cannot bind exported symbols to a "\
"fully linked executable\n");
ret = -ENOEXEC;
goto errout_with_load;
}
/* The entrypoint for a fully linked executable can be found directly */
binp->entrypt = (main_t)(loadinfo.ehdr.e_entry);
}
else
{
berr("Unexpected elf type %d\n", loadinfo.ehdr.e_type);
ret = -ENOEXEC;
}
/* Return the load information */ /* Return the load information */
binp->entrypt = (main_t)(loadinfo.textalloc + loadinfo.ehdr.e_entry);
binp->stacksize = CONFIG_ELF_STACKSIZE; binp->stacksize = CONFIG_ELF_STACKSIZE;
/* Add the ELF allocation to the alloc[] only if there is no address /* Add the ELF allocation to the alloc[] only if there is no address

View File

@ -89,9 +89,10 @@ int elf_verifyheader(FAR const Elf_Ehdr *ehdr)
/* Verify that this is a relocatable file */ /* Verify that this is a relocatable file */
if (ehdr->e_type != ET_REL) if ((ehdr->e_type != ET_REL) && (ehdr->e_type != ET_EXEC))
{ {
berr("Not a relocatable file: e_type=%d\n", ehdr->e_type); berr("Not a relocatable or executable file: e_type=%d\n",
ehdr->e_type);
return -EINVAL; return -EINVAL;
} }