xref: /netbsd-src/sys/arch/usermode/usermode/machdep.c (revision df3dd44824305f5aaedf730a010facfe570c00e2)
1 /* $NetBSD: machdep.c,v 1.58 2022/01/01 21:07:14 andvar Exp $ */
2 
3 /*-
4  * Copyright (c) 2011 Reinoud Zandijk <reinoud@netbsd.org>
5  * Copyright (c) 2007 Jared D. McNeill <jmcneill@invisible.ca>
6  * All rights reserved.
7  *
8  * Redistribution and use in source and binary forms, with or without
9  * modification, are permitted provided that the following conditions
10  * are met:
11  * 1. Redistributions of source code must retain the above copyright
12  *    notice, this list of conditions and the following disclaimer.
13  * 2. Redistributions in binary form must reproduce the above copyright
14  *    notice, this list of conditions and the following disclaimer in the
15  *    documentation and/or other materials provided with the distribution.
16  *
17  * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
18  * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
19  * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
20  * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
21  * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
22  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
23  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
24  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
25  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
26  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
27  * POSSIBILITY OF SUCH DAMAGE.
28  */
29 
30 /*
31  * Note that this machdep.c uses the `dummy' mcontext_t defined for usermode.
32  * This is basically a blob of PAGE_SIZE big. We might want to switch over to
33  * non-generic mcontext_t's one day, but will this break non-NetBSD hosts?
34  */
35 
36 
37 #include "opt_memsize.h"
38 
39 #include <sys/cdefs.h>
40 __KERNEL_RCSID(0, "$NetBSD: machdep.c,v 1.58 2022/01/01 21:07:14 andvar Exp $");
41 
42 #include <sys/types.h>
43 #include <sys/systm.h>
44 #include <sys/param.h>
45 #include <sys/time.h>
46 #include <sys/exec.h>
47 #include <sys/buf.h>
48 #include <sys/boot_flag.h>
49 #include <sys/ucontext.h>
50 #include <sys/utsname.h>
51 #include <machine/pcb.h>
52 #include <machine/psl.h>
53 
54 #include <uvm/uvm_extern.h>
55 #include <uvm/uvm_page.h>
56 
57 #include <dev/mm.h>
58 #include <machine/vmparam.h>
59 #include <machine/machdep.h>
60 #include <machine/mainbus.h>
61 #include <machine/thunk.h>
62 #include <machine/cpu.h>
63 #include <sys/kgdb.h>
64 
65 #include "opt_ddb.h"
66 #include "opt_kgdb.h"
67 
68 #ifndef MAX_DISK_IMAGES
69 #define MAX_DISK_IMAGES	4
70 #endif
71 
72 #ifndef MAX_VDEVS
73 #define MAX_VDEVS 4
74 #endif
75 
76 char machine[_SYS_NMLN] = "";
77 char machine_arch[_SYS_NMLN] = "";
78 char module_machine_usermode[_SYS_NMLN] = "";
79 
80 struct vm_map *phys_map = NULL;
81 
82 static char **saved_argv;
83 
84 char *usermode_disk_image_path[MAX_DISK_IMAGES];
85 int usermode_disk_image_path_count = 0;
86 
87 int   usermode_vdev_type[MAX_VDEVS];
88 char *usermode_vdev_path[MAX_VDEVS];
89 int usermode_vdev_count = 0;
90 
91 static char usermode_tap_devicebuf[PATH_MAX] = "";
92 char *usermode_tap_device = NULL;
93 char *usermode_tap_eaddr = NULL;
94 static char usermode_audio_devicebuf[PATH_MAX] = "";
95 char *usermode_audio_device = NULL;
96 char *usermode_root_device = NULL;
97 int usermode_vnc_width = 0;
98 int usermode_vnc_height = 0;
99 int usermode_vnc_port = -1;
100 
101 void	main(int argc, char *argv[]);
102 void	usermode_reboot(void);
103 
104 static void
usage(const char * pn)105 usage(const char *pn)
106 {
107 	thunk_printf("usage: %s [-acdqsvxz]"
108 	    " [net=<tapdev>,<eaddr>]"
109 	    " [audio=<audiodev>]"
110 	    " [disk=<diskimg> ...]"
111 	    " [root=<device>]"
112 	    " [vnc=<width>x<height>,<port>]"
113 	    " [vdev=atapi,device]\n",
114 	    pn);
115 	thunk_printf("       (ex. \"%s"
116 	    " net=tap0,00:00:be:ef:ca:fe"
117 	    " audio=audio0"
118 	    " disk=root.fs"
119 	    " root=ld0"
120 	    " vnc=640x480,5900"
121 	    " vdev=atapi,/dev/rcd0d\")\n", pn);
122 }
123 
124 
125 static int
vdev_type(const char * type)126 vdev_type(const char *type)
127 {
128 	if (strcasecmp(type, "atapi")==0)
129 		return THUNKBUS_TYPE_VATAPI;
130 #if 0
131 	if (strcasecmp(type, "scsi")==0)
132 		return THUNKBUS_TYPE_VSCSI;
133 #endif
134 	return -1;
135 }
136 
137 
138 void
main(int argc,char * argv[])139 main(int argc, char *argv[])
140 {
141 	extern void ttycons_consinit(void);
142 	extern void pmap_bootstrap(void);
143 	extern void kernmain(void);
144 	int type, i, j, r, tmpopt = 0;
145 
146 	saved_argv = argv;
147 
148 	/* Get machine and machine_arch from host */
149 	thunk_getmachine(machine, sizeof(machine),
150 	    machine_arch, sizeof(machine_arch));
151 	/* Override module_machine to be ${machine}usermode */
152 	snprintf(module_machine_usermode, sizeof(module_machine_usermode),
153 	    "%susermode", machine);
154 
155 	ttycons_consinit();
156 
157 	for (i = 1; i < argc; i++) {
158 		if (argv[i][0] != '-') {
159 			if (strncmp(argv[i], "net=", strlen("net=")) == 0) {
160 				char *tap = argv[i] + strlen("net=");
161 				char *mac = strchr(tap, ',');
162 				char *p = usermode_tap_devicebuf;
163 				if (mac == NULL) {
164 					thunk_printf("bad net= format\n");
165 					return;
166 				}
167 				memset(usermode_tap_devicebuf, 0,
168 				    sizeof(usermode_tap_devicebuf));
169 				if (*tap != '/') {
170 					memcpy(p, "/dev/", strlen("/dev/"));
171 					p += strlen("/dev/");
172 				}
173 				for (; *tap != ','; p++, tap++)
174 					*p = *tap;
175 				usermode_tap_device = usermode_tap_devicebuf;
176 				usermode_tap_eaddr = mac + 1;
177 			} else if (strncmp(argv[i], "audio=",
178 			    strlen("audio=")) == 0) {
179 				char *audio = argv[i] + strlen("audio=");
180 				if (*audio != '/')
181 					snprintf(usermode_audio_devicebuf,
182 					    sizeof(usermode_audio_devicebuf),
183 					    "/dev/%s", audio);
184 				else
185 					snprintf(usermode_audio_devicebuf,
186 					    sizeof(usermode_audio_devicebuf),
187 					    "%s", audio);
188 				usermode_audio_device =
189 				    usermode_audio_devicebuf;
190 			} else if (strncmp(argv[i], "vnc=",
191 			    strlen("vnc=")) == 0) {
192 				char *vnc = argv[i] + strlen("vnc=");
193 				char *w, *h, *p;
194 				w = vnc;
195 				h = strchr(w, 'x');
196 				if (h == NULL) {
197 					thunk_printf("bad vnc= format\n");
198 					return;
199 				}
200 				*h++ = '\0';
201 				p = strchr(h, ',');
202 				if (p == NULL) {
203 					thunk_printf("bad vnc= format\n");
204 					return;
205 				}
206 				*p++ = '\0';
207 				usermode_vnc_width = strtoul(w, NULL, 10);
208 				usermode_vnc_height = strtoul(h, NULL, 10);
209 				usermode_vnc_port = strtoul(p, NULL, 10);
210 			} else if (strncmp(argv[i], "disk=",
211 			    strlen("disk=")) == 0) {
212 				if (usermode_disk_image_path_count ==
213 				    MAX_DISK_IMAGES) {
214 					thunk_printf("too many disk images "
215 					    "(increase MAX_DISK_IMAGES)\n");
216 					usage(argv[0]);
217 					return;
218 				}
219 				usermode_disk_image_path[
220 				    usermode_disk_image_path_count++] =
221 				    argv[i] + strlen("disk=");
222 			} else if (strncmp(argv[i], "vdev=",
223 			    strlen("vdev=")) == 0) {
224 				char *vdev = argv[i] + strlen("vdev=");
225 				char *t, *p;
226 				if (usermode_disk_image_path_count ==
227 				    MAX_VDEVS) {
228 					thunk_printf("too many vdevs "
229 					    "(increase MAX_VDEVS)\n");
230 					usage(argv[0]);
231 					return;
232 				}
233 				t = vdev;
234 				p = strchr(t, ',');
235 				if (p == NULL) {
236 					thunk_printf("bad vdev= format\n");
237 					return;
238 				}
239 				*p++ = '\0';
240 				type = vdev_type(t);
241 				if (type < 0) {
242 					thunk_printf("unknown vdev device type\n");
243 					return;
244 				}
245 				usermode_vdev_type[usermode_vdev_count] = type;
246 				usermode_vdev_path[usermode_vdev_count] = p;
247 				usermode_vdev_count++;
248 			} else if (strncmp(argv[i], "root=",
249 			    strlen("root=")) == 0) {
250 				usermode_root_device = argv[i] +
251 				    strlen("root=");
252 			} else {
253 				thunk_printf("%s: unknown parameter\n", argv[i]);
254 				usage(argv[0]);
255 				return;
256 			}
257 			continue;
258 		}
259 		for (j = 1; argv[i][j] != '\0'; j++) {
260 			r = 0;
261 			BOOT_FLAG(argv[i][j], r);
262 			if (r == 0) {
263 				thunk_printf("unknown kernel boot flag '%c'\n", argv[i][j]);
264 				usage(argv[0]);
265 				return;
266 			}
267 			tmpopt |= r;
268 		}
269 	}
270 	boothowto = tmpopt;
271 
272 	uvm_md_init();
273 	uvmexp.ncolors = 2;
274 
275 	pmap_bootstrap();
276 
277 	splinit();
278 	splraise(IPL_HIGH);
279 
280 #ifdef DDB
281 	if (boothowto & RB_KDB)
282 		Debugger();
283 #endif
284 #ifdef KGDB
285 	if (boothowto & RB_KDB) {
286 		kgdb_port_init();
287 		kgdb_debug_init = 1;
288 		kgdb_connect(1);
289 	}
290 #endif
291 
292 	kernmain();
293 }
294 
295 void
usermode_reboot(void)296 usermode_reboot(void)
297 {
298 	struct thunk_itimerval itimer;
299 
300 	/* make sure the timer is turned off */
301 	memset(&itimer, 0, sizeof(itimer));
302 	thunk_setitimer(ITIMER_REAL, &itimer, NULL);
303 
304 	if (thunk_execv(saved_argv[0], saved_argv) == -1)
305 		thunk_abort();
306 	/* NOTREACHED */
307 }
308 
309 void
setstatclockrate(int arg)310 setstatclockrate(int arg)
311 {
312 }
313 
314 void
consinit(void)315 consinit(void)
316 {
317 //	kgdb_connect(0);
318 	printf("NetBSD/usermode startup\n");
319 }
320 
321 int
mm_md_physacc(paddr_t pa,vm_prot_t prot)322 mm_md_physacc(paddr_t pa, vm_prot_t prot)
323 {
324 	// printf("%s: pa = %p, acc %d\n", __func__, (void *) pa, prot);
325 	if (pa >= physmem * PAGE_SIZE)
326 		return EFAULT;
327 	return 0;
328 }
329 
330 
331 int
mm_md_kernacc(void * ptr,vm_prot_t prot,bool * handled)332 mm_md_kernacc(void *ptr, vm_prot_t prot, bool *handled)
333 {
334 	const vaddr_t va = (vaddr_t)ptr;
335 	extern void *end;
336 
337 	// printf("%s: ptr %p, acc %d\n", __func__, ptr, prot);
338 	if (va < kmem_kvm_start)
339 		return EFAULT;
340 	if ((va >= kmem_kvm_cur_end) && (va < kmem_k_start))
341 		return EFAULT;
342 	if (va > (vaddr_t) end)
343 		return EFAULT;
344 
345 	*handled = true;
346 	return 0;
347 }
348 
349