Files
sysbench/src/sb_thread.c
2018-04-21 15:19:36 +03:00

200 lines
4.6 KiB
C

/*
Copyright (C) 2016-2018 Alexey Kopytov <akopytov@gmail.com>
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 2 of the License, or
(at your option) any later version.
This program 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 General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
*/
/*
Wrappers around pthread_create() and friends to provide necessary
(de-)initialization.
*/
#ifdef HAVE_CONFIG_H
# include "config.h"
#endif
#ifdef HAVE_PTHREAD_H
# include <pthread.h>
#endif
#ifndef HAVE_PTHREAD_CANCEL
#include <signal.h>
#endif
#include "sb_thread.h"
#include "sb_rand.h"
#include "sb_logger.h"
#include "sysbench.h"
#include "sb_ck_pr.h"
pthread_attr_t sb_thread_attr;
/* Thread descriptors */
static sb_thread_ctxt_t *threads;
/* Stack size for each thread */
static int thread_stack_size;
int sb_thread_init(void)
{
thread_stack_size = sb_get_value_size("thread-stack-size");
if (thread_stack_size <= 0)
{
log_text(LOG_FATAL, "Invalid value for thread-stack-size: %d.\n", thread_stack_size);
return 1;
}
/* initialize attr */
pthread_attr_init(&sb_thread_attr);
#ifdef PTHREAD_SCOPE_SYSTEM
pthread_attr_setscope(&sb_thread_attr,PTHREAD_SCOPE_SYSTEM);
#endif
pthread_attr_setstacksize(&sb_thread_attr, thread_stack_size);
#ifdef HAVE_THR_SETCONCURRENCY
/* Set thread concurrency (required on Solaris) */
thr_setconcurrency(sb_globals.threads);
#endif
threads = malloc(sb_globals.threads * sizeof(sb_thread_ctxt_t));
if (threads == NULL)
{
log_text(LOG_FATAL, "Memory allocation failure.\n");
return EXIT_FAILURE;
}
return EXIT_SUCCESS;
}
void sb_thread_done(void)
{
if (threads != NULL)
free(threads);
}
#ifndef HAVE_PTHREAD_CANCEL
#define PTHREAD_CANCELED ((void *) -1)
struct sb_thread_proxy {
void *(*start_routine) (void *);
void *arg;
};
static int thread_cancel_signal = SIGUSR1;
static void thread_cancel_handler(int sig)
{
if (sig == thread_cancel_signal)
pthread_exit(PTHREAD_CANCELED);
}
static int install_thread_signal_handler(void) {
struct sigaction action;
memset(&action, 0, sizeof(action));
sigemptyset(&action.sa_mask);
action.sa_flags = 0;
action.sa_handler = thread_cancel_handler;
return sigaction(thread_cancel_signal, &action, NULL);
}
static void* thread_start_routine_proxy(void *arg) {
struct sb_thread_proxy *proxy = arg;
void *(*start_routine) (void *) = proxy->start_routine;
void *real_arg = proxy->arg;
free(proxy);
install_thread_signal_handler();
return start_routine(real_arg);
}
#endif
int sb_thread_create(pthread_t *thread, const pthread_attr_t *attr,
void *(*start_routine) (void *), void *arg)
{
#ifdef HAVE_PTHREAD_CANCEL
return pthread_create(thread, attr, start_routine, arg);
#else
struct sb_thread_proxy *proxy = malloc(sizeof(struct sb_thread_proxy));
if (!proxy)
{
return EXIT_FAILURE;
}
proxy->start_routine = start_routine;
proxy->arg = arg;
int rv = pthread_create(thread, attr, thread_start_routine_proxy, proxy);
if (rv)
{
free(proxy);
}
return rv;
#endif
}
int sb_thread_join(pthread_t thread, void **retval)
{
return pthread_join(thread, retval);
}
int sb_thread_cancel(pthread_t thread)
{
#ifdef HAVE_PTHREAD_CANCEL
return pthread_cancel(thread);
#else
return pthread_kill(thread, thread_cancel_signal);
#endif
}
int sb_thread_create_workers(void *(*worker_routine)(void*))
{
unsigned int i;
log_text(LOG_NOTICE, "Initializing worker threads...\n");
for(i = 0; i < sb_globals.threads; i++)
{
threads[i].id = i;
}
for(i = 0; i < sb_globals.threads; i++)
{
int err;
if ((err = sb_thread_create(&(threads[i].thread), &sb_thread_attr,
worker_routine, (void*)(threads + i))) != 0)
{
log_errno(LOG_FATAL, "sb_thread_create() for thread #%d failed.", i);
return EXIT_FAILURE;
}
}
return EXIT_SUCCESS;
}
int sb_thread_join_workers(void)
{
for(unsigned i = 0; i < sb_globals.threads; i++)
{
int err;
if((err = sb_thread_join(threads[i].thread, NULL)) != 0)
log_errno(LOG_FATAL, "sb_thread_join() for thread #%d failed.", i);
ck_pr_dec_uint(&sb_globals.threads_running);
}
return EXIT_SUCCESS;
}