#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_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.m)
+ 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 cxMutIterator(
void *array,
size_t elem_size,
size_t elem_count,
bool remove_keeps_order
) {
CxIterator iter;
iter.index =
0;
iter.src_handle.m = array;
iter.elem_handle = 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.mutating = true;
return iter;
}
CxIterator cxIterator(
const void *array,
size_t elem_size,
size_t elem_count
) {
CxIterator iter = cxMutIterator((
void*)array, elem_size, elem_count, false);
iter.base.mutating = false;
return iter;
}