| /* This file is derived from the GAS 2.1.4 assembler control file. | 
 |    The GAS product is under the GNU General Public License, version 2 or later. | 
 |    As such, this file is also under that license. | 
 |  | 
 |    If the file format changes in the COFF object, this file should be | 
 |    subsequently updated to reflect the changes. | 
 |  | 
 |    The actual loader module only uses a few of these structures. The full | 
 |    set is documented here because I received the full set. If you wish | 
 |    more information about COFF, then O'Reilly has a very excellent book. | 
 | */ | 
 |  | 
 | #define  E_SYMNMLEN  8   /* Number of characters in a symbol name         */ | 
 | #define  E_FILNMLEN 14   /* Number of characters in a file name           */ | 
 | #define  E_DIMNUM    4   /* Number of array dimensions in auxiliary entry */ | 
 |  | 
 | /* | 
 |  * These defines are byte order independent. There is no alignment of fields | 
 |  * permitted in the structures. Therefore they are declared as characters | 
 |  * and the values loaded from the character positions. It also makes it | 
 |  * nice to have it "endian" independent. | 
 |  */ | 
 |   | 
 | /* Load a short int from the following tables with little-endian formats */ | 
 | #define COFF_SHORT_L(ps) ((short)(((unsigned short)((unsigned char)ps[1])<<8)|\ | 
 | 				  ((unsigned short)((unsigned char)ps[0])))) | 
 |  | 
 | /* Load a long int from the following tables with little-endian formats */ | 
 | #define COFF_LONG_L(ps) (((long)(((unsigned long)((unsigned char)ps[3])<<24) |\ | 
 | 				 ((unsigned long)((unsigned char)ps[2])<<16) |\ | 
 | 				 ((unsigned long)((unsigned char)ps[1])<<8)  |\ | 
 | 				 ((unsigned long)((unsigned char)ps[0]))))) | 
 |   | 
 | /* Load a short int from the following tables with big-endian formats */ | 
 | #define COFF_SHORT_H(ps) ((short)(((unsigned short)((unsigned char)ps[0])<<8)|\ | 
 | 				  ((unsigned short)((unsigned char)ps[1])))) | 
 |  | 
 | /* Load a long int from the following tables with big-endian formats */ | 
 | #define COFF_LONG_H(ps) (((long)(((unsigned long)((unsigned char)ps[0])<<24) |\ | 
 | 				 ((unsigned long)((unsigned char)ps[1])<<16) |\ | 
 | 				 ((unsigned long)((unsigned char)ps[2])<<8)  |\ | 
 | 				 ((unsigned long)((unsigned char)ps[3]))))) | 
 |  | 
 | /* These may be overridden later by brain dead implementations which generate | 
 |    a big-endian header with little-endian data. In that case, generate a | 
 |    replacement macro which tests a flag and uses either of the two above | 
 |    as appropriate. */ | 
 |  | 
 | #define COFF_LONG(v)   COFF_LONG_L(v) | 
 | #define COFF_SHORT(v)  COFF_SHORT_L(v) | 
 |  | 
 | /*** coff information for Intel 386/486.  */ | 
 |  | 
 | /********************** FILE HEADER **********************/ | 
 |  | 
 | struct COFF_filehdr { | 
 | 	char f_magic[2];	/* magic number			*/ | 
 | 	char f_nscns[2];	/* number of sections		*/ | 
 | 	char f_timdat[4];	/* time & date stamp		*/ | 
 | 	char f_symptr[4];	/* file pointer to symtab	*/ | 
 | 	char f_nsyms[4];	/* number of symtab entries	*/ | 
 | 	char f_opthdr[2];	/* sizeof(optional hdr)		*/ | 
 | 	char f_flags[2];	/* flags			*/ | 
 | }; | 
 |  | 
 | /* | 
 |  *   Bits for f_flags: | 
 |  * | 
 |  *	F_RELFLG	relocation info stripped from file | 
 |  *	F_EXEC		file is executable  (i.e. no unresolved external | 
 |  *			references) | 
 |  *	F_LNNO		line numbers stripped from file | 
 |  *	F_LSYMS		local symbols stripped from file | 
 |  *	F_MINMAL	this is a minimal object file (".m") output of fextract | 
 |  *	F_UPDATE	this is a fully bound update file, output of ogen | 
 |  *	F_SWABD		this file has had its bytes swabbed (in names) | 
 |  *	F_AR16WR	this file has the byte ordering of an AR16WR | 
 |  *			(e.g. 11/70) machine | 
 |  *	F_AR32WR	this file has the byte ordering of an AR32WR machine | 
 |  *			(e.g. vax and iNTEL 386) | 
 |  *	F_AR32W		this file has the byte ordering of an AR32W machine | 
 |  *			(e.g. 3b,maxi) | 
 |  *	F_PATCH		file contains "patch" list in optional header | 
 |  *	F_NODF		(minimal file only) no decision functions for | 
 |  *			replaced functions | 
 |  */ | 
 |  | 
 | #define  COFF_F_RELFLG		0000001 | 
 | #define  COFF_F_EXEC		0000002 | 
 | #define  COFF_F_LNNO		0000004 | 
 | #define  COFF_F_LSYMS		0000010 | 
 | #define  COFF_F_MINMAL		0000020 | 
 | #define  COFF_F_UPDATE		0000040 | 
 | #define  COFF_F_SWABD		0000100 | 
 | #define  COFF_F_AR16WR		0000200 | 
 | #define  COFF_F_AR32WR		0000400 | 
 | #define  COFF_F_AR32W		0001000 | 
 | #define  COFF_F_PATCH		0002000 | 
 | #define  COFF_F_NODF		0002000 | 
 |  | 
 | #define	COFF_I386MAGIC	        0x14c   /* Linux's system    */ | 
 |  | 
 | #if 0   /* Perhaps, someday, these formats may be used.      */ | 
 | #define COFF_I386PTXMAGIC	0x154 | 
 | #define COFF_I386AIXMAGIC	0x175   /* IBM's AIX system  */ | 
 | #define COFF_I386BADMAG(x) ((COFF_SHORT((x).f_magic) != COFF_I386MAGIC) \ | 
 | 			  && COFF_SHORT((x).f_magic) != COFF_I386PTXMAGIC \ | 
 | 			  && COFF_SHORT((x).f_magic) != COFF_I386AIXMAGIC) | 
 | #else | 
 | #define COFF_I386BADMAG(x) (COFF_SHORT((x).f_magic) != COFF_I386MAGIC) | 
 | #endif | 
 |  | 
 | #define	COFF_FILHDR	struct COFF_filehdr | 
 | #define	COFF_FILHSZ	sizeof(COFF_FILHDR) | 
 |  | 
 | /********************** AOUT "OPTIONAL HEADER" **********************/ | 
 |  | 
 | /* Linux COFF must have this "optional" header. Standard COFF has no entry | 
 |    location for the "entry" point. They normally would start with the first | 
 |    location of the .text section. This is not a good idea for linux. So, | 
 |    the use of this "optional" header is not optional. It is required. | 
 |  | 
 |    Do not be tempted to assume that the size of the optional header is | 
 |    a constant and simply index the next byte by the size of this structure. | 
 |    Use the 'f_opthdr' field in the main coff header for the size of the | 
 |    structure actually written to the file!! | 
 | */ | 
 |  | 
 | typedef struct  | 
 | { | 
 |   char 	magic[2];		/* type of file				 */ | 
 |   char	vstamp[2];		/* version stamp			 */ | 
 |   char	tsize[4];		/* text size in bytes, padded to FW bdry */ | 
 |   char	dsize[4];		/* initialized   data "   "		 */ | 
 |   char	bsize[4];		/* uninitialized data "   "		 */ | 
 |   char	entry[4];		/* entry pt.				 */ | 
 |   char 	text_start[4];		/* base of text used for this file       */ | 
 |   char 	data_start[4];		/* base of data used for this file       */ | 
 | } | 
 | COFF_AOUTHDR; | 
 |  | 
 | #define COFF_AOUTSZ (sizeof(COFF_AOUTHDR)) | 
 |  | 
 | #define COFF_STMAGIC	0401 | 
 | #define COFF_OMAGIC     0404 | 
 | #define COFF_JMAGIC     0407    /* dirty text and data image, can't share  */ | 
 | #define COFF_DMAGIC     0410    /* dirty text segment, data aligned        */ | 
 | #define COFF_ZMAGIC     0413    /* The proper magic number for executables  */ | 
 | #define COFF_SHMAGIC	0443	/* shared library header                   */ | 
 |  | 
 | /********************** SECTION HEADER **********************/ | 
 |  | 
 | struct COFF_scnhdr { | 
 |   char		s_name[8];	/* section name			    */ | 
 |   char		s_paddr[4];	/* physical address, aliased s_nlib */ | 
 |   char		s_vaddr[4];	/* virtual address		    */ | 
 |   char		s_size[4];	/* section size			    */ | 
 |   char		s_scnptr[4];	/* file ptr to raw data for section */ | 
 |   char		s_relptr[4];	/* file ptr to relocation	    */ | 
 |   char		s_lnnoptr[4];	/* file ptr to line numbers	    */ | 
 |   char		s_nreloc[2];	/* number of relocation entries	    */ | 
 |   char		s_nlnno[2];	/* number of line number entries    */ | 
 |   char		s_flags[4];	/* flags			    */ | 
 | }; | 
 |  | 
 | #define	COFF_SCNHDR	struct COFF_scnhdr | 
 | #define	COFF_SCNHSZ	sizeof(COFF_SCNHDR) | 
 |  | 
 | /* | 
 |  * names of "special" sections | 
 |  */ | 
 |  | 
 | #define COFF_TEXT	".text" | 
 | #define COFF_DATA	".data" | 
 | #define COFF_BSS	".bss" | 
 | #define COFF_COMMENT    ".comment" | 
 | #define COFF_LIB        ".lib" | 
 |  | 
 | #define COFF_SECT_TEXT  0      /* Section for instruction code             */ | 
 | #define COFF_SECT_DATA  1      /* Section for initialized globals          */ | 
 | #define COFF_SECT_BSS   2      /* Section for un-initialized globals       */ | 
 | #define COFF_SECT_REQD  3      /* Minimum number of sections for good file */ | 
 |  | 
 | #define COFF_STYP_REG     0x00 /* regular segment                          */ | 
 | #define COFF_STYP_DSECT   0x01 /* dummy segment                            */ | 
 | #define COFF_STYP_NOLOAD  0x02 /* no-load segment                          */ | 
 | #define COFF_STYP_GROUP   0x04 /* group segment                            */ | 
 | #define COFF_STYP_PAD     0x08 /* .pad segment                             */ | 
 | #define COFF_STYP_COPY    0x10 /* copy section                             */ | 
 | #define COFF_STYP_TEXT    0x20 /* .text segment                            */ | 
 | #define COFF_STYP_DATA    0x40 /* .data segment                            */ | 
 | #define COFF_STYP_BSS     0x80 /* .bss segment                             */ | 
 | #define COFF_STYP_INFO   0x200 /* .comment section                         */ | 
 | #define COFF_STYP_OVER   0x400 /* overlay section                          */ | 
 | #define COFF_STYP_LIB    0x800 /* library section                          */ | 
 |  | 
 | /* | 
 |  * Shared libraries have the following section header in the data field for | 
 |  * each library. | 
 |  */ | 
 |  | 
 | struct COFF_slib { | 
 |   char		sl_entsz[4];	/* Size of this entry               */ | 
 |   char		sl_pathndx[4];	/* size of the header field         */ | 
 | }; | 
 |  | 
 | #define	COFF_SLIBHD	struct COFF_slib | 
 | #define	COFF_SLIBSZ	sizeof(COFF_SLIBHD) | 
 |  | 
 | /********************** LINE NUMBERS **********************/ | 
 |  | 
 | /* 1 line number entry for every "breakpointable" source line in a section. | 
 |  * Line numbers are grouped on a per function basis; first entry in a function | 
 |  * grouping will have l_lnno = 0 and in place of physical address will be the | 
 |  * symbol table index of the function name. | 
 |  */ | 
 |  | 
 | struct COFF_lineno { | 
 |   union { | 
 |     char l_symndx[4];	/* function name symbol index, iff l_lnno == 0*/ | 
 |     char l_paddr[4];	/* (physical) address of line number	*/ | 
 |   } l_addr; | 
 |   char l_lnno[2];	/* line number		*/ | 
 | }; | 
 |  | 
 | #define	COFF_LINENO	struct COFF_lineno | 
 | #define	COFF_LINESZ	6 | 
 |  | 
 | /********************** SYMBOLS **********************/ | 
 |  | 
 | #define COFF_E_SYMNMLEN	 8	/* # characters in a short symbol name	*/ | 
 | #define COFF_E_FILNMLEN	14	/* # characters in a file name		*/ | 
 | #define COFF_E_DIMNUM	 4	/* # array dimensions in auxiliary entry */ | 
 |  | 
 | /* | 
 |  *  All symbols and sections have the following definition | 
 |  */ | 
 |  | 
 | struct COFF_syment  | 
 | { | 
 |   union { | 
 |     char e_name[E_SYMNMLEN];    /* Symbol name (first 8 characters) */ | 
 |     struct { | 
 |       char e_zeroes[4];         /* Leading zeros */ | 
 |       char e_offset[4];         /* Offset if this is a header section */ | 
 |     } e; | 
 |   } e; | 
 |  | 
 |   char e_value[4];              /* Value (address) of the segment */ | 
 |   char e_scnum[2];              /* Section number */ | 
 |   char e_type[2];               /* Type of section */ | 
 |   char e_sclass[1];             /* Loader class */ | 
 |   char e_numaux[1];             /* Number of auxiliary entries which follow */ | 
 | }; | 
 |  | 
 | #define COFF_N_BTMASK	(0xf)   /* Mask for important class bits */ | 
 | #define COFF_N_TMASK	(0x30)  /* Mask for important type bits  */ | 
 | #define COFF_N_BTSHFT	(4)     /* # bits to shift class field   */ | 
 | #define COFF_N_TSHIFT	(2)     /* # bits to shift type field    */ | 
 |  | 
 | /* | 
 |  *  Auxiliary entries because the main table is too limiting. | 
 |  */ | 
 |    | 
 | union COFF_auxent { | 
 |  | 
 | /* | 
 |  *  Debugger information | 
 |  */ | 
 |  | 
 |   struct { | 
 |     char x_tagndx[4];	        /* str, un, or enum tag indx */ | 
 |     union { | 
 |       struct { | 
 | 	char  x_lnno[2];        /* declaration line number */ | 
 | 	char  x_size[2];        /* str/union/array size */ | 
 |       } x_lnsz; | 
 |       char x_fsize[4];	        /* size of function */ | 
 |     } x_misc; | 
 |  | 
 |     union { | 
 |       struct {		        /* if ISFCN, tag, or .bb */ | 
 | 	char x_lnnoptr[4];	/* ptr to fcn line # */ | 
 | 	char x_endndx[4];	/* entry ndx past block end */ | 
 |       } x_fcn; | 
 |  | 
 |       struct {		        /* if ISARY, up to 4 dimen. */ | 
 | 	char x_dimen[E_DIMNUM][2]; | 
 |       } x_ary; | 
 |     } x_fcnary; | 
 |  | 
 |     char x_tvndx[2];	/* tv index */ | 
 |   } x_sym; | 
 |  | 
 | /* | 
 |  *   Source file names (debugger information) | 
 |  */ | 
 |  | 
 |   union { | 
 |     char x_fname[E_FILNMLEN]; | 
 |     struct { | 
 |       char x_zeroes[4]; | 
 |       char x_offset[4]; | 
 |     } x_n; | 
 |   } x_file; | 
 |  | 
 | /* | 
 |  *   Section information | 
 |  */ | 
 |  | 
 |   struct { | 
 |     char x_scnlen[4];	/* section length */ | 
 |     char x_nreloc[2];	/* # relocation entries */ | 
 |     char x_nlinno[2];	/* # line numbers */ | 
 |   } x_scn; | 
 |  | 
 | /* | 
 |  *   Transfer vector (branch table) | 
 |  */ | 
 |    | 
 |   struct { | 
 |     char x_tvfill[4];	/* tv fill value */ | 
 |     char x_tvlen[2];	/* length of .tv */ | 
 |     char x_tvran[2][2];	/* tv range */ | 
 |   } x_tv;		/* info about .tv section (in auxent of symbol .tv)) */ | 
 | }; | 
 |  | 
 | #define	COFF_SYMENT	struct COFF_syment | 
 | #define	COFF_SYMESZ	18	 | 
 | #define	COFF_AUXENT	union COFF_auxent | 
 | #define	COFF_AUXESZ	18 | 
 |  | 
 | #define COFF_ETEXT	"etext" | 
 |  | 
 | /********************** RELOCATION DIRECTIVES **********************/ | 
 |  | 
 | struct COFF_reloc { | 
 |   char r_vaddr[4];        /* Virtual address of item    */ | 
 |   char r_symndx[4];       /* Symbol index in the symtab */ | 
 |   char r_type[2];         /* Relocation type            */ | 
 | }; | 
 |  | 
 | #define COFF_RELOC struct COFF_reloc | 
 | #define COFF_RELSZ 10 | 
 |  | 
 | #define COFF_DEF_DATA_SECTION_ALIGNMENT  4 | 
 | #define COFF_DEF_BSS_SECTION_ALIGNMENT   4 | 
 | #define COFF_DEF_TEXT_SECTION_ALIGNMENT  4 | 
 |  | 
 | /* For new sections we haven't heard of before */ | 
 | #define COFF_DEF_SECTION_ALIGNMENT       4 |