main.c revision 1.4 1 1.1 is /*
2 1.4 mycroft * $NetBSD: main.c,v 1.4 1997/03/24 18:54:26 mycroft Exp $
3 1.1 is *
4 1.1 is *
5 1.1 is * Copyright (c) 1996 Ignatios Souvatzis
6 1.1 is * Copyright (c) 1994 Michael L. Hitch
7 1.1 is * All rights reserved.
8 1.1 is *
9 1.1 is * Redistribution and use in source and binary forms, with or without
10 1.1 is * modification, are permitted provided that the following conditions
11 1.1 is * are met:
12 1.1 is * 1. Redistributions of source code must retain the above copyright
13 1.1 is * notice, this list of conditions and the following disclaimer.
14 1.1 is * 2. Redistributions in binary form must reproduce the above copyright
15 1.1 is * notice, this list of conditions and the following disclaimer in the
16 1.1 is * documentation and/or other materials provided with the distribution.
17 1.1 is * 3. All advertising materials mentioning features or use of this software
18 1.1 is * must display the following acknowledgement:
19 1.1 is * This product includes software developed by Michael L. Hitch.
20 1.1 is * 4. The name of the authors may not be used to endorse or promote products
21 1.1 is * derived from this software without specific prior written permission
22 1.1 is *
23 1.1 is * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
24 1.1 is * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
25 1.1 is * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
26 1.1 is * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
27 1.1 is * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
28 1.1 is * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
29 1.1 is * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
30 1.1 is * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
31 1.1 is * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
32 1.1 is * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
33 1.1 is *
34 1.1 is */
35 1.1 is
36 1.1 is #include <sys/cdefs.h>
37 1.1 is #include <sys/reboot.h>
38 1.1 is #include <sys/types.h>
39 1.1 is
40 1.1 is #include <sys/exec_aout.h>
41 1.1 is
42 1.1 is #include <amiga/cfdev.h>
43 1.1 is #include <amiga/memlist.h>
44 1.1 is #include <include/cpu.h>
45 1.1 is
46 1.2 is #include <saerrno.h>
47 1.2 is #include <stand.h>
48 1.1 is
49 1.1 is #include "libstubs.h"
50 1.1 is #include "samachdep.h"
51 1.1 is
52 1.4 mycroft #undef __LDPGSZ
53 1.1 is #define __LDPGSZ 8192
54 1.1 is #define __PGSZ 8192
55 1.1 is
56 1.1 is #define DRACOREVISION (*(u_int8_t *)0x02000009)
57 1.1 is #define DRACOMMUMARGIN 0x200000
58 1.1 is #define DRACOZ2OFFSET 0x3000000
59 1.1 is #define DRACOZ2MAX 0x1000000
60 1.1 is
61 1.1 is #define EXECMIN 36
62 1.1 is
63 1.1 is void startit __P((void *, u_long, u_long, void *, u_long, u_long, int, void *,
64 1.3 mhitch int, int, u_long, u_long, u_long, int));
65 1.1 is void startit_end __P((void));
66 1.1 is int get_cpuid __P((u_int32_t *));
67 1.1 is
68 1.1 is /*
69 1.1 is * Kernel startup interface version
70 1.1 is * 1: first version of loadbsd
71 1.1 is * 2: needs esym location passed in a4
72 1.1 is * 3: load kernel image into fastmem rather than chipmem
73 1.1 is * MAX: highest version with backward compatibility.
74 1.1 is */
75 1.1 is
76 1.1 is #define KERNEL_STARTUP_VERSION 3
77 1.1 is #define KERNEL_STARTUP_VERSION_MAX 9
78 1.1 is
79 1.1 is static long get_number(char **);
80 1.1 is
81 1.1 is const char version[] = "2.0";
82 1.3 mhitch char default_command[] = "netbsd -AS";
83 1.1 is
84 1.1 is int
85 1.1 is pain()
86 1.1 is {
87 1.1 is long int io = 0;
88 1.1 is char linebuf[128];
89 1.1 is char *kernel_name = default_command;
90 1.1 is char *path = default_command;
91 1.1 is int boothowto = RB_AUTOBOOT;
92 1.1 is u_int32_t cpuid = 0;
93 1.1 is int amiga_flags = 0;
94 1.1 is u_int32_t I_flag = 0;
95 1.1 is int k_flag = 0;
96 1.1 is int p_flag = 0;
97 1.1 is int Z_flag = 0;
98 1.1 is int m_value = 0;
99 1.1 is int S_flag = 0;
100 1.1 is int t_flag = 0;
101 1.1 is long stringsz;
102 1.1 is
103 1.1 is u_int32_t fmem = 0x0;
104 1.1 is int fmemsz = 0x0;
105 1.1 is int cmemsz = 0x0;
106 1.1 is int eclock = SysBase->EClockFreq;
107 1.1 is /* int skip_chipmem = 0; */
108 1.1 is
109 1.1 is void (*start_it)(void *, u_long, u_long, void *, u_long, u_long, int,
110 1.3 mhitch void *, int, int, u_long, u_long, u_long, int) = startit;
111 1.1 is
112 1.1 is caddr_t kp;
113 1.1 is u_int16_t *kvers;
114 1.1 is struct exec *eh;
115 1.1 is int textsz, ksize;
116 1.1 is void *esym = 0;
117 1.1 is int32_t *nkcd;
118 1.1 is struct cfdev *cd, *kcd;
119 1.1 is struct boot_memseg *kmemseg;
120 1.1 is struct boot_memseg *memseg;
121 1.1 is struct MemHead *mh;
122 1.1 is u_int32_t from, size, vfrom, vsize;
123 1.1 is int contflag, mapped1to1;
124 1.1 is
125 1.1 is int ncd, nseg;
126 1.1 is char c;
127 1.1 is
128 1.1 is extern u_int16_t timelimit;
129 1.1 is
130 1.3 mhitch extern u_int32_t aio_base;
131 1.3 mhitch
132 1.1 is /*
133 1.1 is * we need V36 for: EClock, RDB Bootblocks, CacheClearU
134 1.1 is */
135 1.1 is
136 1.1 is if (SysBase->LibNode.Version < EXECMIN) {
137 1.1 is printf("Exec V%ld, need V%ld\n",
138 1.1 is (long)SysBase->LibNode.Version, (long)EXECMIN);
139 1.1 is goto out;
140 1.1 is }
141 1.1 is
142 1.1 is printf("\2337mNetBSD/Amiga bootblock %s\2330m\n%s :- ",
143 1.1 is version, kernel_name);
144 1.1 is
145 1.1 is timelimit = 3;
146 1.1 is gets(linebuf);
147 1.1 is
148 1.1 is if (*linebuf == 'q')
149 1.1 is return 1;
150 1.1 is
151 1.1 is if (*linebuf)
152 1.1 is path = linebuf;
153 1.1 is
154 1.1 is /*
155 1.1 is * parse boot command for path name and process any options
156 1.1 is */
157 1.1 is while ((c = *path)) {
158 1.1 is while (c == ' ')
159 1.1 is c = *++path;
160 1.1 is if (c == '-') {
161 1.1 is while ((c = *++path) && c != ' ') {
162 1.1 is switch (c) {
163 1.1 is case 'a': /* multi-user state */
164 1.1 is boothowto &= ~RB_SINGLE;
165 1.1 is break;
166 1.1 is case 'b': /* ask for root device */
167 1.1 is boothowto |= RB_ASKNAME;
168 1.1 is break;
169 1.1 is case 'c': /* force machine model */
170 1.1 is cpuid = get_number(&path) << 16;
171 1.1 is break;
172 1.1 is case 'k': /* Reserve first 4M fastmem */
173 1.1 is k_flag++;
174 1.1 is break;
175 1.1 is case 'm': /* Force fastmem size */
176 1.1 is m_value = get_number(&path) * 1024;
177 1.1 is break;
178 1.1 is case 'n': /* non-contiguous memory */
179 1.1 is amiga_flags |=
180 1.1 is (get_number(&path) & 3) << 1;
181 1.1 is break;
182 1.1 is case 'p': /* Select fastmem by priority */
183 1.1 is p_flag++;
184 1.1 is break;
185 1.1 is case 's': /* single-user state */
186 1.1 is boothowto |= RB_SINGLE;
187 1.1 is break;
188 1.1 is case 't': /* test flag */
189 1.1 is t_flag = 1;
190 1.1 is break;
191 1.1 is case 'A': /* enable AGA modes */
192 1.1 is amiga_flags |= 1;
193 1.1 is break;
194 1.1 is case 'D': /* enter Debugger */
195 1.1 is boothowto |= RB_KDB;
196 1.1 is break;
197 1.1 is case 'I': /* inhibit sync negotiation */
198 1.1 is I_flag = get_number(&path);
199 1.1 is break;
200 1.1 is case 'K': /* remove 1st 4MB fastmem */
201 1.1 is break;
202 1.1 is case 'S': /* include debug symbols */
203 1.1 is S_flag = 1;
204 1.1 is break;
205 1.1 is case 'Z': /* force chip memory load */
206 1.1 is Z_flag = 1;
207 1.1 is break;
208 1.1 is }
209 1.1 is }
210 1.1 is } else {
211 1.1 is kernel_name = path;
212 1.1 is while ((c = *++path) && c != ' ')
213 1.1 is ;
214 1.1 is if (c)
215 1.1 is *path++ = 0;
216 1.1 is }
217 1.1 is }
218 1.1 is while ((c = *kernel_name) && c == ' ')
219 1.1 is ++kernel_name;
220 1.1 is path = kernel_name;
221 1.1 is while ((c = *path) && c != ' ')
222 1.1 is ++path;
223 1.1 is if (c)
224 1.1 is *path = 0;
225 1.1 is
226 1.1 is if (get_cpuid(&cpuid))
227 1.1 is goto out;
228 1.1 is
229 1.1 is ExpansionBase = OpenLibrary("expansion.library", 0);
230 1.1 is if (!ExpansionBase) {
231 1.1 is printf("can't open %s\n", "expansion.library");
232 1.1 is return 1;
233 1.1 is }
234 1.1 is
235 1.1 is for (ncd=0, cd=0; (cd = FindConfigDev(cd, -1, -1)); ncd++)
236 1.1 is /* nothing */;
237 1.1 is
238 1.1 is /* find memory list */
239 1.1 is
240 1.1 is memseg = (struct boot_memseg *)alloc(16*sizeof(struct boot_memseg));
241 1.1 is
242 1.1 is /* Forbid(); */
243 1.1 is
244 1.1 is nseg = 0;
245 1.1 is mh = SysBase->MemLst;
246 1.1 is vfrom = mh->Lower & -__PGSZ;
247 1.1 is vsize = (mh->Upper - vfrom) & -__PGSZ;
248 1.1 is contflag = mapped1to1 = 0;
249 1.1 is
250 1.1 is do {
251 1.1 is size = vsize;
252 1.1 is
253 1.1 is if (SysBase->LibNode.Version > 36) {
254 1.1 is from = CachePreDMA(vfrom, &size, contflag);
255 1.1 is contflag = DMAF_Continue;
256 1.1 is mapped1to1 = (from == vfrom);
257 1.1 is vsize -= size;
258 1.1 is vfrom += size;
259 1.1 is } else {
260 1.1 is from = vfrom;
261 1.1 is vsize = 0;
262 1.1 is }
263 1.1 is
264 1.1 is #if DEBUG_MEMORY_LIST
265 1.1 is printf("%lx %lx %lx %ld/%lx %lx\n",
266 1.1 is (long)from, (long)size,
267 1.1 is (long)mh->Attribs, (long)mh->Pri,
268 1.1 is (long)vfrom, (long)vsize);
269 1.1 is #endif
270 1.1 is /* Insert The Evergrowing Kludge List Here: */
271 1.1 is
272 1.1 is /* a) dont load kernel over DraCo MMU table */
273 1.1 is
274 1.1 is if (((cpuid >> 24) == 0x7D) &&
275 1.1 is ((from & -DRACOMMUMARGIN) == 0x40000000) &&
276 1.1 is (size >= DRACOMMUMARGIN)) {
277 1.1 is
278 1.1 is memseg[nseg].ms_start = from & -DRACOMMUMARGIN;
279 1.1 is memseg[nseg].ms_size = DRACOMMUMARGIN;
280 1.1 is memseg[nseg].ms_attrib = mh->Attribs;
281 1.1 is memseg[nseg].ms_pri = mh->Pri;
282 1.1 is
283 1.1 is size -= DRACOMMUMARGIN - (from & (DRACOMMUMARGIN - 1));
284 1.1 is from += DRACOMMUMARGIN - (from & (DRACOMMUMARGIN - 1));
285 1.1 is ++nseg;
286 1.1 is }
287 1.1 is
288 1.1 is if ((mh->Attribs & (MEMF_CHIP|MEMF_FAST)) == MEMF_CHIP) {
289 1.1 is size += from;
290 1.1 is cmemsz = size;;
291 1.1 is from = 0;
292 1.1 is } else if ((fmemsz < size) && mapped1to1) {
293 1.1 is fmem = from;
294 1.1 is fmemsz = size;
295 1.1 is }
296 1.1 is
297 1.1 is memseg[nseg].ms_start = from;
298 1.1 is memseg[nseg].ms_size = size;
299 1.1 is memseg[nseg].ms_attrib = mh->Attribs;
300 1.1 is memseg[nseg].ms_pri = mh->Pri;
301 1.1 is
302 1.1 is if (vsize == 0) {
303 1.1 is mh = mh->next;
304 1.1 is contflag = 0;
305 1.1 is if (mh->next) {
306 1.1 is vfrom = mh->Lower & -__PGSZ;
307 1.1 is vsize = (mh->Upper & -__PGSZ) - vfrom;
308 1.1 is }
309 1.1 is }
310 1.1 is } while ((++nseg <= 16) && vsize);
311 1.1 is
312 1.1 is /* Permit(); */
313 1.1 is
314 1.1 is printf("Loading %s: ", kernel_name);
315 1.1 is io = open(kernel_name, 0);
316 1.1 is if (io < 0)
317 1.1 is goto err;
318 1.1 is
319 1.1 is eh = alloc(sizeof(*eh));
320 1.1 is if (!eh) {
321 1.1 is errno = ENOMEM;
322 1.1 is goto err;
323 1.1 is }
324 1.1 is if (read(io, eh, sizeof(*eh)) != sizeof(*eh)) {
325 1.1 is errno = ENOEXEC;
326 1.1 is goto err;
327 1.1 is }
328 1.1 is
329 1.1 is if ((N_GETMAGIC(*eh) != NMAGIC) || (N_GETMID(*eh) != MID_M68K)) {
330 1.1 is errno = ENOEXEC;
331 1.1 is goto err;
332 1.1 is }
333 1.1 is
334 1.1 is textsz = (eh->a_text + __LDPGSZ - 1) & (-__LDPGSZ);
335 1.1 is esym = 0;
336 1.1 is
337 1.1 is ksize = textsz + eh->a_data + eh->a_bss
338 1.1 is + sizeof(*nkcd) + ncd*sizeof(*cd)
339 1.1 is + sizeof(*nkcd) + nseg * sizeof(struct boot_memseg);
340 1.1 is
341 1.1 is if (S_flag && eh->a_syms) {
342 1.1 is if (lseek(io, eh->a_text+ eh->a_data+ eh->a_syms, SEEK_CUR)
343 1.1 is <= 0
344 1.1 is || read(io, &stringsz, 4) != 4
345 1.1 is || lseek(io, sizeof(*eh), SEEK_SET) < 0)
346 1.1 is goto err;
347 1.1 is ksize += eh->a_syms + 4 + ((stringsz + 3) & ~3);
348 1.1 is }
349 1.1 is
350 1.1 is kp = alloc(ksize + 256 + ((u_char *)startit_end - (u_char *)startit));
351 1.1 is if (kp == 0) {
352 1.1 is errno = ENOMEM;
353 1.1 is goto err;
354 1.1 is }
355 1.1 is
356 1.1 is printf("%ld", eh->a_text);
357 1.1 is if (read(io, kp, eh->a_text) != eh->a_text)
358 1.1 is goto err;
359 1.1 is
360 1.1 is printf("+%ld", eh->a_data);
361 1.1 is if (read(io, kp + textsz, eh->a_data) != eh->a_data)
362 1.1 is goto err;
363 1.1 is
364 1.1 is printf("+%ld", eh->a_bss);
365 1.1 is
366 1.1 is kvers = (u_short *)(kp + eh->a_entry - 2);
367 1.1 is
368 1.1 is if (*kvers > KERNEL_STARTUP_VERSION_MAX && *kvers != 0x4e73) {
369 1.1 is printf("\nnewer bootblock required: %ld\n", (long)*kvers);
370 1.1 is goto freeall;
371 1.1 is }
372 1.1 is #if 0
373 1.1 is if (*kvers > KERNEL_STARTUP_VERSION)
374 1.1 is printf("\nKernel V%ld newer than bootblock V%ld\n",
375 1.1 is (long)*kvers, (long)KERNEL_STARTUP_VERSION);
376 1.1 is #endif
377 1.1 is nkcd = (int *)(kp + textsz + eh->a_data + eh->a_bss);
378 1.1 is if (*kvers != 0x4e73 && *kvers > 1 && S_flag && eh->a_syms) {
379 1.1 is *nkcd++ = eh->a_syms;
380 1.1 is printf("+[%ld", eh->a_syms);
381 1.1 is if (read(io, (char *)nkcd, eh->a_syms) != eh->a_syms)
382 1.1 is goto err;
383 1.1 is nkcd = (int *)((char *)nkcd + eh->a_syms);
384 1.1 is printf("+%ld]", stringsz);
385 1.1 is if (read(io, (char *)nkcd, stringsz) != stringsz)
386 1.1 is goto err;
387 1.1 is nkcd = (int*)((char *)nkcd + ((stringsz + 3) & ~3));
388 1.1 is esym = (char *)(textsz + eh->a_data + eh->a_bss
389 1.1 is + eh->a_syms + 4 + ((stringsz + 3) & ~3));
390 1.1 is }
391 1.1 is putchar('\n');
392 1.1 is
393 1.1 is *nkcd = ncd;
394 1.1 is kcd = (struct cfdev *)(nkcd + 1);
395 1.1 is
396 1.1 is while ((cd = FindConfigDev(cd, -1, -1))) {
397 1.1 is *kcd = *cd;
398 1.1 is if (((cpuid >> 24) == 0x7D) &&
399 1.1 is ((u_long)kcd->addr < 0x1000000)) {
400 1.1 is kcd->addr += 0x3000000;
401 1.1 is }
402 1.1 is ++kcd;
403 1.1 is }
404 1.1 is
405 1.1 is nkcd = (u_int32_t *)kcd;
406 1.1 is *nkcd = nseg;
407 1.1 is
408 1.1 is kmemseg = (struct boot_memseg *)(nkcd + 1);
409 1.1 is
410 1.1 is while (nseg-- > 0)
411 1.1 is *kmemseg++ = *memseg++;
412 1.1 is
413 1.1 is if (*kvers > 2 && Z_flag == 0) {
414 1.1 is /*
415 1.1 is * Kernel supports direct load to fastmem, and the -Z
416 1.1 is * option was not specified. Copy startup code to end
417 1.1 is * of kernel image and set start_it.
418 1.1 is */
419 1.1 is if ((u_int32_t)kp < fmem) {
420 1.1 is errno = EFBIG;
421 1.1 is goto err;
422 1.1 is }
423 1.1 is memcpy(kp + ksize + 256, (char *)startit,
424 1.1 is (char *)startit_end - (char *)startit);
425 1.1 is CacheClearU();
426 1.1 is (caddr_t)start_it = kp + ksize + 256;
427 1.1 is printf("*** Loading from %08lx to Fastmem %08lx ***\n",
428 1.1 is (u_long)kp, (u_long)fmem);
429 1.1 is /* sleep(2); */
430 1.1 is } else {
431 1.1 is /*
432 1.1 is * Either the kernel doesn't suppport loading directly to
433 1.1 is * fastmem or the -Z flag was given. Verify kernel image
434 1.1 is * fits into chipmem.
435 1.1 is */
436 1.1 is if (ksize >= cmemsz) {
437 1.1 is printf("Kernel size %d exceeds Chip Memory of %d\n",
438 1.1 is ksize, cmemsz);
439 1.1 is return 20;
440 1.1 is }
441 1.1 is Z_flag = 1;
442 1.1 is printf("*** Loading from %08lx to Chipmem ***\n", (u_long)kp);
443 1.1 is }
444 1.1 is
445 1.1 is #if 0
446 1.1 is printf("would start(kp=0x%lx, ksize=%ld, entry=0x%lx,\n"
447 1.1 is "fmem=0x%lx, fmemsz=%ld, cmemsz=%ld\n"
448 1.1 is "boothow=0x%lx, esym=0x%lx, cpuid=0x%lx, eclock=%ld\n"
449 1.1 is "amigaflags=0x%lx, I_flags=0x%lx, Zflag=%ld, ok?\n",
450 1.1 is (u_long)kp, (u_long)ksize, eh->a_entry,
451 1.1 is (u_long)fmem, (u_long)fmemsz, (u_long)cmemsz,
452 1.1 is (u_long)boothowto, (u_long)esym, (u_long)cpuid, (u_long)eclock,
453 1.1 is (u_long)amiga_flags, (u_long)I_flag, (u_long)(Z_flag == 0));
454 1.1 is #endif
455 1.1 is timelimit = 2;
456 1.1 is (void)getchar();
457 1.1 is
458 1.1 is start_it(kp, ksize, eh->a_entry, (void *)fmem, fmemsz, cmemsz,
459 1.3 mhitch boothowto, esym, cpuid, eclock, amiga_flags, I_flag,
460 1.3 mhitch aio_base >> 9, Z_flag == 0);
461 1.1 is /*NOTREACHED*/
462 1.1 is
463 1.1 is freeall:
464 1.1 is free(kp, ksize);
465 1.1 is free(eh, sizeof(*eh));
466 1.1 is err:
467 1.1 is printf("\nError %ld\n", (long)errno);
468 1.1 is close(io);
469 1.1 is out:
470 1.1 is timelimit = 10;
471 1.1 is (void)getchar();
472 1.1 is return 1;
473 1.1 is }
474 1.1 is
475 1.1 is static
476 1.1 is long get_number(ptr)
477 1.1 is char **ptr;
478 1.1 is {
479 1.1 is long value = 0;
480 1.1 is int base = 10;
481 1.1 is char *p = *ptr;
482 1.1 is char c;
483 1.1 is char sign = 0;
484 1.1 is
485 1.1 is c = *++p;
486 1.1 is while (c == ' ')
487 1.1 is c = *++p;
488 1.1 is if (c == '-') {
489 1.1 is sign = -1;
490 1.1 is c = *++p;
491 1.1 is }
492 1.1 is if (c == '$') {
493 1.1 is base = 16;
494 1.1 is c = *++p;
495 1.1 is } else if (c == '0') {
496 1.1 is c = *++p;
497 1.1 is if ((c & 0xdf) == 'X') {
498 1.1 is base = 16;
499 1.1 is c = *++p;
500 1.1 is }
501 1.1 is }
502 1.1 is while (c) {
503 1.1 is if (c >= '0' && c <= '9')
504 1.1 is c -= '0';
505 1.1 is else {
506 1.1 is c = (c & 0xdf) - 'A' + 10;
507 1.1 is if (base != 16 || c < 10 || c > 15)
508 1.1 is break;
509 1.1 is }
510 1.1 is value = value * base + c;
511 1.1 is c = *++p;
512 1.1 is }
513 1.1 is *ptr = p - 1;
514 1.1 is #ifdef TEST
515 1.1 is fprintf(stderr, "get_number: got %c0x%x",
516 1.1 is sign ? '-' : '+', value);
517 1.1 is #endif
518 1.1 is return (sign ? -value : value);
519 1.1 is }
520 1.1 is
521 1.1 is /*
522 1.1 is * Try to determine the machine ID by searching the resident module list
523 1.1 is * for modules only present on specific machines. (Thanks, Bill!)
524 1.1 is */
525 1.1 is
526 1.1 is int
527 1.1 is get_cpuid(cpuid)
528 1.1 is u_int32_t *cpuid;
529 1.1 is {
530 1.1 is *cpuid |= SysBase->AttnFlags; /* get FPU and CPU flags */
531 1.1 is if (*cpuid & 0xffff0000) {
532 1.1 is if ((*cpuid >> 24) == 0x7D)
533 1.1 is return 0;
534 1.1 is
535 1.1 is switch (*cpuid >> 16) {
536 1.1 is case 500:
537 1.1 is case 600:
538 1.1 is case 1000:
539 1.1 is case 1200:
540 1.1 is case 2000:
541 1.1 is case 3000:
542 1.1 is case 4000:
543 1.1 is return 0;
544 1.1 is default:
545 1.1 is printf("Amiga %ld ???\n",
546 1.1 is (long)(*cpuid >> 16));
547 1.1 is return(1);
548 1.1 is }
549 1.1 is }
550 1.1 is if (FindResident("A4000 Bonus") || FindResident("A4000 bonus")
551 1.1 is || FindResident("A1000 Bonus"))
552 1.1 is *cpuid |= 4000 << 16;
553 1.1 is else if (FindResident("A3000 Bonus") || FindResident("A3000 bonus")
554 1.1 is || (SysBase->LibNode.Version == 36))
555 1.1 is *cpuid |= 3000 << 16;
556 1.1 is else if (OpenResource("card.resource")) {
557 1.1 is /* Test for AGA? */
558 1.1 is *cpuid |= 1200 << 16;
559 1.1 is } else if (OpenResource("draco.resource")) {
560 1.1 is *cpuid |= (32000 | DRACOREVISION) << 16;
561 1.1 is }
562 1.1 is /*
563 1.1 is * Nothing found, it's probably an A2000 or A500
564 1.1 is */
565 1.1 is if ((*cpuid >> 16) == 0)
566 1.1 is *cpuid |= 2000 << 16;
567 1.1 is
568 1.1 is return 0;
569 1.1 is }
570