196 lines
		
	
	
		
			5.4 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			196 lines
		
	
	
		
			5.4 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
 | |
|  */
 | |
| 
 | |
| #include <stdio.h>
 | |
| #include <stdlib.h>
 | |
| #include <string.h>
 | |
| #include <math.h>
 | |
| #include <time.h>
 | |
| 
 | |
| #include "../../include/hashtable.h"
 | |
| 
 | |
| static void
 | |
| read_lock(HASHTABLE *table)
 | |
| {
 | |
| 	spinlock_acquire(&table->spin);
 | |
| 	while (table->writelock)
 | |
| 	{
 | |
| 		spinlock_release(&table->spin);
 | |
| 		while (table->writelock)
 | |
| 			;
 | |
| 		spinlock_acquire(&table->spin);
 | |
| 	}
 | |
| 	table->n_readers++;
 | |
| 	spinlock_release(&table->spin);
 | |
| }
 | |
| 
 | |
| static void
 | |
| read_unlock(HASHTABLE *table)
 | |
| {
 | |
| 	atomic_add(&table->n_readers, -1);
 | |
| }
 | |
| 
 | |
| static int hfun(void* key);
 | |
| static int cmpfun (void *, void *);
 | |
| 
 | |
| static int hfun(
 | |
|         void* key)
 | |
| {
 | |
|     int *i = (int *)key;
 | |
|     int j = (*i * 23) + 41;
 | |
|     return j;
 | |
|     /*    return *(int *)key;   */
 | |
| }
 | |
| 
 | |
| static int cmpfun(
 | |
|         void* v1,
 | |
|         void* v2)
 | |
| {
 | |
|         int i1;
 | |
|         int i2;
 | |
| 
 | |
|         i1 = *(int *)v1;
 | |
|         i2 = *(int *)v2;
 | |
| 
 | |
|         return (i1 < i2 ? -1 : (i1 > i2 ? 1 : 0));
 | |
| }
 | |
| 
 | |
| static double start;
 | |
| 
 | |
| /**
 | |
|  * 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 bool do_hashtest(
 | |
|         int argelems,
 | |
|         int argsize)
 | |
| {
 | |
|         bool       succp = true;
 | |
|         HASHTABLE* h;
 | |
|         int        nelems;
 | |
|         int        i;
 | |
|         int*       val_arr;
 | |
|         int        hsize;
 | |
|         int        longest;
 | |
|         int*       iter;
 | |
|         
 | |
|         ss_dfprintf(stderr,
 | |
|                     "testhash : creating hash table of size %d, including %d "
 | |
|                     "elements in total, at time %g.",
 | |
|                     argsize,
 | |
|                     argelems,
 | |
|                     (double)clock()-start); 
 | |
|         
 | |
|         val_arr = (int *)malloc(sizeof(void *)*argelems);
 | |
|         
 | |
|         h = hashtable_alloc(argsize, hfun, cmpfun);
 | |
| 
 | |
|         ss_dfprintf(stderr, "\t..done\nAdd %d elements to hash table.", argelems);
 | |
|         
 | |
|         for (i=0; i<argelems; i++) {
 | |
|             val_arr[i] = i;
 | |
|             hashtable_add(h, (void *)&val_arr[i], (void *)&val_arr[i]);
 | |
|         }
 | |
|         if (argelems > 1000) ss_dfprintf(stderr, "\t..done\nOperation took %g", (double)clock()-start);
 | |
|         
 | |
|         ss_dfprintf(stderr, "\t..done\nRead hash table statistics.");
 | |
|         
 | |
|         hashtable_get_stats((void *)h, &hsize, &nelems, &longest);
 | |
| 
 | |
|         ss_dfprintf(stderr, "\t..done\nValidate read values.");
 | |
|         
 | |
|         ss_info_dassert(hsize == (argsize > 0 ? argsize: 1), "Invalid hash size");
 | |
|         ss_info_dassert((nelems == argelems) || (nelems == 0 && argsize == 0),
 | |
|                         "Invalid element count");
 | |
|         ss_info_dassert(longest <= nelems, "Too large longest list value");
 | |
|         if (argelems > 1000) ss_dfprintf(stderr, "\t..done\nOperation took %g", (double)clock()-start);
 | |
| 
 | |
|         ss_dfprintf(stderr, "\t..done\nValidate iterator.");
 | |
|         
 | |
|         HASHITERATOR *iterator = hashtable_iterator(h);
 | |
|         read_lock(h);
 | |
|         for (i=0; i < (argelems+1); i++) {
 | |
|             iter = (int *)hashtable_next(iterator);
 | |
|             if (iter == NULL) break;
 | |
|             if (argelems < 100) ss_dfprintf(stderr, "\nNext item, iter = %d, i = %d", *iter, i);
 | |
|         }
 | |
|         read_unlock(h);
 | |
|         ss_info_dassert((i == argelems) || (i == 0 && argsize == 0), "\nIncorrect number of elements from iterator");
 | |
|         hashtable_iterator_free(iterator);
 | |
|         if (argelems > 1000) ss_dfprintf(stderr, "\t..done\nOperation took %g", (double)clock()-start);
 | |
| 
 | |
|         ss_dfprintf(stderr, "\t\t..done\n\nTest completed successfully.\n\n");
 | |
|         
 | |
|         CHK_HASHTABLE(h);
 | |
|         hashtable_free(h);
 | |
|         
 | |
| return_succp:
 | |
| 		free(val_arr);
 | |
|         return succp;
 | |
| }
 | |
| 
 | |
| /** 
 | |
|  * @node Simple test which creates hashtable and frees it. Size and number of entries
 | |
|  * sre specified by user and passed as arguments.
 | |
|  *
 | |
|  *
 | |
|  * @return 0 if succeed, 1 if failed.
 | |
|  *
 | |
|  * 
 | |
|  * @details (write detailed description here)
 | |
|  *
 | |
|  */
 | |
| int main(void)
 | |
| {
 | |
|         int rc = 1;
 | |
|         start = (double) clock();
 | |
| 
 | |
|         if (!do_hashtest(0, 1))         goto return_rc;
 | |
|         if (!do_hashtest(10, 1))        goto return_rc;
 | |
|         if (!do_hashtest(1000, 10))     goto return_rc;
 | |
|         if (!do_hashtest(10, 0))        goto return_rc;
 | |
|         if (!do_hashtest(10, -5))       goto return_rc;
 | |
|         if (!do_hashtest(1500, 17))     goto return_rc;
 | |
|         if (!do_hashtest(1, 1))         goto return_rc;
 | |
|         if (!do_hashtest(10000, 133))   goto return_rc;
 | |
|         if (!do_hashtest(1000, 1000))   goto return_rc;
 | |
|         if (!do_hashtest(1000, 100000)) goto return_rc;
 | |
|         
 | |
|         rc = 0;
 | |
| return_rc:
 | |
|         return rc;
 | |
| }
 | 
