218 lines
6.0 KiB
C
218 lines
6.0 KiB
C
#ifdef COFF
|
|
# include <filehdr.h>
|
|
# include <aouthdr.h>
|
|
# include <scnhdr.h>
|
|
# include <syms.h>
|
|
# ifndef N_BADMAG
|
|
# define N_BADMAG(x) (0)
|
|
# endif
|
|
#else
|
|
# include AOUT_H
|
|
#endif
|
|
|
|
extern void *sbrk();
|
|
|
|
#if defined(hp9000s300) || defined(__hp9000s300) || defined(__hp9000s300__)
|
|
static int getpagesize () {
|
|
return EXEC_PAGESIZE;
|
|
}
|
|
#endif
|
|
|
|
Object P_Dump (ofile) Object ofile; {
|
|
#ifdef COFF
|
|
static struct scnhdr thdr, dhdr, bhdr, scn;
|
|
static struct filehdr hdr;
|
|
static struct aouthdr ohdr;
|
|
unsigned bias;
|
|
unsigned lnno_start, syms_start;
|
|
unsigned text_scn_start, data_scn_start;
|
|
unsigned data_end;
|
|
int pagemask = COFF_PAGESIZE-1;
|
|
#else
|
|
struct exec hdr, shdr;
|
|
unsigned data_start, data_end;
|
|
int pagemask = getpagesize () - 1;
|
|
#endif
|
|
char *afn;
|
|
register n;
|
|
char buf[BUFSIZ];
|
|
|
|
Dump_Prolog;
|
|
|
|
if (read (afd, (char *)&hdr, sizeof (hdr)) != sizeof (hdr)
|
|
|| N_BADMAG(hdr)) {
|
|
#ifdef COFF
|
|
badaout:
|
|
#endif
|
|
Dump_Finalize;
|
|
Primitive_Error ("corrupt a.out file");
|
|
}
|
|
#ifdef COFF
|
|
data_end = ((unsigned)sbrk (0) + pagemask) & ~pagemask;
|
|
syms_start = sizeof (hdr);
|
|
if (hdr.f_opthdr > 0) {
|
|
if (read (afd, (char *)&ohdr, sizeof (ohdr)) != sizeof (ohdr))
|
|
goto badaout;
|
|
}
|
|
for (n = 0; n < hdr.f_nscns; n++) {
|
|
if (read (afd, (char *)&scn, sizeof (scn)) != sizeof (scn))
|
|
goto badaout;
|
|
if (scn.s_scnptr > 0 && syms_start < scn.s_scnptr + scn.s_size)
|
|
syms_start = scn.s_scnptr + scn.s_size;
|
|
if (strcmp (scn.s_name, ".text") == 0)
|
|
thdr = scn;
|
|
else if (strcmp (scn.s_name, ".data") == 0)
|
|
dhdr = scn;
|
|
else if (strcmp (scn.s_name, ".bss") == 0)
|
|
bhdr = scn;
|
|
}
|
|
hdr.f_flags |= (F_RELFLG|F_EXEC);
|
|
ohdr.dsize = data_end - ohdr.data_start;
|
|
ohdr.bsize = 0;
|
|
thdr.s_size = ohdr.tsize;
|
|
thdr.s_scnptr = sizeof (hdr) + sizeof (ohdr)
|
|
+ hdr.f_nscns * sizeof (thdr);
|
|
lnno_start = thdr.s_lnnoptr;
|
|
text_scn_start = thdr.s_scnptr;
|
|
dhdr.s_paddr = dhdr.s_vaddr = ohdr.data_start;
|
|
dhdr.s_size = ohdr.dsize;
|
|
dhdr.s_scnptr = thdr.s_scnptr + thdr.s_size;
|
|
data_scn_start = dhdr.s_scnptr;
|
|
bhdr.s_paddr = bhdr.s_vaddr = ohdr.data_start + ohdr.dsize;
|
|
bhdr.s_size = ohdr.bsize;
|
|
bhdr.s_scnptr = 0;
|
|
|
|
bias = dhdr.s_scnptr + dhdr.s_size - syms_start;
|
|
if (hdr.f_symptr > 0)
|
|
hdr.f_symptr += bias;
|
|
if (thdr.s_lnnoptr > 0)
|
|
thdr.s_lnnoptr += bias;
|
|
|
|
if (write (ofd, (char *)&hdr, sizeof (hdr)) != sizeof (hdr)) {
|
|
badwrite:
|
|
Dump_Finalize;
|
|
Primitive_Error ("error writing dump file: ~E");
|
|
}
|
|
if (write (ofd, (char *)&ohdr, sizeof (ohdr)) != sizeof (ohdr))
|
|
goto badwrite;
|
|
if (write (ofd, (char *)&thdr, sizeof (thdr)) != sizeof (thdr))
|
|
goto badwrite;
|
|
if (write (ofd, (char *)&dhdr, sizeof (dhdr)) != sizeof (dhdr))
|
|
goto badwrite;
|
|
if (write (ofd, (char *)&bhdr, sizeof (bhdr)) != sizeof (bhdr))
|
|
goto badwrite;
|
|
lseek (ofd, (off_t)text_scn_start, 0);
|
|
if (write (ofd, (char *)ohdr.text_start, ohdr.tsize) != ohdr.tsize)
|
|
goto badwrite;
|
|
Was_Dumped = 1;
|
|
lseek (ofd, (off_t)data_scn_start, 0);
|
|
if (write (ofd, (char *)ohdr.data_start, ohdr.dsize) != ohdr.dsize)
|
|
goto badwrite;
|
|
lseek (afd, lnno_start ? (off_t)lnno_start : (off_t)syms_start, 0);
|
|
#else
|
|
close (afd);
|
|
#if defined(__bsdi__)
|
|
data_start = N_DATADDR(hdr);
|
|
#else
|
|
data_start = hdr.a_text;
|
|
#if defined(sun) || defined(__sun__)
|
|
data_start += pagemask+1;
|
|
#endif
|
|
data_start = (data_start + SEG_SIZ-1) & ~(SEG_SIZ-1);
|
|
#endif
|
|
data_end = (unsigned)sbrk (0);
|
|
#if !defined(__bsdi__)
|
|
data_end = (data_end + pagemask) & ~pagemask;
|
|
#endif
|
|
hdr.a_data = data_end - data_start;
|
|
hdr.a_bss = 0;
|
|
hdr.a_trsize = hdr.a_drsize = 0;
|
|
|
|
afn = Loader_Input;
|
|
if (!afn)
|
|
afn = A_Out_Name;
|
|
if ((afd = open (afn, O_RDONLY|O_BINARY)) == -1) {
|
|
Dump_Finalize;
|
|
Primitive_Error ("cannot open symbol table file: ~E");
|
|
}
|
|
if (read (afd, (char *)&shdr, sizeof (shdr)) != sizeof (shdr)
|
|
|| N_BADMAG(shdr)) {
|
|
Dump_Finalize;
|
|
Primitive_Error ("corrupt symbol table file");
|
|
}
|
|
#if defined(hp9000s300) || defined(__hp9000s300) || defined(__hp9000s300__)
|
|
hdr.a_lesyms = shdr.a_lesyms;
|
|
#else
|
|
hdr.a_syms = shdr.a_syms;
|
|
#endif
|
|
|
|
(void)lseek (ofd, (off_t)FILE_TEXT_START, 0);
|
|
n = hdr.a_text - TEXT_LENGTH_ADJ;
|
|
if (write (ofd, (char *)MEM_TEXT_START, n) != n) {
|
|
badwrite:
|
|
Dump_Finalize;
|
|
Primitive_Error ("error writing dump file: ~E");
|
|
}
|
|
Was_Dumped = 1;
|
|
|
|
#if defined(hp9000s300) || defined(__hp9000s300) || defined(__hp9000s300__)
|
|
(void)lseek (ofd, (off_t)DATA_OFFSET(hdr), 0);
|
|
#endif
|
|
#if defined(__bsdi__)
|
|
(void)lseek (ofd, (off_t)N_DATOFF(hdr), 0);
|
|
#endif
|
|
#ifdef GENERATIONAL_GC
|
|
n = data_end - data_start;
|
|
if (write (ofd, (char *)data_start, n) != n)
|
|
goto badwrite;
|
|
#else
|
|
|
|
if (Heap_Start > Free_Start) {
|
|
n = (unsigned)Free_Start - data_start;
|
|
if (write (ofd, (char *)data_start, n) != n)
|
|
goto badwrite;
|
|
(void)lseek (ofd, (off_t)(Free_End - Free_Start), 1);
|
|
n = Hp - Heap_Start;
|
|
if (write (ofd, Heap_Start, n) != n)
|
|
goto badwrite;
|
|
(void)lseek (ofd, (off_t)(Heap_End - Hp), 1);
|
|
n = data_end - (unsigned)Heap_End;
|
|
if (write (ofd, Heap_End, n) != n)
|
|
goto badwrite;
|
|
} else {
|
|
n = (unsigned)Hp - data_start;
|
|
if (write (ofd, (char *)data_start, n) != n)
|
|
goto badwrite;
|
|
(void)lseek (ofd, (off_t)(Free_End - Hp), 1);
|
|
n = data_end - (unsigned)Free_End;
|
|
if (write (ofd, Free_End, n) != n)
|
|
goto badwrite;
|
|
}
|
|
#endif
|
|
#if defined(hp9000s300) || defined(__hp9000s300) || defined(__hp9000s300__)
|
|
(void)lseek (afd, (off_t)LESYM_OFFSET(shdr), 0);
|
|
(void)lseek (ofd, (off_t)LESYM_OFFSET(hdr), 0);
|
|
#else
|
|
(void)lseek (afd, (off_t)N_SYMOFF(shdr), 0);
|
|
#if defined(__bsdi__)
|
|
(void)lseek (ofd, (off_t)N_SYMOFF(hdr), 0);
|
|
#endif
|
|
#endif
|
|
#endif /* !COFF */
|
|
while ((n = read (afd, buf, BUFSIZ)) > 0) {
|
|
if (write (ofd, buf, n) != n)
|
|
goto badwrite;
|
|
}
|
|
if (n < 0) {
|
|
Dump_Finalize;
|
|
Primitive_Error ("error reading symbol table: ~E");
|
|
}
|
|
#if !defined(COFF)
|
|
(void)lseek (ofd, (off_t)0L, 0);
|
|
if (write (ofd, (char *)&hdr, sizeof (hdr)) != sizeof (hdr))
|
|
goto badwrite;
|
|
#endif
|
|
|
|
Dump_Epilog;
|
|
}
|