
The THREAD type was not used everywhere and pthread_t was used instead. The thread creation function also returned the address of a stack allocated value which isn't guaranteed to be usable.
252 lines
5.8 KiB
C
252 lines
5.8 KiB
C
/*
|
|
* This file is distributed as part of MaxScale. It 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,
|
|
* version 2.
|
|
*
|
|
* 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.
|
|
*
|
|
* Copyright MariaDB Corporation Ab 2014
|
|
*/
|
|
|
|
/**
|
|
*
|
|
* @verbatim
|
|
* Revision History
|
|
*
|
|
* Date Who Description
|
|
* 18/08-2014 Mark Riddoch Initial implementation
|
|
*
|
|
* @endverbatim
|
|
*/
|
|
|
|
// To ensure that ss_info_assert asserts also when builing in non-debug mode.
|
|
#if !defined(SS_DEBUG)
|
|
#define SS_DEBUG
|
|
#endif
|
|
#if defined(NDEBUG)
|
|
#undef NDEBUG
|
|
#endif
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
|
|
#include <spinlock.h>
|
|
#include <thread.h>
|
|
|
|
|
|
/**
|
|
* test1 spinlock_acquire_nowait tests
|
|
*
|
|
* Test that spinlock_acquire_nowait returns false if the spinlock
|
|
* is already taken.
|
|
*
|
|
* Test that spinlock_acquire_nowait returns true if the spinlock
|
|
* is not taken.
|
|
*
|
|
* Test that spinlock_acquire_nowait does hold the spinlock.
|
|
*/
|
|
static int
|
|
test1()
|
|
{
|
|
SPINLOCK lck;
|
|
|
|
spinlock_init(&lck);
|
|
spinlock_acquire(&lck);
|
|
if (spinlock_acquire_nowait(&lck))
|
|
{
|
|
fprintf(stderr, "spinlock_acquire_nowait: test 1.1 failed.\n");
|
|
return 1;
|
|
}
|
|
spinlock_release(&lck);
|
|
if (!spinlock_acquire_nowait(&lck))
|
|
{
|
|
fprintf(stderr, "spinlock_acquire_nowait: test 1.2 failed.\n");
|
|
return 1;
|
|
}
|
|
if (spinlock_acquire_nowait(&lck))
|
|
{
|
|
fprintf(stderr, "spinlock_acquire_nowait: test 1.3 failed.\n");
|
|
return 1;
|
|
}
|
|
spinlock_release(&lck);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int acquire_time;
|
|
|
|
static void
|
|
test2_helper(void *data)
|
|
{
|
|
SPINLOCK *lck = (SPINLOCK *)data;
|
|
unsigned long t1 = time(0);
|
|
|
|
spinlock_acquire(lck);
|
|
acquire_time = time(0) - t1;
|
|
spinlock_release(lck);
|
|
return;
|
|
}
|
|
|
|
/**
|
|
* test2 spinlock_acquire tests
|
|
*
|
|
* Check that spinlock correctly blocks another thread whilst the spinlock
|
|
* is held.
|
|
*
|
|
* Take out a lock.
|
|
* Start a second thread to take the same lock
|
|
* sleep for 10 seconds
|
|
* release lock
|
|
* verify that second thread took at least 8 seconds to obtain the lock
|
|
*/
|
|
static int
|
|
test2()
|
|
{
|
|
SPINLOCK lck;
|
|
THREAD handle;
|
|
struct timespec sleeptime;
|
|
|
|
sleeptime.tv_sec = 10;
|
|
sleeptime.tv_nsec = 0;
|
|
|
|
acquire_time = 0;
|
|
spinlock_init(&lck);
|
|
spinlock_acquire(&lck);
|
|
thread_start(&handle, test2_helper, (void *)&lck);
|
|
nanosleep(&sleeptime, NULL);
|
|
spinlock_release(&lck);
|
|
thread_wait(handle);
|
|
|
|
if (acquire_time < 8)
|
|
{
|
|
fprintf(stderr, "spinlock: test 2 failed.\n");
|
|
return 1;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* test3 spinlock_acquire tests process bound threads
|
|
*
|
|
* Check that spinlock correctly blocks all other threads whilst the spinlock
|
|
* is held.
|
|
*
|
|
* Start multiple threads that obtain spinlock and run process bound
|
|
*/
|
|
#define THREADS 5
|
|
#define ITERATIONS 50000
|
|
#define PROCESS_LOOP 10000
|
|
#define SECONDS 15
|
|
#define NANOTIME 100000
|
|
|
|
static int times_run, failures;
|
|
static volatile int active;
|
|
static int threadrun[THREADS];
|
|
static int nowait[THREADS];
|
|
static SPINLOCK lck;
|
|
static void
|
|
test3_helper(void *data)
|
|
{
|
|
// SPINLOCK *lck = (SPINLOCK *)data;
|
|
int i;
|
|
int n = *(int *)data;
|
|
time_t rawtime;
|
|
|
|
#if defined(ADD_SOME_NANOSLEEP)
|
|
struct timespec sleeptime;
|
|
|
|
sleeptime.tv_sec = 0;
|
|
sleeptime.tv_nsec = 1;
|
|
#endif
|
|
|
|
while (1) {
|
|
if (spinlock_acquire_nowait(&lck)) {
|
|
nowait[n]++;
|
|
}
|
|
else {
|
|
spinlock_acquire(&lck);
|
|
}
|
|
if (times_run++ > ITERATIONS) {
|
|
break;
|
|
}
|
|
threadrun[n]++;
|
|
/*
|
|
if (99 == (times_run % 100)) {
|
|
time ( &rawtime );
|
|
fprintf(stderr, "%s Done %d iterations of test, in thread %d.\n", asctime (localtime ( &rawtime )), times_run, n);
|
|
}
|
|
*/
|
|
if (0 != active) {
|
|
fprintf(stderr, "spinlock: test 3 failed with active non-zero after lock obtained.\n");
|
|
failures++;
|
|
}
|
|
else {
|
|
active = 1;
|
|
for (i=0; i<PROCESS_LOOP; i++);
|
|
}
|
|
active = 0;
|
|
spinlock_release(&lck);
|
|
for (i=0; i<(4*PROCESS_LOOP); i++);
|
|
#if defined(ADD_SOME_NANOSLEEP)
|
|
nanosleep(&sleeptime, NULL);
|
|
#endif
|
|
}
|
|
spinlock_release(&lck);
|
|
}
|
|
|
|
static int
|
|
test3()
|
|
{
|
|
// SPINLOCK lck;
|
|
THREAD handle[THREADS];
|
|
int i;
|
|
int tnum[THREADS];
|
|
time_t rawtime;
|
|
|
|
times_run = 0;
|
|
active = 0;
|
|
failures = 0;
|
|
spinlock_init(&lck);
|
|
time ( &rawtime );
|
|
fprintf(stderr, "%s Starting %d threads.\n", asctime (localtime ( &rawtime )), THREADS);
|
|
for (i = 0; i<THREADS; i++) {
|
|
threadrun[i] = 0;
|
|
tnum[i] = i;
|
|
thread_start(&handle[i], test3_helper, &tnum[i]);
|
|
}
|
|
for (i = 0; i<THREADS; i++) {
|
|
fprintf(stderr, "spinlock_test 3 thread %d ran %d times, no wait %d times before waits.\n", i, threadrun[i], nowait[i]);
|
|
}
|
|
for (i = 0; i<THREADS; i++) {
|
|
time ( &rawtime );
|
|
fprintf(stderr, "%s spinlock_test 3 finished sleeps, about to wait for thread %d.\n", asctime (localtime ( &rawtime )), i);
|
|
thread_wait(handle[i]);
|
|
}
|
|
for (i = 0; i<THREADS; i++) {
|
|
fprintf(stderr, "spinlock_test 3 thread %d ran %d times, no wait %d times.\n", i, threadrun[i], nowait[i]);
|
|
}
|
|
time ( &rawtime );
|
|
fprintf(stderr, "%s spinlock_test 3 completed, %d failures.\n", asctime (localtime ( &rawtime )), failures);
|
|
return 0 == failures ? 0: 1;
|
|
}
|
|
|
|
int main(int argc, char **argv)
|
|
{
|
|
int result = 0;
|
|
|
|
result += test1();
|
|
result += test2();
|
|
result += test3();
|
|
|
|
exit(result);
|
|
}
|
|
|