2001-01-25 10:37:55 +01:00
|
|
|
|
/* vi: set ts=2:
|
|
|
|
|
*
|
2006-04-16 00:10:57 +02:00
|
|
|
|
*
|
|
|
|
|
* Eressea PB(E)M host Copyright (C) 1998-2003
|
2001-01-25 10:37:55 +01:00
|
|
|
|
* Christian Schlittchen (corwin@amber.kn-bremen.de)
|
|
|
|
|
* Katja Zedel (katze@felidae.kn-bremen.de)
|
|
|
|
|
* Henning Peters (faroul@beyond.kn-bremen.de)
|
2007-09-02 20:11:17 +02:00
|
|
|
|
* Enno Rehling (enno@eressea.de)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
* Ingo Wilken (Ingo.Wilken@informatik.uni-oldenburg.de)
|
|
|
|
|
*
|
|
|
|
|
* based on:
|
|
|
|
|
*
|
|
|
|
|
* Atlantis v1.0 13 September 1993 Copyright 1993 by Russell Wallace
|
|
|
|
|
* Atlantis v1.7 Copyright 1996 by Alex Schr<EFBFBD>der
|
|
|
|
|
*
|
|
|
|
|
* This program may not be used, modified or distributed without
|
|
|
|
|
* prior permission by the authors of Eressea.
|
|
|
|
|
* This program may not be sold or used commercially without prior written
|
|
|
|
|
* permission from the authors.
|
|
|
|
|
*/
|
|
|
|
|
|
2006-02-19 23:43:56 +01:00
|
|
|
|
#include <config.h>
|
|
|
|
|
#include "cvector.h"
|
|
|
|
|
#include "rng.h"
|
|
|
|
|
|
2001-01-25 10:37:55 +01:00
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
#include <assert.h>
|
|
|
|
|
#include <limits.h>
|
2006-02-25 01:12:48 +01:00
|
|
|
|
#include <memory.h>
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
cv_init(cvector * cv)
|
|
|
|
|
{
|
2006-04-16 00:10:57 +02:00
|
|
|
|
cv->begin = 0;
|
|
|
|
|
cv->end = 0;
|
|
|
|
|
cv->space = 0;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
cvector *
|
|
|
|
|
cv_kill(cvector * cv)
|
|
|
|
|
{
|
2006-04-16 00:10:57 +02:00
|
|
|
|
if (cv->begin) free(cv->begin);
|
|
|
|
|
cv_init(cv);
|
|
|
|
|
return cv;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
size_t
|
|
|
|
|
cv_size(cvector * cv)
|
|
|
|
|
{
|
2006-04-16 00:10:57 +02:00
|
|
|
|
return cv->end - cv->begin;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
cv_reserve(cvector * cv, size_t size)
|
|
|
|
|
{
|
2006-04-16 00:10:57 +02:00
|
|
|
|
size_t count = cv->end - cv->begin;
|
|
|
|
|
cv->begin = realloc(cv->begin, size * sizeof(void *));
|
2001-01-25 10:37:55 +01:00
|
|
|
|
|
2006-04-16 00:10:57 +02:00
|
|
|
|
cv->space = size;
|
|
|
|
|
cv->end = cv->begin + count;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
cv_pushback(cvector * cv, void *u)
|
|
|
|
|
{
|
2006-04-16 00:10:57 +02:00
|
|
|
|
if (cv->space == cv_size(cv))
|
|
|
|
|
cv_reserve(cv, cv->space ? cv->space * 2 : 2);
|
|
|
|
|
*(cv->end++) = u;
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
__cv_scramblecmp(const void *p1, const void *p2)
|
|
|
|
|
{
|
2006-04-16 00:10:57 +02:00
|
|
|
|
return *((long *) p1) - *((long *) p2);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#define addptr(p,i) ((void *)(((char *)p) + i))
|
|
|
|
|
|
2006-04-16 00:10:57 +02:00
|
|
|
|
/** randomly shuffle an array
|
|
|
|
|
* for correctness, see Donald E. Knuth, The Art of Computer Programming
|
|
|
|
|
*/
|
|
|
|
|
static void
|
2005-06-10 00:10:35 +02:00
|
|
|
|
__cv_scramble(void *v1, size_t n, size_t width)
|
2001-01-25 10:37:55 +01:00
|
|
|
|
{
|
2006-04-16 00:10:57 +02:00
|
|
|
|
size_t i;
|
|
|
|
|
void * temp = malloc(width);
|
|
|
|
|
for (i=0;i!=n;++i) {
|
|
|
|
|
size_t j = i + (rng_int() % (n-i));
|
|
|
|
|
memcpy(temp, addptr(v1, i*width), width);
|
|
|
|
|
memcpy(addptr(v1, i*width), addptr(v1, j*width), width);
|
|
|
|
|
memcpy(addptr(v1, j*width), temp, width);
|
|
|
|
|
}
|
|
|
|
|
free(temp);
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
v_scramble(void **begin, void **end)
|
|
|
|
|
{
|
2006-04-16 00:10:57 +02:00
|
|
|
|
__cv_scramble(begin, end - begin, sizeof(void *));
|
2001-01-25 10:37:55 +01:00
|
|
|
|
}
|