#include "cx/iterator.h"
#include <string.h>
static bool cx_iter_valid(
const void *it) {
const struct cx_iterator_s *iter = it;
return iter->index < iter->elem_count;
}
static void *cx_iter_current(
const void *it) {
const struct cx_iterator_s *iter = it;
return iter->elem_handle;
}
static void *cx_iter_current_ptr(
const void *it) {
const struct cx_iterator_s *iter = it;
return *(
void**)iter->elem_handle;
}
static void cx_iter_next_fast(
void *it) {
struct cx_iterator_s *iter = it;
if (iter->base.remove) {
iter->base.remove = false;
iter->elem_count--;
if (iter->index < iter->elem_count) {
void *last = ((
char *) iter->src_handle)
+ iter->elem_count * iter->elem_size;
memcpy(iter->elem_handle, last, iter->elem_size);
}
}
else {
iter->index++;
iter->elem_handle = ((
char *) iter->elem_handle) + iter->elem_size;
}
}
static void cx_iter_next_slow(
void *it) {
struct cx_iterator_s *iter = it;
if (iter->base.remove) {
iter->base.remove = false;
iter->elem_count--;
size_t remaining = iter->elem_count - iter->index;
if (remaining >
0) {
memmove(
iter->elem_handle,
((
char *) iter->elem_handle) + iter->elem_size,
remaining * iter->elem_size
);
}
}
else {
iter->index++;
iter->elem_handle = ((
char *) iter->elem_handle) + iter->elem_size;
}
}
CxIterator cxIterator(
const void *array,
size_t elem_size,
size_t elem_count,
bool remove_keeps_order
) {
CxIterator iter;
iter.index =
0;
iter.src_handle = (
void*) array;
iter.elem_handle = (
void*) array;
iter.elem_size = elem_size;
iter.elem_count = array ==
NULL ?
0 : elem_count;
iter.base.valid = cx_iter_valid;
iter.base.current = cx_iter_current;
iter.base.next = remove_keeps_order ? cx_iter_next_slow : cx_iter_next_fast;
iter.base.remove = false;
iter.base.allow_remove = true;
return iter;
}
CxIterator cxIteratorPtr(
const void *array,
size_t elem_count,
bool remove_keeps_order
) {
CxIterator iter = cxIterator(array,
sizeof(
void*), elem_count, remove_keeps_order);
iter.base.current = cx_iter_current_ptr;
return iter;
}