xref: /netbsd-src/sys/arch/atari/stand/tostools/libtos/aout.c (revision c363a9cb62ac8888c384fe0ecac5b88d0f0c02dc)
1*c363a9cbScegger /*	$NetBSD: aout.c,v 1.12 2009/03/18 16:00:10 cegger Exp $	*/
2d6d66d66Sleo 
3d6d66d66Sleo /*-
4d6d66d66Sleo  * Copyright (c) 1998 The NetBSD Foundation, Inc.
5d6d66d66Sleo  * All rights reserved.
6d6d66d66Sleo  *
7d6d66d66Sleo  * This code is derived from software contributed to The NetBSD Foundation
8d6d66d66Sleo  * by Leo Weppelman.
9d6d66d66Sleo  *
10d6d66d66Sleo  * Redistribution and use in source and binary forms, with or without
11d6d66d66Sleo  * modification, are permitted provided that the following conditions
12d6d66d66Sleo  * are met:
13d6d66d66Sleo  * 1. Redistributions of source code must retain the above copyright
14d6d66d66Sleo  *    notice, this list of conditions and the following disclaimer.
15d6d66d66Sleo  * 2. Redistributions in binary form must reproduce the above copyright
16d6d66d66Sleo  *    notice, this list of conditions and the following disclaimer in the
17d6d66d66Sleo  *    documentation and/or other materials provided with the distribution.
18d6d66d66Sleo  *
19d6d66d66Sleo  * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
20d6d66d66Sleo  * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
21d6d66d66Sleo  * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
22d6d66d66Sleo  * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
23d6d66d66Sleo  * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
24d6d66d66Sleo  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
25d6d66d66Sleo  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
26d6d66d66Sleo  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
27d6d66d66Sleo  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
28d6d66d66Sleo  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
29d6d66d66Sleo  * POSSIBILITY OF SUCH DAMAGE.
30d6d66d66Sleo  */
3113a1055eSleo 
3295d16475Sleo 
3395d16475Sleo #ifdef TOSTOOLS
34d6d66d66Sleo #include <stdio.h>
35d6d66d66Sleo #include <stdlib.h>
3609bc53b5Sleo #include <unistd.h>
37d6d66d66Sleo #include <string.h>
38d6d66d66Sleo #include <sys/types.h>
39d6d66d66Sleo #include <a_out.h>
4095d16475Sleo 
4195d16475Sleo #define	MALLOC(x)	malloc(x)
4295d16475Sleo 
43d6d66d66Sleo #else
4495d16475Sleo 
45e63501d2Sjunyoung #include <lib/libsa/stand.h>
4695d16475Sleo #include <atari_stand.h>
4795d16475Sleo #include <libkern.h>
48d6d66d66Sleo #include <sys/exec_aout.h>
4995d16475Sleo 
5095d16475Sleo #define	MALLOC(x)	alloc(x)
51d6d66d66Sleo #endif
5295d16475Sleo 
53d6d66d66Sleo #include "libtos.h"
54fa1cc128Sleo #include "kparamb.h"
55fa1cc128Sleo #include "tosdefs.h"
5613a1055eSleo #include "cread.h"
5713a1055eSleo 
58d6d66d66Sleo 
59d6d66d66Sleo #ifdef TOSTOOLS
60d6d66d66Sleo /*
61d6d66d66Sleo  * Assume compiling under TOS or MINT. The page-size will always
62d6d66d66Sleo  * be incorrect then (if it is defined anyway).
63d6d66d66Sleo  */
64e8cc3884Sthorpej #ifdef AOUT_LDPGSZ
65e8cc3884Sthorpej #undef AOUT_LDPGSZ
66d6d66d66Sleo #endif
67d6d66d66Sleo 
68e8cc3884Sthorpej #define AOUT_LDPGSZ	(8*1024)	/* Page size for NetBSD		*/
69d6d66d66Sleo 
70d6d66d66Sleo #endif /* TOSTOOLS */
71d6d66d66Sleo 
72d6d66d66Sleo /*
73d6d66d66Sleo  * Load an a.out image.
74d6d66d66Sleo  * Exit codes:
75d6d66d66Sleo  *	-1      : Not an a.outfile
76d6d66d66Sleo  *	 0      : OK
77d6d66d66Sleo  *	 error# : Error during load (*errp might contain error string).
78d6d66d66Sleo  */
79d6d66d66Sleo int
aout_load(int fd,osdsc_t * od,char ** errp,int loadsyms)80a2e395a4Sjunyoung aout_load(int fd, osdsc_t *od, char **errp, int loadsyms)
81d6d66d66Sleo {
82d6d66d66Sleo 	long		textsz, stringsz;
83d6d66d66Sleo 	struct exec	ehdr;
84d6d66d66Sleo 	int		err;
85d6d66d66Sleo 
86d6d66d66Sleo 	*errp = NULL;
87d6d66d66Sleo 
88d6d66d66Sleo 	lseek(fd, (off_t)0, SEEK_SET);
89d6d66d66Sleo 	if (read(fd, (char *)&ehdr, sizeof(ehdr)) != sizeof(ehdr))
90d6d66d66Sleo 		return -1;
91d6d66d66Sleo 
9295d16475Sleo #ifdef TOSTOOLS
9395d16475Sleo 	if ((ehdr.a_magic & 0xffff) != NMAGIC)
94d6d66d66Sleo 		return -1;
9595d16475Sleo #else
9695d16475Sleo 	if ((N_GETMAGIC(ehdr) != NMAGIC) && (N_GETMAGIC(ehdr) != OMAGIC))
9795d16475Sleo 		return -1;
9895d16475Sleo #endif
99d6d66d66Sleo 
100d6d66d66Sleo 	/*
101d6d66d66Sleo 	 * Extract various sizes from the kernel executable
102d6d66d66Sleo 	 */
103e8cc3884Sthorpej 	textsz     = (ehdr.a_text + AOUT_LDPGSZ - 1) & ~(AOUT_LDPGSZ - 1);
104d6d66d66Sleo 	od->k_esym = 0;
105d6d66d66Sleo 	od->ksize  = textsz + ehdr.a_data + ehdr.a_bss;
106d6d66d66Sleo 	od->kentry = ehdr.a_entry;
107d6d66d66Sleo 
108d6d66d66Sleo 	if (loadsyms && ehdr.a_syms) {
109d6d66d66Sleo 		err = 1;
110a2e395a4Sjunyoung 		if (lseek(fd, ehdr.a_text+ehdr.a_data+ehdr.a_syms+sizeof(ehdr),
111a2e395a4Sjunyoung 			  0) <= 0)
112d6d66d66Sleo 			goto error;
113d6d66d66Sleo 		err = 2;
114d6d66d66Sleo 		if (read(fd, (char *)&stringsz, sizeof(long)) != sizeof(long))
115d6d66d66Sleo 			goto error;
116d6d66d66Sleo 		err = 3;
117d6d66d66Sleo 		if (lseek(fd, sizeof(ehdr), 0) <= 0)
118d6d66d66Sleo 			goto error;
119d6d66d66Sleo 		od->ksize += ehdr.a_syms + sizeof(long) + stringsz;
120d6d66d66Sleo 	}
121d6d66d66Sleo 
122d6d66d66Sleo 	err = 4;
12395d16475Sleo 	if ((od->kstart = (u_char *)MALLOC(od->ksize)) == NULL)
124d6d66d66Sleo 		goto error;
125d6d66d66Sleo 
126d6d66d66Sleo 	/*
127d6d66d66Sleo 	 * Read text & data, clear bss
128d6d66d66Sleo 	 */
129d6d66d66Sleo 	err = 5;
130d6d66d66Sleo 	if ((read(fd, (char *)(od->kstart), ehdr.a_text) != ehdr.a_text)
131d6d66d66Sleo 	    ||(read(fd,(char *)(od->kstart+textsz),ehdr.a_data) != ehdr.a_data))
132d6d66d66Sleo 		goto error;
133*c363a9cbScegger 	memset(od->kstart + textsz + ehdr.a_data, 0, ehdr.a_bss);
134d6d66d66Sleo 
135d6d66d66Sleo 	/*
136d6d66d66Sleo 	 * Read symbol and string table
137d6d66d66Sleo 	 */
138d6d66d66Sleo 	if (loadsyms && ehdr.a_syms) {
139d6d66d66Sleo 		long	*p;
140d6d66d66Sleo 
141d6d66d66Sleo 		p = (long *)((od->kstart) + textsz + ehdr.a_data + ehdr.a_bss);
142d6d66d66Sleo 		*p++ = ehdr.a_syms;
143d6d66d66Sleo 		err = 6;
144d6d66d66Sleo 		if (read(fd, (char *)p, ehdr.a_syms) != ehdr.a_syms)
145d6d66d66Sleo 			goto error;
146d6d66d66Sleo 		p = (long *)((char *)p + ehdr.a_syms);
147d6d66d66Sleo 		err = 7;
148d6d66d66Sleo 		if (read(fd, (char *)p, stringsz) != stringsz)
149d6d66d66Sleo 			goto error;
150d6d66d66Sleo 		od->k_esym = (long)((char *)p-(char *)od->kstart +stringsz);
151d6d66d66Sleo 	}
152d6d66d66Sleo 	return 0;
153d6d66d66Sleo 
154d6d66d66Sleo error:
155d6d66d66Sleo #ifdef TOSTOOLS
156d6d66d66Sleo 	{
157d6d66d66Sleo 		static char *errs[] = {
158d6d66d66Sleo 			/* 1 */ "Cannot seek to string table",
159d6d66d66Sleo 			/* 2 */ "Cannot read string-table size",
160d6d66d66Sleo 			/* 3 */ "Cannot seek back to text start",
161d6d66d66Sleo 			/* 4 */ "Cannot malloc kernel image space",
162d6d66d66Sleo 			/* 5 */ "Unable to read kernel image",
163d6d66d66Sleo 			/* 6 */ "Cannot read symbol table",
164d6d66d66Sleo 			/* 7 */ "Cannot read string table"
165d6d66d66Sleo 		};
166d6d66d66Sleo 		*errp = errs[err];
167d6d66d66Sleo 	}
168d6d66d66Sleo #endif /* TOSTOOLS */
169d6d66d66Sleo 
170d6d66d66Sleo 	return err;
171d6d66d66Sleo }
172