summaryrefslogtreecommitdiffstats
path: root/sys/boot
diff options
context:
space:
mode:
authorgrehan <grehan@FreeBSD.org>2014-11-24 01:25:19 +0000
committergrehan <grehan@FreeBSD.org>2014-11-24 01:25:19 +0000
commit2c63daa4b3c51ef90b98764664f4254c69cea0dd (patch)
tree6bcbb371222165055a7a9765b2e843b77557573c /sys/boot
parenta3fd5530e74784b21a4ccedadde1180f84802e48 (diff)
downloadFreeBSD-src-2c63daa4b3c51ef90b98764664f4254c69cea0dd.zip
FreeBSD-src-2c63daa4b3c51ef90b98764664f4254c69cea0dd.tar.gz
MFC r274407
Fix incorrect reading of 32-bit modinfo by 64-bit loaders. The various structures in the mod_metadata set of a FreeBSD kernel and modules contain pointers. The FreeBSD loader correctly deals with a mismatch in loader and kernel pointer size (e.g. 32-bit i386/ppc loader, loading 64-bit amd64/ppc64 kernels), but wasn't dealing with the inverse case where a 64-bit loader was loading a 32-bit kernel.
Diffstat (limited to 'sys/boot')
-rw-r--r--sys/boot/common/load_elf.c22
1 files changed, 22 insertions, 0 deletions
diff --git a/sys/boot/common/load_elf.c b/sys/boot/common/load_elf.c
index 8990d90..672c566 100644
--- a/sys/boot/common/load_elf.c
+++ b/sys/boot/common/load_elf.c
@@ -610,6 +610,14 @@ struct mod_metadata64 {
u_int64_t md_cval; /* common string label */
};
#endif
+#if defined(__amd64__) && __ELF_WORD_SIZE == 32
+struct mod_metadata32 {
+ int md_version; /* structure version MDTV_* */
+ int md_type; /* type of entry MDT_* */
+ u_int32_t md_data; /* specific data */
+ u_int32_t md_cval; /* common string label */
+};
+#endif
int
__elfN(parse_modmetadata)(struct preloaded_file *fp, elf_file_t ef)
@@ -617,6 +625,8 @@ __elfN(parse_modmetadata)(struct preloaded_file *fp, elf_file_t ef)
struct mod_metadata md;
#if (defined(__i386__) || defined(__powerpc__)) && __ELF_WORD_SIZE == 64
struct mod_metadata64 md64;
+#elif defined(__amd64__) && __ELF_WORD_SIZE == 32
+ struct mod_metadata32 md32;
#endif
struct mod_depend *mdepend;
struct mod_version mver;
@@ -652,6 +662,18 @@ __elfN(parse_modmetadata)(struct preloaded_file *fp, elf_file_t ef)
md.md_type = md64.md_type;
md.md_cval = (const char *)(uintptr_t)md64.md_cval;
md.md_data = (void *)(uintptr_t)md64.md_data;
+#elif defined(__amd64__) && __ELF_WORD_SIZE == 32
+ COPYOUT(v, &md32, sizeof(md32));
+ error = __elfN(reloc_ptr)(fp, ef, v, &md32, sizeof(md32));
+ if (error == EOPNOTSUPP) {
+ md32.md_cval += ef->off;
+ md32.md_data += ef->off;
+ } else if (error != 0)
+ return (error);
+ md.md_version = md32.md_version;
+ md.md_type = md32.md_type;
+ md.md_cval = (const char *)(uintptr_t)md32.md_cval;
+ md.md_data = (void *)(uintptr_t)md32.md_data;
#else
COPYOUT(v, &md, sizeof(md));
error = __elfN(reloc_ptr)(fp, ef, v, &md, sizeof(md));
OpenPOWER on IntegriCloud