1 1.16 christos /* $NetBSD: tls.c,v 1.16 2024/11/30 01:04:04 christos Exp $ */ 2 1.1 joerg 3 1.1 joerg /*- 4 1.1 joerg * Copyright (c) 2011 The NetBSD Foundation, Inc. 5 1.1 joerg * All rights reserved. 6 1.1 joerg * 7 1.1 joerg * This code is derived from software contributed to The NetBSD Foundation 8 1.1 joerg * by Joerg Sonnenberger. 9 1.1 joerg * 10 1.1 joerg * Redistribution and use in source and binary forms, with or without 11 1.1 joerg * modification, are permitted provided that the following conditions 12 1.1 joerg * are met: 13 1.1 joerg * 1. Redistributions of source code must retain the above copyright 14 1.1 joerg * notice, this list of conditions and the following disclaimer. 15 1.1 joerg * 2. Redistributions in binary form must reproduce the above copyright 16 1.1 joerg * notice, this list of conditions and the following disclaimer in the 17 1.1 joerg * documentation and/or other materials provided with the distribution. 18 1.1 joerg * 19 1.1 joerg * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS 20 1.1 joerg * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED 21 1.1 joerg * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR 22 1.1 joerg * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS 23 1.1 joerg * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR 24 1.1 joerg * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF 25 1.1 joerg * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS 26 1.1 joerg * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN 27 1.1 joerg * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) 28 1.1 joerg * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE 29 1.1 joerg * POSSIBILITY OF SUCH DAMAGE. 30 1.1 joerg */ 31 1.1 joerg 32 1.1 joerg #include <sys/cdefs.h> 33 1.16 christos __RCSID("$NetBSD: tls.c,v 1.16 2024/11/30 01:04:04 christos Exp $"); 34 1.1 joerg 35 1.1 joerg #include "namespace.h" 36 1.1 joerg 37 1.2 joerg #define _rtld_tls_allocate __libc_rtld_tls_allocate 38 1.2 joerg #define _rtld_tls_free __libc_rtld_tls_free 39 1.2 joerg 40 1.1 joerg #include <sys/tls.h> 41 1.1 joerg 42 1.1 joerg #if defined(__HAVE_TLS_VARIANT_I) || defined(__HAVE_TLS_VARIANT_II) 43 1.1 joerg 44 1.1 joerg #include <sys/param.h> 45 1.1 joerg #include <sys/mman.h> 46 1.1 joerg #include <link_elf.h> 47 1.1 joerg #include <lwp.h> 48 1.9 joerg #include <stdbool.h> 49 1.11 joerg #include <stdalign.h> 50 1.1 joerg #include <stddef.h> 51 1.1 joerg #include <stdlib.h> 52 1.1 joerg #include <string.h> 53 1.1 joerg #include <unistd.h> 54 1.1 joerg 55 1.16 christos #include <machine/lwp_private.h> 56 1.16 christos 57 1.1 joerg __dso_hidden void __libc_static_tls_setup(void); 58 1.1 joerg 59 1.9 joerg static bool is_dynamic; 60 1.1 joerg static const void *tls_initaddr; 61 1.1 joerg static size_t tls_initsize; 62 1.1 joerg static size_t tls_size; 63 1.1 joerg static size_t tls_allocation; 64 1.1 joerg static void *initial_thread_tcb; 65 1.1 joerg 66 1.4 skrll void * __libc_tls_get_addr(void); 67 1.4 skrll 68 1.4 skrll __weak_alias(__tls_get_addr, __libc_tls_get_addr) 69 1.1 joerg #ifdef __i386__ 70 1.4 skrll __weak_alias(___tls_get_addr, __libc_tls_get_addr) 71 1.1 joerg #endif 72 1.1 joerg 73 1.4 skrll void * 74 1.4 skrll __libc_tls_get_addr(void) 75 1.4 skrll { 76 1.4 skrll 77 1.4 skrll abort(); 78 1.5 he /* NOTREACHED */ 79 1.4 skrll } 80 1.4 skrll 81 1.1 joerg __weak_alias(_rtld_tls_allocate, __libc_rtld_tls_allocate) 82 1.1 joerg 83 1.1 joerg struct tls_tcb * 84 1.1 joerg _rtld_tls_allocate(void) 85 1.1 joerg { 86 1.1 joerg struct tls_tcb *tcb; 87 1.1 joerg uint8_t *p; 88 1.1 joerg 89 1.1 joerg if (initial_thread_tcb == NULL) { 90 1.13 nakayama #ifdef __HAVE_TLS_VARIANT_I 91 1.13 nakayama tls_allocation = tls_size; 92 1.13 nakayama #else 93 1.12 joerg tls_allocation = roundup2(tls_size, alignof(max_align_t)); 94 1.1 joerg #endif 95 1.1 joerg 96 1.12 joerg initial_thread_tcb = p = mmap(NULL, 97 1.12 joerg tls_allocation + sizeof(*tcb), PROT_READ | PROT_WRITE, 98 1.12 joerg MAP_ANON, -1, 0); 99 1.14 riastrad if (p == MAP_FAILED) 100 1.14 riastrad initial_thread_tcb = p = NULL; 101 1.1 joerg } else { 102 1.12 joerg p = calloc(1, tls_allocation + sizeof(*tcb)); 103 1.1 joerg } 104 1.1 joerg if (p == NULL) { 105 1.1 joerg static const char msg[] = "TLS allocation failed, terminating\n"; 106 1.1 joerg write(STDERR_FILENO, msg, sizeof(msg)); 107 1.1 joerg _exit(127); 108 1.1 joerg } 109 1.1 joerg #ifdef __HAVE_TLS_VARIANT_I 110 1.1 joerg /* LINTED */ 111 1.1 joerg tcb = (struct tls_tcb *)p; 112 1.1 joerg p += sizeof(struct tls_tcb); 113 1.1 joerg #else 114 1.1 joerg /* LINTED tls_size is rounded above */ 115 1.12 joerg tcb = (struct tls_tcb *)(p + tls_allocation); 116 1.12 joerg p = (uint8_t *)tcb - tls_size; 117 1.1 joerg tcb->tcb_self = tcb; 118 1.1 joerg #endif 119 1.1 joerg memcpy(p, tls_initaddr, tls_initsize); 120 1.1 joerg 121 1.1 joerg return tcb; 122 1.1 joerg } 123 1.1 joerg 124 1.1 joerg __weak_alias(_rtld_tls_free, __libc_rtld_tls_free) 125 1.1 joerg 126 1.1 joerg void 127 1.1 joerg _rtld_tls_free(struct tls_tcb *tcb) 128 1.1 joerg { 129 1.1 joerg uint8_t *p; 130 1.1 joerg 131 1.1 joerg #ifdef __HAVE_TLS_VARIANT_I 132 1.1 joerg /* LINTED */ 133 1.1 joerg p = (uint8_t *)tcb; 134 1.1 joerg #else 135 1.1 joerg /* LINTED */ 136 1.12 joerg p = (uint8_t *)tcb - tls_allocation; 137 1.1 joerg #endif 138 1.1 joerg if (p == initial_thread_tcb) 139 1.12 joerg munmap(p, tls_allocation + sizeof(*tcb)); 140 1.1 joerg else 141 1.1 joerg free(p); 142 1.1 joerg } 143 1.1 joerg 144 1.7 matt static int __section(".text.startup") 145 1.15 christos /*ARGSUSED*/ 146 1.1 joerg __libc_static_tls_setup_cb(struct dl_phdr_info *data, size_t len, void *cookie) 147 1.1 joerg { 148 1.1 joerg const Elf_Phdr *phdr = data->dlpi_phdr; 149 1.1 joerg const Elf_Phdr *phlimit = data->dlpi_phdr + data->dlpi_phnum; 150 1.1 joerg 151 1.1 joerg for (; phdr < phlimit; ++phdr) { 152 1.9 joerg if (phdr->p_type == PT_INTERP) { 153 1.9 joerg is_dynamic = true; 154 1.9 joerg return -1; 155 1.9 joerg } 156 1.1 joerg if (phdr->p_type != PT_TLS) 157 1.1 joerg continue; 158 1.1 joerg tls_initaddr = (void *)(phdr->p_vaddr + data->dlpi_addr); 159 1.1 joerg tls_initsize = phdr->p_filesz; 160 1.13 nakayama #ifdef __HAVE_TLS_VARIANT_I 161 1.1 joerg tls_size = phdr->p_memsz; 162 1.13 nakayama #else 163 1.13 nakayama tls_size = roundup2(phdr->p_memsz, phdr->p_align); 164 1.13 nakayama #endif 165 1.1 joerg } 166 1.1 joerg return 0; 167 1.1 joerg } 168 1.1 joerg 169 1.1 joerg void 170 1.1 joerg __libc_static_tls_setup(void) 171 1.1 joerg { 172 1.1 joerg struct tls_tcb *tcb; 173 1.1 joerg 174 1.9 joerg dl_iterate_phdr(__libc_static_tls_setup_cb, NULL); 175 1.9 joerg if (is_dynamic) 176 1.1 joerg return; 177 1.1 joerg 178 1.1 joerg tcb = _rtld_tls_allocate(); 179 1.3 matt #ifdef __HAVE___LWP_SETTCB 180 1.3 matt __lwp_settcb(tcb); 181 1.3 matt #else 182 1.1 joerg _lwp_setprivate(tcb); 183 1.3 matt #endif 184 1.1 joerg } 185 1.1 joerg 186 1.1 joerg #endif /* __HAVE_TLS_VARIANT_I || __HAVE_TLS_VARIANT_II */ 187