Add support for reading msdos MZ executables.

See email thread starting here:  https://www.sourceware.org/ml/binutils/2018-01/msg00001.html

include	* coff/msdos.h: New header.
	* coff/pe.h: Move common defines to msdos.h.
	* coff/powerpc.h: Likewise.

bfd	* i386msdos.c (msdos_mkobject); New function.
	(msdos_object_p): New function.
	(i386_msdos_vec): Use msdos_object_p as the check_format
	function.
	* peicode.h: Rename external_PEI_DOS_hdr, DOSMAGIC, and
	NT_SIGNATURE to external_DOS_hdr, IMAGE_DOS_SIGNATURE, and
	IMAGE_NT_SIGNATURE.
	* peXXigen.c: Likewise.
	* coff-ia64.c: Likewise.
This commit is contained in:
Zebediah Figura 2018-02-12 13:12:45 +00:00 committed by Nick Clifton
parent 7e784da543
commit 830db0485e
9 changed files with 170 additions and 43 deletions

View File

@ -1,3 +1,15 @@
2018-02-12 Zebediah Figura <z.figura12@gmail.com>
* i386msdos.c (msdos_mkobject); New function.
(msdos_object_p): New function.
(i386_msdos_vec): Use msdos_object_p as the check_format
function.
* peicode.h: Rename external_PEI_DOS_hdr, DOSMAGIC, and
NT_SIGNATURE to external_DOS_hdr, IMAGE_DOS_SIGNATURE, and
IMAGE_NT_SIGNATURE.
* peXXigen.c: Likewise.
* coff-ia64.c: Likewise.
2018-02-12 Nick Clifton <nickc@redhat.com>
* elf32-nds32.c (nds32_elf_relax_longjump3): Remove redundant

View File

@ -72,7 +72,7 @@ ia64coff_object_p (bfd *abfd)
{
#ifdef COFF_IMAGE_WITH_PE
{
struct external_PEI_DOS_hdr dos_hdr;
struct external_DOS_hdr dos_hdr;
struct external_PEI_IMAGE_hdr image_hdr;
file_ptr offset;
@ -87,7 +87,7 @@ ia64coff_object_p (bfd *abfd)
/* There are really two magic numbers involved; the magic number
that says this is a NT executable (PEI) and the magic number
that determines the architecture. The former is DOSMAGIC,
that determines the architecture. The former is IMAGE_DOS_SIGNATURE,
stored in the e_magic field. The latter is stored in the
f_magic field. If the NT magic number isn't valid, the
architecture magic number could be mimicked by some other
@ -95,7 +95,7 @@ ia64coff_object_p (bfd *abfd)
this routine can only be called correctly for a PEI file, check
the e_magic number here, and, if it doesn't match, clobber the
f_magic number so that we don't get a false match. */
if (H_GET_16 (abfd, dos_hdr.e_magic) != DOSMAGIC)
if (H_GET_16 (abfd, dos_hdr.e_magic) != IMAGE_DOS_SIGNATURE)
{
bfd_set_error (bfd_error_wrong_format);
return NULL;

View File

@ -27,12 +27,94 @@
#include "bfd.h"
#include "libbfd.h"
#include "libaout.h"
#include "coff/msdos.h"
#define EXE_MAGIC 0x5a4d
#define EXE_LOAD_HIGH 0x0000
#define EXE_LOAD_LOW 0xffff
#define EXE_PAGE_SIZE 512
static bfd_boolean
msdos_mkobject (bfd *abfd)
{
bfd_default_set_arch_mach (abfd, bfd_arch_i386, bfd_mach_i386_i8086);
return aout_32_mkobject (abfd);
}
static const bfd_target *
msdos_object_p (bfd *abfd)
{
struct external_DOS_hdr hdr;
bfd_byte buffer[2];
asection *section;
unsigned int size;
if (bfd_seek (abfd, (file_ptr) 0, SEEK_SET) != 0
|| bfd_bread (&hdr, (bfd_size_type) sizeof (hdr), abfd) < DOS_HDR_SIZE)
{
if (bfd_get_error () != bfd_error_system_call)
bfd_set_error (bfd_error_wrong_format);
return NULL;
}
if (H_GET_16 (abfd, hdr.e_magic) != IMAGE_DOS_SIGNATURE)
{
bfd_set_error (bfd_error_wrong_format);
return NULL;
}
/* Check that this isn't actually a PE, NE, or LE file. If it is, the
e_lfanew field will be valid and point to a header beginning with one of
the relevant signatures. If not, e_lfanew might point to anything, so
don't bail if we can't read there. */
if (H_GET_16 (abfd, hdr.e_cparhdr) < 4
|| bfd_seek (abfd, (file_ptr) H_GET_32 (abfd, hdr.e_lfanew), SEEK_SET) != 0
|| bfd_bread (buffer, (bfd_size_type) 2, abfd) != 2)
{
if (bfd_get_error () == bfd_error_system_call)
return NULL;
}
else
{
if (H_GET_16 (abfd, buffer) == IMAGE_NT_SIGNATURE
|| H_GET_16 (abfd, buffer) == IMAGE_OS2_SIGNATURE
|| H_GET_16 (abfd, buffer) == IMAGE_OS2_SIGNATURE_LE
|| H_GET_16 (abfd, buffer) == IMAGE_OS2_SIGNATURE_LX)
{
bfd_set_error (bfd_error_wrong_format);
return NULL;
}
}
if (!msdos_mkobject (abfd))
return NULL;
abfd->flags = EXEC_P;
abfd->start_address = H_GET_16 (abfd, hdr.e_ip);
section = bfd_make_section (abfd, ".text");
if (section == NULL)
return NULL;
section->flags = (SEC_ALLOC | SEC_LOAD | SEC_CODE | SEC_HAS_CONTENTS);
section->filepos = H_GET_16 (abfd, hdr.e_cparhdr) * 16;
size = (H_GET_16 (abfd, hdr.e_cp) - 1) * EXE_PAGE_SIZE - section->filepos;
size += H_GET_16 (abfd, hdr.e_cblp);
/* Check that the size is valid. */
if (bfd_seek (abfd, (file_ptr) (section->filepos + size), SEEK_SET) != 0)
{
if (bfd_get_error () != bfd_error_system_call)
bfd_set_error (bfd_error_wrong_format);
return NULL;
}
bfd_set_section_size (abfd, section, size);
section->alignment_power = 4;
return abfd->xvec;
}
static int
msdos_sizeof_headers (bfd *abfd ATTRIBUTE_UNUSED,
struct bfd_link_info *info ATTRIBUTE_UNUSED)
@ -77,7 +159,7 @@ msdos_write_object_contents (bfd *abfd)
}
/* Constants. */
H_PUT_16 (abfd, EXE_MAGIC, &hdr[0]);
H_PUT_16 (abfd, IMAGE_DOS_SIGNATURE, &hdr[0]);
H_PUT_16 (abfd, EXE_PAGE_SIZE / 16, &hdr[8]);
H_PUT_16 (abfd, EXE_LOAD_LOW, &hdr[12]);
H_PUT_16 (abfd, 0x3e, &hdr[24]);
@ -127,7 +209,6 @@ msdos_set_section_contents (bfd *abfd,
#define msdos_mkobject aout_32_mkobject
#define msdos_make_empty_symbol aout_32_make_empty_symbol
#define msdos_bfd_reloc_type_lookup aout_32_reloc_type_lookup
#define msdos_bfd_reloc_name_lookup aout_32_reloc_name_lookup
@ -203,7 +284,7 @@ const bfd_target i386_msdos_vec =
{
_bfd_dummy_target,
_bfd_dummy_target, /* bfd_check_format */
msdos_object_p, /* bfd_check_format */
_bfd_dummy_target,
_bfd_dummy_target,
},

View File

@ -828,7 +828,7 @@ _bfd_XXi_only_swap_filehdr_out (bfd * abfd, void * in, void * out)
if (pe_data (abfd)->dll)
filehdr_in->f_flags |= F_DLL;
filehdr_in->pe.e_magic = DOSMAGIC;
filehdr_in->pe.e_magic = IMAGE_DOS_SIGNATURE;
filehdr_in->pe.e_cblp = 0x90;
filehdr_in->pe.e_cp = 0x3;
filehdr_in->pe.e_crlc = 0x0;
@ -872,7 +872,7 @@ _bfd_XXi_only_swap_filehdr_out (bfd * abfd, void * in, void * out)
filehdr_in->pe.dos_message[13] = 0x0a0d0d2e;
filehdr_in->pe.dos_message[14] = 0x24;
filehdr_in->pe.dos_message[15] = 0x0;
filehdr_in->pe.nt_signature = NT_SIGNATURE;
filehdr_in->pe.nt_signature = IMAGE_NT_SIGNATURE;
H_PUT_16 (abfd, filehdr_in->f_magic, filehdr_out->f_magic);
H_PUT_16 (abfd, filehdr_in->f_nscns, filehdr_out->f_nscns);

View File

@ -1386,7 +1386,7 @@ static const bfd_target *
pe_bfd_object_p (bfd * abfd)
{
bfd_byte buffer[6];
struct external_PEI_DOS_hdr dos_hdr;
struct external_DOS_hdr dos_hdr;
struct external_PEI_IMAGE_hdr image_hdr;
struct internal_filehdr internal_f;
struct internal_aouthdr internal_a;
@ -1420,7 +1420,7 @@ pe_bfd_object_p (bfd * abfd)
/* There are really two magic numbers involved; the magic number
that says this is a NT executable (PEI) and the magic number that
determines the architecture. The former is DOSMAGIC, stored in
determines the architecture. The former is IMAGE_DOS_SIGNATURE, stored in
the e_magic field. The latter is stored in the f_magic field.
If the NT magic number isn't valid, the architecture magic number
could be mimicked by some other field (specifically, the number
@ -1428,7 +1428,7 @@ pe_bfd_object_p (bfd * abfd)
correctly for a PEI file, check the e_magic number here, and, if
it doesn't match, clobber the f_magic number so that we don't get
a false match. */
if (H_GET_16 (abfd, dos_hdr.e_magic) != DOSMAGIC)
if (H_GET_16 (abfd, dos_hdr.e_magic) != IMAGE_DOS_SIGNATURE)
{
bfd_set_error (bfd_error_wrong_format);
return NULL;

View File

@ -1,3 +1,9 @@
2018-02-12 Zebediah Figura <z.figura12@gmail.com>
* coff/msdos.h: New header.
* coff/pe.h: Move common defines to msdos.h.
* coff/powerpc.h: Likewise.
2018-01-13 Nick Clifton <nickc@redhat.com>
2.30 branch created.

57
include/coff/msdos.h Normal file
View File

@ -0,0 +1,57 @@
/* msdos.h - MS-DOS and derived executable header information
Copyright (C) 1999-2018 Free Software Foundation, Inc.
This file is part of BFD, the Binary File Descriptor library.
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software Foundation,
Inc., 51 Franklin Street - Fifth Floor, Boston, MA 02110-1301, USA. */
#ifndef _MSDOS_H
#define _MSDOS_H
#define IMAGE_DOS_SIGNATURE 0x5a4d /* MZ */
#define IMAGE_OS2_SIGNATURE 0x454e /* NE */
#define IMAGE_OS2_SIGNATURE_LE 0x454c /* LE */
#define IMAGE_OS2_SIGNATURE_LX 0x584c /* LX */
#define IMAGE_NT_SIGNATURE 0x00004550 /* PE\0\0 */
struct external_DOS_hdr
{
/* DOS header fields - always at offset zero in the EXE file. */
char e_magic[2]; /* Magic number. */
char e_cblp[2]; /* Bytes on last page of file. */
char e_cp[2]; /* Pages in file. */
char e_crlc[2]; /* Relocations. */
char e_cparhdr[2]; /* Size of header in paragraphs. */
char e_minalloc[2]; /* Minimum extra paragraphs needed. */
char e_maxalloc[2]; /* Maximum extra paragraphs needed. */
char e_ss[2]; /* Initial (relative) SS value. */
char e_sp[2]; /* Initial SP value. */
char e_csum[2]; /* Checksum. */
char e_ip[2]; /* Initial IP value. */
char e_cs[2]; /* Initial (relative) CS value. */
char e_lfarlc[2]; /* File address of relocation table. */
char e_ovno[2]; /* Overlay number. */
char e_res[4][2]; /* Reserved words, all 0x0. */
char e_oemid[2]; /* OEM identifier. */
char e_oeminfo[2]; /* OEM information. */
char e_res2[10][2]; /* Reserved words, all 0x0. */
char e_lfanew[4]; /* File address of new exe header, usually 0x80. */
char dos_message[16][4]; /* Other stuff, always follow DOS header. */
};
/* The actual DOS header only includes up to the e_ovno field. */
#define DOS_HDR_SIZE (offsetof (struct external_DOS_hdr, e_res))
#endif /* _MSDOS_H */

View File

@ -20,6 +20,8 @@
#ifndef _PE_H
#define _PE_H
#include "msdos.h"
/* NT specific file attributes. */
#define IMAGE_FILE_RELOCS_STRIPPED 0x0001
#define IMAGE_FILE_EXECUTABLE_IMAGE 0x0002
@ -168,40 +170,11 @@
#define IMAGE_SUBSYSTEM_SAL_RUNTIME_DRIVER 13
#define IMAGE_SUBSYSTEM_XBOX 14
/* Magic values that are true for all dos/nt implementations. */
#define DOSMAGIC 0x5a4d
#define NT_SIGNATURE 0x00004550
/* NT allows long filenames, we want to accommodate this.
This may break some of the bfd functions. */
#undef FILNMLEN
#define FILNMLEN 18 /* # characters in a file name. */
struct external_PEI_DOS_hdr
{
/* DOS header fields - always at offset zero in the EXE file. */
char e_magic[2]; /* Magic number, 0x5a4d. */
char e_cblp[2]; /* Bytes on last page of file, 0x90. */
char e_cp[2]; /* Pages in file, 0x3. */
char e_crlc[2]; /* Relocations, 0x0. */
char e_cparhdr[2]; /* Size of header in paragraphs, 0x4. */
char e_minalloc[2]; /* Minimum extra paragraphs needed, 0x0. */
char e_maxalloc[2]; /* Maximum extra paragraphs needed, 0xFFFF. */
char e_ss[2]; /* Initial (relative) SS value, 0x0. */
char e_sp[2]; /* Initial SP value, 0xb8. */
char e_csum[2]; /* Checksum, 0x0. */
char e_ip[2]; /* Initial IP value, 0x0. */
char e_cs[2]; /* Initial (relative) CS value, 0x0. */
char e_lfarlc[2]; /* File address of relocation table, 0x40. */
char e_ovno[2]; /* Overlay number, 0x0. */
char e_res[4][2]; /* Reserved words, all 0x0. */
char e_oemid[2]; /* OEM identifier (for e_oeminfo), 0x0. */
char e_oeminfo[2]; /* OEM information; e_oemid specific, 0x0. */
char e_res2[10][2]; /* Reserved words, all 0x0. */
char e_lfanew[4]; /* File address of new exe header, usually 0x80. */
char dos_message[16][4]; /* Other stuff, always follow DOS header. */
};
struct external_PEI_IMAGE_hdr
{
char nt_signature[4]; /* Required NT signature, 0x4550. */

View File

@ -37,8 +37,6 @@
/* extra NT defines */
#define PPCMAGIC 0760 /* peeked on aa PowerPC Windows NT box */
#define DOSMAGIC 0x5a4d /* from arm.h, i386.h */
#define NT_SIGNATURE 0x00004550 /* from arm.h, i386.h */
/* from winnt.h */
#define IMAGE_NT_OPTIONAL_HDR_MAGIC 0x10b