1*58a2b000SEvgeniy Ivanov /* $NetBSD: devopen.c,v 1.8 2010/12/24 20:40:42 jakllsch Exp $ */
2*58a2b000SEvgeniy Ivanov
3*58a2b000SEvgeniy Ivanov /*-
4*58a2b000SEvgeniy Ivanov * Copyright (c) 2005 The NetBSD Foundation, Inc.
5*58a2b000SEvgeniy Ivanov * All rights reserved.
6*58a2b000SEvgeniy Ivanov *
7*58a2b000SEvgeniy Ivanov * This code is derived from software contributed to The NetBSD Foundation
8*58a2b000SEvgeniy Ivanov * by Bang Jun-Young.
9*58a2b000SEvgeniy Ivanov *
10*58a2b000SEvgeniy Ivanov * Redistribution and use in source and binary forms, with or without
11*58a2b000SEvgeniy Ivanov * modification, are permitted provided that the following conditions
12*58a2b000SEvgeniy Ivanov * are met:
13*58a2b000SEvgeniy Ivanov * 1. Redistributions of source code must retain the above copyright
14*58a2b000SEvgeniy Ivanov * notice, this list of conditions and the following disclaimer.
15*58a2b000SEvgeniy Ivanov * 2. Redistributions in binary form must reproduce the above copyright
16*58a2b000SEvgeniy Ivanov * notice, this list of conditions and the following disclaimer in the
17*58a2b000SEvgeniy Ivanov * documentation and/or other materials provided with the distribution.
18*58a2b000SEvgeniy Ivanov *
19*58a2b000SEvgeniy Ivanov * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
20*58a2b000SEvgeniy Ivanov * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
21*58a2b000SEvgeniy Ivanov * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
22*58a2b000SEvgeniy Ivanov * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
23*58a2b000SEvgeniy Ivanov * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
24*58a2b000SEvgeniy Ivanov * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
25*58a2b000SEvgeniy Ivanov * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
26*58a2b000SEvgeniy Ivanov * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
27*58a2b000SEvgeniy Ivanov * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
28*58a2b000SEvgeniy Ivanov * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
29*58a2b000SEvgeniy Ivanov * POSSIBILITY OF SUCH DAMAGE.
30*58a2b000SEvgeniy Ivanov */
31*58a2b000SEvgeniy Ivanov
32*58a2b000SEvgeniy Ivanov /*
33*58a2b000SEvgeniy Ivanov * Copyright (c) 1996, 1997
34*58a2b000SEvgeniy Ivanov * Matthias Drochner. All rights reserved.
35*58a2b000SEvgeniy Ivanov *
36*58a2b000SEvgeniy Ivanov * Redistribution and use in source and binary forms, with or without
37*58a2b000SEvgeniy Ivanov * modification, are permitted provided that the following conditions
38*58a2b000SEvgeniy Ivanov * are met:
39*58a2b000SEvgeniy Ivanov * 1. Redistributions of source code must retain the above copyright
40*58a2b000SEvgeniy Ivanov * notice, this list of conditions and the following disclaimer.
41*58a2b000SEvgeniy Ivanov * 2. Redistributions in binary form must reproduce the above copyright
42*58a2b000SEvgeniy Ivanov * notice, this list of conditions and the following disclaimer in the
43*58a2b000SEvgeniy Ivanov * documentation and/or other materials provided with the distribution.
44*58a2b000SEvgeniy Ivanov *
45*58a2b000SEvgeniy Ivanov * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
46*58a2b000SEvgeniy Ivanov * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
47*58a2b000SEvgeniy Ivanov * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
48*58a2b000SEvgeniy Ivanov * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
49*58a2b000SEvgeniy Ivanov * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
50*58a2b000SEvgeniy Ivanov * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
51*58a2b000SEvgeniy Ivanov * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
52*58a2b000SEvgeniy Ivanov * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
53*58a2b000SEvgeniy Ivanov * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
54*58a2b000SEvgeniy Ivanov * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
55*58a2b000SEvgeniy Ivanov */
56*58a2b000SEvgeniy Ivanov
57*58a2b000SEvgeniy Ivanov
58*58a2b000SEvgeniy Ivanov #include <sys/types.h>
59*58a2b000SEvgeniy Ivanov
60*58a2b000SEvgeniy Ivanov #include <lib/libsa/stand.h>
61*58a2b000SEvgeniy Ivanov #include <lib/libkern/libkern.h>
62*58a2b000SEvgeniy Ivanov
63*58a2b000SEvgeniy Ivanov #include <libi386.h>
64*58a2b000SEvgeniy Ivanov #include <biosdisk.h>
65*58a2b000SEvgeniy Ivanov #include "devopen.h"
66*58a2b000SEvgeniy Ivanov #ifdef _STANDALONE
67*58a2b000SEvgeniy Ivanov #include <bootinfo.h>
68*58a2b000SEvgeniy Ivanov #endif
69*58a2b000SEvgeniy Ivanov #ifdef SUPPORT_PS2
70*58a2b000SEvgeniy Ivanov #include <biosmca.h>
71*58a2b000SEvgeniy Ivanov #endif
72*58a2b000SEvgeniy Ivanov
73*58a2b000SEvgeniy Ivanov static int dev2bios(char *, int, int *);
74*58a2b000SEvgeniy Ivanov
75*58a2b000SEvgeniy Ivanov static int
dev2bios(char * devname,int unit,int * biosdev)76*58a2b000SEvgeniy Ivanov dev2bios(char *devname, int unit, int *biosdev)
77*58a2b000SEvgeniy Ivanov {
78*58a2b000SEvgeniy Ivanov
79*58a2b000SEvgeniy Ivanov if (strcmp(devname, "hd") == 0)
80*58a2b000SEvgeniy Ivanov *biosdev = 0x80 + unit;
81*58a2b000SEvgeniy Ivanov else if (strcmp(devname, "fd") == 0)
82*58a2b000SEvgeniy Ivanov *biosdev = 0x00 + unit;
83*58a2b000SEvgeniy Ivanov else if (strcmp(devname, "cd") == 0)
84*58a2b000SEvgeniy Ivanov *biosdev = boot_biosdev;
85*58a2b000SEvgeniy Ivanov else
86*58a2b000SEvgeniy Ivanov return ENXIO;
87*58a2b000SEvgeniy Ivanov
88*58a2b000SEvgeniy Ivanov return 0;
89*58a2b000SEvgeniy Ivanov }
90*58a2b000SEvgeniy Ivanov
91*58a2b000SEvgeniy Ivanov void
bios2dev(int biosdev,daddr_t sector,char ** devname,int * unit,int * partition)92*58a2b000SEvgeniy Ivanov bios2dev(int biosdev, daddr_t sector, char **devname, int *unit, int *partition)
93*58a2b000SEvgeniy Ivanov {
94*58a2b000SEvgeniy Ivanov
95*58a2b000SEvgeniy Ivanov /* set default */
96*58a2b000SEvgeniy Ivanov *unit = biosdev & 0x7f;
97*58a2b000SEvgeniy Ivanov
98*58a2b000SEvgeniy Ivanov if (biosdev & 0x80) {
99*58a2b000SEvgeniy Ivanov /*
100*58a2b000SEvgeniy Ivanov * There seems to be no standard way of numbering BIOS
101*58a2b000SEvgeniy Ivanov * CD-ROM drives. The following method is a little tricky
102*58a2b000SEvgeniy Ivanov * but works nicely.
103*58a2b000SEvgeniy Ivanov */
104*58a2b000SEvgeniy Ivanov if (biosdev >= 0x80 + get_harddrives()) {
105*58a2b000SEvgeniy Ivanov *devname = "cd";
106*58a2b000SEvgeniy Ivanov *unit = 0; /* override default */
107*58a2b000SEvgeniy Ivanov } else
108*58a2b000SEvgeniy Ivanov *devname = "hd";
109*58a2b000SEvgeniy Ivanov } else
110*58a2b000SEvgeniy Ivanov *devname = "fd";
111*58a2b000SEvgeniy Ivanov
112*58a2b000SEvgeniy Ivanov *partition = biosdisk_findpartition(biosdev, sector);
113*58a2b000SEvgeniy Ivanov }
114*58a2b000SEvgeniy Ivanov
115*58a2b000SEvgeniy Ivanov #ifdef _STANDALONE
116*58a2b000SEvgeniy Ivanov struct btinfo_bootpath bibp;
117*58a2b000SEvgeniy Ivanov extern bool kernel_loaded;
118*58a2b000SEvgeniy Ivanov #endif
119*58a2b000SEvgeniy Ivanov
120*58a2b000SEvgeniy Ivanov /*
121*58a2b000SEvgeniy Ivanov * Open the BIOS disk device
122*58a2b000SEvgeniy Ivanov */
123*58a2b000SEvgeniy Ivanov int
devopen(struct open_file * f,const char * fname,char ** file)124*58a2b000SEvgeniy Ivanov devopen(struct open_file *f, const char *fname, char **file)
125*58a2b000SEvgeniy Ivanov {
126*58a2b000SEvgeniy Ivanov char *fsname, *devname;
127*58a2b000SEvgeniy Ivanov int unit, partition;
128*58a2b000SEvgeniy Ivanov int biosdev;
129*58a2b000SEvgeniy Ivanov int error;
130*58a2b000SEvgeniy Ivanov
131*58a2b000SEvgeniy Ivanov if ((error = parsebootfile(fname, &fsname, &devname,
132*58a2b000SEvgeniy Ivanov &unit, &partition, (const char **) file))
133*58a2b000SEvgeniy Ivanov || (error = dev2bios(devname, unit, &biosdev)))
134*58a2b000SEvgeniy Ivanov return error;
135*58a2b000SEvgeniy Ivanov
136*58a2b000SEvgeniy Ivanov f->f_dev = &devsw[0]; /* must be biosdisk */
137*58a2b000SEvgeniy Ivanov
138*58a2b000SEvgeniy Ivanov #ifdef _STANDALONE
139*58a2b000SEvgeniy Ivanov if (!kernel_loaded) {
140*58a2b000SEvgeniy Ivanov strncpy(bibp.bootpath, *file, sizeof(bibp.bootpath));
141*58a2b000SEvgeniy Ivanov BI_ADD(&bibp, BTINFO_BOOTPATH, sizeof(bibp));
142*58a2b000SEvgeniy Ivanov }
143*58a2b000SEvgeniy Ivanov #endif
144*58a2b000SEvgeniy Ivanov
145*58a2b000SEvgeniy Ivanov return biosdisk_open(f, biosdev, partition);
146*58a2b000SEvgeniy Ivanov }
147