mirror of
https://sourceware.org/git/glibc.git
synced 2025-01-11 20:00:07 +00:00
Linux: Avoid calling malloc indirectly from __get_nprocs
malloc initialization depends on __get_nprocs, so using scratch buffers in __get_nprocs may result in infinite recursion. Reviewed-by: Carlos O'Donell <carlos@redhat.com> Reviewed-by: Siddhesh Poyarekar <siddhesh@sourceware.org>
This commit is contained in:
parent
38f58041ba
commit
eb68d7d23c
@ -17,42 +17,73 @@
|
|||||||
License along with the GNU C Library; if not, see
|
License along with the GNU C Library; if not, see
|
||||||
<https://www.gnu.org/licenses/>. */
|
<https://www.gnu.org/licenses/>. */
|
||||||
|
|
||||||
|
#include <array_length.h>
|
||||||
#include <dirent.h>
|
#include <dirent.h>
|
||||||
|
#include <errno.h>
|
||||||
|
#include <ldsodefs.h>
|
||||||
|
#include <limits.h>
|
||||||
#include <not-cancel.h>
|
#include <not-cancel.h>
|
||||||
#include <scratch_buffer.h>
|
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
#include <stdio_ext.h>
|
#include <stdio_ext.h>
|
||||||
|
#include <sys/mman.h>
|
||||||
#include <sys/sysinfo.h>
|
#include <sys/sysinfo.h>
|
||||||
#include <sysdep.h>
|
#include <sysdep.h>
|
||||||
|
|
||||||
|
/* Compute the population count of the entire array. */
|
||||||
|
static int
|
||||||
|
__get_nprocs_count (const unsigned long int *array, size_t length)
|
||||||
|
{
|
||||||
|
int count = 0;
|
||||||
|
for (size_t i = 0; i < length; ++i)
|
||||||
|
if (__builtin_add_overflow (count, __builtin_popcountl (array[i]),
|
||||||
|
&count))
|
||||||
|
return INT_MAX;
|
||||||
|
return count;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* __get_nprocs with a large buffer. */
|
||||||
|
static int
|
||||||
|
__get_nprocs_large (void)
|
||||||
|
{
|
||||||
|
/* This code cannot use scratch_buffer because it is used during
|
||||||
|
malloc initialization. */
|
||||||
|
size_t pagesize = GLRO (dl_pagesize);
|
||||||
|
unsigned long int *page = __mmap (0, pagesize, PROT_READ | PROT_WRITE,
|
||||||
|
MAP_ANONYMOUS | MAP_PRIVATE, -1, 0);
|
||||||
|
if (page == MAP_FAILED)
|
||||||
|
return 2;
|
||||||
|
int r = INTERNAL_SYSCALL_CALL (sched_getaffinity, 0, pagesize, page);
|
||||||
|
int count;
|
||||||
|
if (r > 0)
|
||||||
|
count = __get_nprocs_count (page, pagesize / sizeof (unsigned long int));
|
||||||
|
else if (r == -EINVAL)
|
||||||
|
/* One page is still not enough to store the bits. A more-or-less
|
||||||
|
arbitrary value. This assumes t hat such large systems never
|
||||||
|
happen in practice. */
|
||||||
|
count = GLRO (dl_pagesize) * CHAR_BIT;
|
||||||
|
else
|
||||||
|
count = 2;
|
||||||
|
__munmap (page, GLRO (dl_pagesize));
|
||||||
|
return count;
|
||||||
|
}
|
||||||
|
|
||||||
int
|
int
|
||||||
__get_nprocs (void)
|
__get_nprocs (void)
|
||||||
{
|
{
|
||||||
struct scratch_buffer set;
|
/* Fast path for most systems. The kernel expects a buffer size
|
||||||
scratch_buffer_init (&set);
|
that is a multiple of 8. */
|
||||||
|
unsigned long int small_buffer[1024 / CHAR_BIT / sizeof (unsigned long int)];
|
||||||
int r;
|
int r = INTERNAL_SYSCALL_CALL (sched_getaffinity, 0,
|
||||||
while (true)
|
sizeof (small_buffer), small_buffer);
|
||||||
{
|
if (r > 0)
|
||||||
/* The possible error are EFAULT for an invalid buffer or ESRCH for
|
return __get_nprocs_count (small_buffer, r / sizeof (unsigned long int));
|
||||||
invalid pid, none could happen. */
|
else if (r == -EINVAL)
|
||||||
r = INTERNAL_SYSCALL_CALL (sched_getaffinity, 0, set.length,
|
/* The kernel requests a larger buffer to store the data. */
|
||||||
set.data);
|
return __get_nprocs_large ();
|
||||||
if (r > 0)
|
else
|
||||||
break;
|
/* Some other error. 2 is conservative (not a uniprocessor
|
||||||
|
system, so atomics are needed). */
|
||||||
if (!scratch_buffer_grow (&set))
|
return 2;
|
||||||
/* Default to an SMP system in case we cannot obtain an accurate
|
|
||||||
number. */
|
|
||||||
return 2;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* The scratch_buffer is aligned to max_align_t. */
|
|
||||||
r = __sched_cpucount (r, (const cpu_set_t *) set.data);
|
|
||||||
|
|
||||||
scratch_buffer_free (&set);
|
|
||||||
|
|
||||||
return r;
|
|
||||||
}
|
}
|
||||||
libc_hidden_def (__get_nprocs)
|
libc_hidden_def (__get_nprocs)
|
||||||
weak_alias (__get_nprocs, get_nprocs)
|
weak_alias (__get_nprocs, get_nprocs)
|
||||||
|
Loading…
Reference in New Issue
Block a user