259 lines
		
	
	
		
			5.9 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			259 lines
		
	
	
		
			5.9 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
/*
 | 
						|
 * Copyright (c) 2016 MariaDB Corporation Ab
 | 
						|
 *
 | 
						|
 * Use of this software is governed by the Business Source License included
 | 
						|
 * in the LICENSE.TXT file and at www.mariadb.com/bsl11.
 | 
						|
 *
 | 
						|
 * Change Date: 2022-01-01
 | 
						|
 *
 | 
						|
 * On the date above, in accordance with the Business Source License, use
 | 
						|
 * of this software will be governed by version 2 or later of the General
 | 
						|
 * Public License.
 | 
						|
 */
 | 
						|
 | 
						|
/**
 | 
						|
 *
 | 
						|
 * @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 <maxscale/spinlock.h>
 | 
						|
#include <maxscale/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, 0);
 | 
						|
    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], 0);
 | 
						|
    }
 | 
						|
    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);
 | 
						|
}
 | 
						|
 |