xref: /netbsd-src/sys/arch/x68k/x68k/autoconf.c (revision 6b664a713479c31d4f17b38b42182a5d5fa21802)
1 /*	$NetBSD: autoconf.c,v 1.69 2024/01/07 07:58:35 isaki Exp $	*/
2 
3 /*
4  * Copyright (c) 1995 Leo Weppelman
5  * Copyright (c) 1994 Christian E. Hopps
6  *
7  * Redistribution and use in source and binary forms, with or without
8  * modification, are permitted provided that the following conditions
9  * are met:
10  * 1. Redistributions of source code must retain the above copyright
11  *    notice, this list of conditions and the following disclaimer.
12  * 2. Redistributions in binary form must reproduce the above copyright
13  *    notice, this list of conditions and the following disclaimer in the
14  *    documentation and/or other materials provided with the distribution.
15  * 3. All advertising materials mentioning features or use of this software
16  *    must display the following acknowledgement:
17  *      This product includes software developed by Christian E. Hopps.
18  * 4. The name of the author may not be used to endorse or promote products
19  *    derived from this software without specific prior written permission
20  *
21  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
22  * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
23  * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
24  * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
25  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
26  * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
27  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
28  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
29  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
30  * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
31  */
32 
33 #include <sys/cdefs.h>
34 __KERNEL_RCSID(0, "$NetBSD: autoconf.c,v 1.69 2024/01/07 07:58:35 isaki Exp $");
35 
36 #include "opt_compat_netbsd.h"
37 #include "scsibus.h"
38 
39 #include <sys/param.h>
40 #include <sys/systm.h>
41 #include <sys/reboot.h>
42 #include <sys/conf.h>
43 #include <sys/device.h>
44 #include <sys/disk.h>
45 #include <sys/disklabel.h>
46 #include <machine/cpu.h>
47 #include <machine/bootinfo.h>
48 #include <machine/autoconf.h>
49 
50 #include <dev/scsipi/scsi_all.h>
51 #include <dev/scsipi/scsipi_all.h>
52 #include <dev/scsipi/scsiconf.h>
53 
54 static void findroot(void);
55 static device_t scsi_find(dev_t);
56 
57 int x68k_realconfig;
58 
59 /*
60  * called at boot time, configure all devices on system
61  */
62 void
cpu_configure(void)63 cpu_configure(void)
64 {
65 	x68k_realconfig = 1;
66 
67 	if (config_rootfound("mainbus", NULL) == NULL)
68 		panic("no mainbus found");
69 
70 	/* Turn on interrupts */
71 	printf("enabling interrupts\n");
72 	(void) spl0();
73 }
74 
75 void
cpu_rootconf(void)76 cpu_rootconf(void)
77 {
78 	findroot();
79 
80 	printf("boot device: %s\n",
81 	    booted_device ? device_xname(booted_device) : "<unknown>");
82 
83 	rootconf();
84 }
85 
86 void
config_console(void)87 config_console(void)
88 {
89 	mfp_config_console();
90 	grf_config_console();
91 	ite_config_console();
92 }
93 
94 uint32_t bootdev = 0;
95 
96 static void
findroot(void)97 findroot(void)
98 {
99 	int majdev, unit, part;
100 	const char *name;
101 
102 	if (booted_device)
103 		return;
104 
105 	if (boothowto & RB_ASKNAME)
106 		return;		/* Don't bother looking */
107 
108 	if ((bootdev & B_MAGICMASK) != (u_long)B_DEVMAGIC)
109 		return;
110 
111 	majdev = B_TYPE(bootdev);
112 	if (X68K_BOOT_DEV_IS_SCSI(majdev)) {
113 		/*
114 		 * SCSI device
115 		 */
116 		if ((booted_device = scsi_find(bootdev)) != NULL)
117 			booted_partition = B_X68K_SCSI_PART(bootdev);
118 		return;
119 	}
120 	name = devsw_blk2name(majdev);
121 	if (name == NULL)
122 		return;
123 
124 	part = B_PARTITION(bootdev);
125 	unit = B_UNIT(bootdev);
126 
127 	if ((booted_device = device_find_by_driver_unit(name, unit)) != NULL)
128 		booted_partition = part;
129 }
130 
131 static const char *const name_netif[] = { X68K_BOOT_NETIF_STRINGS };
132 
133 void
device_register(device_t dev,void * aux)134 device_register(device_t dev, void *aux)
135 {
136 	int majdev;
137 	char tname[16];
138 
139 	/*
140 	 * Handle network interfaces here, the attachment information is
141 	 * not available driver independently later.
142 	 * For disks, there is nothing useful available at attach time.
143 	 */
144 	if (device_class(dev) == DV_IFNET) {
145 		majdev = B_TYPE(bootdev);
146 		if (X68K_BOOT_DEV_IS_NETIF(majdev)) {
147 			snprintf(tname, sizeof(tname), "%s%d",
148 				name_netif[255 - majdev], B_UNIT(bootdev));
149 			if (!strcmp(tname, device_xname(dev)))
150 				goto found;
151 		}
152 	}
153 	return;
154 
155 found:
156 	if (booted_device) {
157 		/* XXX should be a "panic()" */
158 		printf("warning: double match for boot device (%s, %s)\n",
159 		       device_xname(booted_device), device_xname(dev));
160 		return;
161 	}
162 	booted_device = dev;
163 }
164 
165 static const char *const name_scsiif[] = { X68K_BOOT_SCSIIF_STRINGS };
166 
167 static device_t
scsi_find(dev_t bdev)168 scsi_find(dev_t bdev)
169 {
170 #if defined(NSCSIBUS) && NSCSIBUS > 0
171 	int ifid;
172 	char tname[16];
173 	device_t scsibus;
174 	deviter_t di;
175 	struct scsibus_softc *sbsc;
176 	struct scsipi_periph *periph;
177 
178 	ifid = B_X68K_SCSI_IF(bdev);
179 	if (ifid >= sizeof name_scsiif/sizeof name_scsiif[0] ||
180 					!name_scsiif[ifid]) {
181 #ifdef COMPAT_13
182 		/*
183 		 * old boot didn't pass interface type
184 		 * try "scsibus0"
185 		 */
186 		printf("warning: scsi_find: can't get boot interface -- "
187 		       "update boot loader\n");
188 		scsibus = device_find_by_xname("scsibus0");
189 #else
190 		/* can't determine interface type */
191 		return NULL;
192 #endif
193 	} else {
194 		/*
195 		 * search for the scsibus whose parent is
196 		 * the specified SCSI interface
197 		 */
198 		snprintf(tname, sizeof(tname), "%s%" PRIu64,
199 			name_scsiif[ifid], B_X68K_SCSI_IF_UN(bdev));
200 
201 		for (scsibus = deviter_first(&di, DEVITER_F_ROOT_FIRST);
202 		     scsibus != NULL;
203 		     scsibus = deviter_next(&di)) {
204 			if (device_parent(scsibus)
205 			    && strcmp(tname, device_xname(device_parent(scsibus))) == 0)
206 				break;
207 		}
208 		deviter_release(&di);
209 	}
210 	if (scsibus == NULL)
211 		return NULL;
212 	sbsc = device_private(scsibus);
213 	periph = scsipi_lookup_periph(sbsc->sc_channel,
214 	    B_X68K_SCSI_ID(bdev), B_X68K_SCSI_LUN(bdev));
215 
216 	return periph ? periph->periph_dev : NULL;
217 #else
218 	return NULL;
219 #endif /* NSCSIBUS > 0 */
220 }
221