summaryrefslogtreecommitdiffstats
path: root/qemu/roms/openhackware/src/libexec/elf.c
blob: ae9f8e00e4e33bc51c4b3fc92d09d59005f5d136 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
/*
 * <elf.c>
 *
 * Open Hack'Ware BIOS ELF executable file loader
 * 
 * Copyright (c) 2004-2005 Jocelyn Mayer
 * 
 *   This program is free software; you can redistribute it and/or
 *   modify it under the terms of the GNU General Public License V2
 *   as published by the Free Software Foundation
 *
 *   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., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
 */

#include <stdlib.h>
#include <stdio.h>
#include "bios.h"
#include "exec.h"

uint32_t fs_inode_get_size (inode_t *inode);

/* ELF executable loader */
typedef uint16_t Elf32_Half;
typedef uint32_t Elf32_Word;
typedef uint32_t Elf32_Off;
typedef uint32_t Elf32_Addr;

#define EI_NIDENT	16

typedef struct elf32_hdr_t {
  unsigned char	e_ident[EI_NIDENT];
  Elf32_Half	e_type;
  Elf32_Half	e_machine;
  Elf32_Word	e_version;
  Elf32_Addr	e_entry;  /* Entry point */
  Elf32_Off	e_phoff;
  Elf32_Off	e_shoff;
  Elf32_Word	e_flags;
  Elf32_Half	e_ehsize;
  Elf32_Half	e_phentsize;
  Elf32_Half	e_phnum;
  Elf32_Half	e_shentsize;
  Elf32_Half	e_shnum;
  Elf32_Half	e_shstrndx;
} Elf32_Ehdr_t;

typedef struct elf32_phdr_t {
  Elf32_Word	p_type;
  Elf32_Off	p_offset;
  Elf32_Addr	p_vaddr;
  Elf32_Addr	p_paddr;
  Elf32_Word	p_filesz;
  Elf32_Word	p_memsz;
  Elf32_Word	p_flags;
  Elf32_Word	p_align;
} Elf32_Phdr_t;

#define	EI_MAG0		0		/* e_ident[] indexes */
#define	EI_MAG1		1
#define	EI_MAG2		2
#define	EI_MAG3		3
#define	EI_CLASS	4
#define	EI_DATA		5
#define	EI_VERSION	6
#define	EI_OSABI	7
#define	EI_PAD		8

#define	ELFMAG0		0x7f		/* EI_MAG */
#define	ELFMAG1		'E'
#define	ELFMAG2		'L'
#define	ELFMAG3		'F'

#define	ELFCLASSNONE	0		/* EI_CLASS */
#define	ELFCLASS32	1
#define	ELFCLASS64	2
#define	ELFCLASSNUM	3

#define ELFDATANONE	0		/* e_ident[EI_DATA] */
#define ELFDATA2LSB	1
#define ELFDATA2MSB	2

#define EV_NONE		0		/* e_version, EI_VERSION */
#define EV_CURRENT	1
#define EV_NUM		2

/* These constants define the different elf file types */
#define ET_NONE   0
#define ET_REL    1
#define ET_EXEC   2
#define ET_DYN    3
#define ET_CORE   4
#define ET_LOPROC 0xff00
#define ET_HIPROC 0xffff

/* These constants define the various ELF target machines */
#define EM_NONE  0
#define EM_M32   1
#define EM_SPARC 2
#define EM_386   3
#define EM_68K   4
#define EM_88K   5
#define EM_486   6   /* Perhaps disused */
#define EM_860   7
#define EM_MIPS		8	/* MIPS R3000 (officially, big-endian only) */
#define EM_MIPS_RS4_BE 10	/* MIPS R4000 big-endian */
#define EM_PARISC      15	/* HPPA */
#define EM_SPARC32PLUS 18	/* Sun's "v8plus" */
#define EM_PPC	       20	/* PowerPC */
#define EM_PPC64       21       /* PowerPC64 */
#define EM_SH	       42	/* SuperH */
#define EM_SPARCV9     43	/* SPARC v9 64-bit */
#define EM_IA_64	50	/* HP/Intel IA-64 */
#define EM_X86_64	62	/* AMD x86-64 */
#define EM_S390		22	/* IBM S/390 */
#define EM_CRIS         76      /* Axis Communications 32-bit embedded processor */
#define EM_V850		87	/* NEC v850 */
#define EM_H8_300H      47      /* Hitachi H8/300H */
#define EM_H8S          48      /* Hitachi H8S     */
/*
 * This is an interim value that we will use until the committee comes
 * up with a final number.
 */
#define EM_ALPHA	0x9026
/* Bogus old v850 magic number, used by old tools.  */
#define EM_CYGNUS_V850	0x9080
/*
 * This is the old interim value for S/390 architecture
 */
#define EM_S390_OLD     0xA390

int exec_load_elf (inode_t *file, void **dest, void **entry, void **end,
                   uint32_t loffset)
{
    Elf32_Ehdr_t ehdr;
    Elf32_Phdr_t phdr;
    void *address, *first, *last;
    uint32_t offset, fsize, msize;
    int i;

    file_seek(file, loffset);
    if (fs_read(file, &ehdr, sizeof(Elf32_Ehdr_t)) < 0) {
        ERROR("Cannot load first bloc of file...\n");
        return -1;
    }
    DPRINTF("Check ELF file\n");
    /* Check ident */
    if (ehdr.e_ident[EI_MAG0] != ELFMAG0 ||
        ehdr.e_ident[EI_MAG1] != ELFMAG1 ||
        ehdr.e_ident[EI_MAG2] != ELFMAG2 ||
        ehdr.e_ident[EI_MAG3] != ELFMAG3) {
        DPRINTF("Not an ELF file %0x\n", *(uint32_t *)ehdr.e_ident);
        return -2;
    }
    if (ehdr.e_ident[EI_CLASS] != ELFCLASS32) {
        ERROR("Not a 32 bits ELF file\n");
        return -2;
    }
    if (ehdr.e_ident[EI_DATA] != ELFDATA2MSB) {
        ERROR("Not a big-endian ELF file\n");
        return -2;
    }
    if (ehdr.e_ident[EI_VERSION] != EV_CURRENT /*||
        ehdr->e_version != EV_CURRENT*/) {
        ERROR("Invalid ELF executable version %d %08x\n",
              ehdr.e_ident[EI_VERSION], ehdr.e_version);
        return -2;
    }
    if (ehdr.e_type != ET_EXEC) {
        ERROR("Not an executable ELF file\n");
        return -2;
    }
    if (ehdr.e_machine != EM_PPC) {
        ERROR("Not a PPC ELF executable\n");
        return -2;
    }
    /* All right, seems to be a regular ELF program for PPC */
    *entry = (void *)ehdr.e_entry;
    DPRINTF("ELF file found entry = %p\n", *entry);
    last = NULL;
    first = last - 4;
    fsize = msize = 0;
    offset = ehdr.e_phoff;
    for (i = 0; i < ehdr.e_phnum; i++) {
#if 0
        if (offset > fs_inode_get_size(file)) {
            ERROR("ELF program header %d offset > file size %d %d\n", i,
                  offset, fs_inode_get_size(file));
            return -1;
        }
#endif
        DPRINTF("Load program header %d from %08x\n", i, offset);
        file_seek(file, offset + loffset);
        if (fs_read(file, &phdr, sizeof(Elf32_Phdr_t)) < 0) {
            ERROR("Cannot load ELF program header %d...\n", i);
            return -1;
        }
        DPRINTF("Load program header %d %08x %08x %08x %08x\n", i,
                phdr.p_offset, phdr.p_vaddr, phdr.p_filesz, phdr.p_memsz);
#if 0
        if (phdr.p_offset > fs_inode_get_size(file)) {
            ERROR("ELF program %d offset > file size %d %d\n",
                  i, phdr.p_offset, fs_inode_get_size(file));
            return -1;
        }
#endif
        /* As we won't remap memory, load it at it's virtual address (!) */
        address = (void *)phdr.p_vaddr;
        if (address < first)
            first = address;
        fsize = phdr.p_filesz;
        msize = phdr.p_memsz;
        if (address + msize > last)
            last = address + msize;
        file_seek(file, phdr.p_offset + loffset);
        set_loadinfo((void *)first, last - first);
        if (fs_read(file, address, fsize) < 0) {
            ERROR("Cannot load ELF program %d...\n", i);
            return -1;
        }
        if (msize > fsize) {
            memset(address + fsize, 0, msize - fsize);
        }
        offset += ehdr.e_phentsize;
    }
    *dest = (void *)first;
    *end = (void *)last;
    DPRINTF("ELF file loaded at %p => %p fsize %08x msize %08x "
            "(%08x %08x)\n", *dest, *entry, fsize, msize,
            *(uint32_t *)entry, *((uint32_t *)entry + 1));

    return 0;
}