/*
* Copyright 2008 Google Inc. All Rights Reserved.
* Author: md@google.com (Michael Davidson)
*/
#define _GNU_SOURCE
#include <stdio.h>
#include <string.h>
#include <errno.h>
#include <sched.h>
#include <pthread.h>
#include "logging.h"
#include "threads.h"
#define MAX_CPUS CPU_SETSIZE
#define MAX_THREADS MAX_CPUS
typedef struct thread {
pthread_t thread;
cpu_set_t cpus;
thread_func_t func;
void *arg;
} thread_t;
static thread_t threads[MAX_THREADS];
static int num_threads;
/*
* Helper function to run a thread on a specific set of CPUs.
*/
static void *run_thread(void *arg)
{
thread_t *thread = arg;
void *result;
if (sched_setaffinity(0, sizeof thread->cpus, &thread->cpus) < 0)
WARN(errno, "sched_setaffinity() failed");
result = thread->func(thread->arg);
return result;
}
/*
* Create a set of threads each of which is bound to one of
* the CPUs specified by cpus.
* Returns the number of threads created.
*/
int create_per_cpu_threads(cpu_set_t *cpus, thread_func_t func, void *arg)
{
int cpu;
for (cpu = 0; cpu < MAX_CPUS; cpu++) {
int err;
thread_t *thread;
if (!CPU_ISSET(cpu, cpus))
continue;
if (num_threads >= MAX_THREADS)
break;
thread = &threads[num_threads++];
thread->func = func;
thread->arg = arg;
CPU_ZERO(&thread->cpus);
CPU_SET(cpu, &thread->cpus);
err = pthread_create(&thread->thread, NULL, run_thread, thread);
if (err) {
WARN(err, "pthread_create() failed");
--num_threads;
break;
}
}
return num_threads;
}
/*
* Create nthreads threads.
* Returns the number of threads created.
*/
int create_threads(int nthreads, thread_func_t func, void *arg)
{
if (nthreads > MAX_THREADS)
nthreads = MAX_THREADS;
while (--nthreads >= 0) {
int err;
thread_t *thread;
thread = &threads[num_threads++];
thread->func = func;
thread->arg = arg;
CPU_ZERO(&thread->cpus);
err = pthread_create(&thread->thread, NULL, func, arg);
if (err) {
WARN(err, "pthread_create() failed");
--num_threads;
break;
}
}
return num_threads;
}
/*
* Join with the set of previsouly created threads.
*/
void join_threads(void)
{
while (num_threads > 0)
pthread_join(threads[--num_threads].thread, NULL);
}