blob: 5d8436b5ad497c6e5566ee2557b3c63eb35192f9 [file] [log] [blame]
/* Definition of `errno' variable. Canonical version.
Copyright (C) 2002, 2004 Free Software Foundation, Inc.
This file is part of the GNU C Library.
The GNU C Library is free software; you can redistribute it and/or
modify it under the terms of the GNU Lesser General Public
License as published by the Free Software Foundation; either
version 2.1 of the License, or (at your option) any later version.
The GNU C Library is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
Lesser General Public License for more details.
You should have received a copy of the GNU Lesser General Public
License along with the GNU C Library; if not, write to the Free
Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA
02111-1307 USA. */
#include <errno.h>
#include <tls.h>
#include <dl-sysdep.h>
#undef errno
#if RTLD_PRIVATE_ERRNO
/* Code compiled for rtld has errno #defined to rtld_errno. */
int rtld_errno attribute_hidden;
#define errno rtld_errno
char __errstr_tls[MAX_ERRSTR_LEN] = {0};
#else
__thread int errno;
extern __thread int __libc_errno __attribute__ ((alias ("errno")))
attribute_hidden;
__thread char __errstr_tls[MAX_ERRSTR_LEN] = {0};
#endif
/* this is the glibc default, exported so uthread.c can use them */
int *__errno_location_tls(void)
{
return &errno;
}
char *__errstr_location_tls(void)
{
return __errstr_tls;
}
/* These func ptrs will be used to access errno_loc and errstr_loc. can be
* overriden at runtime (uthread code will do this) */
int *(*ros_errno_loc)(void) = __errno_location_tls;
char *(*ros_errstr_loc)(void) = __errstr_location_tls;
/* errno calls this (and derefs the result). it's also called internally */
int *__errno_location(void)
{
return ros_errno_loc();
}
/* libc doesn't really know about errstr, but we'll use it in syscall.c */
char *errstr(void)
{
return ros_errstr_loc();
}
/* Using this instead of strcpy/memcpy to avoid dependencies. */
static void simple_strcpy(char *dst, char *src, size_t size)
{
for (size_t i = 0; (i < size) && *src; i++)
*dst++ = *src++;
}
/* Helpers to save and restore errno and errstr into temporary locations. Make
* sure str is MAX_ERRSTR_LEN bytes long. */
void save_err(int *no, char *str)
{
*no = errno;
simple_strcpy(str, errstr(), MAX_ERRSTR_LEN);
}
void restore_err(int *no, char *str)
{
errno = *no;
simple_strcpy(errstr(), str, MAX_ERRSTR_LEN);
}
/* Don't try to hidden_data_def the function pointers. Won't allow us to
* switch, or otherwise track the right location. */
libc_hidden_def(__errno_location_tls)
libc_hidden_def(__errstr_location_tls)
libc_hidden_def(__errno_location)
libc_hidden_def(errstr)