BFS B+tree test: style cleanup.

This commit is contained in:
Axel Dörfler 2015-01-13 17:37:28 +01:00
parent f982c9ed88
commit 758cfd808a

View File

@ -1,21 +1,23 @@
/* test - BFS B+Tree torture test
**
** Initial version by Axel Dörfler, axeld@pinc-software.de
** This file may be used under the terms of the OpenBeOS License.
*/
/*
* Copyright 2001-2015, Axel Dörfler, axeld@pinc-software.de.
* This file may be used under the terms of the MIT License.
*/
#include "Volume.h"
#include "Inode.h"
#include "BPlusTree.h"
//! BFS B+Tree torture test
#include <ctype.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/stat.h>
#include <List.h>
#include <sys/stat.h>
#include <string.h>
#include <stdlib.h>
#include <stdio.h>
#include <ctype.h>
#include "BPlusTree.h"
#include "Inode.h"
#include "Volume.h"
#define DEFAULT_ITERATIONS 10
@ -26,21 +28,21 @@
#define MAX_STRING 256
struct key {
void *data;
void* data;
uint32 length;
int32 in;
int32 count;
off_t value;
};
key *gKeys;
key* gKeys;
int32 gNum = DEFAULT_NUM_KEYS;
int32 gType = DEFAULT_KEY_TYPE;
int32 gTreeCount = 0;
bool gVerbose, gExcessive;
int32 gIterations = DEFAULT_ITERATIONS;
int32 gHard = 1;
Volume *gVolume;
Volume* gVolume;
int32 gSeed = 42;
// from cache.cpp (yes, we are that mean)
@ -49,9 +51,9 @@ extern BList gBlocks;
// prototypes
void bailOut();
void bailOutWithKey(void *key, uint16 length);
void bailOutWithKey(void* key, uint16 length);
void dumpTree();
void dumpKey(void *key, int32 length);
void dumpKey(void* key, int32 length);
void dumpKeys();
@ -60,15 +62,15 @@ dumpTree()
{
puts("\n*** Tree-Dump:\n");
bplustree_header *header = (bplustree_header *)gBlocks.ItemAt(0);
bplustree_header* header = (bplustree_header*)gBlocks.ItemAt(0);
dump_bplustree_header(header);
for (int32 i = 1;i < gBlocks.CountItems();i++) {
bplustree_node *node = (bplustree_node *)gBlocks.ItemAt(i);
for (int32 i = 1; i < gBlocks.CountItems(); i++) {
bplustree_node* node = (bplustree_node*)gBlocks.ItemAt(i);
printf("\n--- %s node at %ld --------------------------------------\n",
node->overflow_link == BPLUSTREE_NULL ? "leaf" : "index",
i * BPLUSTREE_NODE_SIZE);
dump_bplustree_node(node,header,gVolume);
dump_bplustree_node(node, header, gVolume);
}
}
@ -82,13 +84,13 @@ bailOut()
}
// in any case, write the tree back to disk
shutdown_cache(gVolume->Device(),gVolume->BlockSize());
shutdown_cache(gVolume->Device(), gVolume->BlockSize());
exit(-1);
}
void
bailOutWithKey(void *key, uint16 length)
bailOutWithKey(void* key, uint16 length)
{
dumpKey(key, length);
putchar('\n');
@ -97,44 +99,44 @@ bailOutWithKey(void *key, uint16 length)
void
dumpKey(void *key, int32 length)
dumpKey(void* key, int32 length)
{
switch (gType) {
case S_STR_INDEX:
printf("\"%s\" (%ld bytes)", (char *)key, length);
printf("\"%s\" (%ld bytes)", (char*)key, length);
break;
case S_INT_INDEX:
printf("%ld", *(int32 *)key);
printf("%ld", *(int32*)key);
break;
case S_UINT_INDEX:
printf("%lu", *(uint32 *)key);
printf("%lu", *(uint32*)key);
break;
case S_LONG_LONG_INDEX:
printf("%Ld", *(int64 *)key);
printf("%Ld", *(int64*)key);
break;
case S_ULONG_LONG_INDEX:
printf("%Lu", *(uint64 *)key);
printf("%Lu", *(uint64*)key);
break;
case S_FLOAT_INDEX:
printf("%g", *(float *)key);
printf("%g", *(float*)key);
break;
case S_DOUBLE_INDEX:
printf("%g", *(double *)key);
printf("%g", *(double*)key);
break;
}
if ((gType == S_INT_INDEX || gType == S_UINT_INDEX || gType == S_FLOAT_INDEX)
&& length != 4)
printf(" (wrong length %ld)",length);
else if ((gType == S_LONG_LONG_INDEX || gType == S_ULONG_LONG_INDEX || gType == S_DOUBLE_INDEX)
&& length != 8)
printf(" (wrong length %ld)",length);
if ((gType == S_INT_INDEX || gType == S_UINT_INDEX
|| gType == S_FLOAT_INDEX) && length != 4)
printf(" (wrong length %ld)", length);
else if ((gType == S_LONG_LONG_INDEX || gType == S_ULONG_LONG_INDEX
|| gType == S_DOUBLE_INDEX) && length != 8)
printf(" (wrong length %ld)", length);
}
void
dumpKeys()
{
const char *type;
const char* type;
switch (gType) {
case S_STR_INDEX:
type = "string";
@ -161,24 +163,21 @@ dumpKeys()
debugger("unknown type in gType");
return;
}
printf("Dumping %ld keys of type %s\n",gNum,type);
for (int32 i = 0;i < gNum;i++) {
printf("% 8ld. (%3ld) key = ",i,gKeys[i].in);
dumpKey(gKeys[i].data,gKeys[i].length);
printf("Dumping %ld keys of type %s\n", gNum, type);
for (int32 i = 0; i < gNum; i++) {
printf("% 8ld. (%3ld) key = ", i, gKeys[i].in);
dumpKey(gKeys[i].data, gKeys[i].length);
putchar('\n');
}
}
// #pragma mark -
//
// Functions to generate the keys in every available type
//
// #pragma mark - Key generation
void
generateName(int32 i,char *name,int32 *_length)
generateName(int32 i, char* name, int32* _length)
{
// We're using the index position as a hint for the length
// of the string - this way, it's much less expansive to
@ -186,7 +185,7 @@ generateName(int32 i,char *name,int32 *_length)
// We don't want to sort the strings to have more realistic
// access patterns to the tree (only true for the strings test).
int32 length = i % (MAX_STRING - MIN_STRING) + MIN_STRING;
for (int32 i = 0;i < length;i++) {
for (int32 i = 0; i < length; i++) {
int32 c = int32(52.0 * rand() / RAND_MAX);
if (c >= 26)
name[i] = 'A' + c - 26;
@ -199,43 +198,43 @@ generateName(int32 i,char *name,int32 *_length)
void
fillBuffer(void *buffer,int32 start)
fillBuffer(void* buffer, int32 start)
{
for (int32 i = 0;i < gNum;i++) {
for (int32 i = 0; i < gNum; i++) {
switch (gType) {
case S_INT_INDEX:
{
int32 *array = (int32 *)buffer;
int32* array = (int32*)buffer;
array[i] = start + i;
break;
}
case S_UINT_INDEX:
{
uint32 *array = (uint32 *)buffer;
uint32* array = (uint32*)buffer;
array[i] = start + i;
break;
}
case S_LONG_LONG_INDEX:
{
int64 *array = (int64 *)buffer;
int64* array = (int64*)buffer;
array[i] = start + i;
break;
}
case S_ULONG_LONG_INDEX:
{
uint64 *array = (uint64 *)buffer;
uint64* array = (uint64*)buffer;
array[i] = start + i;
break;
}
case S_FLOAT_INDEX:
{
float *array = (float *)buffer;
float* array = (float*)buffer;
array[i] = start + i * 1.0001;
break;
}
case S_DOUBLE_INDEX:
{
double *array = (double *)buffer;
double* array = (double*)buffer;
array[i] = start + i * 1.0001;
break;
}
@ -246,9 +245,9 @@ fillBuffer(void *buffer,int32 start)
bool
findKey(void *key, int32 length, int32 maxIndex)
findKey(void* key, int32 length, int32 maxIndex)
{
for (int32 i = length;i < maxIndex;i += MAX_STRING - MIN_STRING) {
for (int32 i = length; i < maxIndex; i += MAX_STRING - MIN_STRING) {
if (length == (int32)gKeys[i].length
&& !memcpy(key, gKeys[i].data, length))
return true;
@ -260,21 +259,22 @@ findKey(void *key, int32 length, int32 maxIndex)
status_t
createKeys()
{
gKeys = (key *)malloc(gNum * sizeof(key));
gKeys = (key*)malloc(gNum * sizeof(key));
if (gKeys == NULL)
return B_NO_MEMORY;
if (gType == S_STR_INDEX) {
for (int32 i = 0;i < gNum;i++) {
for (int32 i = 0; i < gNum; i++) {
char name[B_FILE_NAME_LENGTH];
int32 length,tries = 0;
int32 length;
int32 tries = 0;
bool last;
// create unique keys!
do {
generateName(i,name,&length);
} while ((last = findKey(name,length,i)) && tries++ < 100);
generateName(i, name, &length);
} while ((last = findKey(name, length, i)) && tries++ < 100);
if (last) {
printf("Couldn't create unique key list!\n");
dumpKeys();
@ -282,7 +282,7 @@ createKeys()
}
gKeys[i].data = malloc(length + 1);
memcpy(gKeys[i].data,name,length + 1);
memcpy(gKeys[i].data, name, length + 1);
gKeys[i].length = length;
gKeys[i].in = 0;
gKeys[i].count = 0;
@ -307,58 +307,60 @@ createKeys()
default:
return B_BAD_VALUE;
}
uint8 *buffer = (uint8 *)malloc(length * gNum);
uint8* buffer = (uint8*)malloc(length * gNum);
if (buffer == NULL)
return B_NO_MEMORY;
for (int32 i = 0;i < gNum;i++) {
gKeys[i].data = (void *)(buffer + i * length);
for (int32 i = 0; i < gNum; i++) {
gKeys[i].data = (void*)(buffer + i * length);
gKeys[i].length = length;
gKeys[i].in = 0;
gKeys[i].count = 0;
}
fillBuffer(buffer,start);
fillBuffer(buffer, start);
}
return B_OK;
}
// #pragma mark -
//
// Tree validity checker
//
// #pragma mark - Validity checker
void
checkTreeContents(BPlusTree *tree)
checkTreeContents(BPlusTree* tree)
{
// reset counter
for (int32 i = 0;i < gNum;i++)
for (int32 i = 0; i < gNum; i++)
gKeys[i].count = 0;
TreeIterator iterator(tree);
char key[B_FILE_NAME_LENGTH];
uint16 length,duplicate;
uint16 length;
uint16 duplicate;
off_t value;
status_t status;
while ((status = iterator.GetNextEntry(key,&length,B_FILE_NAME_LENGTH,&value,&duplicate)) == B_OK) {
while ((status = iterator.GetNextEntry(key, &length, B_FILE_NAME_LENGTH,
&value, &duplicate)) == B_OK) {
if (value < 0 || value >= gNum) {
iterator.Dump();
printf("\ninvalid value %Ld in tree: ",value);
bailOutWithKey(key,length);
printf("\ninvalid value %Ld in tree: ", value);
bailOutWithKey(key, length);
}
if (gKeys[value].value != value) {
iterator.Dump();
printf("\nkey pointing to the wrong value %Ld (should be %Ld)\n",value,gKeys[value].value);
bailOutWithKey(key,length);
printf("\nkey pointing to the wrong value %Ld (should be %Ld)\n",
value, gKeys[value].value);
bailOutWithKey(key, length);
}
if (length != gKeys[value].length
|| memcmp(key,gKeys[value].data,length)) {
|| memcmp(key, gKeys[value].data, length)) {
iterator.Dump();
printf("\nkeys don't match (key index = %Ld, %ld times in tree, %ld. occassion):\n\tfound: ",value,gKeys[value].in,gKeys[value].count + 1);
dumpKey(key,length);
printf("\nkeys don't match (key index = %Ld, %ld times in tree, "
"%ld. occassion):\n\tfound: ", value, gKeys[value].in,
gKeys[value].count + 1);
dumpKey(key, length);
printf("\n\texpected: ");
dumpKey(gKeys[value].data,gKeys[value].length);
dumpKey(gKeys[value].data, gKeys[value].length);
putchar('\n');
bailOut();
}
@ -366,46 +368,46 @@ checkTreeContents(BPlusTree *tree)
gKeys[value].count++;
}
if (status != B_ENTRY_NOT_FOUND) {
printf("TreeIterator::GetNext() returned: %s\n",strerror(status));
printf("TreeIterator::GetNext() returned: %s\n", strerror(status));
iterator.Dump();
bailOut();
}
for (int32 i = 0;i < gNum;i++) {
for (int32 i = 0; i < gNum; i++) {
if (gKeys[i].in != gKeys[i].count) {
printf("Key ");
dumpKey(gKeys[i].data,gKeys[i].length);
printf(" found only %ld from %ld\n",gKeys[i].count,gKeys[i].in);
dumpKey(gKeys[i].data, gKeys[i].length);
printf(" found only %ld from %ld\n", gKeys[i].count, gKeys[i].in);
bailOut();
}
}
}
/*! Simple test, just seeks down to every key - if it's in and couldn't
be found or it's not in and can be found, something must be wrong
*/
void
checkTreeIntegrity(BPlusTree *tree)
checkTreeIntegrity(BPlusTree* tree)
{
// simple test, just seeks down to every key - if it's in and couldn't
// be found or it's not in and can be found, something must be wrong
TreeIterator iterator(tree);
for (int32 i = 0;i < gNum;i++) {
status_t status = iterator.Find((uint8 *)gKeys[i].data,gKeys[i].length);
for (int32 i = 0; i < gNum; i++) {
status_t status = iterator.Find((uint8*)gKeys[i].data, gKeys[i].length);
if (gKeys[i].in == 0) {
if (status == B_OK) {
printf("found key %" B_PRId32 " even though it's not in!\n", i);
bailOutWithKey(gKeys[i].data, gKeys[i].length);
}
} else if (status != B_OK) {
printf("TreeIterator::Find() returned: %s\n",strerror(status));
bailOutWithKey(gKeys[i].data,gKeys[i].length);
printf("TreeIterator::Find() returned: %s\n", strerror(status));
bailOutWithKey(gKeys[i].data, gKeys[i].length);
}
}
}
void
checkTree(BPlusTree *tree)
checkTree(BPlusTree* tree)
{
if (!gExcessive)
printf("* Check tree...\n");
@ -422,25 +424,22 @@ checkTree(BPlusTree *tree)
}
// #pragma mark -
//
// The tree "torture" functions
//
// #pragma mark - "Torture" functions
void
addAllKeys(Transaction &transaction, BPlusTree *tree)
addAllKeys(Transaction& transaction, BPlusTree* tree)
{
printf("*** Adding all keys to the tree...\n");
for (int32 i = 0;i < gNum;i++) {
status_t status = tree->Insert(transaction,(uint8 *)gKeys[i].data,gKeys[i].length,gKeys[i].value);
if (status < B_OK) {
printf("BPlusTree::Insert() returned: %s\n",strerror(status));
for (int32 i = 0; i < gNum; i++) {
status_t status = tree->Insert(transaction, (uint8*)gKeys[i].data,
gKeys[i].length, gKeys[i].value);
if (status != B_OK) {
printf("BPlusTree::Insert() returned: %s\n", strerror(status));
printf("key: ");
dumpKey(gKeys[i].data,gKeys[i].length);
dumpKey(gKeys[i].data, gKeys[i].length);
putchar('\n');
}
else {
} else {
gKeys[i].in++;
gTreeCount++;
}
@ -450,54 +449,56 @@ addAllKeys(Transaction &transaction, BPlusTree *tree)
void
removeAllKeys(Transaction &transaction, BPlusTree *tree)
removeAllKeys(Transaction& transaction, BPlusTree* tree)
{
printf("*** Removing all keys from the tree...\n");
for (int32 i = 0;i < gNum;i++) {
for (int32 i = 0; i < gNum; i++) {
while (gKeys[i].in > 0) {
status_t status = tree->Remove(transaction, (uint8 *)gKeys[i].data,
status_t status = tree->Remove(transaction, (uint8*)gKeys[i].data,
gKeys[i].length, gKeys[i].value);
if (status < B_OK) {
if (status != B_OK) {
printf("BPlusTree::Remove() returned: %s\n", strerror(status));
printf("key: ");
dumpKey(gKeys[i].data, gKeys[i].length);
putchar('\n');
}
else {
} else {
gKeys[i].in--;
gTreeCount--;
}
}
}
checkTree(tree);
}
void
duplicateTest(Transaction &transaction,BPlusTree *tree)
duplicateTest(Transaction& transaction, BPlusTree* tree)
{
int32 index = int32(1.0 * gNum * rand() / RAND_MAX);
if (index == gNum)
index = gNum - 1;
printf("*** Duplicate test with key ");
dumpKey(gKeys[index].data,gKeys[index].length);
dumpKey(gKeys[index].data, gKeys[index].length);
puts("...");
status_t status;
int32 insertTotal = 0;
for (int32 i = 0;i < 8;i++) {
for (int32 i = 0; i < 8; i++) {
int32 insertCount = int32(1000.0 * rand() / RAND_MAX);
if (gVerbose)
printf("* insert %ld to %ld old entries...\n",insertCount,insertTotal + gKeys[index].in);
if (gVerbose) {
printf("* insert %ld to %ld old entries...\n", insertCount,
insertTotal + gKeys[index].in);
}
for (int32 j = 0;j < insertCount;j++) {
status = tree->Insert(transaction,(uint8 *)gKeys[index].data,gKeys[index].length,gKeys[index].value);
if (status < B_OK) {
printf("BPlusTree::Insert() returned: %s\n",strerror(status));
bailOutWithKey(gKeys[index].data,gKeys[index].length);
for (int32 j = 0; j < insertCount; j++) {
status = tree->Insert(transaction, (uint8*)gKeys[index].data,
gKeys[index].length, gKeys[index].value);
if (status != B_OK) {
printf("BPlusTree::Insert() returned: %s\n", strerror(status));
bailOutWithKey(gKeys[index].data, gKeys[index].length);
}
insertTotal++;
gTreeCount++;
@ -515,14 +516,18 @@ duplicateTest(Transaction &transaction,BPlusTree *tree)
} else
count = insertTotal;
if (gVerbose)
printf("* remove %ld from %ld entries...\n",count,insertTotal + gKeys[index].in);
if (gVerbose) {
printf("* remove %ld from %ld entries...\n", count,
insertTotal + gKeys[index].in);
}
for (int32 j = 0;j < count;j++) {
status_t status = tree->Remove(transaction,(uint8 *)gKeys[index].data,gKeys[index].length,gKeys[index].value);
if (status < B_OK) {
printf("BPlusTree::Remove() returned: %s\n",strerror(status));
bailOutWithKey(gKeys[index].data,gKeys[index].length);
for (int32 j = 0; j < count; j++) {
status_t status = tree->Remove(transaction,
(uint8*)gKeys[index].data, gKeys[index].length,
gKeys[index].value);
if (status != B_OK) {
printf("BPlusTree::Remove() returned: %s\n", strerror(status));
bailOutWithKey(gKeys[index].data, gKeys[index].length);
}
insertTotal--;
gTreeCount--;
@ -539,22 +544,26 @@ duplicateTest(Transaction &transaction,BPlusTree *tree)
void
addRandomSet(Transaction &transaction,BPlusTree *tree,int32 num)
addRandomSet(Transaction& transaction, BPlusTree* tree, int32 num)
{
printf("*** Add random set to tree (%ld to %ld old entries)...\n",num,gTreeCount);
printf("*** Add random set to tree (%ld to %ld old entries)...\n",
num, gTreeCount);
for (int32 i = 0;i < num;i++) {
for (int32 i = 0; i < num; i++) {
int32 index = int32(1.0 * gNum * rand() / RAND_MAX);
if (index == gNum)
index = gNum - 1;
if (gVerbose)
printf("adding key %ld (%ld times in the tree)\n",index,gKeys[index].in);
if (gVerbose) {
printf("adding key %ld (%ld times in the tree)\n", index,
gKeys[index].in);
}
status_t status = tree->Insert(transaction,(uint8 *)gKeys[index].data,gKeys[index].length,gKeys[index].value);
if (status < B_OK) {
printf("BPlusTree::Insert() returned: %s\n",strerror(status));
bailOutWithKey(gKeys[index].data,gKeys[index].length);
status_t status = tree->Insert(transaction, (uint8*)gKeys[index].data,
gKeys[index].length, gKeys[index].value);
if (status != B_OK) {
printf("BPlusTree::Insert() returned: %s\n", strerror(status));
bailOutWithKey(gKeys[index].data, gKeys[index].length);
}
gKeys[index].in++;
gTreeCount++;
@ -568,13 +577,14 @@ addRandomSet(Transaction &transaction,BPlusTree *tree,int32 num)
void
removeRandomSet(Transaction &transaction,BPlusTree *tree,int32 num)
removeRandomSet(Transaction& transaction, BPlusTree* tree, int32 num)
{
printf("*** Remove random set from tree (%ld from %ld entries)...\n",num,gTreeCount);
printf("*** Remove random set from tree (%ld from %ld entries)...\n",
num, gTreeCount);
int32 tries = 500;
for (int32 i = 0;i < num;i++) {
for (int32 i = 0; i < num; i++) {
if (gTreeCount < 1)
break;
@ -589,13 +599,16 @@ removeRandomSet(Transaction &transaction,BPlusTree *tree,int32 num)
continue;
}
if (gVerbose)
printf("removing key %ld (%ld times in the tree)\n",index,gKeys[index].in);
if (gVerbose) {
printf("removing key %ld (%ld times in the tree)\n", index,
gKeys[index].in);
}
status_t status = tree->Remove(transaction,(uint8 *)gKeys[index].data,gKeys[index].length,gKeys[index].value);
if (status < B_OK) {
printf("BPlusTree::Remove() returned: %s\n",strerror(status));
bailOutWithKey(gKeys[index].data,gKeys[index].length);
status_t status = tree->Remove(transaction, (uint8*)gKeys[index].data,
gKeys[index].length, gKeys[index].value);
if (status != B_OK) {
printf("BPlusTree::Remove() returned: %s\n", strerror(status));
bailOutWithKey(gKeys[index].data, gKeys[index].length);
}
gKeys[index].in--;
gTreeCount--;
@ -612,11 +625,12 @@ removeRandomSet(Transaction &transaction,BPlusTree *tree,int32 num)
void
usage(char *program)
usage(char* program)
{
if (strrchr(program,'/'))
program = strrchr(program,'/') + 1;
fprintf(stderr,"usage: %s [-veh] [-t type] [-n keys] [-i iterations] [-h times] [-r seed]\n"
if (strrchr(program, '/'))
program = strrchr(program, '/') + 1;
fprintf(stderr, "usage: %s [-veh] [-t type] [-n keys] [-i iterations] "
"[-h times] [-r seed]\n"
"BFS B+Tree torture test\n"
"\t-t\ttype is one of string, int32, uint32, int64, uint64, float,\n"
"\t\tor double; defaults to string.\n"
@ -625,7 +639,8 @@ usage(char *program)
"\t-i\titerations is the number of the test cycles, defaults to %d.\n"
"\t-r\tthe seed for the random function, defaults to %ld.\n"
"\t-h\tremoves the keys and start over again for x times.\n"
"\t-e\texcessive validity tests: tree contents will be tested after every operation\n"
"\t-e\texcessive validity tests: tree contents will be tested after "
"every operation\n"
"\t-v\tfor verbose output.\n",
program, DEFAULT_NUM_KEYS, DEFAULT_ITERATIONS, gSeed);
exit(0);
@ -633,22 +648,18 @@ usage(char *program)
int
main(int argc,char **argv)
main(int argc, char** argv)
{
char *program = argv[0];
char* program = argv[0];
while (*++argv)
{
char *arg = *argv;
if (*arg == '-')
{
while (*++argv) {
char* arg = *argv;
if (*arg == '-') {
if (arg[1] == '-')
usage(program);
while (*++arg && isalpha(*arg))
{
switch (*arg)
{
while (*++arg && isalpha(*arg)) {
switch (*arg) {
case 'v':
gVerbose = true;
break;
@ -659,23 +670,23 @@ main(int argc,char **argv)
if (*++argv == NULL)
usage(program);
if (!strcmp(*argv,"string"))
if (!strcmp(*argv, "string"))
gType = S_STR_INDEX;
else if (!strcmp(*argv,"int32")
|| !strcmp(*argv,"int"))
else if (!strcmp(*argv, "int32")
|| !strcmp(*argv, "int"))
gType = S_INT_INDEX;
else if (!strcmp(*argv,"uint32")
|| !strcmp(*argv,"uint"))
else if (!strcmp(*argv, "uint32")
|| !strcmp(*argv, "uint"))
gType = S_UINT_INDEX;
else if (!strcmp(*argv,"int64")
|| !strcmp(*argv,"llong"))
else if (!strcmp(*argv, "int64")
|| !strcmp(*argv, "llong"))
gType = S_LONG_LONG_INDEX;
else if (!strcmp(*argv,"uint64")
|| !strcmp(*argv,"ullong"))
else if (!strcmp(*argv, "uint64")
|| !strcmp(*argv, "ullong"))
gType = S_ULONG_LONG_INDEX;
else if (!strcmp(*argv,"float"))
else if (!strcmp(*argv, "float"))
gType = S_FLOAT_INDEX;
else if (!strcmp(*argv,"double"))
else if (!strcmp(*argv, "double"))
gType = S_DOUBLE_INDEX;
else
usage(program);
@ -683,7 +694,7 @@ main(int argc,char **argv)
case 'n':
if (*++argv == NULL || !isdigit(**argv))
usage(program);
gNum = atoi(*argv);
if (gNum < 1)
gNum = 1;
@ -707,62 +718,62 @@ main(int argc,char **argv)
case 'r':
if (*++argv == NULL || !isdigit(**argv))
usage(program);
gSeed = atoi(*argv);
break;
}
}
}
else
} else
break;
}
// we do want to have reproducible random keys
if (gVerbose)
printf("Set seed to %ld\n",gSeed);
printf("Set seed to %ld\n", gSeed);
srand(gSeed);
Inode inode("tree.data",gType | S_ALLOW_DUPS);
Inode inode("tree.data", gType | S_ALLOW_DUPS);
rw_lock_write_lock(&inode.Lock());
gVolume = inode.GetVolume();
Transaction transaction(gVolume,0);
Transaction transaction(gVolume, 0);
init_cache(gVolume->Device(),gVolume->BlockSize());
init_cache(gVolume->Device(), gVolume->BlockSize());
//
// Create the tree, the keys, and add all keys to the tree initially
//
BPlusTree tree(transaction,&inode);
status_t status;
if ((status = tree.InitCheck()) < B_OK) {
fprintf(stderr,"creating tree failed: %s\n",strerror(status));
BPlusTree tree(transaction, &inode);
status_t status = tree.InitCheck();
if (status != B_OK) {
fprintf(stderr, "creating tree failed: %s\n", strerror(status));
bailOut();
}
printf("*** Creating %ld keys...\n",gNum);
if ((status = createKeys()) < B_OK) {
fprintf(stderr,"creating keys failed: %s\n",strerror(status));
printf("*** Creating %ld keys...\n", gNum);
status = createKeys();
if (status != B_OK) {
fprintf(stderr, "creating keys failed: %s\n", strerror(status));
bailOut();
}
if (gVerbose)
dumpKeys();
for (int32 j = 0; j < gHard; j++ ) {
for (int32 j = 0; j < gHard; j++) {
addAllKeys(transaction, &tree);
//
// Run the tests (they will exit the app, if an error occurs)
//
for (int32 i = 0;i < gIterations;i++) {
printf("---------- Test iteration %ld ---------------------------------\n",i+1);
addRandomSet(transaction,&tree,int32(1.0 * gNum * rand() / RAND_MAX));
removeRandomSet(transaction,&tree,int32(1.0 * gNum * rand() / RAND_MAX));
duplicateTest(transaction,&tree);
for (int32 i = 0; i < gIterations; i++) {
printf("---------- Test iteration %ld --------------------------\n",
i + 1);
addRandomSet(transaction, &tree,
int32(1.0 * gNum * rand() / RAND_MAX));
removeRandomSet(transaction, &tree,
int32(1.0 * gNum * rand() / RAND_MAX));
duplicateTest(transaction, &tree);
}
removeAllKeys(transaction, &tree);
}
@ -771,7 +782,7 @@ main(int argc,char **argv)
// of course, we would have to free all our memory in a real application here...
// write the cache back to the tree
shutdown_cache(gVolume->Device(),gVolume->BlockSize());
shutdown_cache(gVolume->Device(), gVolume->BlockSize());
return 0;
}