/* varray - a variable sized array
 * Copyright (c) 2003 Michael B. Allen <mba2000 ioplex.com>
 *
 * The MIT License
 * 
 * Permission is hereby granted, free of charge, to any person obtaining a
 * copy of this software and associated documentation files (the "Software"),
 * to deal in the Software without restriction, including without limitation
 * the rights to use, copy, modify, merge, publish, distribute, sublicense,
 * and/or sell copies of the Software, and to permit persons to whom the
 * Software is furnished to do so, subject to the following conditions:
 * 
 * The above copyright notice and this permission notice shall be included
 * in all copies or substantial portions of the Software.
 * 
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
 * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR
 * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
 * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
 * OTHER DEALINGS IN THE SOFTWARE.
 */

#include <stdlib.h>
#include <string.h>
#include <errno.h>

#include "mba/iterator.h"
#include "mba/varray.h"
#include "mba/msgno.h"

#define VAAL(va) ((struct allocator *)((va)->al ? (char *)(va) - (ptrdiff_t)(va)->al : NULL))
#define BINSIZ(i) ((i) ? 1 << ((i) + (VARRAY_INIT_SIZE - 1)) : (1 << VARRAY_INIT_SIZE))

int
varray_init(struct varray *va, size_t membsize, struct allocator *al)
{
	if (va == NULL || membsize == 0) {
		PMNO(errno = EINVAL);
		return -1;
	}

	memset(va, 0, sizeof *va);
	va->size = membsize;
	if (al && al->tail) { /* al is a suba allocator */
		va->al = (char *)va - (char *)al;
	}

	return 0;
}
int
varray_deinit(struct varray *va)
{
	if (varray_release(va, 0) == -1) {
		AMSG("");
		return -1;
	}
	return 0;
}
struct varray *
varray_new(size_t membsize, struct allocator *al)
{
	struct varray *va;

	if ((va = allocator_alloc(al, sizeof *va, 0)) == NULL) {
		AMSG("");
		return NULL;
	}
	if (varray_init(va, membsize, al) == -1) {
		AMSG("");
		allocator_free(al, va);
		return NULL;
	}

	return va;
}
int
varray_del(void *va)
{
	int ret = 0;

	if (va) {
		ret += varray_release(va, 0);
		ret += allocator_free(VAAL((struct varray *)va), va);
	}

	if (ret) {
		AMSG("");
		return -1;
	}

	return 0;
}
int
varray_release(struct varray *va, unsigned int from)
{
	unsigned int r, i;
	int ret = 0;

	if (va == NULL) {
		return 0;
	}

	r = (1 << VARRAY_INIT_SIZE);
	for (i = 0; i < 16; i++) {
		if (from <= r) {
			break;
		}
		r *= 2;
	}
	if (from != 0) i++;
	for ( ; i < 16; i++) {
		if (va->bins[i]) {
			struct allocator *al = VAAL(va);
			ret += allocator_free(al, ALADR(al, va->bins[i]));
			va->bins[i] = 0;
		}
	}

	if (ret) {
		AMSG("");
		return -1;
	}

	return 0;
}
void *
varray_get(struct varray *va, unsigned int idx)
{
	unsigned int r, i, n;
	struct allocator *al;

	if (va == NULL) {
		PMNO(errno = EINVAL);
		return NULL;
	}

	r = (1 << VARRAY_INIT_SIZE);          /* First and second bins hold 32 then 64,128,256,... */
	for (i = 0; i < 16; i++) {
		if (r > idx) {
			break;
		}
		r *= 2;
	}
	if (i == 16) {
		PMNO(errno = ERANGE);
		return NULL;
	}

	al = VAAL(va);
	n = BINSIZ(i);                                                      /* n is nmemb in bin i */

	if (va->bins[i] == 0) {
		char *mem = allocator_alloc(al, n * va->size, 1);
		if (mem == NULL) {
			AMSG("");
			return NULL;
		}
		va->bins[i] = ALREF(al, mem);
	}

	return (char *)ALADR(al, va->bins[i]) + (i ? idx - n : idx) * va->size;
}
int
varray_index(struct varray *va, void *elem)
{
	ref_t er = ALREF(VAAL(va), elem);
	int i;

	for (i = 0; i < 16; i++) {
		if (va->bins[i]) {
			size_t n = BINSIZ(i);
			ref_t start = va->bins[i];
			ref_t end = start + n * va->size;
			if (er >= start && er < end) {
				return (i ? n : 0) + ((er - start) / va->size);
			}
		}
	}

	PMNO(errno = EFAULT);
	return -1;
}
void
varray_iterate(void *va, iter_t *iter)
{
	if (va && iter) {
		iter->i1 = iter->i2 = 0;
	}
}
void *
varray_next(void *va0, iter_t *iter)
{
	struct varray *va = va0;
	unsigned int n;

	if (va == NULL || iter == NULL) {
		PMNO(errno = EINVAL);
		return NULL;
	}

	/* n is nmemb in iter->i1 */
	n = iter->i1 == 0 ? (1 << VARRAY_INIT_SIZE) :
			1 << (iter->i1 + (VARRAY_INIT_SIZE - 1));

	if (iter->i2 == n) {
		iter->i2 = 0;
		iter->i1++;
	}
	while (va->bins[iter->i1] == 0) {
		iter->i1++;
		if (iter->i1 == 16) {
			return NULL;
		}
	}

	return (char *)ALADR(VAAL(va), va->bins[iter->i1]) + iter->i2++ * va->size;
}



syntax highlighted by Code2HTML, v. 0.9.1