/* * Copyright (c) 2003,2004,2008 The DragonFly Project. All rights reserved. * * This code is derived from software contributed to The DragonFly Project * by David Xu and Matthew Dillon * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in * the documentation and/or other materials provided with the * distribution. * 3. Neither the name of The DragonFly Project nor the names of its * contributors may be used to endorse or promote products derived * from this software without specific, prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, * INCIDENTAL, SPECIAL, EXEMPLARY OR CONSEQUENTIAL DAMAGES (INCLUDING, * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF * SUCH DAMAGE. */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include /* pcb.h included via sys/user.h */ #include /* CPU_prvspace */ #include #include /* * set a TLS descriptor. For x86_64 descriptor 0 identifies %fs and * descriptor 1 identifies %gs, and 0 is returned in sysmsg_result. * * Returns the value userland needs to load into %gs representing the * TLS descriptor or -1 on error. * * (int which, struct tls_info *info, size_t infosize) * * MPSAFE */ int sys_set_tls_area(struct set_tls_area_args *uap) { struct tls_info info; int error; int i; /* * Sanity checks * * which 0 == %fs, which 1 == %gs */ i = uap->which; if (i < 0 || i > 1) return (ERANGE); if (uap->infosize < 0) return (EINVAL); /* * Maintain forwards compatibility with future extensions. */ if (uap->infosize != sizeof(info)) { bzero(&info, sizeof(info)); error = copyin(uap->info, &info, min(sizeof(info), uap->infosize)); } else { error = copyin(uap->info, &info, sizeof(info)); } if (error) return (error); if (info.size < -1) return (EINVAL); /* * For x86_64 we can only adjust FSBASE and GSBASE */ curthread->td_tls.info[i] = info; set_user_TLS(); uap->sysmsg_result = 0; /* segment descriptor $0 */ return(0); } /* * Return the specified TLS descriptor to userland. * * Returns the value userland needs to load into %gs representing the * TLS descriptor or -1 on error. * * (int which, struct tls_info *info, size_t infosize) * * MPSAFE */ int sys_get_tls_area(struct get_tls_area_args *uap) { struct tls_info info; int error; int i; /* * Sanity checks */ i = uap->which; if (i < 0 || i > 1) return (ERANGE); if (uap->infosize < 0) return (EINVAL); info = curthread->td_tls.info[i]; error = copyout(&info, uap->info, min(sizeof(info), uap->infosize)); return(error); } /* * Install the TLS. * * It shouldn't be possible for a preemptive thread switch to do anything * more than set gd_user_fs and wrmsr for us. Even though there is a window * where gd_user_fs/gd_user_gs do not match the MSRs no preemptive thread * switch should ever switch to any heavy weight thread other than our own. * * Still, use a critical section to be safe. * * MPSAFE */ void set_user_TLS(void) { struct mdglobaldata *gd = mdcpu; thread_t td = gd->mi.gd_curthread; crit_enter_quick(td); td->td_pcb->pcb_fsbase = (register_t)td->td_tls.info[0].base; td->td_pcb->pcb_gsbase = (register_t)td->td_tls.info[1].base; if (gd->gd_user_fs != td->td_pcb->pcb_fsbase) { gd->gd_user_fs = td->td_pcb->pcb_fsbase; wrmsr(MSR_FSBASE, gd->gd_user_fs); } if (gd->gd_user_gs != td->td_pcb->pcb_gsbase) { gd->gd_user_gs = td->td_pcb->pcb_gsbase; wrmsr(MSR_KGSBASE, gd->gd_user_gs); } clear_quickret(); crit_exit_quick(td); }