]> CyberLeo.Net >> Repos - FreeBSD/releng/8.1.git/blob - sys/boot/powerpc/boot1.chrp/boot1.c
Copy stable/8 to releng/8.1 in preparation for 8.1-RC1.
[FreeBSD/releng/8.1.git] / sys / boot / powerpc / boot1.chrp / boot1.c
1 /*-
2  * Copyright (c) 1998 Robert Nordier
3  * All rights reserved.
4  * Copyright (c) 2001 Robert Drehmel
5  * All rights reserved.
6  *
7  * Redistribution and use in source and binary forms are freely
8  * permitted provided that the above copyright notice and this
9  * paragraph and the following disclaimer are duplicated in all
10  * such forms.
11  *
12  * This software is provided "AS IS" and without any express or
13  * implied warranties, including, without limitation, the implied
14  * warranties of merchantability and fitness for a particular
15  * purpose.
16  */
17
18 #include <sys/cdefs.h>
19 __FBSDID("$FreeBSD$");
20
21 #include <sys/param.h>
22 #include <sys/dirent.h>
23 #include <machine/elf.h>
24 #include <machine/stdarg.h>
25
26 #define _PATH_LOADER    "/boot/loader"
27 #define _PATH_KERNEL    "/boot/kernel/kernel"
28
29 #define BSIZEMAX        16384
30
31 typedef int putc_func_t(char c, void *arg);
32 typedef int32_t ofwh_t;
33
34 struct sp_data {
35         char    *sp_buf;
36         u_int   sp_len;
37         u_int   sp_size;
38 };
39
40 static const char digits[] = "0123456789abcdef";
41
42 static char bootpath[128];
43 static char bootargs[128];
44
45 static ofwh_t bootdev;
46
47 static struct fs fs;
48 static ino_t inomap;
49 static char blkbuf[BSIZEMAX];
50 static unsigned int fsblks;
51
52 static uint32_t fs_off;
53
54 int main(int ac, char **av);
55
56 static void exit(int) __dead2;
57 static void load(const char *);
58 static int dskread(void *, u_int64_t, int);
59
60 static void usage(void);
61
62 static void bcopy(const void *src, void *dst, size_t len);
63 static void bzero(void *b, size_t len);
64
65 static int mount(const char *device, int quiet);
66
67 static void panic(const char *fmt, ...) __dead2;
68 static int printf(const char *fmt, ...);
69 static int putchar(char c, void *arg);
70 static int vprintf(const char *fmt, va_list ap);
71 static int vsnprintf(char *str, size_t sz, const char *fmt, va_list ap);
72
73 static int __printf(const char *fmt, putc_func_t *putc, void *arg, va_list ap);
74 static int __putc(char c, void *arg);
75 static int __puts(const char *s, putc_func_t *putc, void *arg);
76 static int __sputc(char c, void *arg);
77 static char *__uitoa(char *buf, u_int val, int base);
78 static char *__ultoa(char *buf, u_long val, int base);
79
80 /*
81  * Open Firmware interface functions
82  */
83 typedef u_int32_t       ofwcell_t;
84 typedef u_int32_t       u_ofwh_t;
85 typedef int (*ofwfp_t)(void *);
86 ofwfp_t ofw;                    /* the prom Open Firmware entry */
87 ofwh_t chosenh;
88
89 void ofw_init(void *, int, int (*)(void *), char *, int);
90 static ofwh_t ofw_finddevice(const char *);
91 static ofwh_t ofw_open(const char *);
92 static int ofw_close(ofwh_t);
93 static int ofw_getprop(ofwh_t, const char *, void *, size_t);
94 static int ofw_setprop(ofwh_t, const char *, void *, size_t);
95 static int ofw_read(ofwh_t, void *, size_t);
96 static int ofw_write(ofwh_t, const void *, size_t);
97 static int ofw_claim(void *virt, size_t len, u_int align);
98 static int ofw_seek(ofwh_t, u_int64_t);
99 static void ofw_exit(void) __dead2;
100
101 ofwh_t bootdevh;
102 ofwh_t stdinh, stdouth;
103
104 __asm("                         \n\
105         .data                   \n\
106 stack:                          \n\
107         .space  16384           \n\
108                                 \n\
109         .text                   \n\
110         .globl  _start          \n\
111 _start:                         \n\
112         lis     %r1,stack@ha    \n\
113         addi    %r1,%r1,stack@l \n\
114         addi    %r1,%r1,8192    \n\
115                                 \n\
116         b       ofw_init        \n\
117 ");
118
119 void
120 ofw_init(void *vpd, int res, int (*openfirm)(void *), char *arg, int argl)
121 {
122         char *av[16];
123         char *p;
124         int ac;
125
126         ofw = openfirm;
127
128         chosenh = ofw_finddevice("/chosen");
129         ofw_getprop(chosenh, "stdin", &stdinh, sizeof(stdinh));
130         ofw_getprop(chosenh, "stdout", &stdouth, sizeof(stdouth));
131         ofw_getprop(chosenh, "bootargs", bootargs, sizeof(bootargs));
132         ofw_getprop(chosenh, "bootpath", bootpath, sizeof(bootpath));
133
134         bootargs[sizeof(bootargs) - 1] = '\0';
135         bootpath[sizeof(bootpath) - 1] = '\0';
136
137         p = bootpath;
138         while (*p != '\0') {
139                 if (*p == ':') {
140                         *(++p) = '\0';
141                         break;
142                 }
143                 p++;
144         }
145
146         ac = 0;
147         p = bootargs;
148         for (;;) {
149                 while (*p == ' ' && *p != '\0')
150                         p++;
151                 if (*p == '\0' || ac >= 16)
152                         break;
153                 av[ac++] = p;
154                 while (*p != ' ' && *p != '\0')
155                         p++;
156                 if (*p != '\0')
157                         *p++ = '\0';
158         }
159
160         exit(main(ac, av));
161 }
162
163 static ofwh_t
164 ofw_finddevice(const char *name)
165 {
166         ofwcell_t args[] = {
167                 (ofwcell_t)"finddevice",
168                 1,
169                 1,
170                 (ofwcell_t)name,
171                 0
172         };
173
174         if ((*ofw)(args)) {
175                 printf("ofw_finddevice: name=\"%s\"\n", name);
176                 return (1);
177         }
178         return (args[4]);
179 }
180
181 static int
182 ofw_getprop(ofwh_t ofwh, const char *name, void *buf, size_t len)
183 {
184         ofwcell_t args[] = {
185                 (ofwcell_t)"getprop",
186                 4,
187                 1,
188                 (u_ofwh_t)ofwh,
189                 (ofwcell_t)name,
190                 (ofwcell_t)buf,
191                 len,
192         0
193         };
194
195         if ((*ofw)(args)) {
196                 printf("ofw_getprop: ofwh=0x%x buf=%p len=%u\n",
197                         ofwh, buf, len);
198                 return (1);
199         }
200         return (0);
201 }
202
203 static int
204 ofw_setprop(ofwh_t ofwh, const char *name, void *buf, size_t len)
205 {
206         ofwcell_t args[] = {
207                 (ofwcell_t)"setprop",
208                 4,
209                 1,
210                 (u_ofwh_t)ofwh,
211                 (ofwcell_t)name,
212                 (ofwcell_t)buf,
213                 len,
214         0
215         };
216
217         if ((*ofw)(args)) {
218                 printf("ofw_setprop: ofwh=0x%x buf=%p len=%u\n",
219                         ofwh, buf, len);
220                 return (1);
221         }
222         return (0);
223 }
224
225 static ofwh_t
226 ofw_open(const char *path)
227 {
228         ofwcell_t args[] = {
229                 (ofwcell_t)"open",
230                 1,
231                 1,
232                 (ofwcell_t)path,
233                 0
234         };
235
236         if ((*ofw)(args)) {
237                 printf("ofw_open: path=\"%s\"\n", path);
238                 return (-1);
239         }
240         return (args[4]);
241 }
242
243 static int
244 ofw_close(ofwh_t devh)
245 {
246         ofwcell_t args[] = {
247                 (ofwcell_t)"close",
248                 1,
249                 0,
250                 (u_ofwh_t)devh
251         };
252
253         if ((*ofw)(args)) {
254                 printf("ofw_close: devh=0x%x\n", devh);
255                 return (1);
256         }
257         return (0);
258 }
259
260 static int
261 ofw_claim(void *virt, size_t len, u_int align)
262 {
263         ofwcell_t args[] = {
264                 (ofwcell_t)"claim",
265                 3,
266                 1,
267                 (ofwcell_t)virt,
268                 len,
269                 align,
270                 0,
271                 0
272         };
273
274         if ((*ofw)(args)) {
275                 printf("ofw_claim: virt=%p len=%u\n", virt, len);
276                 return (1);
277         }
278
279         return (0);
280 }
281
282 static int
283 ofw_read(ofwh_t devh, void *buf, size_t len)
284 {
285         ofwcell_t args[] = {
286                 (ofwcell_t)"read",
287                 3,
288                 1,
289                 (u_ofwh_t)devh,
290                 (ofwcell_t)buf,
291                 len,
292                 0
293         };
294
295         if ((*ofw)(args)) {
296                 printf("ofw_read: devh=0x%x buf=%p len=%u\n", devh, buf, len);
297                 return (1);
298         }
299         return (0);
300 }
301
302 static int
303 ofw_write(ofwh_t devh, const void *buf, size_t len)
304 {
305         ofwcell_t args[] = {
306                 (ofwcell_t)"write",
307                 3,
308                 1,
309                 (u_ofwh_t)devh,
310                 (ofwcell_t)buf,
311                 len,
312                 0
313         };
314
315         if ((*ofw)(args)) {
316                 printf("ofw_write: devh=0x%x buf=%p len=%u\n", devh, buf, len);
317                 return (1);
318         }
319         return (0);
320 }
321
322 static int
323 ofw_seek(ofwh_t devh, u_int64_t off)
324 {
325         ofwcell_t args[] = {
326                 (ofwcell_t)"seek",
327                 3,
328                 1,
329                 (u_ofwh_t)devh,
330                 off >> 32,
331                 off,
332                 0
333         };
334
335         if ((*ofw)(args)) {
336                 printf("ofw_seek: devh=0x%x off=0x%lx\n", devh, off);
337                 return (1);
338         }
339         return (0);
340 }
341
342 static void
343 ofw_exit(void)
344 {
345         ofwcell_t args[3];
346
347         args[0] = (ofwcell_t)"exit";
348         args[1] = 0;
349         args[2] = 0;
350
351         for (;;)
352                 (*ofw)(args);
353 }
354
355 static void
356 bcopy(const void *src, void *dst, size_t len)
357 {
358         const char *s = src;
359         char *d = dst;
360
361         while (len-- != 0)
362                 *d++ = *s++;
363 }
364
365 static void
366 memcpy(void *dst, const void *src, size_t len)
367 {
368         bcopy(src, dst, len);
369 }
370
371 static void
372 bzero(void *b, size_t len)
373 {
374         char *p = b;
375
376         while (len-- != 0)
377                 *p++ = 0;
378 }
379
380 static int
381 strcmp(const char *s1, const char *s2)
382 {
383         for (; *s1 == *s2 && *s1; s1++, s2++)
384                 ;
385         return ((u_char)*s1 - (u_char)*s2);
386 }
387
388 #include "ufsread.c"
389
390 int
391 main(int ac, char **av)
392 {
393         const char *path;
394         char bootpath_full[255];
395         int i, len;
396
397         path = _PATH_LOADER;
398         for (i = 0; i < ac; i++) {
399                 switch (av[i][0]) {
400                 case '-':
401                         switch (av[i][1]) {
402                         default:
403                                 usage();
404                         }
405                         break;
406                 default:
407                         path = av[i];
408                         break;
409                 }
410         }
411
412         printf(" \n>> FreeBSD/powerpc Open Firmware boot block\n"
413         "   Boot path:   %s\n"
414         "   Boot loader: %s\n", bootpath, path);
415
416         len = 0;
417         while (bootpath[len] != '\0') len++;
418
419         memcpy(bootpath_full,bootpath,len+1);
420
421         if (bootpath_full[len-1] == ':') {
422                 for (i = 0; i < 16; i++) {
423                         if (i < 10) {
424                                 bootpath_full[len] = i + '0';
425                                 bootpath_full[len+1] = '\0';
426                         } else {
427                                 bootpath_full[len] = '1';
428                                 bootpath_full[len+1] = i - 10 + '0';
429                                 bootpath_full[len+2] = '\0';
430                         }
431                                 
432                         if (mount(bootpath_full,1) >= 0)
433                                 break;
434
435                         if (bootdev > 0)
436                                 ofw_close(bootdev);
437                 }
438
439                 if (i >= 16)
440                         panic("mount");
441         } else {
442                 if (mount(bootpath_full,0) == -1)
443                         panic("mount");
444         }
445
446         printf("   Boot volume:   %s\n",bootpath_full);
447         ofw_setprop(chosenh, "bootargs", bootpath_full, len+2);
448         load(path);
449         return (1);
450 }
451
452 static void
453 usage(void)
454 {
455
456         printf("usage: boot device [/path/to/loader]\n");
457         exit(1);
458 }
459
460 static void
461 exit(int code)
462 {
463
464         ofw_exit();
465 }
466
467 static struct dmadat __dmadat;
468
469 static int
470 mount(const char *device, int quiet)
471 {
472
473         dmadat = &__dmadat;
474         if ((bootdev = ofw_open(device)) == -1) {
475                 printf("mount: can't open device\n");
476                 return (-1);
477         }
478         if (fsread(0, NULL, 0)) {
479                 if (!quiet)
480                         printf("mount: can't read superblock\n");
481                 return (-1);
482         }
483         return (0);
484 }
485
486 static void
487 load(const char *fname)
488 {
489         Elf32_Ehdr eh;
490         Elf32_Phdr ph;
491         caddr_t p;
492         ino_t ino;
493         int i;
494
495         if ((ino = lookup(fname)) == 0) {
496                 printf("File %s not found\n", fname);
497                 return;
498         }
499         if (fsread(ino, &eh, sizeof(eh)) != sizeof(eh)) {
500                 printf("Can't read elf header\n");
501                 return;
502         }
503         if (!IS_ELF(eh)) {
504                 printf("Not an ELF file\n");
505                 return;
506         }
507         for (i = 0; i < eh.e_phnum; i++) {
508                 fs_off = eh.e_phoff + i * eh.e_phentsize;
509                 if (fsread(ino, &ph, sizeof(ph)) != sizeof(ph)) {
510                         printf("Can't read program header %d\n", i);
511                         return;
512                 }
513                 if (ph.p_type != PT_LOAD)
514                         continue;
515                 fs_off = ph.p_offset;
516                 p = (caddr_t)ph.p_vaddr;
517                 ofw_claim(p,(ph.p_filesz > ph.p_memsz) ? 
518                     ph.p_filesz : ph.p_memsz,0);
519                 if (fsread(ino, p, ph.p_filesz) != ph.p_filesz) {
520                         printf("Can't read content of section %d\n", i);
521                         return;
522                 }
523                 if (ph.p_filesz != ph.p_memsz)
524                         bzero(p + ph.p_filesz, ph.p_memsz - ph.p_filesz);
525         }
526         ofw_close(bootdev);
527         (*(void (*)(void *, int, ofwfp_t, char *, int))eh.e_entry)(NULL, 0, 
528             ofw,NULL,0);
529 }
530
531 static int
532 dskread(void *buf, u_int64_t lba, int nblk)
533 {
534         /*
535          * The Open Firmware should open the correct partition for us.
536          * That means, if we read from offset zero on an open instance handle,
537          * we should read from offset zero of that partition.
538          */
539         ofw_seek(bootdev, lba * DEV_BSIZE);
540         ofw_read(bootdev, buf, nblk * DEV_BSIZE);
541         return (0);
542 }
543
544 static void
545 panic(const char *fmt, ...)
546 {
547         char buf[128];
548         va_list ap;
549
550         va_start(ap, fmt);
551         vsnprintf(buf, sizeof buf, fmt, ap);
552         printf("panic: %s\n", buf);
553         va_end(ap);
554
555         exit(1);
556 }
557
558 static int
559 printf(const char *fmt, ...)
560 {
561         va_list ap;
562         int ret;
563
564         /* Don't annoy the user as we probe for partitions */
565         if (strcmp(fmt,"Not ufs\n") == 0)
566                 return 0;
567
568         va_start(ap, fmt);
569         ret = vprintf(fmt, ap);
570         va_end(ap);
571         return (ret);
572 }
573
574 static int
575 putchar(char c, void *arg)
576 {
577         char buf;
578
579         if (c == '\n') {
580                 buf = '\r';
581                 ofw_write(stdouth, &buf, 1);
582         }
583         buf = c;
584         ofw_write(stdouth, &buf, 1);
585         return (1);
586 }
587
588 static int
589 vprintf(const char *fmt, va_list ap)
590 {
591         int ret;
592
593         ret = __printf(fmt, putchar, 0, ap);
594         return (ret);
595 }
596
597 static int
598 vsnprintf(char *str, size_t sz, const char *fmt, va_list ap)
599 {
600         struct sp_data sp;
601         int ret;
602
603         sp.sp_buf = str;
604         sp.sp_len = 0;
605         sp.sp_size = sz;
606         ret = __printf(fmt, __sputc, &sp, ap);
607         return (ret);
608 }
609
610 static int
611 __printf(const char *fmt, putc_func_t *putc, void *arg, va_list ap)
612 {
613         char buf[(sizeof(long) * 8) + 1];
614         char *nbuf;
615         u_long ul;
616         u_int ui;
617         int lflag;
618         int sflag;
619         char *s;
620         int pad;
621         int ret;
622         int c;
623
624         nbuf = &buf[sizeof buf - 1];
625         ret = 0;
626         while ((c = *fmt++) != 0) {
627                 if (c != '%') {
628                         ret += putc(c, arg);
629                         continue;
630                 }
631                 lflag = 0;
632                 sflag = 0;
633                 pad = 0;
634 reswitch:       c = *fmt++;
635                 switch (c) {
636                 case '#':
637                         sflag = 1;
638                         goto reswitch;
639                 case '%':
640                         ret += putc('%', arg);
641                         break;
642                 case 'c':
643                         c = va_arg(ap, int);
644                         ret += putc(c, arg);
645                         break;
646                 case 'd':
647                         if (lflag == 0) {
648                                 ui = (u_int)va_arg(ap, int);
649                                 if (ui < (int)ui) {
650                                         ui = -ui;
651                                         ret += putc('-', arg);
652                                 }
653                                 s = __uitoa(nbuf, ui, 10);
654                         } else {
655                                 ul = (u_long)va_arg(ap, long);
656                                 if (ul < (long)ul) {
657                                         ul = -ul;
658                                         ret += putc('-', arg);
659                                 }
660                                 s = __ultoa(nbuf, ul, 10);
661                         }
662                         ret += __puts(s, putc, arg);
663                         break;
664                 case 'l':
665                         lflag = 1;
666                         goto reswitch;
667                 case 'o':
668                         if (lflag == 0) {
669                                 ui = (u_int)va_arg(ap, u_int);
670                                 s = __uitoa(nbuf, ui, 8);
671                         } else {
672                                 ul = (u_long)va_arg(ap, u_long);
673                                 s = __ultoa(nbuf, ul, 8);
674                         }
675                         ret += __puts(s, putc, arg);
676                         break;
677                 case 'p':
678                         ul = (u_long)va_arg(ap, void *);
679                         s = __ultoa(nbuf, ul, 16);
680                         ret += __puts("0x", putc, arg);
681                         ret += __puts(s, putc, arg);
682                         break;
683                 case 's':
684                         s = va_arg(ap, char *);
685                         ret += __puts(s, putc, arg);
686                         break;
687                 case 'u':
688                         if (lflag == 0) {
689                                 ui = va_arg(ap, u_int);
690                                 s = __uitoa(nbuf, ui, 10);
691                         } else {
692                                 ul = va_arg(ap, u_long);
693                                 s = __ultoa(nbuf, ul, 10);
694                         }
695                         ret += __puts(s, putc, arg);
696                         break;
697                 case 'x':
698                         if (lflag == 0) {
699                                 ui = va_arg(ap, u_int);
700                                 s = __uitoa(nbuf, ui, 16);
701                         } else {
702                                 ul = va_arg(ap, u_long);
703                                 s = __ultoa(nbuf, ul, 16);
704                         }
705                         if (sflag)
706                                 ret += __puts("0x", putc, arg);
707                         ret += __puts(s, putc, arg);
708                         break;
709                 case '0': case '1': case '2': case '3': case '4':
710                 case '5': case '6': case '7': case '8': case '9':
711                         pad = pad * 10 + c - '0';
712                         goto reswitch;
713                 default:
714                         break;
715                 }
716         }
717         return (ret);
718 }
719
720 static int
721 __sputc(char c, void *arg)
722 {
723         struct sp_data *sp;
724
725         sp = arg;
726         if (sp->sp_len < sp->sp_size)
727                 sp->sp_buf[sp->sp_len++] = c;
728         sp->sp_buf[sp->sp_len] = '\0';
729         return (1);
730 }
731
732 static int
733 __puts(const char *s, putc_func_t *putc, void *arg)
734 {
735         const char *p;
736         int ret;
737
738         ret = 0;
739         for (p = s; *p != '\0'; p++)
740                 ret += putc(*p, arg);
741         return (ret);
742 }
743
744 static char *
745 __uitoa(char *buf, u_int ui, int base)
746 {
747         char *p;
748
749         p = buf;
750         *p = '\0';
751         do
752                 *--p = digits[ui % base];
753         while ((ui /= base) != 0);
754         return (p);
755 }
756
757 static char *
758 __ultoa(char *buf, u_long ul, int base)
759 {
760         char *p;
761
762         p = buf;
763         *p = '\0';
764         do
765                 *--p = digits[ul % base];
766         while ((ul /= base) != 0);
767         return (p);
768 }