Annotation of src/sys/compat/linux/arch/i386/linux_machdep.c, Revision 1.99
1.99 ! christos 1: /* $NetBSD: linux_machdep.c,v 1.98 2003/09/21 19:29:10 jdolecek Exp $ */
1.1 fvdl 2:
1.45 fvdl 3: /*-
1.52 fvdl 4: * Copyright (c) 1995, 2000 The NetBSD Foundation, Inc.
1.1 fvdl 5: * All rights reserved.
6: *
1.45 fvdl 7: * This code is derived from software contributed to The NetBSD Foundation
8: * by Frank van der Linden.
9: *
1.1 fvdl 10: * Redistribution and use in source and binary forms, with or without
11: * modification, are permitted provided that the following conditions
12: * are met:
13: * 1. Redistributions of source code must retain the above copyright
14: * notice, this list of conditions and the following disclaimer.
15: * 2. Redistributions in binary form must reproduce the above copyright
16: * notice, this list of conditions and the following disclaimer in the
17: * documentation and/or other materials provided with the distribution.
18: * 3. All advertising materials mentioning features or use of this software
19: * must display the following acknowledgement:
1.45 fvdl 20: * This product includes software developed by the NetBSD
21: * Foundation, Inc. and its contributors.
22: * 4. Neither the name of The NetBSD Foundation nor the names of its
23: * contributors may be used to endorse or promote products derived
24: * from this software without specific prior written permission.
1.1 fvdl 25: *
1.45 fvdl 26: * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
27: * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
28: * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
29: * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
30: * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
31: * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
32: * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
33: * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
34: * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
35: * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
36: * POSSIBILITY OF SUCH DAMAGE.
1.1 fvdl 37: */
1.68 lukem 38:
39: #include <sys/cdefs.h>
1.99 ! christos 40: __KERNEL_RCSID(0, "$NetBSD: linux_machdep.c,v 1.98 2003/09/21 19:29:10 jdolecek Exp $");
1.35 thorpej 41:
1.63 mrg 42: #if defined(_KERNEL_OPT)
1.35 thorpej 43: #include "opt_vm86.h"
1.36 thorpej 44: #include "opt_user_ldt.h"
1.51 jdolecek 45: #endif
1.1 fvdl 46:
47: #include <sys/param.h>
48: #include <sys/systm.h>
49: #include <sys/signalvar.h>
50: #include <sys/kernel.h>
51: #include <sys/proc.h>
52: #include <sys/user.h>
53: #include <sys/buf.h>
54: #include <sys/reboot.h>
55: #include <sys/conf.h>
1.31 thorpej 56: #include <sys/exec.h>
1.1 fvdl 57: #include <sys/file.h>
58: #include <sys/callout.h>
59: #include <sys/malloc.h>
60: #include <sys/mbuf.h>
61: #include <sys/msgbuf.h>
62: #include <sys/mount.h>
63: #include <sys/vnode.h>
64: #include <sys/device.h>
1.85 thorpej 65: #include <sys/sa.h>
1.1 fvdl 66: #include <sys/syscallargs.h>
1.13 fvdl 67: #include <sys/filedesc.h>
1.39 mycroft 68: #include <sys/exec_elf.h>
1.52 fvdl 69: #include <sys/disklabel.h>
1.62 manu 70: #include <sys/ioctl.h>
1.52 fvdl 71: #include <miscfs/specfs/specdev.h>
1.7 mycroft 72:
1.44 christos 73: #include <compat/linux/common/linux_types.h>
74: #include <compat/linux/common/linux_signal.h>
75: #include <compat/linux/common/linux_util.h>
76: #include <compat/linux/common/linux_ioctl.h>
1.52 fvdl 77: #include <compat/linux/common/linux_hdio.h>
1.44 christos 78: #include <compat/linux/common/linux_exec.h>
79: #include <compat/linux/common/linux_machdep.h>
1.98 jdolecek 80: #include <compat/linux/common/linux_errno.h>
1.44 christos 81:
1.1 fvdl 82: #include <compat/linux/linux_syscallargs.h>
83:
84: #include <machine/cpu.h>
85: #include <machine/cpufunc.h>
86: #include <machine/psl.h>
87: #include <machine/reg.h>
1.7 mycroft 88: #include <machine/segments.h>
1.1 fvdl 89: #include <machine/specialreg.h>
1.7 mycroft 90: #include <machine/sysarch.h>
1.26 mycroft 91: #include <machine/vm86.h>
1.34 mycroft 92: #include <machine/vmparam.h>
1.1 fvdl 93:
94: /*
1.50 veego 95: * To see whether wscons is configured (for virtual console ioctl calls).
1.13 fvdl 96: */
1.63 mrg 97: #if defined(_KERNEL_OPT)
1.41 drochner 98: #include "wsdisplay.h"
1.51 jdolecek 99: #endif
1.41 drochner 100: #if (NWSDISPLAY > 0)
1.52 fvdl 101: #include <dev/wscons/wsconsio.h>
1.41 drochner 102: #include <dev/wscons/wsdisplay_usl_io.h>
1.63 mrg 103: #if defined(_KERNEL_OPT)
1.41 drochner 104: #include "opt_xserver.h"
1.51 jdolecek 105: #endif
1.41 drochner 106: #endif
107:
1.29 christos 108: #ifdef USER_LDT
109: #include <machine/cpu.h>
1.85 thorpej 110: int linux_read_ldt __P((struct lwp *, struct linux_sys_modify_ldt_args *,
1.29 christos 111: register_t *));
1.85 thorpej 112: int linux_write_ldt __P((struct lwp *, struct linux_sys_modify_ldt_args *,
1.29 christos 113: register_t *));
114: #endif
115:
1.72 christos 116: #ifdef DEBUG_LINUX
117: #define DPRINTF(a) uprintf a
118: #else
119: #define DPRINTF(a)
120: #endif
121:
1.52 fvdl 122: static struct biosdisk_info *fd2biosinfo __P((struct proc *, struct file *));
123: extern struct disklist *i386_alldisks;
1.90 christos 124: static void linux_save_ucontext __P((struct lwp *, struct trapframe *,
1.99 ! christos 125: const sigset_t *, struct sigaltstack *, struct linux_ucontext *));
1.90 christos 126: static void linux_save_sigcontext __P((struct lwp *, struct trapframe *,
1.99 ! christos 127: const sigset_t *, struct linux_sigcontext *));
1.90 christos 128: static int linux_restore_sigcontext __P((struct lwp *,
129: struct linux_sigcontext *, register_t *));
1.99 ! christos 130: static void linux_rt_sendsig __P((const ksiginfo_t *, const sigset_t *));
! 131: static void linux_old_sendsig __P((const ksiginfo_t *, const sigset_t *));
1.52 fvdl 132:
1.83 christos 133: extern char linux_sigcode[], linux_rt_sigcode[];
1.13 fvdl 134: /*
1.1 fvdl 135: * Deal with some i386-specific things in the Linux emulation code.
136: */
1.37 mycroft 137:
138: void
1.85 thorpej 139: linux_setregs(l, epp, stack)
140: struct lwp *l;
1.37 mycroft 141: struct exec_package *epp;
142: u_long stack;
143: {
1.85 thorpej 144: struct pcb *pcb = &l->l_addr->u_pcb;
1.73 christos 145: struct trapframe *tf;
146:
147: #if NNPX > 0
148: /* If we were using the FPU, forget about it. */
1.85 thorpej 149: if (npxproc == l)
1.73 christos 150: npxdrop();
151: #endif
152:
153: #ifdef USER_LDT
1.85 thorpej 154: pmap_ldt_cleanup(l);
1.73 christos 155: #endif
1.37 mycroft 156:
1.85 thorpej 157: l->l_md.md_flags &= ~MDP_USEDFPU;
1.73 christos 158:
159: if (i386_use_fxsave) {
1.67 thorpej 160: pcb->pcb_savefpu.sv_xmm.sv_env.en_cw = __Linux_NPXCW__;
1.73 christos 161: pcb->pcb_savefpu.sv_xmm.sv_env.en_mxcsr = __INITIAL_MXCSR__;
162: } else
1.67 thorpej 163: pcb->pcb_savefpu.sv_87.sv_env.en_cw = __Linux_NPXCW__;
1.73 christos 164:
1.85 thorpej 165: tf = l->l_md.md_regs;
1.73 christos 166: tf->tf_gs = GSEL(GUDATA_SEL, SEL_UPL);
167: tf->tf_fs = GSEL(GUDATA_SEL, SEL_UPL);
168: tf->tf_es = GSEL(GUDATA_SEL, SEL_UPL);
169: tf->tf_ds = GSEL(GUDATA_SEL, SEL_UPL);
170: tf->tf_edi = 0;
171: tf->tf_esi = 0;
172: tf->tf_ebp = 0;
1.85 thorpej 173: tf->tf_ebx = (int)l->l_proc->p_psstr;
1.73 christos 174: tf->tf_edx = 0;
175: tf->tf_ecx = 0;
176: tf->tf_eax = 0;
177: tf->tf_eip = epp->ep_entry;
1.95 chs 178: tf->tf_cs = GSEL(GUCODEBIG_SEL, SEL_UPL);
1.73 christos 179: tf->tf_eflags = PSL_USERSET;
180: tf->tf_esp = stack;
181: tf->tf_ss = GSEL(GUDATA_SEL, SEL_UPL);
1.37 mycroft 182: }
1.1 fvdl 183:
184: /*
185: * Send an interrupt to process.
186: *
187: * Stack is set up to allow sigcode stored
188: * in u. to call routine, followed by kcall
189: * to sigreturn routine below. After sigreturn
190: * resets the signal mask, the stack, and the
191: * frame pointer, it returns to the user
192: * specified pc, psl.
193: */
194:
195: void
1.99 ! christos 196: linux_sendsig(const ksiginfo_t *ksi, const sigset_t *mask)
1.1 fvdl 197: {
1.96 christos 198: if (SIGACTION(curproc, ksi->ksi_signo).sa_flags & SA_SIGINFO)
1.97 christos 199: linux_rt_sendsig(ksi, mask);
1.83 christos 200: else
1.97 christos 201: linux_old_sendsig(ksi, mask);
1.83 christos 202: }
203:
204:
205: static void
1.90 christos 206: linux_save_ucontext(l, tf, mask, sas, uc)
207: struct lwp *l;
208: struct trapframe *tf;
1.99 ! christos 209: const sigset_t *mask;
1.90 christos 210: struct sigaltstack *sas;
211: struct linux_ucontext *uc;
212: {
213: uc->uc_flags = 0;
214: uc->uc_link = NULL;
215: native_to_linux_sigaltstack(&uc->uc_stack, sas);
216: linux_save_sigcontext(l, tf, mask, &uc->uc_mcontext);
217: native_to_linux_sigset(&uc->uc_sigmask, mask);
218: (void)memset(&uc->uc_fpregs_mem, 0, sizeof(uc->uc_fpregs_mem));
219: }
220:
221: static void
222: linux_save_sigcontext(l, tf, mask, sc)
1.85 thorpej 223: struct lwp *l;
1.83 christos 224: struct trapframe *tf;
1.99 ! christos 225: const sigset_t *mask;
1.83 christos 226: struct linux_sigcontext *sc;
227: {
228: /* Save register context. */
229: #ifdef VM86
230: if (tf->tf_eflags & PSL_VM) {
231: sc->sc_gs = tf->tf_vm86_gs;
232: sc->sc_fs = tf->tf_vm86_fs;
233: sc->sc_es = tf->tf_vm86_es;
234: sc->sc_ds = tf->tf_vm86_ds;
1.85 thorpej 235: sc->sc_eflags = get_vflags(l);
1.83 christos 236: } else
237: #endif
238: {
239: sc->sc_gs = tf->tf_gs;
240: sc->sc_fs = tf->tf_fs;
241: sc->sc_es = tf->tf_es;
242: sc->sc_ds = tf->tf_ds;
243: sc->sc_eflags = tf->tf_eflags;
244: }
245: sc->sc_edi = tf->tf_edi;
246: sc->sc_esi = tf->tf_esi;
247: sc->sc_esp = tf->tf_esp;
248: sc->sc_ebp = tf->tf_ebp;
249: sc->sc_ebx = tf->tf_ebx;
250: sc->sc_edx = tf->tf_edx;
251: sc->sc_ecx = tf->tf_ecx;
252: sc->sc_eax = tf->tf_eax;
253: sc->sc_eip = tf->tf_eip;
254: sc->sc_cs = tf->tf_cs;
255: sc->sc_esp_at_signal = tf->tf_esp;
256: sc->sc_ss = tf->tf_ss;
257: sc->sc_err = tf->tf_err;
258: sc->sc_trapno = tf->tf_trapno;
1.85 thorpej 259: sc->sc_cr2 = l->l_addr->u_pcb.pcb_cr2;
1.83 christos 260: sc->sc_387 = NULL;
261:
262: /* Save signal stack. */
263: /* Linux doesn't save the onstack flag in sigframe */
264:
265: /* Save signal mask. */
266: native_to_linux_old_sigset(&sc->sc_mask, mask);
267: }
268:
269: static void
1.99 ! christos 270: linux_rt_sendsig(const ksiginfo_t *ksi, const sigset_t *mask)
1.83 christos 271: {
1.85 thorpej 272: struct lwp *l = curlwp;
273: struct proc *p = l->l_proc;
1.48 augustss 274: struct trapframe *tf;
1.83 christos 275: struct linux_rt_sigframe *fp, frame;
1.66 jdolecek 276: int onstack;
1.97 christos 277: linux_siginfo_t *lsi;
1.98 jdolecek 278: int sig = ksi->ksi_signo;
1.78 thorpej 279: sig_t catcher = SIGACTION(p, sig).sa_handler;
1.82 christos 280: struct sigaltstack *sas = &p->p_sigctx.ps_sigstk;
1.1 fvdl 281:
1.85 thorpej 282: tf = l->l_md.md_regs;
1.66 jdolecek 283: /* Do we need to jump onto the signal stack? */
1.82 christos 284: onstack = (sas->ss_flags & (SS_DISABLE | SS_ONSTACK)) == 0 &&
1.66 jdolecek 285: (SIGACTION(p, sig).sa_flags & SA_ONSTACK) != 0;
286:
1.83 christos 287:
1.42 mycroft 288: /* Allocate space for the signal handler context. */
1.66 jdolecek 289: if (onstack)
1.83 christos 290: fp = (struct linux_rt_sigframe *)((caddr_t)sas->ss_sp +
291: sas->ss_size);
1.66 jdolecek 292: else
1.83 christos 293: fp = (struct linux_rt_sigframe *)tf->tf_esp;
1.42 mycroft 294: fp--;
1.1 fvdl 295:
1.92 christos 296: DPRINTF(("rt: onstack = %d, fp = %p sig = %d eip = 0x%x cr2 = 0x%x\n",
297: onstack, fp, sig, tf->tf_eip, l->l_addr->u_pcb.pcb_cr2));
1.83 christos 298:
1.42 mycroft 299: /* Build stack frame for signal trampoline. */
1.7 mycroft 300: frame.sf_handler = catcher;
1.74 christos 301: frame.sf_sig = native_to_linux_signo[sig];
1.82 christos 302: frame.sf_sip = &fp->sf_si;
1.90 christos 303: frame.sf_ucp = &fp->sf_uc;
1.83 christos 304:
1.82 christos 305: /*
1.97 christos 306: * XXX: the following code assumes that the constants for
307: * siginfo are the same between linux and NetBSD.
1.82 christos 308: */
1.97 christos 309: (void)memset(lsi = &frame.sf_si, 0, sizeof(frame.sf_si));
310: lsi->lsi_errno = native_to_linux_errno[ksi->ksi_errno];
311: lsi->lsi_code = ksi->ksi_code;
312: switch (lsi->lsi_signo = frame.sf_sig) {
313: case LINUX_SIGILL:
314: case LINUX_SIGFPE:
1.90 christos 315: case LINUX_SIGSEGV:
316: case LINUX_SIGBUS:
317: case LINUX_SIGTRAP:
1.97 christos 318: lsi->lsi_addr = ksi->ksi_addr;
1.90 christos 319: break;
320: case LINUX_SIGCHLD:
1.97 christos 321: lsi->lsi_uid = ksi->ksi_uid;
322: lsi->lsi_pid = ksi->ksi_pid;
323: lsi->lsi_status = ksi->ksi_status;
324: lsi->lsi_utime = ksi->ksi_utime;
325: lsi->lsi_stime = ksi->ksi_stime;
326: break;
1.90 christos 327: case LINUX_SIGIO:
1.97 christos 328: lsi->lsi_band = ksi->ksi_band;
329: lsi->lsi_fd = ksi->ksi_fd;
330: break;
1.90 christos 331: default:
1.97 christos 332: lsi->lsi_uid = ksi->ksi_uid;
333: lsi->lsi_pid = ksi->ksi_pid;
334: if (lsi->lsi_signo == LINUX_SIGALRM ||
335: lsi->lsi_signo >= LINUX_SIGRTMIN)
1.98 jdolecek 336: lsi->lsi_value.sival_ptr = ksi->ksi_sigval.sival_ptr;
1.90 christos 337: break;
338: }
1.1 fvdl 339:
1.42 mycroft 340: /* Save register context. */
1.90 christos 341: linux_save_ucontext(l, tf, mask, sas, &frame.sf_uc);
1.83 christos 342:
343: if (copyout(&frame, fp, sizeof(frame)) != 0) {
344: /*
345: * Process has trashed its stack; give it an illegal
346: * instruction to halt it in its tracks.
347: */
1.85 thorpej 348: sigexit(l, SIGILL);
1.83 christos 349: /* NOTREACHED */
1.4 mycroft 350: }
1.1 fvdl 351:
1.83 christos 352: /*
353: * Build context to run handler in.
354: */
355: tf->tf_gs = GSEL(GUDATA_SEL, SEL_UPL);
356: tf->tf_fs = GSEL(GUDATA_SEL, SEL_UPL);
357: tf->tf_es = GSEL(GUDATA_SEL, SEL_UPL);
358: tf->tf_ds = GSEL(GUDATA_SEL, SEL_UPL);
359: tf->tf_eip = ((int)p->p_sigctx.ps_sigcode) +
360: (linux_rt_sigcode - linux_sigcode);
361: tf->tf_cs = GSEL(GUCODE_SEL, SEL_UPL);
362: tf->tf_eflags &= ~(PSL_T|PSL_VM|PSL_AC);
363: tf->tf_esp = (int)fp;
364: tf->tf_ss = GSEL(GUDATA_SEL, SEL_UPL);
365:
366: /* Remember that we're now on the signal stack. */
367: if (onstack)
368: sas->ss_flags |= SS_ONSTACK;
369: }
370:
371: static void
1.99 ! christos 372: linux_old_sendsig(const ksiginfo_t *ksi, const sigset_t *mask)
1.83 christos 373: {
1.85 thorpej 374: struct lwp *l = curlwp;
375: struct proc *p = l->l_proc;
1.83 christos 376: struct trapframe *tf;
377: struct linux_sigframe *fp, frame;
378: int onstack;
1.97 christos 379: int sig = ksi->ksi_signo;
1.83 christos 380: sig_t catcher = SIGACTION(p, sig).sa_handler;
381: struct sigaltstack *sas = &p->p_sigctx.ps_sigstk;
382:
1.85 thorpej 383: tf = l->l_md.md_regs;
1.83 christos 384:
385: /* Do we need to jump onto the signal stack? */
386: onstack = (sas->ss_flags & (SS_DISABLE | SS_ONSTACK)) == 0 &&
387: (SIGACTION(p, sig).sa_flags & SA_ONSTACK) != 0;
388:
389: /* Allocate space for the signal handler context. */
390: if (onstack)
391: fp = (struct linux_sigframe *) ((caddr_t)sas->ss_sp +
392: sas->ss_size);
393: else
394: fp = (struct linux_sigframe *)tf->tf_esp;
395: fp--;
396:
1.92 christos 397: DPRINTF(("old: onstack = %d, fp = %p sig = %d eip = 0x%x cr2 = 0x%x\n",
398: onstack, fp, sig, tf->tf_eip, l->l_addr->u_pcb.pcb_cr2));
1.83 christos 399:
400: /* Build stack frame for signal trampoline. */
401: frame.sf_handler = catcher;
402: frame.sf_sig = native_to_linux_signo[sig];
1.42 mycroft 403:
1.99 ! christos 404: /*###404 [cc] warning: passing arg 3 of `linux_save_sigcontext' discards qualifiers from pointer target type%%%*/
1.90 christos 405: linux_save_sigcontext(l, tf, mask, &frame.sf_sc);
1.42 mycroft 406:
1.1 fvdl 407: if (copyout(&frame, fp, sizeof(frame)) != 0) {
408: /*
409: * Process has trashed its stack; give it an illegal
410: * instruction to halt it in its tracks.
411: */
1.85 thorpej 412: sigexit(l, SIGILL);
1.1 fvdl 413: /* NOTREACHED */
414: }
415:
416: /*
417: * Build context to run handler in.
418: */
1.75 christos 419: tf->tf_gs = GSEL(GUDATA_SEL, SEL_UPL);
420: tf->tf_fs = GSEL(GUDATA_SEL, SEL_UPL);
1.26 mycroft 421: tf->tf_es = GSEL(GUDATA_SEL, SEL_UPL);
422: tf->tf_ds = GSEL(GUDATA_SEL, SEL_UPL);
1.56 jdolecek 423: tf->tf_eip = (int)p->p_sigctx.ps_sigcode;
1.95 chs 424: tf->tf_cs = GSEL(GUCODEBIG_SEL, SEL_UPL);
1.28 mycroft 425: tf->tf_eflags &= ~(PSL_T|PSL_VM|PSL_AC);
1.26 mycroft 426: tf->tf_esp = (int)fp;
1.23 mycroft 427: tf->tf_ss = GSEL(GUDATA_SEL, SEL_UPL);
1.42 mycroft 428:
429: /* Remember that we're now on the signal stack. */
1.66 jdolecek 430: if (onstack)
1.82 christos 431: sas->ss_flags |= SS_ONSTACK;
1.1 fvdl 432: }
433:
434: /*
435: * System call to cleanup state after a signal
436: * has been taken. Reset signal mask and
437: * stack state from context left by sendsig (above).
438: * Return to previous pc and psl as specified by
439: * context left by sendsig. Check carefully to
440: * make sure that the user has not modified the
441: * psl to gain improper privileges or to cause
442: * a machine fault.
443: */
1.43 erh 444: int
1.85 thorpej 445: linux_sys_rt_sigreturn(l, v, retval)
446: struct lwp *l;
1.43 erh 447: void *v;
448: register_t *retval;
449: {
1.90 christos 450: struct linux_sys_rt_sigreturn_args /* {
451: syscallarg(struct linux_ucontext *) ucp;
452: } */ *uap = v;
453: struct linux_ucontext context, *ucp = SCARG(uap, ucp);
454: int error;
455:
456: /*
457: * The trampoline code hands us the context.
458: * It is unsafe to keep track of it ourselves, in the event that a
459: * program jumps out of a signal handler.
460: */
461: if ((error = copyin(ucp, &context, sizeof(*ucp))) != 0)
462: return error;
463:
464: /* XXX XAX we can do better here by using more of the ucontext */
465: return linux_restore_sigcontext(l, &context.uc_mcontext, retval);
1.43 erh 466: }
467:
1.1 fvdl 468: int
1.85 thorpej 469: linux_sys_sigreturn(l, v, retval)
470: struct lwp *l;
1.19 thorpej 471: void *v;
472: register_t *retval;
473: {
1.20 mycroft 474: struct linux_sys_sigreturn_args /* {
1.1 fvdl 475: syscallarg(struct linux_sigcontext *) scp;
1.19 thorpej 476: } */ *uap = v;
1.90 christos 477: struct linux_sigcontext context, *scp = SCARG(uap, scp);
478: int error;
1.1 fvdl 479:
480: /*
481: * The trampoline code hands us the context.
482: * It is unsafe to keep track of it ourselves, in the event that a
483: * program jumps out of a signal handler.
484: */
1.90 christos 485: if ((error = copyin((caddr_t)scp, &context, sizeof(*scp))) != 0)
486: return error;
487: return linux_restore_sigcontext(l, &context, retval);
488: }
1.1 fvdl 489:
1.90 christos 490: static int
491: linux_restore_sigcontext(l, scp, retval)
492: struct lwp *l;
493: struct linux_sigcontext *scp;
494: register_t *retval;
495: {
496: struct proc *p = l->l_proc;
497: struct sigaltstack *sas = &p->p_sigctx.ps_sigstk;
498: struct trapframe *tf;
499: sigset_t mask;
500: ssize_t ss_gap;
1.42 mycroft 501: /* Restore register context. */
1.85 thorpej 502: tf = l->l_md.md_regs;
1.83 christos 503:
504: DPRINTF(("sigreturn enter esp=%x eip=%x\n", tf->tf_esp, tf->tf_eip));
1.4 mycroft 505: #ifdef VM86
1.90 christos 506: if (scp->sc_eflags & PSL_VM) {
1.94 hannken 507: void syscall_vm86 __P((struct trapframe *));
1.83 christos 508:
1.90 christos 509: tf->tf_vm86_gs = scp->sc_gs;
510: tf->tf_vm86_fs = scp->sc_fs;
511: tf->tf_vm86_es = scp->sc_es;
512: tf->tf_vm86_ds = scp->sc_ds;
513: set_vflags(l, scp->sc_eflags);
1.83 christos 514: p->p_md.md_syscall = syscall_vm86;
1.4 mycroft 515: } else
516: #endif
517: {
1.26 mycroft 518: /*
519: * Check for security violations. If we're returning to
520: * protected mode, the CPU will validate the segment registers
521: * automatically and generate a trap on violations. We handle
522: * the trap, rather than doing all of the checking here.
523: */
1.90 christos 524: if (((scp->sc_eflags ^ tf->tf_eflags) & PSL_USERSTATIC) != 0 ||
525: !USERMODE(scp->sc_cs, scp->sc_eflags))
1.83 christos 526: return EINVAL;
1.26 mycroft 527:
1.90 christos 528: tf->tf_gs = scp->sc_gs;
529: tf->tf_fs = scp->sc_fs;
530: tf->tf_es = scp->sc_es;
531: tf->tf_ds = scp->sc_ds;
1.83 christos 532: #ifdef VM86
533: if (tf->tf_eflags & PSL_VM)
534: (*p->p_emul->e_syscall_intern)(p);
535: #endif
1.90 christos 536: tf->tf_eflags = scp->sc_eflags;
1.4 mycroft 537: }
1.90 christos 538: tf->tf_edi = scp->sc_edi;
539: tf->tf_esi = scp->sc_esi;
540: tf->tf_ebp = scp->sc_ebp;
541: tf->tf_ebx = scp->sc_ebx;
542: tf->tf_edx = scp->sc_edx;
543: tf->tf_ecx = scp->sc_ecx;
544: tf->tf_eax = scp->sc_eax;
545: tf->tf_eip = scp->sc_eip;
546: tf->tf_cs = scp->sc_cs;
547: tf->tf_esp = scp->sc_esp_at_signal;
548: tf->tf_ss = scp->sc_ss;
1.26 mycroft 549:
1.42 mycroft 550: /* Restore signal stack. */
1.66 jdolecek 551: /*
552: * Linux really does it this way; it doesn't have space in sigframe
553: * to save the onstack flag.
554: */
555: ss_gap = (ssize_t)
1.90 christos 556: ((caddr_t) scp->sc_esp_at_signal - (caddr_t) sas->ss_sp);
1.82 christos 557: if (ss_gap >= 0 && ss_gap < sas->ss_size)
558: sas->ss_flags |= SS_ONSTACK;
1.66 jdolecek 559: else
1.82 christos 560: sas->ss_flags &= ~SS_ONSTACK;
1.42 mycroft 561:
562: /* Restore signal mask. */
1.90 christos 563: linux_old_to_native_sigset(&mask, &scp->sc_mask);
1.42 mycroft 564: (void) sigprocmask1(p, SIG_SETMASK, &mask, 0);
1.83 christos 565: DPRINTF(("sigreturn exit esp=%x eip=%x\n", tf->tf_esp, tf->tf_eip));
566: return EJUSTRETURN;
1.6 mycroft 567: }
568:
1.7 mycroft 569: #ifdef USER_LDT
570:
571: int
1.85 thorpej 572: linux_read_ldt(l, uap, retval)
573: struct lwp *l;
1.20 mycroft 574: struct linux_sys_modify_ldt_args /* {
1.7 mycroft 575: syscallarg(int) func;
576: syscallarg(void *) ptr;
577: syscallarg(size_t) bytecount;
578: } */ *uap;
579: register_t *retval;
580: {
1.85 thorpej 581: struct proc *p = l->l_proc;
1.7 mycroft 582: struct i386_get_ldt_args gl;
583: int error;
584: caddr_t sg;
585: char *parms;
586:
1.72 christos 587: DPRINTF(("linux_read_ldt!"));
1.71 christos 588: sg = stackgap_init(p, 0);
1.7 mycroft 589:
590: gl.start = 0;
591: gl.desc = SCARG(uap, ptr);
592: gl.num = SCARG(uap, bytecount) / sizeof(union descriptor);
593:
1.71 christos 594: parms = stackgap_alloc(p, &sg, sizeof(gl));
1.7 mycroft 595:
1.29 christos 596: if ((error = copyout(&gl, parms, sizeof(gl))) != 0)
1.7 mycroft 597: return (error);
598:
1.85 thorpej 599: if ((error = i386_get_ldt(l, parms, retval)) != 0)
1.7 mycroft 600: return (error);
601:
602: *retval *= sizeof(union descriptor);
603: return (0);
604: }
605:
606: struct linux_ldt_info {
607: u_int entry_number;
608: u_long base_addr;
609: u_int limit;
610: u_int seg_32bit:1;
611: u_int contents:2;
612: u_int read_exec_only:1;
613: u_int limit_in_pages:1;
614: u_int seg_not_present:1;
1.72 christos 615: u_int useable:1;
1.7 mycroft 616: };
617:
618: int
1.85 thorpej 619: linux_write_ldt(l, uap, retval)
620: struct lwp *l;
1.20 mycroft 621: struct linux_sys_modify_ldt_args /* {
1.7 mycroft 622: syscallarg(int) func;
623: syscallarg(void *) ptr;
624: syscallarg(size_t) bytecount;
625: } */ *uap;
626: register_t *retval;
627: {
1.85 thorpej 628: struct proc *p = l->l_proc;
1.7 mycroft 629: struct linux_ldt_info ldt_info;
630: struct segment_descriptor sd;
631: struct i386_set_ldt_args sl;
632: int error;
633: caddr_t sg;
634: char *parms;
1.72 christos 635: int oldmode = (int)retval[0];
1.7 mycroft 636:
1.72 christos 637: DPRINTF(("linux_write_ldt %d\n", oldmode));
1.7 mycroft 638: if (SCARG(uap, bytecount) != sizeof(ldt_info))
639: return (EINVAL);
1.29 christos 640: if ((error = copyin(SCARG(uap, ptr), &ldt_info, sizeof(ldt_info))) != 0)
1.7 mycroft 641: return error;
1.72 christos 642: if (ldt_info.entry_number >= 8192)
1.7 mycroft 643: return (EINVAL);
1.72 christos 644: if (ldt_info.contents == 3) {
645: if (oldmode)
646: return (EINVAL);
647: if (ldt_info.seg_not_present)
648: return (EINVAL);
649: }
1.7 mycroft 650:
1.72 christos 651: if (ldt_info.base_addr == 0 && ldt_info.limit == 0 &&
652: (oldmode || (ldt_info.contents == 0 &&
653: ldt_info.read_exec_only == 1 && ldt_info.seg_32bit == 0 &&
654: ldt_info.limit_in_pages == 0 && ldt_info.seg_not_present == 1 &&
655: ldt_info.useable == 0))) {
1.70 christos 656: /* this means you should zero the ldt */
657: (void)memset(&sd, 0, sizeof(sd));
658: } else {
659: sd.sd_lobase = ldt_info.base_addr & 0xffffff;
660: sd.sd_hibase = (ldt_info.base_addr >> 24) & 0xff;
661: sd.sd_lolimit = ldt_info.limit & 0xffff;
662: sd.sd_hilimit = (ldt_info.limit >> 16) & 0xf;
663: sd.sd_type = 16 | (ldt_info.contents << 2) |
664: (!ldt_info.read_exec_only << 1);
665: sd.sd_dpl = SEL_UPL;
666: sd.sd_p = !ldt_info.seg_not_present;
667: sd.sd_def32 = ldt_info.seg_32bit;
668: sd.sd_gran = ldt_info.limit_in_pages;
1.72 christos 669: if (!oldmode)
670: sd.sd_xx = ldt_info.useable;
1.73 christos 671: else
672: sd.sd_xx = 0;
1.70 christos 673: }
1.71 christos 674: sg = stackgap_init(p, 0);
1.7 mycroft 675: sl.start = ldt_info.entry_number;
1.71 christos 676: sl.desc = stackgap_alloc(p, &sg, sizeof(sd));
1.7 mycroft 677: sl.num = 1;
678:
1.72 christos 679: DPRINTF(("linux_write_ldt: idx=%d, base=0x%lx, limit=0x%x\n",
680: ldt_info.entry_number, ldt_info.base_addr, ldt_info.limit));
1.7 mycroft 681:
1.71 christos 682: parms = stackgap_alloc(p, &sg, sizeof(sl));
1.7 mycroft 683:
1.29 christos 684: if ((error = copyout(&sd, sl.desc, sizeof(sd))) != 0)
1.7 mycroft 685: return (error);
1.29 christos 686: if ((error = copyout(&sl, parms, sizeof(sl))) != 0)
1.7 mycroft 687: return (error);
688:
1.85 thorpej 689: if ((error = i386_set_ldt(l, parms, retval)) != 0)
1.7 mycroft 690: return (error);
691:
692: *retval = 0;
693: return (0);
694: }
695:
696: #endif /* USER_LDT */
697:
1.6 mycroft 698: int
1.85 thorpej 699: linux_sys_modify_ldt(l, v, retval)
700: struct lwp *l;
1.19 thorpej 701: void *v;
702: register_t *retval;
703: {
1.20 mycroft 704: struct linux_sys_modify_ldt_args /* {
1.6 mycroft 705: syscallarg(int) func;
706: syscallarg(void *) ptr;
707: syscallarg(size_t) bytecount;
1.19 thorpej 708: } */ *uap = v;
1.6 mycroft 709:
710: switch (SCARG(uap, func)) {
1.7 mycroft 711: #ifdef USER_LDT
1.6 mycroft 712: case 0:
1.85 thorpej 713: return linux_read_ldt(l, uap, retval);
1.6 mycroft 714: case 1:
1.72 christos 715: retval[0] = 1;
1.85 thorpej 716: return linux_write_ldt(l, uap, retval);
1.72 christos 717: case 2:
718: #ifdef notyet
1.85 thorpej 719: return (linux_read_default_ldt(l, uap, retval);
1.72 christos 720: #else
721: return (ENOSYS);
722: #endif
723: case 0x11:
724: retval[0] = 0;
1.85 thorpej 725: return linux_write_ldt(l, uap, retval);
1.7 mycroft 726: #endif /* USER_LDT */
727:
1.6 mycroft 728: default:
729: return (ENOSYS);
730: }
1.13 fvdl 731: }
732:
733: /*
734: * XXX Pathetic hack to make svgalib work. This will fake the major
735: * device number of an opened VT so that svgalib likes it. grmbl.
736: * Should probably do it 'wrong the right way' and use a mapping
737: * array for all major device numbers, and map linux_mknod too.
738: */
739: dev_t
1.69 christos 740: linux_fakedev(dev, raw)
1.13 fvdl 741: dev_t dev;
1.69 christos 742: int raw;
1.13 fvdl 743: {
1.69 christos 744: if (raw) {
1.41 drochner 745: #if (NWSDISPLAY > 0)
1.79 gehenna 746: extern const struct cdevsw wsdisplay_cdevsw;
747: if (cdevsw_lookup(dev) == &wsdisplay_cdevsw)
1.69 christos 748: return makedev(LINUX_CONS_MAJOR, (minor(dev) + 1));
1.41 drochner 749: #endif
1.69 christos 750: }
1.77 jdolecek 751:
752: return dev;
1.13 fvdl 753: }
754:
1.49 jhawk 755: #if (NWSDISPLAY > 0)
1.41 drochner 756: /*
757: * That's not complete, but enough to get an X server running.
758: */
759: #define NR_KEYS 128
1.61 jdolecek 760: static const u_short plain_map[NR_KEYS] = {
1.41 drochner 761: 0x0200, 0x001b, 0x0031, 0x0032, 0x0033, 0x0034, 0x0035, 0x0036,
762: 0x0037, 0x0038, 0x0039, 0x0030, 0x002d, 0x003d, 0x007f, 0x0009,
763: 0x0b71, 0x0b77, 0x0b65, 0x0b72, 0x0b74, 0x0b79, 0x0b75, 0x0b69,
764: 0x0b6f, 0x0b70, 0x005b, 0x005d, 0x0201, 0x0702, 0x0b61, 0x0b73,
765: 0x0b64, 0x0b66, 0x0b67, 0x0b68, 0x0b6a, 0x0b6b, 0x0b6c, 0x003b,
766: 0x0027, 0x0060, 0x0700, 0x005c, 0x0b7a, 0x0b78, 0x0b63, 0x0b76,
767: 0x0b62, 0x0b6e, 0x0b6d, 0x002c, 0x002e, 0x002f, 0x0700, 0x030c,
768: 0x0703, 0x0020, 0x0207, 0x0100, 0x0101, 0x0102, 0x0103, 0x0104,
769: 0x0105, 0x0106, 0x0107, 0x0108, 0x0109, 0x0208, 0x0209, 0x0307,
770: 0x0308, 0x0309, 0x030b, 0x0304, 0x0305, 0x0306, 0x030a, 0x0301,
771: 0x0302, 0x0303, 0x0300, 0x0310, 0x0206, 0x0200, 0x003c, 0x010a,
772: 0x010b, 0x0200, 0x0200, 0x0200, 0x0200, 0x0200, 0x0200, 0x0200,
773: 0x030e, 0x0702, 0x030d, 0x001c, 0x0701, 0x0205, 0x0114, 0x0603,
774: 0x0118, 0x0601, 0x0602, 0x0117, 0x0600, 0x0119, 0x0115, 0x0116,
775: 0x011a, 0x010c, 0x010d, 0x011b, 0x011c, 0x0110, 0x0311, 0x011d,
776: 0x0200, 0x0200, 0x0200, 0x0200, 0x0200, 0x0200, 0x0200, 0x0200,
777: }, shift_map[NR_KEYS] = {
778: 0x0200, 0x001b, 0x0021, 0x0040, 0x0023, 0x0024, 0x0025, 0x005e,
779: 0x0026, 0x002a, 0x0028, 0x0029, 0x005f, 0x002b, 0x007f, 0x0009,
780: 0x0b51, 0x0b57, 0x0b45, 0x0b52, 0x0b54, 0x0b59, 0x0b55, 0x0b49,
781: 0x0b4f, 0x0b50, 0x007b, 0x007d, 0x0201, 0x0702, 0x0b41, 0x0b53,
782: 0x0b44, 0x0b46, 0x0b47, 0x0b48, 0x0b4a, 0x0b4b, 0x0b4c, 0x003a,
783: 0x0022, 0x007e, 0x0700, 0x007c, 0x0b5a, 0x0b58, 0x0b43, 0x0b56,
784: 0x0b42, 0x0b4e, 0x0b4d, 0x003c, 0x003e, 0x003f, 0x0700, 0x030c,
785: 0x0703, 0x0020, 0x0207, 0x010a, 0x010b, 0x010c, 0x010d, 0x010e,
786: 0x010f, 0x0110, 0x0111, 0x0112, 0x0113, 0x0213, 0x0203, 0x0307,
787: 0x0308, 0x0309, 0x030b, 0x0304, 0x0305, 0x0306, 0x030a, 0x0301,
788: 0x0302, 0x0303, 0x0300, 0x0310, 0x0206, 0x0200, 0x003e, 0x010a,
789: 0x010b, 0x0200, 0x0200, 0x0200, 0x0200, 0x0200, 0x0200, 0x0200,
790: 0x030e, 0x0702, 0x030d, 0x0200, 0x0701, 0x0205, 0x0114, 0x0603,
791: 0x020b, 0x0601, 0x0602, 0x0117, 0x0600, 0x020a, 0x0115, 0x0116,
792: 0x011a, 0x010c, 0x010d, 0x011b, 0x011c, 0x0110, 0x0311, 0x011d,
793: 0x0200, 0x0200, 0x0200, 0x0200, 0x0200, 0x0200, 0x0200, 0x0200,
794: }, altgr_map[NR_KEYS] = {
795: 0x0200, 0x0200, 0x0200, 0x0040, 0x0200, 0x0024, 0x0200, 0x0200,
796: 0x007b, 0x005b, 0x005d, 0x007d, 0x005c, 0x0200, 0x0200, 0x0200,
797: 0x0b71, 0x0b77, 0x0918, 0x0b72, 0x0b74, 0x0b79, 0x0b75, 0x0b69,
798: 0x0b6f, 0x0b70, 0x0200, 0x007e, 0x0201, 0x0702, 0x0914, 0x0b73,
799: 0x0917, 0x0919, 0x0b67, 0x0b68, 0x0b6a, 0x0b6b, 0x0b6c, 0x0200,
800: 0x0200, 0x0200, 0x0700, 0x0200, 0x0b7a, 0x0b78, 0x0916, 0x0b76,
801: 0x0915, 0x0b6e, 0x0b6d, 0x0200, 0x0200, 0x0200, 0x0700, 0x030c,
802: 0x0703, 0x0200, 0x0207, 0x050c, 0x050d, 0x050e, 0x050f, 0x0510,
803: 0x0511, 0x0512, 0x0513, 0x0514, 0x0515, 0x0208, 0x0202, 0x0911,
804: 0x0912, 0x0913, 0x030b, 0x090e, 0x090f, 0x0910, 0x030a, 0x090b,
805: 0x090c, 0x090d, 0x090a, 0x0310, 0x0206, 0x0200, 0x007c, 0x0516,
806: 0x0517, 0x0200, 0x0200, 0x0200, 0x0200, 0x0200, 0x0200, 0x0200,
807: 0x030e, 0x0702, 0x030d, 0x0200, 0x0701, 0x0205, 0x0114, 0x0603,
808: 0x0118, 0x0601, 0x0602, 0x0117, 0x0600, 0x0119, 0x0115, 0x0116,
809: 0x011a, 0x010c, 0x010d, 0x011b, 0x011c, 0x0110, 0x0311, 0x011d,
810: 0x0200, 0x0200, 0x0200, 0x0200, 0x0200, 0x0200, 0x0200, 0x0200,
811: }, ctrl_map[NR_KEYS] = {
812: 0x0200, 0x0200, 0x0200, 0x0000, 0x001b, 0x001c, 0x001d, 0x001e,
813: 0x001f, 0x007f, 0x0200, 0x0200, 0x001f, 0x0200, 0x0008, 0x0200,
814: 0x0011, 0x0017, 0x0005, 0x0012, 0x0014, 0x0019, 0x0015, 0x0009,
815: 0x000f, 0x0010, 0x001b, 0x001d, 0x0201, 0x0702, 0x0001, 0x0013,
816: 0x0004, 0x0006, 0x0007, 0x0008, 0x000a, 0x000b, 0x000c, 0x0200,
817: 0x0007, 0x0000, 0x0700, 0x001c, 0x001a, 0x0018, 0x0003, 0x0016,
818: 0x0002, 0x000e, 0x000d, 0x0200, 0x020e, 0x007f, 0x0700, 0x030c,
819: 0x0703, 0x0000, 0x0207, 0x0100, 0x0101, 0x0102, 0x0103, 0x0104,
820: 0x0105, 0x0106, 0x0107, 0x0108, 0x0109, 0x0208, 0x0204, 0x0307,
821: 0x0308, 0x0309, 0x030b, 0x0304, 0x0305, 0x0306, 0x030a, 0x0301,
822: 0x0302, 0x0303, 0x0300, 0x0310, 0x0206, 0x0200, 0x0200, 0x010a,
823: 0x010b, 0x0200, 0x0200, 0x0200, 0x0200, 0x0200, 0x0200, 0x0200,
824: 0x030e, 0x0702, 0x030d, 0x001c, 0x0701, 0x0205, 0x0114, 0x0603,
825: 0x0118, 0x0601, 0x0602, 0x0117, 0x0600, 0x0119, 0x0115, 0x0116,
826: 0x011a, 0x010c, 0x010d, 0x011b, 0x011c, 0x0110, 0x0311, 0x011d,
827: 0x0200, 0x0200, 0x0200, 0x0200, 0x0200, 0x0200, 0x0200, 0x0200,
828: };
829:
1.61 jdolecek 830: const u_short * const linux_keytabs[] = {
1.41 drochner 831: plain_map, shift_map, altgr_map, altgr_map, ctrl_map
832: };
833: #endif
834:
1.52 fvdl 835: static struct biosdisk_info *
836: fd2biosinfo(p, fp)
837: struct proc *p;
838: struct file *fp;
839: {
840: struct vnode *vp;
841: const char *blkname;
842: char diskname[16];
843: int i;
844: struct nativedisk_info *nip;
845: struct disklist *dl = i386_alldisks;
846:
847: if (fp->f_type != DTYPE_VNODE)
848: return NULL;
849: vp = (struct vnode *)fp->f_data;
850:
851: if (vp->v_type != VBLK)
852: return NULL;
853:
1.79 gehenna 854: blkname = devsw_blk2name(major(vp->v_rdev));
1.52 fvdl 855: snprintf(diskname, sizeof diskname, "%s%u", blkname,
856: DISKUNIT(vp->v_rdev));
857:
858: for (i = 0; i < dl->dl_nnativedisks; i++) {
859: nip = &dl->dl_nativedisks[i];
860: if (strcmp(diskname, nip->ni_devname))
861: continue;
862: if (nip->ni_nmatches != 0)
863: return &dl->dl_biosdisks[nip->ni_biosmatches[0]];
864: }
865:
866: return NULL;
867: }
868:
869:
1.13 fvdl 870: /*
871: * We come here in a last attempt to satisfy a Linux ioctl() call
872: */
873: int
1.89 fvdl 874: linux_machdepioctl(p, v, retval)
875: struct proc *p;
1.19 thorpej 876: void *v;
877: register_t *retval;
878: {
1.20 mycroft 879: struct linux_sys_ioctl_args /* {
1.13 fvdl 880: syscallarg(int) fd;
881: syscallarg(u_long) com;
882: syscallarg(caddr_t) data;
1.19 thorpej 883: } */ *uap = v;
1.29 christos 884: struct sys_ioctl_args bia;
1.15 fvdl 885: u_long com;
1.52 fvdl 886: int error, error1;
1.50 veego 887: #if (NWSDISPLAY > 0)
1.13 fvdl 888: struct vt_mode lvt;
889: caddr_t bvtp, sg;
1.41 drochner 890: struct kbentry kbe;
891: #endif
1.52 fvdl 892: struct linux_hd_geometry hdg;
893: struct linux_hd_big_geometry hdg_big;
894: struct biosdisk_info *bip;
895: struct filedesc *fdp;
896: struct file *fp;
897: int fd;
898: struct disklabel label, *labp;
899: struct partinfo partp;
1.89 fvdl 900: int (*ioctlf)(struct file *, u_long, void *, struct proc *);
1.52 fvdl 901: u_long start, biostotal, realtotal;
902: u_char heads, sectors;
903: u_int cylinders;
1.55 fvdl 904: struct ioctl_pt pt;
1.13 fvdl 905:
1.52 fvdl 906: fd = SCARG(uap, fd);
907: SCARG(&bia, fd) = fd;
1.13 fvdl 908: SCARG(&bia, data) = SCARG(uap, data);
909: com = SCARG(uap, com);
910:
1.52 fvdl 911: fdp = p->p_fd;
912:
1.64 thorpej 913: if ((fp = fd_getfile(fdp, fd)) == NULL)
1.53 thorpej 914: return (EBADF);
1.52 fvdl 915:
1.86 yamt 916: FILE_USE(fp);
917:
1.13 fvdl 918: switch (com) {
1.50 veego 919: #if (NWSDISPLAY > 0)
1.13 fvdl 920: case LINUX_KDGKBMODE:
1.21 fvdl 921: com = KDGKBMODE;
922: break;
1.13 fvdl 923: case LINUX_KDSKBMODE:
924: com = KDSKBMODE;
925: if ((unsigned)SCARG(uap, data) == LINUX_K_MEDIUMRAW)
926: SCARG(&bia, data) = (caddr_t)K_RAW;
927: break;
1.59 fvdl 928: case LINUX_KIOCSOUND:
929: SCARG(&bia, data) =
930: (caddr_t)(((unsigned long)SCARG(&bia, data)) & 0xffff);
931: /* fall through */
1.13 fvdl 932: case LINUX_KDMKTONE:
933: com = KDMKTONE;
934: break;
935: case LINUX_KDSETMODE:
936: com = KDSETMODE;
1.58 fvdl 937: break;
938: case LINUX_KDGETMODE:
939: /* KD_* values are equal to the wscons numbers */
940: com = WSDISPLAYIO_GMODE;
1.13 fvdl 941: break;
942: case LINUX_KDENABIO:
943: com = KDENABIO;
944: break;
945: case LINUX_KDDISABIO:
946: com = KDDISABIO;
947: break;
948: case LINUX_KDGETLED:
949: com = KDGETLED;
950: break;
951: case LINUX_KDSETLED:
952: com = KDSETLED;
953: break;
954: case LINUX_VT_OPENQRY:
955: com = VT_OPENQRY;
956: break;
957: case LINUX_VT_GETMODE:
958: SCARG(&bia, com) = VT_GETMODE;
1.85 thorpej 959: /* XXX NJWLWP */
960: if ((error = sys_ioctl(curlwp, &bia, retval)))
1.86 yamt 961: goto out;
1.13 fvdl 962: if ((error = copyin(SCARG(uap, data), (caddr_t)&lvt,
963: sizeof (struct vt_mode))))
1.86 yamt 964: goto out;
1.74 christos 965: lvt.relsig = native_to_linux_signo[lvt.relsig];
966: lvt.acqsig = native_to_linux_signo[lvt.acqsig];
967: lvt.frsig = native_to_linux_signo[lvt.frsig];
1.86 yamt 968: error = copyout((caddr_t)&lvt, SCARG(uap, data),
1.13 fvdl 969: sizeof (struct vt_mode));
1.86 yamt 970: goto out;
1.13 fvdl 971: case LINUX_VT_SETMODE:
972: com = VT_SETMODE;
973: if ((error = copyin(SCARG(uap, data), (caddr_t)&lvt,
974: sizeof (struct vt_mode))))
1.86 yamt 975: goto out;
1.74 christos 976: lvt.relsig = linux_to_native_signo[lvt.relsig];
977: lvt.acqsig = linux_to_native_signo[lvt.acqsig];
978: lvt.frsig = linux_to_native_signo[lvt.frsig];
1.71 christos 979: sg = stackgap_init(p, 0);
980: bvtp = stackgap_alloc(p, &sg, sizeof (struct vt_mode));
1.13 fvdl 981: if ((error = copyout(&lvt, bvtp, sizeof (struct vt_mode))))
1.86 yamt 982: goto out;
1.13 fvdl 983: SCARG(&bia, data) = bvtp;
984: break;
1.52 fvdl 985: case LINUX_VT_DISALLOCATE:
986: /* XXX should use WSDISPLAYIO_DELSCREEN */
1.86 yamt 987: error = 0;
988: goto out;
1.13 fvdl 989: case LINUX_VT_RELDISP:
990: com = VT_RELDISP;
991: break;
992: case LINUX_VT_ACTIVATE:
993: com = VT_ACTIVATE;
994: break;
995: case LINUX_VT_WAITACTIVE:
996: com = VT_WAITACTIVE;
997: break;
1.41 drochner 998: case LINUX_VT_GETSTATE:
999: com = VT_GETSTATE;
1000: break;
1001: case LINUX_KDGKBTYPE:
1.91 jdolecek 1002: {
1003: static const u_int8_t kb101 = KB_101;
1004:
1.41 drochner 1005: /* This is what Linux does. */
1.91 jdolecek 1006: error = copyout(&kb101, SCARG(uap, data), 1);
1.86 yamt 1007: goto out;
1.91 jdolecek 1008: }
1.41 drochner 1009: case LINUX_KDGKBENT:
1010: /*
1011: * The Linux KDGKBENT ioctl is different from the
1012: * SYSV original. So we handle it in machdep code.
1013: * XXX We should use keyboard mapping information
1014: * from wsdisplay, but this would be expensive.
1015: */
1016: if ((error = copyin(SCARG(uap, data), &kbe,
1017: sizeof(struct kbentry))))
1.86 yamt 1018: goto out;
1.41 drochner 1019: if (kbe.kb_table >= sizeof(linux_keytabs) / sizeof(u_short *)
1.86 yamt 1020: || kbe.kb_index >= NR_KEYS) {
1021: error = EINVAL;
1022: goto out;
1023: }
1.41 drochner 1024: kbe.kb_value = linux_keytabs[kbe.kb_table][kbe.kb_index];
1.86 yamt 1025: error = copyout(&kbe, SCARG(uap, data),
1026: sizeof(struct kbentry));
1027: goto out;
1.54 fvdl 1028: #endif
1.52 fvdl 1029: case LINUX_HDIO_GETGEO:
1030: case LINUX_HDIO_GETGEO_BIG:
1031: /*
1032: * Try to mimic Linux behaviour: return the BIOS geometry
1033: * if possible (extending its # of cylinders if it's beyond
1034: * the 1023 limit), fall back to the MI geometry (i.e.
1035: * the real geometry) if not found, by returning an
1036: * error. See common/linux_hdio.c
1037: */
1038: bip = fd2biosinfo(p, fp);
1039: ioctlf = fp->f_ops->fo_ioctl;
1.89 fvdl 1040: error = ioctlf(fp, DIOCGDEFLABEL, (caddr_t)&label, p);
1041: error1 = ioctlf(fp, DIOCGPART, (caddr_t)&partp, p);
1.86 yamt 1042: if (error != 0 && error1 != 0) {
1043: error = error1;
1044: goto out;
1045: }
1.52 fvdl 1046: labp = error != 0 ? &label : partp.disklab;
1047: start = error1 != 0 ? partp.part->p_offset : 0;
1048: if (bip != NULL && bip->bi_head != 0 && bip->bi_sec != 0
1049: && bip->bi_cyl != 0) {
1050: heads = bip->bi_head;
1051: sectors = bip->bi_sec;
1052: cylinders = bip->bi_cyl;
1053: biostotal = heads * sectors * cylinders;
1054: realtotal = labp->d_ntracks * labp->d_nsectors *
1055: labp->d_ncylinders;
1056: if (realtotal > biostotal)
1057: cylinders = realtotal / (heads * sectors);
1058: } else {
1059: heads = labp->d_ntracks;
1060: cylinders = labp->d_ncylinders;
1061: sectors = labp->d_nsectors;
1062: }
1063: if (com == LINUX_HDIO_GETGEO) {
1064: hdg.start = start;
1065: hdg.heads = heads;
1066: hdg.cylinders = cylinders;
1067: hdg.sectors = sectors;
1.86 yamt 1068: error = copyout(&hdg, SCARG(uap, data), sizeof hdg);
1069: goto out;
1.52 fvdl 1070: } else {
1071: hdg_big.start = start;
1072: hdg_big.heads = heads;
1073: hdg_big.cylinders = cylinders;
1074: hdg_big.sectors = sectors;
1.86 yamt 1075: error = copyout(&hdg_big, SCARG(uap, data),
1.52 fvdl 1076: sizeof hdg_big);
1.86 yamt 1077: goto out;
1.52 fvdl 1078: }
1079:
1.13 fvdl 1080: default:
1.54 fvdl 1081: /*
1.55 fvdl 1082: * Unknown to us. If it's on a device, just pass it through
1083: * using PTIOCLINUX, the device itself might be able to
1084: * make some sense of it.
1.57 fvdl 1085: * XXX hack: if the function returns EJUSTRETURN,
1086: * it has stuffed a sysctl return value in pt.data.
1.54 fvdl 1087: */
1.55 fvdl 1088: FILE_USE(fp);
1089: ioctlf = fp->f_ops->fo_ioctl;
1090: pt.com = SCARG(uap, com);
1091: pt.data = SCARG(uap, data);
1.89 fvdl 1092: error = ioctlf(fp, PTIOCLINUX, (caddr_t)&pt, p);
1093: FILE_UNUSE(fp, p);
1.57 fvdl 1094: if (error == EJUSTRETURN) {
1095: retval[0] = (register_t)pt.data;
1096: error = 0;
1097: }
1.55 fvdl 1098:
1.60 fvdl 1099: if (error == ENOTTY)
1.72 christos 1100: DPRINTF(("linux_machdepioctl: invalid ioctl %08lx\n",
1101: com));
1.86 yamt 1102: goto out;
1.13 fvdl 1103: }
1104: SCARG(&bia, com) = com;
1.85 thorpej 1105: /* XXX NJWLWP */
1.86 yamt 1106: error = sys_ioctl(curlwp, &bia, retval);
1107: out:
1.89 fvdl 1108: FILE_UNUSE(fp ,p);
1.86 yamt 1109: return error;
1.13 fvdl 1110: }
1111:
1112: /*
1113: * Set I/O permissions for a process. Just set the maximum level
1114: * right away (ignoring the argument), otherwise we would have
1115: * to rely on I/O permission maps, which are not implemented.
1116: */
1117: int
1.85 thorpej 1118: linux_sys_iopl(l, v, retval)
1119: struct lwp *l;
1.19 thorpej 1120: void *v;
1121: register_t *retval;
1122: {
1.29 christos 1123: #if 0
1.20 mycroft 1124: struct linux_sys_iopl_args /* {
1.13 fvdl 1125: syscallarg(int) level;
1.19 thorpej 1126: } */ *uap = v;
1.29 christos 1127: #endif
1.85 thorpej 1128: struct proc *p = l->l_proc;
1129: struct trapframe *fp = l->l_md.md_regs;
1.13 fvdl 1130:
1131: if (suser(p->p_ucred, &p->p_acflag) != 0)
1132: return EPERM;
1133: fp->tf_eflags |= PSL_IOPL;
1134: *retval = 0;
1135: return 0;
1136: }
1137:
1138: /*
1139: * See above. If a root process tries to set access to an I/O port,
1140: * just let it have the whole range.
1141: */
1142: int
1.85 thorpej 1143: linux_sys_ioperm(l, v, retval)
1144: struct lwp *l;
1.19 thorpej 1145: void *v;
1146: register_t *retval;
1147: {
1.20 mycroft 1148: struct linux_sys_ioperm_args /* {
1.13 fvdl 1149: syscallarg(unsigned int) lo;
1150: syscallarg(unsigned int) hi;
1151: syscallarg(int) val;
1.19 thorpej 1152: } */ *uap = v;
1.85 thorpej 1153: struct proc *p = l->l_proc;
1154: struct trapframe *fp = l->l_md.md_regs;
1.13 fvdl 1155:
1156: if (suser(p->p_ucred, &p->p_acflag) != 0)
1157: return EPERM;
1158: if (SCARG(uap, val))
1159: fp->tf_eflags |= PSL_IOPL;
1160: *retval = 0;
1.92 christos 1161: return 0;
1162: }
1163:
1164: int
1165: linux_exec_setup_stack(struct proc *p, struct exec_package *epp)
1166: {
1167: u_long max_stack_size;
1168: u_long access_linear_min, access_size;
1169: u_long noaccess_linear_min, noaccess_size;
1170:
1171: #ifndef USRSTACK32
1172: #define USRSTACK32 (0x00000000ffffffffL&~PGOFSET)
1173: #endif
1174:
1175: if (epp->ep_flags & EXEC_32) {
1176: epp->ep_minsaddr = USRSTACK32;
1177: max_stack_size = MAXSSIZ;
1178: } else {
1179: epp->ep_minsaddr = USRSTACK;
1180: max_stack_size = MAXSSIZ;
1181: }
1182:
1183: if (epp->ep_minsaddr > LINUX_USRSTACK)
1184: epp->ep_minsaddr = LINUX_USRSTACK;
1.93 christos 1185: #ifdef DEBUG_LINUX
1.92 christos 1186: else {
1187: /*
1188: * Someone needs to make KERNBASE and TEXTADDR
1.93 christos 1189: * java versions < 1.4.2 need the stack to be
1190: * at 0xC0000000
1.92 christos 1191: */
1192: uprintf("Cannot setup stack to 0xC0000000, "
1193: "java will not work properly\n");
1194: }
1195: #endif
1196: epp->ep_maxsaddr = (u_long)STACK_GROW(epp->ep_minsaddr,
1197: max_stack_size);
1198: epp->ep_ssize = p->p_rlimit[RLIMIT_STACK].rlim_cur;
1199:
1200: /*
1201: * set up commands for stack. note that this takes *two*, one to
1202: * map the part of the stack which we can access, and one to map
1203: * the part which we can't.
1204: *
1205: * arguably, it could be made into one, but that would require the
1206: * addition of another mapping proc, which is unnecessary
1207: */
1208: access_size = epp->ep_ssize;
1209: access_linear_min = (u_long)STACK_ALLOC(epp->ep_minsaddr, access_size);
1210: noaccess_size = max_stack_size - access_size;
1211: noaccess_linear_min = (u_long)STACK_ALLOC(STACK_GROW(epp->ep_minsaddr,
1212: access_size), noaccess_size);
1213: NEW_VMCMD(&epp->ep_vmcmds, vmcmd_map_zero, noaccess_size,
1214: noaccess_linear_min, NULLVP, 0, VM_PROT_NONE);
1215: NEW_VMCMD(&epp->ep_vmcmds, vmcmd_map_zero, access_size,
1216: access_linear_min, NULLVP, 0,
1217: VM_PROT_READ|VM_PROT_WRITE|VM_PROT_EXECUTE);
1218:
1.13 fvdl 1219: return 0;
1.1 fvdl 1220: }
CVSweb <webmaster@jp.NetBSD.org>