2022-11-19 20:28:13 +01:00
|
|
|
/******************************************************************************/
|
|
|
|
/* HashSet for C99, by LoRd_MuldeR <MuldeR2@GMX.de> */
|
|
|
|
/* This work has been released under the CC0 1.0 Universal license! */
|
|
|
|
/******************************************************************************/
|
|
|
|
|
2022-11-19 17:34:56 +01:00
|
|
|
#define _CRT_RAND_S 1
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <time.h>
|
2022-11-19 22:34:33 +01:00
|
|
|
#include <string.h>
|
2022-11-19 17:34:56 +01:00
|
|
|
|
|
|
|
#include "hash_set.h"
|
|
|
|
|
|
|
|
static uint64_t next_rand(void)
|
|
|
|
{
|
|
|
|
uint32_t a, b;
|
|
|
|
if (rand_s(&a) || rand_s(&b))
|
|
|
|
{
|
|
|
|
abort();
|
|
|
|
}
|
|
|
|
return (((uint64_t)a) << 32) | b;
|
|
|
|
}
|
2022-11-21 17:14:11 +01:00
|
|
|
#define MAXIMUM 393216U
|
2022-11-19 17:34:56 +01:00
|
|
|
|
|
|
|
int main()
|
|
|
|
{
|
2022-11-21 17:14:11 +01:00
|
|
|
uint64_t rnd;
|
|
|
|
size_t capacity, valid, deleted, limit;
|
2022-11-19 22:34:33 +01:00
|
|
|
uint8_t spinner = 0U;
|
2022-11-20 13:59:41 +01:00
|
|
|
clock_t last_update = clock();
|
2022-11-19 22:34:33 +01:00
|
|
|
|
2022-11-19 23:21:14 +01:00
|
|
|
hash_set_t *const hash_set = hash_set_create(0U, -1.0, HASHSET_OPT_FAILFAST);
|
2022-11-19 22:34:33 +01:00
|
|
|
if (!hash_set)
|
2022-11-19 17:34:56 +01:00
|
|
|
{
|
|
|
|
puts("Allocation has failed!");
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
}
|
|
|
|
|
2022-11-21 17:14:11 +01:00
|
|
|
for (size_t r = 0; r < 5; ++r)
|
|
|
|
{
|
|
|
|
for (uint64_t i = 0; i < MAXIMUM; ++i)
|
|
|
|
{
|
|
|
|
if ((i != 3167U) && (i != 9887U) && (i != 185903U) && (i != 387083U))
|
|
|
|
{
|
|
|
|
const errno_t error = hash_set_insert(hash_set, i);
|
|
|
|
if (error)
|
|
|
|
{
|
|
|
|
printf("Insert operation has failed! (error: %d)\n", error);
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!hash_set_info(hash_set, &capacity, &valid, &deleted, &limit))
|
|
|
|
{
|
|
|
|
printf("%010zu, %010zu, %010zu, %010zu\n", capacity, valid, deleted, limit);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (hash_set_size(hash_set) != MAXIMUM - 4U)
|
|
|
|
{
|
|
|
|
puts("Invalid size!");
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (uint64_t i = 0; i < MAXIMUM; ++i)
|
|
|
|
{
|
|
|
|
if ((i != 3167U) && (i != 9887U) && (i != 387083U))
|
|
|
|
{
|
|
|
|
const errno_t error = hash_set_insert(hash_set, i);
|
|
|
|
if (error != ((i != 185903U) ? EEXIST : 0))
|
|
|
|
{
|
|
|
|
printf("Insert operation has failed! (error: %d)\n", error);
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (hash_set_size(hash_set) != MAXIMUM - 3U)
|
|
|
|
{
|
|
|
|
puts("Invalid size!");
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (uint64_t i = 0; i < MAXIMUM; ++i)
|
|
|
|
{
|
|
|
|
const errno_t error = hash_set_contains(hash_set, i);
|
|
|
|
if (error != ((i != 3167U) && (i != 9887U) && (i != 387083U)) ? 0 : ENOENT)
|
|
|
|
{
|
|
|
|
printf("Contains operation has failed! (error: %d)\n", error);
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (uint64_t i = 0; i < MAXIMUM; ++i)
|
|
|
|
{
|
|
|
|
if ((i != 3167U) && (i != 9887U) && (i != 216263U) && (i != 387083U))
|
|
|
|
{
|
|
|
|
const errno_t error = hash_set_remove(hash_set, i);
|
|
|
|
if (error)
|
|
|
|
{
|
|
|
|
printf("Remove operation has failed! (error: %d)\n", error);
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!hash_set_info(hash_set, &capacity, &valid, &deleted, &limit))
|
|
|
|
{
|
|
|
|
printf("%010zu, %010zu, %010zu, %010zu\n", capacity, valid, deleted, limit);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (hash_set_size(hash_set) != 1U)
|
|
|
|
{
|
|
|
|
puts("Invalid size!");
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (uint64_t i = 0; i < MAXIMUM; ++i)
|
|
|
|
{
|
|
|
|
const errno_t error = hash_set_contains(hash_set, i);
|
|
|
|
if (error != ((i != 216263U) ? ENOENT : 0))
|
|
|
|
{
|
|
|
|
printf("Contains operation has failed! (error: %d)\n", error);
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!hash_set_remove(hash_set, 9887U))
|
|
|
|
{
|
|
|
|
puts("Final remove operation has failed!");
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (hash_set_remove(hash_set, 216263U))
|
|
|
|
{
|
|
|
|
puts("Final remove operation has failed!");
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (hash_set_size(hash_set) != 0U)
|
|
|
|
{
|
|
|
|
puts("Invalid size!");
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!hash_set_info(hash_set, &capacity, &valid, &deleted, &limit))
|
|
|
|
{
|
|
|
|
printf("%010zu, %010zu, %010zu, %010zu\n", capacity, valid, deleted, limit);
|
|
|
|
}
|
|
|
|
|
|
|
|
puts("-----");
|
|
|
|
}
|
|
|
|
|
2022-11-19 17:34:56 +01:00
|
|
|
for (;;)
|
|
|
|
{
|
2022-11-21 17:14:11 +01:00
|
|
|
const errno_t error = hash_set_insert(hash_set, rnd = next_rand() & 0x3FFFFFFFFFFFFFFllu);
|
2022-11-19 20:28:13 +01:00
|
|
|
if (error)
|
2022-11-19 17:34:56 +01:00
|
|
|
{
|
2022-11-21 17:14:11 +01:00
|
|
|
if (error != EEXIST)
|
|
|
|
{
|
|
|
|
printf("Insert operation has failed! (error: %d)\n", error);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
printf("Collision detected! [%016llX]\n", rnd);
|
|
|
|
}
|
2022-11-19 17:34:56 +01:00
|
|
|
break;
|
|
|
|
}
|
2022-11-19 18:42:09 +01:00
|
|
|
if (!(++spinner & 0x7F))
|
2022-11-19 17:34:56 +01:00
|
|
|
{
|
2022-11-19 18:42:09 +01:00
|
|
|
const clock_t now = clock();
|
2022-11-19 22:34:33 +01:00
|
|
|
if ((now < last_update) || (now >= last_update + CLOCKS_PER_SEC))
|
2022-11-19 18:42:09 +01:00
|
|
|
{
|
2022-11-21 15:02:07 +01:00
|
|
|
if (!hash_set_info(hash_set, &capacity, &valid, &deleted, &limit))
|
|
|
|
{
|
|
|
|
printf("%010zu, %010zu, %010zu, %010zu\n", capacity, valid, deleted, limit);
|
|
|
|
}
|
2022-11-19 22:34:33 +01:00
|
|
|
last_update = now;
|
2022-11-19 18:42:09 +01:00
|
|
|
}
|
2022-11-19 17:34:56 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-11-21 17:14:11 +01:00
|
|
|
if (!hash_set_info(hash_set, &capacity, &valid, &deleted, &limit))
|
|
|
|
{
|
|
|
|
printf("%010zu, %010zu, %010zu, %010zu\n", capacity, valid, deleted, limit);
|
|
|
|
}
|
|
|
|
|
2022-11-19 22:34:33 +01:00
|
|
|
hash_set_destroy(hash_set);
|
2022-11-21 17:14:11 +01:00
|
|
|
puts("Test completed successfully.");
|
|
|
|
|
2022-11-19 17:34:56 +01:00
|
|
|
return EXIT_SUCCESS;
|
|
|
|
}
|