Commit 04d82a6d authored by Damien Le Moal's avatar Damien Le Moal Committed by Greg Ungerer

binfmt_flat: allow not offsetting data start

Commit 2217b982 ("binfmt_flat: revert "binfmt_flat: don't offset
the data start"") restored offsetting the start of the data section by
a number of words defined by MAX_SHARED_LIBS. As a result, since
MAX_SHARED_LIBS is never 0, a gap between the text and data sections
always exists. For architectures which cannot support a such gap
between the text and data sections (e.g. riscv nommu), flat binary
programs cannot be executed.

To allow an architecture to request no data start offset to allow for
contiguous text and data sections for binaries flagged with
FLAT_FLAG_RAM, introduce the new config option
CONFIG_BINFMT_FLAT_NO_DATA_START_OFFSET. Using this new option, the
macro DATA_START_OFFSET_WORDS is conditionally defined in binfmt_flat.c
to MAX_SHARED_LIBS for architectures tolerating or needing the data
start offset (CONFIG_BINFMT_FLAT_NO_DATA_START_OFFSET disabled case)
and to 0 when CONFIG_BINFMT_FLAT_NO_DATA_START_OFFSET is enabled.
DATA_START_OFFSET_WORDS is used in load_flat_file() to calculate the
data section length and start position.
Signed-off-by: default avatarDamien Le Moal <damien.lemoal@wdc.com>
Signed-off-by: default avatarGreg Ungerer <gerg@linux-m68k.org>
parent bf05bf16
...@@ -112,6 +112,9 @@ config BINFMT_FLAT_ARGVP_ENVP_ON_STACK ...@@ -112,6 +112,9 @@ config BINFMT_FLAT_ARGVP_ENVP_ON_STACK
config BINFMT_FLAT_OLD_ALWAYS_RAM config BINFMT_FLAT_OLD_ALWAYS_RAM
bool bool
config BINFMT_FLAT_NO_DATA_START_OFFSET
bool
config BINFMT_FLAT_OLD config BINFMT_FLAT_OLD
bool "Enable support for very old legacy flat binaries" bool "Enable support for very old legacy flat binaries"
depends on BINFMT_FLAT depends on BINFMT_FLAT
......
...@@ -74,6 +74,12 @@ ...@@ -74,6 +74,12 @@
#define MAX_SHARED_LIBS (1) #define MAX_SHARED_LIBS (1)
#endif #endif
#ifdef CONFIG_BINFMT_FLAT_NO_DATA_START_OFFSET
#define DATA_START_OFFSET_WORDS (0)
#else
#define DATA_START_OFFSET_WORDS (MAX_SHARED_LIBS)
#endif
struct lib_info { struct lib_info {
struct { struct {
unsigned long start_code; /* Start of text segment */ unsigned long start_code; /* Start of text segment */
...@@ -576,7 +582,8 @@ static int load_flat_file(struct linux_binprm *bprm, ...@@ -576,7 +582,8 @@ static int load_flat_file(struct linux_binprm *bprm,
goto err; goto err;
} }
len = data_len + extra + MAX_SHARED_LIBS * sizeof(unsigned long); len = data_len + extra +
DATA_START_OFFSET_WORDS * sizeof(unsigned long);
len = PAGE_ALIGN(len); len = PAGE_ALIGN(len);
realdatastart = vm_mmap(NULL, 0, len, realdatastart = vm_mmap(NULL, 0, len,
PROT_READ|PROT_WRITE|PROT_EXEC, MAP_PRIVATE, 0); PROT_READ|PROT_WRITE|PROT_EXEC, MAP_PRIVATE, 0);
...@@ -591,7 +598,7 @@ static int load_flat_file(struct linux_binprm *bprm, ...@@ -591,7 +598,7 @@ static int load_flat_file(struct linux_binprm *bprm,
goto err; goto err;
} }
datapos = ALIGN(realdatastart + datapos = ALIGN(realdatastart +
MAX_SHARED_LIBS * sizeof(unsigned long), DATA_START_OFFSET_WORDS * sizeof(unsigned long),
FLAT_DATA_ALIGN); FLAT_DATA_ALIGN);
pr_debug("Allocated data+bss+stack (%u bytes): %lx\n", pr_debug("Allocated data+bss+stack (%u bytes): %lx\n",
...@@ -622,7 +629,8 @@ static int load_flat_file(struct linux_binprm *bprm, ...@@ -622,7 +629,8 @@ static int load_flat_file(struct linux_binprm *bprm,
memp_size = len; memp_size = len;
} else { } else {
len = text_len + data_len + extra + MAX_SHARED_LIBS * sizeof(u32); len = text_len + data_len + extra +
DATA_START_OFFSET_WORDS * sizeof(u32);
len = PAGE_ALIGN(len); len = PAGE_ALIGN(len);
textpos = vm_mmap(NULL, 0, len, textpos = vm_mmap(NULL, 0, len,
PROT_READ | PROT_EXEC | PROT_WRITE, MAP_PRIVATE, 0); PROT_READ | PROT_EXEC | PROT_WRITE, MAP_PRIVATE, 0);
...@@ -638,7 +646,7 @@ static int load_flat_file(struct linux_binprm *bprm, ...@@ -638,7 +646,7 @@ static int load_flat_file(struct linux_binprm *bprm,
realdatastart = textpos + ntohl(hdr->data_start); realdatastart = textpos + ntohl(hdr->data_start);
datapos = ALIGN(realdatastart + datapos = ALIGN(realdatastart +
MAX_SHARED_LIBS * sizeof(u32), DATA_START_OFFSET_WORDS * sizeof(u32),
FLAT_DATA_ALIGN); FLAT_DATA_ALIGN);
reloc = (__be32 __user *) reloc = (__be32 __user *)
...@@ -714,7 +722,7 @@ static int load_flat_file(struct linux_binprm *bprm, ...@@ -714,7 +722,7 @@ static int load_flat_file(struct linux_binprm *bprm,
ret = result; ret = result;
pr_err("Unable to read code+data+bss, errno %d\n", ret); pr_err("Unable to read code+data+bss, errno %d\n", ret);
vm_munmap(textpos, text_len + data_len + extra + vm_munmap(textpos, text_len + data_len + extra +
MAX_SHARED_LIBS * sizeof(u32)); DATA_START_OFFSET_WORDS * sizeof(u32));
goto err; goto err;
} }
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment