elk/src/dump-vanilla.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;
}