/* * Copyright 2013-2015, Michael Ellerman, IBM Corp. * Licensed under GPLv2. */ #define _GNU_SOURCE /* For CPU_ZERO etc. */ #include #include #include #include #include #include #include #include #include #include #include "utils.h" static char auxv[4096]; void *get_auxv_entry(int type) { ElfW(auxv_t) *p; void *result; ssize_t num; int fd; fd = open("/proc/self/auxv", O_RDONLY); if (fd == -1) { perror("open"); return NULL; } result = NULL; num = read(fd, auxv, sizeof(auxv)); if (num < 0) { perror("read"); goto out; } if (num > sizeof(auxv)) { printf("Overflowed auxv buffer\n"); goto out; } p = (ElfW(auxv_t) *)auxv; while (p->a_type != AT_NULL) { if (p->a_type == type) { result = (void *)p->a_un.a_val; break; } p++; } out: close(fd); return result; } int pick_online_cpu(void) { int ncpus, cpu = -1; cpu_set_t *mask; size_t size; ncpus = get_nprocs_conf(); size = CPU_ALLOC_SIZE(ncpus); mask = CPU_ALLOC(ncpus); if (!mask) { perror("malloc"); return -1; } CPU_ZERO_S(size, mask); if (sched_getaffinity(0, size, mask)) { perror("sched_getaffinity"); goto done; } /* We prefer a primary thread, but skip 0 */ for (cpu = 8; cpu < ncpus; cpu += 8) if (CPU_ISSET_S(cpu, size, mask)) goto done; /* Search for anything, but in reverse */ for (cpu = ncpus - 1; cpu >= 0; cpu--) if (CPU_ISSET_S(cpu, size, mask)) goto done; printf("No cpus in affinity mask?!\n"); done: CPU_FREE(mask); return cpu; }