]> git.saurik.com Git - apple/boot.git/blob - i386/libsaio/load.c
2103c14d6c88e11304191eafe0ec75bcbc508223
[apple/boot.git] / i386 / libsaio / load.c
1 /*
2 * Copyright (c) 2003 Apple Computer, Inc. All rights reserved.
3 *
4 * @APPLE_LICENSE_HEADER_START@
5 *
6 * The contents of this file constitute Original Code as defined in and
7 * are subject to the Apple Public Source License Version 1.2 (the
8 * "License"). You may not use this file except in compliance with the
9 * License. Please obtain a copy of the License at
10 * http://www.apple.com/publicsource and read it before using this file.
11 *
12 * This Original Code and all software distributed under the License are
13 * distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY KIND, EITHER
14 * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
15 * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
16 * FITNESS FOR A PARTICULAR PURPOSE OR NON-INFRINGEMENT. Please see the
17 * License for the specific language governing rights and limitations
18 * under the License.
19 *
20 * @APPLE_LICENSE_HEADER_END@
21 */
22 /*
23 * load.c - Functions for decoding a Mach-o Kernel.
24 *
25 * Copyright (c) 1998-2003 Apple Computer, Inc.
26 *
27 */
28
29 #include <mach-o/fat.h>
30 #include <mach-o/loader.h>
31 #include <mach/machine/thread_status.h>
32
33 #include <sl.h>
34
35 static long DecodeSegment(long cmdBase, unsigned int*load_addr, unsigned int *load_size);
36 static long DecodeUnixThread(long cmdBase, unsigned int *entry);
37
38
39 static unsigned long gBinaryAddress;
40 BOOL gHaveKernelCache;
41
42 // Public Functions
43
44 long ThinFatFile(void **binary, unsigned long *length)
45 {
46 unsigned long nfat, swapped, size = 0;
47 struct fat_header *fhp = (struct fat_header *)*binary;
48 struct fat_arch *fap =
49 (struct fat_arch *)((unsigned long)*binary + sizeof(struct fat_header));
50
51 if (fhp->magic == FAT_MAGIC) {
52 nfat = fhp->nfat_arch;
53 swapped = 0;
54 } else if (fhp->magic == FAT_CIGAM) {
55 nfat = NXSwapInt(fhp->nfat_arch);
56 swapped = 1;
57 } else {
58 return -1;
59 }
60
61 for (; nfat > 0; nfat--, fap++) {
62 if (swapped) {
63 fap->cputype = NXSwapInt(fap->cputype);
64 fap->offset = NXSwapInt(fap->offset);
65 fap->size = NXSwapInt(fap->size);
66 }
67
68 if (fap->cputype == CPU_TYPE_I386) {
69 *binary = (void *) ((unsigned long)*binary + fap->offset);
70 size = fap->size;
71 break;
72 }
73 }
74
75 if (length != 0) *length = size;
76
77 return 0;
78 }
79
80 long DecodeMachO(void *binary, entry_t *rentry, char **raddr, int *rsize)
81 {
82 struct mach_header *mH;
83 unsigned long ncmds, cmdBase, cmd, cmdsize;
84 // long headerBase, headerAddr, headerSize;
85 unsigned int vmaddr = ~0;
86 unsigned int vmend = 0;
87 unsigned long cnt;
88 long ret = -1;
89 unsigned int entry;
90
91 gBinaryAddress = (unsigned long)binary;
92
93 // headerBase = gBinaryAddress;
94 cmdBase = (unsigned long)gBinaryAddress + sizeof(struct mach_header);
95
96 mH = (struct mach_header *)(gBinaryAddress);
97 if (mH->magic != MH_MAGIC) {
98 error("Mach-O file has bad magic number\n");
99 return -1;
100 }
101
102 #if NOTDEF
103 printf("magic: %x\n", (unsigned)mH->magic);
104 printf("cputype: %x\n", (unsigned)mH->cputype);
105 printf("cpusubtype: %x\n", (unsigned)mH->cpusubtype);
106 printf("filetype: %x\n", (unsigned)mH->filetype);
107 printf("ncmds: %x\n", (unsigned)mH->ncmds);
108 printf("sizeofcmds: %x\n", (unsigned)mH->sizeofcmds);
109 printf("flags: %x\n", (unsigned)mH->flags);
110 getc();
111 #endif
112
113 ncmds = mH->ncmds;
114
115 for (cnt = 0; cnt < ncmds; cnt++) {
116 cmd = ((long *)cmdBase)[0];
117 cmdsize = ((long *)cmdBase)[1];
118 unsigned int load_addr;
119 unsigned int load_size;
120
121 switch (cmd) {
122
123 case LC_SEGMENT:
124 ret = DecodeSegment(cmdBase, &load_addr, &load_size);
125 if (ret == 0 && load_size != 0) {
126 vmaddr = min(vmaddr, load_addr);
127 vmend = max(vmend, load_addr + load_size);
128 }
129 break;
130
131 case LC_UNIXTHREAD:
132 ret = DecodeUnixThread(cmdBase, &entry);
133 break;
134
135 default:
136 #if NOTDEF
137 printf("Ignoring cmd type %d.\n", (unsigned)cmd);
138 #endif
139 break;
140 }
141
142 if (ret != 0) return -1;
143
144 cmdBase += cmdsize;
145 }
146
147 *rentry = (entry_t)( (unsigned long) entry & 0x3fffffff );
148 *rsize = vmend - vmaddr;
149 *raddr = (char *)vmaddr;
150
151 return ret;
152 }
153
154 // Private Functions
155
156 static long DecodeSegment(long cmdBase, unsigned int *load_addr, unsigned int *load_size)
157 {
158 struct segment_command *segCmd;
159 unsigned long vmaddr, fileaddr;
160 long vmsize, filesize;
161
162 segCmd = (struct segment_command *)cmdBase;
163
164 vmaddr = (segCmd->vmaddr & 0x3fffffff);
165 vmsize = segCmd->vmsize;
166
167 fileaddr = (gBinaryAddress + segCmd->fileoff);
168 filesize = segCmd->filesize;
169
170 if (filesize == 0) {
171 *load_addr = ~0;
172 *load_size = 0;
173 return 0;
174 }
175
176 #if NOTDEF
177 printf("segname: %s, vmaddr: %x, vmsize: %x, fileoff: %x, filesize: %x, nsects: %d, flags: %x.\n",
178 segCmd->segname, (unsigned)vmaddr, (unsigned)vmsize, (unsigned)fileaddr, (unsigned)filesize,
179 (unsigned) segCmd->nsects, (unsigned)segCmd->flags);
180 getc();
181 #endif
182
183 if (vmaddr < KERNEL_ADDR ||
184 (vmaddr + vmsize) > (KERNEL_ADDR + KERNEL_LEN)) {
185 stop("Kernel overflows available space");
186 }
187
188 if (vmsize && (strcmp(segCmd->segname, "__PRELINK") == 0)) {
189 gHaveKernelCache = 1;
190 }
191
192 // Copy from file load area.
193 bcopy((char *)fileaddr, (char *)vmaddr, filesize);
194
195 // Zero space at the end of the segment.
196 bzero((char *)(vmaddr + filesize), vmsize - filesize);
197
198 *load_addr = vmaddr;
199 *load_size = vmsize;
200
201 return 0;
202 }
203
204
205 static long DecodeUnixThread(long cmdBase, unsigned int *entry)
206 {
207 i386_thread_state_t *i386ThreadState;
208
209 i386ThreadState = (i386_thread_state_t *)
210 (cmdBase + sizeof(struct thread_command) + 8);
211
212 *entry = i386ThreadState->eip;
213
214 return 0;
215 }
216