static int __thread_start( struct thread_entry_args *entry_args ) { thread_entry *entry = entry_args->entry; void *arg = entry_args->arg; free(entry_args); __libc_tsd_set(LOCALE,&_nl_global_locale); exit_thread(entry(arg)); /* exit_thread() should never return. Log it. */ dbprintf("%s, %s: panic: exit_thread() returned!\n",__FILE__,__FUNCTION__); return 1; }
/* Switch the current thread's locale to DATASET. If DATASET is null, instead just return the current setting. The special value LC_GLOBAL_LOCALE is the initial setting for all threads, and means the thread uses the global setting controlled by `setlocale'. */ locale_t __uselocale (locale_t newloc) { locale_t oldloc = _NL_CURRENT_LOCALE; if (newloc != NULL) { const locale_t locobj = newloc == LC_GLOBAL_LOCALE ? &_nl_global_locale : newloc; __libc_tsd_set (__locale_t, LOCALE, locobj); #ifdef NL_CURRENT_INDIRECT /* Now we must update all the per-category thread-local variables to point into the new current locale for this thread. The magic symbols _nl_current_LC_FOO_used are defined to meaningless values if _nl_current_LC_FOO was linked in. By using weak references to both symbols and testing the address of _nl_current_LC_FOO_used, we can avoid accessing the _nl_current_LC_FOO thread-local variable at all when no code referring to it was linked in. We need the special bogus symbol because while TLS symbols can be weak, there is no reasonable way to test for the default-zero value as with a heap symbol (taking the address would just use some bogus offset from our thread pointer). */ # define DEFINE_CATEGORY(category, category_name, items, a) \ { \ extern char _nl_current_##category##_used; \ weak_extern (_nl_current_##category##_used) \ weak_extern (_nl_current_##category) \ if (&_nl_current_##category##_used != 0) \ _nl_current_##category = &locobj->__locales[category]; \ } # include "categories.def" # undef DEFINE_CATEGORY #endif /* Update the special tsd cache of some locale data. */ __libc_tsd_set (const uint16_t *, CTYPE_B, (void *) locobj->__ctype_b); __libc_tsd_set (const int32_t *, CTYPE_TOLOWER, (void *) locobj->__ctype_tolower); __libc_tsd_set (const int32_t *, CTYPE_TOUPPER, (void *) locobj->__ctype_toupper); } return oldloc == &_nl_global_locale ? LC_GLOBAL_LOCALE : oldloc; }
/* * Task-variable destructor */ void __rpc_thread_destroy (void) { struct rpc_thread_variables *tvp = __libc_tsd_get (RPC_VARS); if (tvp != NULL && tvp != &__libc_tsd_RPC_VARS_mem) { __rpc_thread_svc_cleanup (); __rpc_thread_clnt_cleanup (); /*__rpc_thread_key_cleanup (); */ free (tvp->clnt_perr_buf_s); free (tvp->clntraw_private_s); free (tvp->svcraw_private_s); free (tvp->authdes_cache_s); free (tvp->authdes_lru_s); free (tvp); __libc_tsd_set (RPC_VARS, NULL); } }
struct rpc_thread_variables attribute_hidden * __rpc_thread_variables (void) { __libc_once_define (static, once); struct rpc_thread_variables *tvp; tvp = __libc_tsd_get (RPC_VARS); if (tvp == NULL) { __libc_once (once, rpc_thread_multi); tvp = __libc_tsd_get (RPC_VARS); if (tvp == NULL) { tvp = calloc (1, sizeof *tvp); if (tvp != NULL) __libc_tsd_set (RPC_VARS, tvp); else tvp = __libc_tsd_get (RPC_VARS); } } return tvp; }
void _nl_postload_ctype (void) { #define current(type,x,offset) \ ((const type *) _NL_CURRENT (LC_CTYPE, _NL_CTYPE_##x) + offset) const union locale_data_value *const ctypes = _nl_global_locale.__locales[LC_CTYPE]->values; /* These thread-local variables are defined in ctype-info.c. The declarations here must match those in localeinfo.h. These point into arrays of 384, so they can be indexed by any `unsigned char' value [0,255]; by EOF (-1); or by any `signed char' value [-128,-1). ISO C requires that the ctype functions work for `unsigned char' values and for EOF; we also support negative `signed char' values for broken old programs. The case conversion arrays are of `int's rather than `unsigned char's because tolower (EOF) must be EOF, which doesn't fit into an `unsigned char'. But today more important is that the arrays are also used for multi-byte character sets. First we update the special members of _nl_global_locale as newlocale would. This is necessary for uselocale (LC_GLOBAL_LOCALE) to find these values properly. */ _nl_global_locale.__ctype_b = (const unsigned short int *) ctypes[_NL_ITEM_INDEX (_NL_CTYPE_CLASS)].string + 128; _nl_global_locale.__ctype_tolower = (const int *) ctypes[_NL_ITEM_INDEX (_NL_CTYPE_TOLOWER)].string + 128; _nl_global_locale.__ctype_toupper = (const int *) ctypes[_NL_ITEM_INDEX (_NL_CTYPE_TOUPPER)].string + 128; /* Next we must set the thread-local caches if and only if this thread is in fact using the global locale. */ if (_NL_CURRENT_LOCALE == &_nl_global_locale) { __libc_tsd_set (CTYPE_B, (void *) _nl_global_locale.__ctype_b); __libc_tsd_set (CTYPE_TOUPPER, (void *) _nl_global_locale.__ctype_toupper); __libc_tsd_set (CTYPE_TOLOWER, (void *) _nl_global_locale.__ctype_tolower); } #include <shlib-compat.h> #if SHLIB_COMPAT (libc, GLIBC_2_0, GLIBC_2_3) /* We must use the exported names to access these so we are sure to be accessing the main executable's copy if it has COPY relocs. */ extern __const unsigned short int *__ctype_b; /* Characteristics. */ extern __const __int32_t *__ctype_tolower; /* Case conversions. */ extern __const __int32_t *__ctype_toupper; /* Case conversions. */ extern const uint32_t *__ctype32_b; extern const uint32_t *__ctype32_toupper; extern const uint32_t *__ctype32_tolower; /* We need the .symver declarations these macros generate so that our references are explicitly bound to the versioned symbol names rather than the unadorned names that are not exported. When the linker sees these bound to local symbols (as the unexported names are) then it doesn't generate a proper relocation to the global symbols. We need those relocations so that a versioned definition with a COPY reloc in an executable will override the libc.so definition. */ compat_symbol (libc, __ctype_b, __ctype_b, GLIBC_2_0); compat_symbol (libc, __ctype_tolower, __ctype_tolower, GLIBC_2_0); compat_symbol (libc, __ctype_toupper, __ctype_toupper, GLIBC_2_0); compat_symbol (libc, __ctype32_b, __ctype32_b, GLIBC_2_0); compat_symbol (libc, __ctype32_tolower, __ctype32_tolower, GLIBC_2_2); compat_symbol (libc, __ctype32_toupper, __ctype32_toupper, GLIBC_2_2); __ctype_b = current (uint16_t, CLASS, 128); __ctype_toupper = current (int32_t, TOUPPER, 128); __ctype_tolower = current (int32_t, TOLOWER, 128); __ctype32_b = current (uint32_t, CLASS32, 0); __ctype32_toupper = current (uint32_t, TOUPPER32, 0); __ctype32_tolower = current (uint32_t, TOLOWER32, 0); #endif }
/* * Initialize RPC multi-threaded operation */ static void rpc_thread_multi (void) { __libc_tsd_set (RPC_VARS, &__libc_tsd_RPC_VARS_mem); }