/** * @file db_debug.c * Debug routines for libdb * * @remark Copyright 2002 OProfile authors * @remark Read the file COPYING * * @author Philippe Elie */ #include #include #include #include "odb.h" static int check_circular_list(odb_data_t const * data) { odb_node_nr_t pos; int do_abort = 0; unsigned char * bitmap = malloc(data->descr->current_size); memset(bitmap, '\0', data->descr->current_size); for (pos = 0 ; pos < data->descr->size * BUCKET_FACTOR ; ++pos) { odb_index_t index = data->hash_base[pos]; if (index && !do_abort) { while (index) { if (bitmap[index]) do_abort = 1; bitmap[index] = 1; index = data->node_base[index].next; } } if (do_abort) { printf("circular list detected size: %d\n", data->descr->current_size); memset(bitmap, '\0', data->descr->current_size); index = data->hash_base[pos]; while (index) { printf("%d ", index); if (bitmap[index]) exit(1); bitmap[index] = 1; index = data->node_base[index].next; } } /* purely an optimization: intead of memset the map reset only * the needed part: not my use to optimize test but here the * test was so slow it was useless */ index = data->hash_base[pos]; while (index) { bitmap[index] = 1; index = data->node_base[index].next; } } free(bitmap); return do_abort; } static int check_redundant_key(odb_data_t const * data, odb_key_t max) { odb_node_nr_t pos; unsigned char * bitmap = malloc(max + 1); memset(bitmap, '\0', max + 1); for (pos = 1 ; pos < data->descr->current_size ; ++pos) { if (bitmap[data->node_base[pos].key]) { printf("redundant key found %lld\n", (unsigned long long)data->node_base[pos].key); return 1; } bitmap[data->node_base[pos].key] = 1; } free(bitmap); return 0; } int odb_check_hash(odb_t const * odb) { odb_node_nr_t pos; odb_node_nr_t nr_node = 0; odb_node_nr_t nr_node_out_of_bound = 0; int ret = 0; odb_key_t max = 0; odb_data_t * data = odb->data; for (pos = 0 ; pos < data->descr->size * BUCKET_FACTOR ; ++pos) { odb_index_t index = data->hash_base[pos]; while (index) { if (index >= data->descr->current_size) { nr_node_out_of_bound++; break; } ++nr_node; if (data->node_base[index].key > max) max = data->node_base[index].key; index = data->node_base[index].next; } } if (nr_node != data->descr->current_size - 1) { printf("hash table walk found %d node expect %d node\n", nr_node, data->descr->current_size - 1); ret = 1; } if (nr_node_out_of_bound) { printf("out of bound node index: %d\n", nr_node_out_of_bound); ret = 1; } if (ret == 0) ret = check_circular_list(data); if (ret == 0) ret = check_redundant_key(data, max); return ret; }