/*********************************************************************
 *
 * Copyright (C) 2001-2002,  Simon Kagstrom
 *
 * Filename:      hash_test.c
 * Description:   A sample program to demonstrate the usage of the
 *                generic hash table.
 *
 * This program is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Library 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 Library General Public
 * License along with this program; if not, write to the Free Software
 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
 * 02111-1307, USA.
 *
 * $Id: hash_test.c 8461 2006-06-04 08:13:06Z ska $
 *
 ********************************************************************/

#include <stdlib.h> /* malloc */
#include <time.h>   /* time */
#include <errno.h>  /* errno */
#include <stdio.h>  /* perror etc */

#include "ght_hash_table.h" /* Include the generic hash table */

void fn(void *data, const void *key)
{
  free(data);
}

int main(int argc, char *argv[])
{
  ght_hash_table_t *p_table;
  ght_hash_table_t *p_table2;
  ght_iterator_t iterator;
  int i_loops = 100000;
  int i_removed = 0;
  const void *p_key;
  void *p_e;
  int i;

  /* Initialise the random seed */
  srand(1000);

  if (argc > 1)
    {
      /* Create two hash table with move to front heuristics, specifying the size for one. Both use automatic rehashing  */
      p_table = ght_create(atoi(argv[1]));
      ght_set_rehash(p_table, TRUE);
      ght_set_heuristics(p_table, GHT_HEURISTICS_MOVE_TO_FRONT);

      if (argc > 2)
	i_loops = atoi(argv[2]);

      p_table2 = ght_create(5000);

      ght_set_rehash(p_table2, TRUE);
      ght_set_heuristics(p_table2, GHT_HEURISTICS_MOVE_TO_FRONT);
    }
  else
    {
      /* Create two hash table with transpose heuristics, size
	 5000. Hash function is one_at_a_time_hash (default with NULL).
      */
      p_table = ght_create(5000);
      ght_set_rehash(p_table, FALSE);
      ght_set_heuristics(p_table, GHT_HEURISTICS_TRANSPOSE);

      p_table2 = ght_create(5000);
      ght_set_rehash(p_table2, FALSE);
      ght_set_heuristics(p_table2, GHT_HEURISTICS_TRANSPOSE);
    }

  /* This is a bounded bucket implementation */
  ght_set_bounded_buckets(p_table, 3, fn);

  /* Enter lots of stuff to the tables. */
  for (i=0; i<i_loops; i++)
    {
      int *p_entry_data;

      /* Allocate data for the entry as well (in this case an integer,
	 more often some structure).
      */
      if ( !(p_entry_data = (int*)malloc(sizeof(int))) )
	{
	  perror("malloc");
	  return -errno;
	}

      /* Set the data for the key and the entry */
      *p_entry_data = i_loops-i;

      /* Insert into table1 or table2 at random */
      if (rand()%2 == 1)
	{
	  if (ght_insert(p_table2,
			 p_entry_data,
			 sizeof(int), &i) < 0)
	    fprintf(stderr, "ERROR: Could not insert into table\n");
	}
      else
	{
	  if (ght_insert(p_table,
			 p_entry_data,
			 sizeof(int), &i) < 0)
	    fprintf(stderr, "ERROR: Could not insert into table\n");
	}

      /* If we didn't use automatic rehashing here, we could do a
       * manual rehash by using something like:
       *
       * if (p_table->i_items > p_table->i_size*2)
       *   ght_he_rehash(p_he);
       */
    }
  printf("Inserted %d elements.\n", i_loops);

  /* Get the same stuff FROM the table */
  for (i=0; i<i_loops; i++)
    {
      int *p_he;
      int i_key_data;

      /* Here we create a key for temporary usage (with one of the
	 numbers used previously)
      */
      i_key_data = rand()%i_loops;

      /* Get some element from one of the hash tables */
      if (rand()%2 == 1)
	p_he = (int*)ght_get(p_table,
			     sizeof(int), &i_key_data);
      else
	p_he = (int*)ght_get(p_table2,
			     sizeof(int), &i_key_data);

      /* Check if anything was found (might be in the other table) */
      if (p_he)
	{
	  /* Check if it's correct */
	  if (*p_he != i_loops-i_key_data)
	    {
	      printf("Found %d for key %d. WRONG!\n", *p_he, i_key_data);
	      return -1;
	    }
	}
    }

#ifdef USE_PROFILING
  printf("Hash table 1:\n");
  ght_print(p_table); /* Do not rely on this function, only available when built without optimization */
  printf("Hash table 2:\n");
  ght_print(p_table2);
  printf("\n");
#endif
  printf("Fetched %d elements. All tested OK\n", i_loops);

  /* Remove the same stuff FROM the table */
  for (i=0; i<i_loops; i++)
    {
      int *p_he;
      int i_key_data;

      /* Again, use a temporary (non-allocated) key */
      i_key_data = rand()%i_loops;

      /* Remove an element from one of the hash tables */
      if (rand()%2 == 1)
	p_he = (int*)ght_remove(p_table,
				sizeof(int), &i_key_data);
      else
	p_he = (int*)ght_remove(p_table2,
				sizeof(int), &i_key_data);

      /* Check if something could be removed */
      if (p_he)
	{
	  if (*p_he != i_loops-i_key_data)
	    {
	      printf("Found %d for key %d. WRONG!\n", *p_he, i_key_data);
	      return -1;
	    }

	  free(p_he);
	  p_he = NULL;
	  i_removed++;
	}
    }
  printf("Removed %d elements. All tested OK\n", i_removed);

#ifdef USE_PROFILING
  printf("Hash table 1:\n");
  ght_print(p_table);
  printf("Hash table 2:\n");
  ght_print(p_table2);
  printf("\n");
#endif


  /* Finally, remove the hash tables and all data within them (this
     will take some time if the tables are large). */
  printf("Finalizing the hash tables (remove all data)...\n");

  /* Free the entry data in the tables */
  for(p_e = ght_first(p_table, &iterator, &p_key); p_e; p_e = ght_next(p_table, &iterator, &p_key))
    {
      void *p;

      /* Remove the current entry from the table as well (will be done
       * in ght_finalize below so this is not strictly necessary).
       */
      p = ght_remove(p_table, sizeof(int), p_key);
      if (!p)
	printf("Removing the current iterated entry failed! This is a BUG\n");

      free(p_e);
    }
  for(p_e = ght_first(p_table2, &iterator, &p_key); p_e; p_e = ght_next(p_table2, &iterator, &p_key))
    {
      free(p_e);
    }

  /* Free the tables */
  ght_finalize(p_table);
  ght_finalize(p_table2);
  printf("Finalizing done\n");

  return 0;
}


syntax highlighted by Code2HTML, v. 0.9.1