diff options
author | ian <ian@138bc75d-0d04-0410-961f-82ee72b054a4> | 2012-08-10 06:08:11 +0000 |
---|---|---|
committer | ian <ian@138bc75d-0d04-0410-961f-82ee72b054a4> | 2012-08-10 06:08:11 +0000 |
commit | 773630dc6f83af190a690e1bf2e7f396cf0148d2 (patch) | |
tree | aa24175caf108038c2ecde81b10654a851f55416 /libgo | |
parent | 99274008f535c39a12d8d1132c6572ca36ed76bd (diff) | |
download | gcc-773630dc6f83af190a690e1bf2e7f396cf0148d2.tar.gz |
runtime: use sched_getaffinity for runtime.NumCPU() on Linux
Fixes Go issue 3921 for gccgo.
From Shenghou Ma.
git-svn-id: svn+ssh://gcc.gnu.org/svn/gcc/trunk@190282 138bc75d-0d04-0410-961f-82ee72b054a4
Diffstat (limited to 'libgo')
-rw-r--r-- | libgo/runtime/getncpu-linux.c | 53 |
1 files changed, 21 insertions, 32 deletions
diff --git a/libgo/runtime/getncpu-linux.c b/libgo/runtime/getncpu-linux.c index 05bd4a37e52..0122b77c9ff 100644 --- a/libgo/runtime/getncpu-linux.c +++ b/libgo/runtime/getncpu-linux.c @@ -2,46 +2,35 @@ // Use of this source code is governed by a BSD-style // license that can be found in the LICENSE file. -#include <string.h> -#include <sys/types.h> -#include <fcntl.h> -#include <unistd.h> +#include <features.h> +#include <sched.h> -#include "runtime.h" -#include "defs.h" +// CPU_COUNT is only provided by glibc 2.6 or higher +#if !defined(__GLIBC_PREREQ) || !__GLIBC_PREREQ(2, 6) +#define CPU_COUNT(set) _CPU_COUNT((unsigned int *)(set), sizeof(*(set))/sizeof(unsigned int)) +static int _CPU_COUNT(unsigned int *set, size_t len) { + int cnt; -#ifndef O_CLOEXEC -#define O_CLOEXEC 0 + cnt = 0; + while (len--) + cnt += __builtin_popcount(*set++); + return cnt; +} #endif +#include "runtime.h" +#include "defs.h" + int32 getproccount(void) { - int32 fd, rd, cnt, cpustrlen; - const char *cpustr; - const byte *pos; - byte *bufpos; - byte buf[256]; + cpu_set_t set; + int32 r, cnt; - fd = open("/proc/stat", O_RDONLY|O_CLOEXEC, 0); - if(fd == -1) - return 1; cnt = 0; - bufpos = buf; - cpustr = "\ncpu"; - cpustrlen = strlen(cpustr); - for(;;) { - rd = read(fd, bufpos, sizeof(buf)-cpustrlen); - if(rd == -1) - break; - bufpos[rd] = 0; - for(pos=buf; (pos=(const byte*)strstr((const char*)pos, cpustr)) != nil; cnt++, pos++) { - } - if(rd < cpustrlen) - break; - memmove(buf, bufpos+rd-cpustrlen+1, cpustrlen-1); - bufpos = buf+cpustrlen-1; - } - close(fd); + r = sched_getaffinity(0, sizeof(set), &set); + if(r == 0) + cnt += CPU_COUNT(&set); + return cnt ? cnt : 1; } |