diff --git a/flecs.c b/flecs.c index 31bfa2e27b..7e51d4766a 100644 --- a/flecs.c +++ b/flecs.c @@ -547,6 +547,61 @@ void flecs_table_diff_build_noalloc( #endif +/** + * @file table_data.h + * @brief Table data implementation. + */ + +#ifndef FLECS_TABLE_DATA_H +#define FLECS_TABLE_DATA_H + +int32_t flecs_table_data_append( + ecs_world_t *world, + ecs_table_t *table, + ecs_entity_t entity, + ecs_record_t *record, + bool construct, + bool on_add); + +int32_t flecs_table_data_appendn( + ecs_world_t *world, + ecs_table_t *table, + int32_t to_add, + const ecs_entity_t *ids); + +void flecs_table_data_move( + ecs_world_t *world, + ecs_entity_t dst_entity, + ecs_entity_t src_entity, + ecs_table_t *dst_table, + int32_t dst_index, + ecs_table_t *src_table, + int32_t src_index, + bool construct); + +int32_t flecs_table_data_delete( + ecs_world_t *world, + ecs_table_t *table, + int32_t index, + bool destruct); + +void flecs_table_data_swap( + ecs_world_t *world, + ecs_table_t *table, + int32_t row_1, + int32_t row_2); + +void flecs_table_data_merge( + ecs_world_t *world, + ecs_table_t *dst_table, + ecs_table_t *src_table); + +bool flecs_table_data_shrink( + ecs_world_t *world, + ecs_table_t *table); + +#endif + /* Table event type for notifying tables of world events */ typedef enum ecs_table_eventkind_t { @@ -576,6 +631,8 @@ typedef struct ecs_table__t { uint64_t hash; /* Type hash */ int32_t lock; /* Prevents modifications */ int32_t traversable_count; /* Traversable relationship targets in table */ + int16_t bs_offset; /* First bitset id in type */ + int16_t ft_offset; /* First flattened id in type */ uint16_t generation; /* Used for table cleanup */ int16_t record_count; /* Table record count including wildcards */ @@ -583,26 +640,31 @@ typedef struct ecs_table__t { ecs_hashmap_t *name_index; /* Cached pointer to name index */ } ecs_table__t; -/** Table column */ +/** Component column */ typedef struct ecs_column_t { ecs_vec_t data; /* Vector with component data */ - ecs_id_t id; /* Component id */ + ecs_id_t id; /* Column id */ ecs_type_info_t *ti; /* Component type info */ ecs_size_t size; /* Component size */ } ecs_column_t; +/** Bitset column */ +typedef struct ecs_bitset_column_t { + ecs_bitset_t data; /* Bitset columns */ + ecs_id_t id; /* Column id */ +} ecs_bitset_column_t; + /** Table data */ -struct ecs_data_t { +struct ecs_table_data_t { ecs_vec_t entities; /* Entity ids */ ecs_vec_t records; /* Ptrs to records in entity index */ ecs_column_t *columns; /* Component data */ int16_t column_count; /* Number of components (excluding tags) */ + ecs_flags32_t flags; /* Flags for testing table data properties */ int32_t *dirty_state; /* Keep track of changes in columns */ - ecs_bitset_t *bs_columns; /* Bitset columns */ - int16_t bs_count; - int16_t bs_offset; - int16_t ft_offset; + ecs_bitset_column_t *bitsets; /* Bitset columns */ + int16_t bs_count; /* Number of bitset columns */ }; /** A table is the Flecs equivalent of an archetype. Tables store all entities @@ -614,7 +676,7 @@ struct ecs_table_t { ecs_flags32_t flags; /* Flags for testing table properties */ ecs_type_t type; /* Vector with component ids */ - ecs_data_t *data; /* Component storage */ + ecs_table_data_t *data; /* Component storage */ ecs_graph_node_t node; /* Graph node */ int32_t *column_map; /* Map type index <-> column @@ -626,7 +688,7 @@ struct ecs_table_t { }; /* Get table data */ -ecs_data_t* flecs_table_data( +ecs_table_data_t* flecs_table_data( const ecs_table_t *table); /* Get table columns */ @@ -2847,7 +2909,7 @@ ecs_table_t* flecs_bootstrap_component_table( }; ecs_table_t *result = flecs_table_find_or_create(world, &array); - ecs_data_t *data = flecs_table_data(result); + ecs_table_data_t *data = flecs_table_data(result); /* Preallocate enough memory for initial components */ ecs_allocator_t *a = &world->allocator; @@ -3478,7 +3540,7 @@ void flecs_instantiate_children( } ecs_type_t type = child_table->type; - ecs_data_t *child_data = flecs_table_data(child_table); + ecs_table_data_t *child_data = flecs_table_data(child_table); ecs_entity_t slot_of = 0; ecs_entity_t *ids = type.array; @@ -3983,7 +4045,7 @@ const ecs_entity_t* flecs_bulk_new( component_array.count = type.count; } - ecs_data_t *data = flecs_table_data(table); + ecs_table_data_t *data = flecs_table_data(table); int32_t row = flecs_table_appendn(world, table, count, entities); /* Update entity index. */ @@ -4206,7 +4268,7 @@ void flecs_notify_on_set( bool owned) { ecs_assert(ids != NULL, ECS_INTERNAL_ERROR, NULL); - ecs_data_t *data = flecs_table_data(table); + ecs_table_data_t *data = flecs_table_data(table); ecs_entity_t *entities = &flecs_table_entities_array(table)[row]; ecs_assert(entities != NULL, ECS_INTERNAL_ERROR, NULL); @@ -6367,11 +6429,11 @@ void ecs_enable_id( } ecs_assert(table->_ != NULL, ECS_INTERNAL_ERROR, NULL); - index -= flecs_table_data(table)->bs_offset; + index -= table->_->bs_offset; ecs_assert(index >= 0, ECS_INTERNAL_ERROR, NULL); /* Data cannot be NULl, since entity is stored in the table */ - ecs_bitset_t *bs = &flecs_table_data(table)->bs_columns[index]; + ecs_bitset_t *bs = &flecs_table_data(table)->bitsets[index].data; ecs_assert(bs != NULL, ECS_INTERNAL_ERROR, NULL); flecs_bitset_set(bs, ECS_RECORD_TO_ROW(r->row), enable); @@ -6407,9 +6469,9 @@ bool ecs_is_enabled_id( } ecs_assert(table->_ != NULL, ECS_INTERNAL_ERROR, NULL); - index -= flecs_table_data(table)->bs_offset; + index -= table->_->bs_offset; ecs_assert(index >= 0, ECS_INTERNAL_ERROR, NULL); - ecs_bitset_t *bs = &flecs_table_data(table)->bs_columns[index]; + ecs_bitset_t *bs = &flecs_table_data(table)->bitsets[index].data; return flecs_bitset_get(bs, ECS_RECORD_TO_ROW(r->row)); error: @@ -8038,7 +8100,7 @@ int flecs_entity_filter_bitset_next( int32_t i, count = ecs_vec_count(&iter->entity_filter->bs_terms); flecs_bitset_term_t *terms = ecs_vec_first(&iter->entity_filter->bs_terms); - int32_t bs_offset = flecs_table_data(table)->bs_offset; + int32_t bs_offset = table->_->bs_offset; int32_t first = iter->bs_offset; int32_t last = 0; @@ -8049,7 +8111,7 @@ int flecs_entity_filter_bitset_next( if (!bs) { int32_t index = column->column_index; ecs_assert((index - bs_offset >= 0), ECS_INTERNAL_ERROR, NULL); - bs = &flecs_table_data(table)->bs_columns[index - bs_offset]; + bs = &flecs_table_data(table)->bitsets[index - bs_offset].data; terms[i].bs_column = bs; } @@ -8200,7 +8262,7 @@ int32_t flecs_get_flattened_target( } if (table->flags & EcsTableHasTarget) { - int32_t col = table->column_map[flecs_table_data(table)->ft_offset]; + int32_t col = table->column_map[table->_->ft_offset]; ecs_assert(col != -1, ECS_INTERNAL_ERROR, NULL); EcsTarget *next = flecs_table_column(table, col)->data.array; next = ECS_ELEM_T(next, EcsTarget, ECS_RECORD_TO_ROW(r->row)); @@ -11089,7 +11151,7 @@ bool flecs_term_match_table( if ((column == -1) && (src->flags & EcsUp) && (table->flags & EcsTableHasTarget)) { ecs_assert(table->_ != NULL, ECS_INTERNAL_ERROR, NULL); - ecs_id_t rel = ECS_PAIR_SECOND(table->type.array[flecs_table_data(table)->ft_offset]); + ecs_id_t rel = ECS_PAIR_SECOND(table->type.array[table->_->ft_offset]); if (rel == (uint32_t)src->trav) { result = true; } @@ -14410,7 +14472,7 @@ void flecs_emit( const ecs_event_record_t *er_onset = flecs_event_record_get_if(observable, EcsOnSet); const ecs_event_record_t *er_unset = flecs_event_record_get_if(observable, EcsUnSet); - ecs_data_t *storage = NULL; + ecs_table_data_t *storage = NULL; ecs_column_t *columns = NULL; if (count) { storage = table->data; @@ -40723,7 +40785,7 @@ void flecs_fini_id_records( #ifdef FLECS_SANITIZE static void flecs_table_check_sanity(ecs_table_t *table) { - ecs_data_t *data = table->data; + ecs_table_data_t *data = table->data; int32_t size = ecs_vec_size(&data->entities); int32_t count = ecs_vec_count(&data->entities); @@ -40869,7 +40931,7 @@ void flecs_table_init_data( ecs_table_t *table, int32_t column_count) { - ecs_data_t *data = table->data; + ecs_table_data_t *data = table->data; data->column_count = flecs_ito(int16_t, column_count); ecs_vec_init_t(NULL, &data->entities, ecs_entity_t, 0); ecs_vec_init_t(NULL, &data->records, ecs_record_t*, 0); @@ -40879,14 +40941,16 @@ void flecs_table_init_data( int32_t i, bs_count = data->bs_count; if (bs_count) { - data->bs_columns = flecs_wcalloc_n(world, ecs_bitset_t, bs_count); + data->bitsets = flecs_wcalloc_n(world, ecs_bitset_column_t, bs_count); for (i = 0; i < bs_count; i ++) { - flecs_bitset_init(&data->bs_columns[i]); + flecs_bitset_init(&data->bitsets[i].data); } } + + data->flags = table->flags; // TODO } -ecs_data_t* flecs_table_data( +ecs_table_data_t* flecs_table_data( const ecs_table_t *table) { ecs_assert(table != NULL, ECS_INTERNAL_ERROR, NULL); @@ -40982,19 +41046,18 @@ void flecs_table_init_flags( } else if (id == ecs_pair_t(EcsIdentifier, EcsName)) { table->flags |= EcsTableHasName; } else if (r == ecs_id(EcsTarget)) { - ecs_data_t *data = table->data; table->flags |= EcsTableHasTarget; - data->ft_offset = flecs_ito(int16_t, i); + table->_->ft_offset = flecs_ito(int16_t, i); } else if (r == ecs_id(EcsPoly)) { table->flags |= EcsTableHasBuiltins; } } else { if (ECS_HAS_ID_FLAG(id, TOGGLE)) { - ecs_data_t *data = table->data; + ecs_table_data_t *data = table->data; table->flags |= EcsTableHasToggle; if (!data->bs_count) { - data->bs_offset = flecs_ito(int16_t, i); + table->_->bs_offset = flecs_ito(int16_t, i); } data->bs_count ++; } @@ -41039,7 +41102,7 @@ void flecs_table_init( ecs_table_t *table, ecs_table_t *from) { - table->data = ecs_os_calloc_t(ecs_data_t); + table->data = ecs_os_calloc_t(ecs_table_data_t); /* Make sure table->flags is initialized */ flecs_table_init_flags(world, table); @@ -41475,7 +41538,7 @@ void flecs_table_dtor_all( /* Can't delete and not update the entity index */ ecs_assert(!is_delete || update_entity_index, ECS_INTERNAL_ERROR, NULL); - ecs_data_t *data = flecs_table_data(table); + ecs_table_data_t *data = flecs_table_data(table); int32_t ids_count = data->column_count; ecs_record_t **records = data->records.array; ecs_entity_t *entities = data->entities.array; @@ -41583,7 +41646,7 @@ void flecs_table_fini_data( { ecs_assert(!table->_->lock, ECS_LOCKED_STORAGE, NULL); ecs_assert(table != NULL, ECS_INTERNAL_ERROR, NULL); - ecs_data_t *data = table->data; + ecs_table_data_t *data = table->data; ecs_assert(data != NULL, ECS_INTERNAL_ERROR, NULL); int32_t count = ecs_table_count(table); @@ -41614,14 +41677,14 @@ void flecs_table_fini_data( data->columns = NULL; } - ecs_bitset_t *bs_columns = data->bs_columns; + ecs_bitset_column_t *bs_columns = data->bitsets; if (bs_columns) { int32_t c, column_count = data->bs_count; for (c = 0; c < column_count; c ++) { - flecs_bitset_fini(&bs_columns[c]); + flecs_bitset_fini(&bs_columns[c].data); } flecs_wfree_n(world, ecs_bitset_t, column_count, bs_columns); - data->bs_columns = NULL; + data->bitsets = NULL; } ecs_vec_fini_t(&world->allocator, &data->entities, ecs_entity_t); @@ -41764,20 +41827,6 @@ void flecs_table_traversable_add( } } -/* Mark table column dirty. This usually happens as the result of a set - * operation, or iteration of a query with [out] fields. */ -static -void flecs_table_mark_table_dirty( - ecs_world_t *world, - ecs_table_t *table, - int32_t index) -{ - (void)world; - if (flecs_table_data(table)->dirty_state) { - flecs_table_data(table)->dirty_state[index] ++; - } -} - /* Mark table component dirty */ void flecs_table_mark_dirty( ecs_world_t *world, @@ -41821,154 +41870,6 @@ int32_t* flecs_table_get_dirty_state( return flecs_table_data(table)->dirty_state; } -/* Table move logic for bitset (toggle component) column */ -static -void flecs_table_move_bitset_columns( - ecs_table_t *dst_table, - int32_t dst_index, - ecs_table_t *src_table, - int32_t src_index, - int32_t count, - bool clear) -{ - ecs_data_t *dst_data = flecs_table_data(dst_table); - ecs_data_t *src_data = flecs_table_data(src_table); - if (!dst_data && !src_data) { - return; - } - - int32_t i_old = 0, src_column_count = src_data ? src_data->bs_count : 0; - int32_t i_new = 0, dst_column_count = dst_data ? dst_data->bs_count : 0; - - if (!src_column_count && !dst_column_count) { - return; - } - - ecs_bitset_t *src_columns = src_data ? src_data->bs_columns : NULL; - ecs_bitset_t *dst_columns = dst_data ? dst_data->bs_columns : NULL; - - ecs_type_t dst_type = dst_table->type; - ecs_type_t src_type = src_table->type; - - int32_t offset_new = dst_data ? dst_data->bs_offset : 0; - int32_t offset_old = src_data ? src_data->bs_offset : 0; - - ecs_id_t *dst_ids = dst_type.array; - ecs_id_t *src_ids = src_type.array; - - for (; (i_new < dst_column_count) && (i_old < src_column_count);) { - ecs_id_t dst_id = dst_ids[i_new + offset_new]; - ecs_id_t src_id = src_ids[i_old + offset_old]; - - if (dst_id == src_id) { - ecs_bitset_t *src_bs = &src_columns[i_old]; - ecs_bitset_t *dst_bs = &dst_columns[i_new]; - - flecs_bitset_ensure(dst_bs, dst_index + count); - - int i; - for (i = 0; i < count; i ++) { - uint64_t value = flecs_bitset_get(src_bs, src_index + i); - flecs_bitset_set(dst_bs, dst_index + i, value); - } - - if (clear) { - ecs_assert(count == flecs_bitset_count(src_bs), - ECS_INTERNAL_ERROR, NULL); - flecs_bitset_fini(src_bs); - } - } else if (dst_id > src_id) { - ecs_bitset_t *src_bs = &src_columns[i_old]; - flecs_bitset_fini(src_bs); - } - - i_new += dst_id <= src_id; - i_old += dst_id >= src_id; - } - - /* Clear remaining columns */ - if (clear) { - for (; (i_old < src_column_count); i_old ++) { - ecs_bitset_t *src_bs = &src_columns[i_old]; - ecs_assert(count == flecs_bitset_count(src_bs), - ECS_INTERNAL_ERROR, NULL); - flecs_bitset_fini(src_bs); - } - } -} - -/* Grow table column. When a column needs to be reallocated this function takes - * care of correctly invoking ctor/move/dtor hooks. */ -static -void* flecs_table_grow_column( - ecs_world_t *world, - ecs_column_t *column, - int32_t to_add, - int32_t dst_size, - bool construct) -{ - ecs_assert(column != NULL, ECS_INTERNAL_ERROR, NULL); - - ecs_type_info_t *ti = column->ti; - int32_t size = column->size; - int32_t count = column->data.count; - int32_t src_size = column->data.size; - int32_t dst_count = count + to_add; - bool can_realloc = dst_size != src_size; - void *result = NULL; - - ecs_assert(dst_size >= dst_count, ECS_INTERNAL_ERROR, NULL); - - /* If the array could possibly realloc and the component has a move action - * defined, move old elements manually */ - ecs_move_t move_ctor; - if (count && can_realloc && (move_ctor = ti->hooks.ctor_move_dtor)) { - ecs_xtor_t ctor = ti->hooks.ctor; - ecs_assert(ctor != NULL, ECS_INTERNAL_ERROR, NULL); - ecs_assert(move_ctor != NULL, ECS_INTERNAL_ERROR, NULL); - - /* Create vector */ - ecs_vec_t dst; - ecs_vec_init(&world->allocator, &dst, size, dst_size); - dst.count = dst_count; - - void *src_buffer = column->data.array; - void *dst_buffer = dst.array; - - /* Move (and construct) existing elements to new vector */ - move_ctor(dst_buffer, src_buffer, count, ti); - - if (construct) { - /* Construct new element(s) */ - result = ECS_ELEM(dst_buffer, size, count); - ctor(result, to_add, ti); - } - - /* Free old vector */ - ecs_vec_fini(&world->allocator, &column->data, size); - - column->data = dst; - } else { - /* If array won't realloc or has no move, simply add new elements */ - if (can_realloc) { - ecs_vec_set_size(&world->allocator, &column->data, size, dst_size); - } - - result = ecs_vec_grow(&world->allocator, &column->data, size, to_add); - - ecs_xtor_t ctor; - if (construct && (ctor = ti->hooks.ctor)) { - /* If new elements need to be constructed and component has a - * constructor, construct */ - ctor(result, to_add, ti); - } - } - - ecs_assert(column->data.size == dst_size, ECS_INTERNAL_ERROR, NULL); - - return result; -} - /* Grow all data structures in a table */ int32_t flecs_table_appendn( ecs_world_t *world, @@ -41978,59 +41879,12 @@ int32_t flecs_table_appendn( { ecs_assert(table != NULL, ECS_INTERNAL_ERROR, NULL); ecs_assert(!table->_->lock, ECS_LOCKED_STORAGE, NULL); - ecs_data_t *data = table->data; - ecs_assert(data != NULL, ECS_INTERNAL_ERROR, NULL); flecs_table_check_sanity(table); int32_t cur_count = ecs_table_count(table); - int32_t size = to_add + cur_count; - int32_t column_count = table->data->column_count; - - /* Add record to record ptr array */ - ecs_vec_set_size_t(&world->allocator, &data->records, ecs_record_t*, size); - ecs_record_t **r = ecs_vec_last_t(&data->records, ecs_record_t*) + 1; - data->records.count += to_add; - if (data->records.size > size) { - size = data->records.size; - } - - /* Add entity to column with entity ids */ - ecs_vec_set_size_t(&world->allocator, &data->entities, ecs_entity_t, size); - ecs_entity_t *e = ecs_vec_last_t(&data->entities, ecs_entity_t) + 1; - data->entities.count += to_add; - ecs_assert(data->entities.size == size, ECS_INTERNAL_ERROR, NULL); - - /* Initialize entity ids and record ptrs */ - int32_t i; - if (ids) { - ecs_os_memcpy_n(e, ids, ecs_entity_t, to_add); - } else { - ecs_os_memset(e, 0, ECS_SIZEOF(ecs_entity_t) * to_add); - } - ecs_os_memset(r, 0, ECS_SIZEOF(ecs_record_t*) * to_add); - - /* Add elements to each column array */ - ecs_column_t *columns = data->columns; - for (i = 0; i < column_count; i ++) { - flecs_table_grow_column(world, &columns[i], to_add, size, true); - ecs_assert(columns[i].data.size == size, ECS_INTERNAL_ERROR, NULL); - flecs_table_invoke_add_hooks(world, table, &columns[i], e, - cur_count, to_add, false); - } - - int32_t bs_count = data->bs_count; - ecs_bitset_t *bs_columns = data->bs_columns; - - /* Add elements to each bitset column */ - for (i = 0; i < bs_count; i ++) { - ecs_bitset_t *bs = &bs_columns[i]; - flecs_bitset_addn(bs, to_add); - } - - /* If the table is monitored indicate that there has been a change */ - flecs_table_mark_table_dirty(world, table, 0); + flecs_table_data_appendn(world, table, to_add, ids); if (!(world->flags & EcsWorldReadonly) && !cur_count) { flecs_table_set_empty(world, table); } @@ -42041,21 +41895,6 @@ int32_t flecs_table_appendn( return cur_count; } -/* Append operation for tables that don't have any complex logic */ -static -void flecs_table_fast_append( - ecs_world_t *world, - ecs_column_t *columns, - int32_t count) -{ - /* Add elements to each column array */ - int32_t i; - for (i = 0; i < count; i ++) { - ecs_column_t *column = &columns[i]; - ecs_vec_append(&world->allocator, &column->data, column->size); - } -} - /* Append entity to table */ int32_t flecs_table_append( ecs_world_t *world, @@ -42072,108 +41911,17 @@ int32_t flecs_table_append( flecs_table_check_sanity(table); - /* Get count & size before growing entities array. This tells us whether the - * arrays will realloc */ - ecs_data_t *data = table->data; - int32_t count = data->entities.count; - int32_t column_count = table->data->column_count; - ecs_column_t *columns = flecs_table_columns(table); - - /* Grow buffer with entity ids, set new element to new entity */ - ecs_entity_t *e = ecs_vec_append_t(&world->allocator, - &data->entities, ecs_entity_t); - ecs_assert(e != NULL, ECS_INTERNAL_ERROR, NULL); - *e = entity; - - /* Add record ptr to array with record ptrs */ - ecs_record_t **r = ecs_vec_append_t(&world->allocator, - &data->records, ecs_record_t*); - ecs_assert(r != NULL, ECS_INTERNAL_ERROR, NULL); - *r = record; - - /* If the table is monitored indicate that there has been a change */ - flecs_table_mark_table_dirty(world, table, 0); - ecs_assert(count >= 0, ECS_INTERNAL_ERROR, NULL); - - /* Fast path: no switch columns, no lifecycle actions */ - if (!(table->flags & EcsTableIsComplex)) { - flecs_table_fast_append(world, columns, column_count); - if (!count) { - flecs_table_set_empty(world, table); /* See below */ - } - return count; - } - - ecs_entity_t *entities = data->entities.array; - - /* Reobtain size to ensure that the columns have the same size as the - * entities and record vectors. This keeps reasoning about when allocations - * occur easier. */ - int32_t size = data->entities.size; - - /* Grow component arrays with 1 element */ - int32_t i; - for (i = 0; i < column_count; i ++) { - ecs_column_t *column = &columns[i]; - flecs_table_grow_column(world, column, 1, size, construct); - - ecs_iter_action_t on_add_hook; - if (on_add && (on_add_hook = column->ti->hooks.on_add)) { - flecs_table_invoke_hook(world, table, on_add_hook, EcsOnAdd, column, - &entities[count], count, 1); - } - - ecs_assert(columns[i].data.size == - data->entities.size, ECS_INTERNAL_ERROR, NULL); - ecs_assert(columns[i].data.count == - data->entities.count, ECS_INTERNAL_ERROR, NULL); - } - - int32_t bs_count = data->bs_count; - ecs_bitset_t *bs_columns = data->bs_columns; - - /* Add element to each bitset column */ - for (i = 0; i < bs_count; i ++) { - ecs_assert(bs_columns != NULL, ECS_INTERNAL_ERROR, NULL); - ecs_bitset_t *bs = &bs_columns[i]; - flecs_bitset_addn(bs, 1); - } - - /* If this is the first entity in this table, signal queries so that the - * table moves from an inactive table to an active table. */ - if (!count) { + int32_t row = flecs_table_data_append(world, table, + entity, record, construct, on_add); + if (row == 0) { + /* If this is the first entity in this table, signal queries so that the + * table moves from an inactive table to an active table. */ flecs_table_set_empty(world, table); } flecs_table_check_sanity(table); - return count; -} - -/* Delete last operation for tables that don't have any complex logic */ -static -void flecs_table_fast_delete_last( - ecs_column_t *columns, - int32_t column_count) -{ - int i; - for (i = 0; i < column_count; i ++) { - ecs_vec_remove_last(&columns[i].data); - } -} - -/* Delete operation for tables that don't have any complex logic */ -static -void flecs_table_fast_delete( - ecs_column_t *columns, - int32_t column_count, - int32_t index) -{ - int i; - for (i = 0; i < column_count; i ++) { - ecs_column_t *column = &columns[i]; - ecs_vec_remove(&column->data, column->size, index); - } + return row; } /* Delete entity from table */ @@ -42191,149 +41939,13 @@ void flecs_table_delete( flecs_table_check_sanity(table); - ecs_data_t *data = table->data; - int32_t count = data->entities.count; - - ecs_assert(count > 0, ECS_INTERNAL_ERROR, NULL); - count --; - ecs_assert(index <= count, ECS_INTERNAL_ERROR, NULL); - - /* Move last entity id to index */ - ecs_entity_t *entities = data->entities.array; - ecs_entity_t entity_to_move = entities[count]; - ecs_entity_t entity_to_delete = entities[index]; - entities[index] = entity_to_move; - ecs_vec_remove_last(&data->entities); - - /* Move last record ptr to index */ - ecs_assert(count < data->records.count, ECS_INTERNAL_ERROR, NULL); - - ecs_record_t **records = data->records.array; - ecs_record_t *record_to_move = records[count]; - records[index] = record_to_move; - ecs_vec_remove_last(&data->records); - - /* Update record of moved entity in entity index */ - if (index != count) { - if (record_to_move) { - uint32_t row_flags = record_to_move->row & ECS_ROW_FLAGS_MASK; - record_to_move->row = ECS_ROW_TO_RECORD(index, row_flags); - ecs_assert(record_to_move->table != NULL, ECS_INTERNAL_ERROR, NULL); - ecs_assert(record_to_move->table == table, ECS_INTERNAL_ERROR, NULL); - } - } - - /* If the table is monitored indicate that there has been a change */ - flecs_table_mark_table_dirty(world, table, 0); - - /* If table is empty, deactivate it */ - if (!count) { + if (!flecs_table_data_delete(world, table, index, destruct)) { flecs_table_set_empty(world, table); } - /* Destruct component data */ - ecs_column_t *columns = data->columns; - int32_t column_count = table->data->column_count; - int32_t i; - - /* If this is a table without lifecycle callbacks or special columns, take - * fast path that just remove an element from the array(s) */ - if (!(table->flags & EcsTableIsComplex)) { - if (index == count) { - flecs_table_fast_delete_last(columns, column_count); - } else { - flecs_table_fast_delete(columns, column_count, index); - } - - flecs_table_check_sanity(table); - return; - } - - /* Last element, destruct & remove */ - if (index == count) { - /* If table has component destructors, invoke */ - if (destruct && (table->flags & EcsTableHasDtors)) { - for (i = 0; i < column_count; i ++) { - flecs_table_invoke_remove_hooks(world, table, &columns[i], - &entity_to_delete, index, 1, true); - } - } - - flecs_table_fast_delete_last(columns, column_count); - - /* Not last element, move last element to deleted element & destruct */ - } else { - /* If table has component destructors, invoke */ - if ((table->flags & (EcsTableHasDtors | EcsTableHasMove))) { - for (i = 0; i < column_count; i ++) { - ecs_column_t *column = &columns[i]; - ecs_type_info_t *ti = column->ti; - ecs_size_t size = column->size; - void *dst = ecs_vec_get(&column->data, size, index); - void *src = ecs_vec_last(&column->data, size); - - ecs_iter_action_t on_remove = ti->hooks.on_remove; - if (destruct && on_remove) { - flecs_table_invoke_hook(world, table, on_remove, EcsOnRemove, - column, &entity_to_delete, index, 1); - } - - ecs_move_t move_dtor = ti->hooks.move_dtor; - if (move_dtor) { - move_dtor(dst, src, 1, ti); - } else { - ecs_os_memcpy(dst, src, size); - } - - ecs_vec_remove_last(&column->data); - } - } else { - flecs_table_fast_delete(columns, column_count, index); - } - } - - /* Remove elements from bitset columns */ - ecs_bitset_t *bs_columns = data->bs_columns; - int32_t bs_count = data->bs_count; - for (i = 0; i < bs_count; i ++) { - flecs_bitset_remove(&bs_columns[i], index); - } - flecs_table_check_sanity(table); } -/* Move operation for tables that don't have any complex logic */ -static -void flecs_table_fast_move( - ecs_table_t *dst_table, - int32_t dst_index, - ecs_table_t *src_table, - int32_t src_index) -{ - int32_t i_new = 0, dst_column_count = dst_table->data->column_count; - int32_t i_old = 0, src_column_count = src_table->data->column_count; - - ecs_column_t *src_columns = flecs_table_columns(src_table); - ecs_column_t *dst_columns = flecs_table_columns(dst_table); - - for (; (i_new < dst_column_count) && (i_old < src_column_count);) { - ecs_column_t *dst_column = &dst_columns[i_new]; - ecs_column_t *src_column = &src_columns[i_old]; - ecs_id_t dst_id = dst_column->id; - ecs_id_t src_id = src_column->id; - - if (dst_id == src_id) { - int32_t size = dst_column->size; - void *dst = ecs_vec_get(&dst_column->data, size, dst_index); - void *src = ecs_vec_get(&src_column->data, size, src_index); - ecs_os_memcpy(dst, src, size); - } - - i_new += dst_id <= src_id; - i_old += dst_id >= src_id; - } -} - /* Move entity from src to dst table */ void flecs_table_move( ecs_world_t *world, @@ -42356,140 +41968,13 @@ void flecs_table_move( flecs_table_check_sanity(dst_table); flecs_table_check_sanity(src_table); - if (!((dst_table->flags | src_table->flags) & EcsTableIsComplex)) { - flecs_table_fast_move(dst_table, dst_index, src_table, src_index); - flecs_table_check_sanity(dst_table); - flecs_table_check_sanity(src_table); - return; - } - - flecs_table_move_bitset_columns(dst_table, dst_index, src_table, src_index, 1, false); - - /* If the source and destination entities are the same, move component - * between tables. If the entities are not the same (like when cloning) use - * a copy. */ - bool same_entity = dst_entity == src_entity; - - /* Call move_dtor for moved away from storage only if the entity is at the - * last index in the source table. If it isn't the last entity, the last - * entity in the table will be moved to the src storage, which will take - * care of cleaning up resources. */ - bool use_move_dtor = ecs_table_count(src_table) == (src_index + 1); - - int32_t i_new = 0, dst_column_count = dst_table->data->column_count; - int32_t i_old = 0, src_column_count = src_table->data->column_count; - - ecs_column_t *src_columns = flecs_table_columns(src_table); - ecs_column_t *dst_columns = flecs_table_columns(dst_table); - - for (; (i_new < dst_column_count) && (i_old < src_column_count); ) { - ecs_column_t *dst_column = &dst_columns[i_new]; - ecs_column_t *src_column = &src_columns[i_old]; - ecs_id_t dst_id = dst_column->id; - ecs_id_t src_id = src_column->id; - - if (dst_id == src_id) { - int32_t size = dst_column->size; - - ecs_assert(size != 0, ECS_INTERNAL_ERROR, NULL); - void *dst = ecs_vec_get(&dst_column->data, size, dst_index); - void *src = ecs_vec_get(&src_column->data, size, src_index); - ecs_type_info_t *ti = dst_column->ti; - - if (same_entity) { - ecs_move_t move = ti->hooks.move_ctor; - if (use_move_dtor || !move) { - /* Also use move_dtor if component doesn't have a move_ctor - * registered, to ensure that the dtor gets called to - * cleanup resources. */ - move = ti->hooks.ctor_move_dtor; - } - - if (move) { - move(dst, src, 1, ti); - } else { - ecs_os_memcpy(dst, src, size); - } - } else { - ecs_copy_t copy = ti->hooks.copy_ctor; - if (copy) { - copy(dst, src, 1, ti); - } else { - ecs_os_memcpy(dst, src, size); - } - } - } else { - if (dst_id < src_id) { - flecs_table_invoke_add_hooks(world, dst_table, - dst_column, &dst_entity, dst_index, 1, construct); - } else { - flecs_table_invoke_remove_hooks(world, src_table, - src_column, &src_entity, src_index, 1, use_move_dtor); - } - } - - i_new += dst_id <= src_id; - i_old += dst_id >= src_id; - } - - for (; (i_new < dst_column_count); i_new ++) { - flecs_table_invoke_add_hooks(world, dst_table, &dst_columns[i_new], - &dst_entity, dst_index, 1, construct); - } - - for (; (i_old < src_column_count); i_old ++) { - flecs_table_invoke_remove_hooks(world, src_table, &src_columns[i_old], - &src_entity, src_index, 1, use_move_dtor); - } + flecs_table_data_move(world, dst_entity, src_entity, dst_table, dst_index, + src_table, src_index, construct); flecs_table_check_sanity(dst_table); flecs_table_check_sanity(src_table); } -/* Shrink table storage to fit number of entities */ -bool flecs_table_shrink( - ecs_world_t *world, - ecs_table_t *table) -{ - ecs_assert(table != NULL, ECS_LOCKED_STORAGE, NULL); - ecs_assert(!table->_->lock, ECS_LOCKED_STORAGE, NULL); - (void)world; - - flecs_table_check_sanity(table); - - ecs_data_t *data = table->data; - bool has_payload = data->entities.array != NULL; - ecs_vec_reclaim_t(&world->allocator, &data->entities, ecs_entity_t); - ecs_vec_reclaim_t(&world->allocator, &data->records, ecs_record_t*); - - int32_t i, count = table->data->column_count; - for (i = 0; i < count; i ++) { - ecs_column_t *column = &data->columns[i]; - ecs_vec_reclaim(&world->allocator, &column->data, column->size); - } - - return has_payload; -} - -/* Swap operation for bitset (toggle component) columns */ -static -void flecs_table_swap_bitset_columns( - ecs_table_t *table, - int32_t row_1, - int32_t row_2) -{ - int32_t i = 0, column_count = flecs_table_data(table)->bs_count; - if (!column_count) { - return; - } - - ecs_bitset_t *columns = flecs_table_data(table)->bs_columns; - for (i = 0; i < column_count; i ++) { - ecs_bitset_t *bs = &columns[i]; - flecs_bitset_swap(bs, row_1, row_2); - } -} - /* Swap two rows in a table. Used for table sorting. */ void flecs_table_swap( ecs_world_t *world, @@ -42504,192 +41989,12 @@ void flecs_table_swap( ecs_assert(row_2 >= 0, ECS_INTERNAL_ERROR, NULL); flecs_table_check_sanity(table); - - if (row_1 == row_2) { - return; - } - - /* If the table is monitored indicate that there has been a change */ - flecs_table_mark_table_dirty(world, table, 0); - - ecs_entity_t *entities = flecs_table_entities_array(table); - ecs_entity_t e1 = entities[row_1]; - ecs_entity_t e2 = entities[row_2]; - - ecs_record_t **records = flecs_table_records_array(table); - ecs_record_t *record_ptr_1 = records[row_1]; - ecs_record_t *record_ptr_2 = records[row_2]; - - ecs_assert(record_ptr_1 != NULL, ECS_INTERNAL_ERROR, NULL); - ecs_assert(record_ptr_2 != NULL, ECS_INTERNAL_ERROR, NULL); - - /* Keep track of whether entity is watched */ - uint32_t flags_1 = ECS_RECORD_TO_ROW_FLAGS(record_ptr_1->row); - uint32_t flags_2 = ECS_RECORD_TO_ROW_FLAGS(record_ptr_2->row); - - /* Swap entities & records */ - entities[row_1] = e2; - entities[row_2] = e1; - record_ptr_1->row = ECS_ROW_TO_RECORD(row_2, flags_1); - record_ptr_2->row = ECS_ROW_TO_RECORD(row_1, flags_2); - records[row_1] = record_ptr_2; - records[row_2] = record_ptr_1; - - flecs_table_swap_bitset_columns(table, row_1, row_2); - - ecs_column_t *columns = flecs_table_columns(table); - if (!columns) { - flecs_table_check_sanity(table); - return; - } - - /* Find the maximum size of column elements - * and allocate a temporary buffer for swapping */ - int32_t i, temp_buffer_size = ECS_SIZEOF(uint64_t); - int32_t column_count = table->data->column_count; - for (i = 0; i < column_count; i++) { - temp_buffer_size = ECS_MAX(temp_buffer_size, columns[i].size); - } - - void* tmp = ecs_os_alloca(temp_buffer_size); - - /* Swap columns */ - for (i = 0; i < column_count; i ++) { - int32_t size = columns[i].size; - void *ptr = columns[i].data.array; - - void *el_1 = ECS_ELEM(ptr, size, row_1); - void *el_2 = ECS_ELEM(ptr, size, row_2); - ecs_os_memcpy(tmp, el_1, size); - ecs_os_memcpy(el_1, el_2, size); - ecs_os_memcpy(el_2, tmp, size); - } + flecs_table_data_swap(world, table, row_1, row_2); flecs_table_check_sanity(table); } -/* Merge data from one table column into other table column */ -static -void flecs_table_merge_column( - ecs_world_t *world, - ecs_column_t *dst, - ecs_column_t *src, - int32_t column_size) -{ - ecs_size_t size = dst->size; - int32_t dst_count = dst->data.count; - - if (!dst_count) { - ecs_vec_fini(&world->allocator, &dst->data, size); - *dst = *src; - src->data.array = NULL; - src->data.count = 0; - src->data.size = 0; - - /* If the new table is not empty, copy the contents from the - * src into the dst. */ - } else { - int32_t src_count = src->data.count; - - flecs_table_grow_column(world, dst, src_count, column_size, true); - void *dst_ptr = ECS_ELEM(dst->data.array, size, dst_count); - void *src_ptr = src->data.array; - - /* Move values into column */ - ecs_type_info_t *ti = dst->ti; - ecs_assert(ti != NULL, ECS_INTERNAL_ERROR, NULL); - ecs_move_t move = ti->hooks.move_dtor; - if (move) { - move(dst_ptr, src_ptr, src_count, ti); - } else { - ecs_os_memcpy(dst_ptr, src_ptr, size * src_count); - } - - ecs_vec_fini(&world->allocator, &src->data, size); - } -} - -/* Merge storage of two tables. */ -static -void flecs_table_merge_data( - ecs_world_t *world, - ecs_table_t *dst_table, - ecs_table_t *src_table, - int32_t src_count, - int32_t dst_count, - ecs_data_t *src_data, - ecs_data_t *dst_data) -{ - int32_t i_new = 0, dst_column_count = dst_table->data->column_count; - int32_t i_old = 0, src_column_count = src_table->data->column_count; - ecs_column_t *src_columns = src_data->columns; - ecs_column_t *dst_columns = dst_data->columns; - - ecs_assert(!dst_column_count || dst_columns, ECS_INTERNAL_ERROR, NULL); - - if (!src_count) { - return; - } - - /* Merge entities & records vectors */ - ecs_allocator_t *a = &world->allocator; - ecs_vec_merge_t(a, &dst_data->entities, &src_data->entities, ecs_entity_t); - ecs_assert(dst_data->entities.count == src_count + dst_count, - ECS_INTERNAL_ERROR, NULL); - ecs_vec_merge_t(a, &dst_data->records, &src_data->records, ecs_record_t*); - - int32_t column_size = dst_data->entities.size; - for (; (i_new < dst_column_count) && (i_old < src_column_count); ) { - ecs_column_t *dst_column = &dst_columns[i_new]; - ecs_column_t *src_column = &src_columns[i_old]; - ecs_id_t dst_id = dst_column->id; - ecs_id_t src_id = src_column->id; - - if (dst_id == src_id) { - flecs_table_merge_column(world, dst_column, src_column, column_size); - flecs_table_mark_table_dirty(world, dst_table, i_new + 1); - i_new ++; - i_old ++; - } else if (dst_id < src_id) { - /* New column, make sure vector is large enough. */ - ecs_size_t size = dst_column->size; - ecs_vec_set_size(a, &dst_column->data, size, column_size); - ecs_vec_set_count(a, &dst_column->data, size, src_count + dst_count); - flecs_table_invoke_ctor(dst_column, dst_count, src_count); - i_new ++; - } else if (dst_id > src_id) { - /* Old column does not occur in new table, destruct */ - flecs_table_invoke_dtor(src_column, 0, src_count); - ecs_vec_fini(a, &src_column->data, src_column->size); - i_old ++; - } - } - - flecs_table_move_bitset_columns( - dst_table, dst_count, src_table, 0, src_count, true); - - /* Initialize remaining columns */ - for (; i_new < dst_column_count; i_new ++) { - ecs_column_t *column = &dst_columns[i_new]; - int32_t size = column->size; - ecs_assert(size != 0, ECS_INTERNAL_ERROR, NULL); - ecs_vec_set_size(a, &column->data, size, column_size); - ecs_vec_set_count(a, &column->data, size, src_count + dst_count); - flecs_table_invoke_ctor(column, dst_count, src_count); - } - - /* Destruct remaining columns */ - for (; i_old < src_column_count; i_old ++) { - ecs_column_t *column = &src_columns[i_old]; - flecs_table_invoke_dtor(column, 0, src_count); - ecs_vec_fini(a, &column->data, column->size); - } - - /* Mark entity column as dirty */ - flecs_table_mark_table_dirty(world, dst_table, 0); -} - /* Merge source table into destination table. This typically happens as result * of a bulk operation, like when a component is removed from all entities in * the source table (like for the Remove OnDelete policy). */ @@ -42703,11 +42008,9 @@ void flecs_table_merge( flecs_table_check_sanity(src_table); flecs_table_check_sanity(dst_table); - - bool move_data = false; - ecs_data_t *dst_data = dst_table->data; - ecs_data_t *src_data = src_table->data; + ecs_table_data_t *dst_data = dst_table->data; + ecs_table_data_t *src_data = src_table->data; ecs_assert(dst_data != NULL, ECS_INTERNAL_ERROR, NULL); ecs_assert(src_data != NULL, ECS_INTERNAL_ERROR, NULL); @@ -42720,46 +42023,10 @@ void flecs_table_merge( ecs_assert(!dst_table->_->lock, ECS_LOCKED_STORAGE, NULL); } - /* If there is no data to merge, drop out */ - if (!src_data) { - return; - } - - if (!dst_data) { - dst_data = dst_table->data; - if (dst_table == src_table) { - move_data = true; - } - } - - ecs_entity_t *src_entities = src_data->entities.array; int32_t src_count = src_data->entities.count; int32_t dst_count = dst_data->entities.count; - ecs_record_t **src_records = src_data->records.array; - - /* First, update entity index so old entities point to new type */ - int32_t i; - for(i = 0; i < src_count; i ++) { - ecs_record_t *record; - if (dst_table != src_table) { - record = src_records[i]; - ecs_assert(record != NULL, ECS_INTERNAL_ERROR, NULL); - } else { - record = flecs_entities_ensure(world, src_entities[i]); - } - - uint32_t flags = ECS_RECORD_TO_ROW_FLAGS(record->row); - record->row = ECS_ROW_TO_RECORD(dst_count + i, flags); - record->table = dst_table; - } - /* Merge table columns */ - if (move_data) { - *dst_data = *src_data; - } else { - flecs_table_merge_data(world, dst_table, src_table, src_count, dst_count, - src_data, dst_data); - } + flecs_table_data_merge(world, dst_table, src_table); if (src_count) { if (!dst_count) { @@ -42776,6 +42043,24 @@ void flecs_table_merge( flecs_table_check_sanity(dst_table); } +/* Shrink table storage to fit number of entities */ +bool flecs_table_shrink( + ecs_world_t *world, + ecs_table_t *table) +{ + ecs_assert(table != NULL, ECS_LOCKED_STORAGE, NULL); + ecs_assert(!table->_->lock, ECS_LOCKED_STORAGE, NULL); + (void)world; + + flecs_table_check_sanity(table); + + bool has_payload = flecs_table_data_shrink(world, table); + + flecs_table_check_sanity(table); + + return has_payload; +} + /* Internal mechanism for propagating information to tables */ void flecs_table_notify( ecs_world_t *world, @@ -43340,6 +42625,965 @@ ecs_table_cache_hdr_t* flecs_table_cache_next_( return next; } +/** + * @file table_data.c + * @brief Table data implementation. + */ + + +/* Construct components */ +static +void flecs_table_data_invoke_ctor( + ecs_column_t *column, + int32_t row, + int32_t count) +{ + ecs_type_info_t *ti = column->ti; + ecs_assert(ti != NULL, ECS_INTERNAL_ERROR, NULL); + ecs_xtor_t ctor = ti->hooks.ctor; + if (ctor) { + void *ptr = ecs_vec_get(&column->data, column->size, row); + ctor(ptr, count, ti); + } +} + +/* Destruct components */ +static +void flecs_table_data_invoke_dtor( + ecs_column_t *column, + int32_t row, + int32_t count) +{ + ecs_type_info_t *ti = column->ti; + ecs_assert(ti != NULL, ECS_INTERNAL_ERROR, NULL); + ecs_xtor_t dtor = ti->hooks.dtor; + if (dtor) { + void *ptr = ecs_vec_get(&column->data, column->size, row); + dtor(ptr, count, ti); + } +} + +/* Invoke type hook for entities in table */ +static +void flecs_table_data_invoke_hook( + ecs_world_t *world, + ecs_table_t *table, + ecs_iter_action_t callback, + ecs_entity_t event, + ecs_column_t *column, + ecs_entity_t *entities, + int32_t row, + int32_t count) +{ + void *ptr = ecs_vec_get(&column->data, column->size, row); + flecs_invoke_hook(world, table, count, row, entities, ptr, column->id, + column->ti, event, callback); +} + +/* Run hooks that get invoked when component is added to entity */ +static +void flecs_table_data_invoke_add_hooks( + ecs_world_t *world, + ecs_table_t *table, + ecs_column_t *column, + ecs_entity_t *entities, + int32_t row, + int32_t count, + bool construct) +{ + ecs_type_info_t *ti = column->ti; + ecs_assert(ti != NULL, ECS_INTERNAL_ERROR, NULL); + + if (construct) { + flecs_table_data_invoke_ctor(column, row, count); + } + + ecs_iter_action_t on_add = ti->hooks.on_add; + if (on_add) { + flecs_table_data_invoke_hook(world, table, on_add, EcsOnAdd, column, + entities, row, count); + } +} + +/* Run hooks that get invoked when component is removed from entity */ +static +void flecs_table_data_invoke_remove_hooks( + ecs_world_t *world, + ecs_table_t *table, + ecs_column_t *column, + ecs_entity_t *entities, + int32_t row, + int32_t count, + bool dtor) +{ + ecs_type_info_t *ti = column->ti; + ecs_assert(ti != NULL, ECS_INTERNAL_ERROR, NULL); + + ecs_iter_action_t on_remove = ti->hooks.on_remove; + if (on_remove) { + flecs_table_data_invoke_hook(world, table, on_remove, EcsOnRemove, + column, entities, row, count); + } + + if (dtor) { + flecs_table_data_invoke_dtor(column, row, count); + } +} + +/* Mark table column dirty. This usually happens as the result of a set + * operation, or iteration of a query with [out] fields. */ +static +void flecs_table_data_mark_table_dirty( + ecs_table_data_t *data, + int32_t index) +{ + if (data->dirty_state) { + data->dirty_state[index] ++; + } +} + +/* Append operation for tables that don't have any complex logic */ +static +void flecs_table_data_fast_append( + ecs_world_t *world, + ecs_column_t *columns, + int32_t count) +{ + /* Add elements to each column array */ + int32_t i; + for (i = 0; i < count; i ++) { + ecs_column_t *column = &columns[i]; + ecs_vec_append(&world->allocator, &column->data, column->size); + } +} + +/* Grow table column. When a column needs to be reallocated this function takes + * care of correctly invoking ctor/move/dtor hooks. */ +static +void* flecs_table_data_column_append( + ecs_world_t *world, + ecs_column_t *column, + int32_t to_add, + int32_t dst_size, + bool construct) +{ + ecs_assert(column != NULL, ECS_INTERNAL_ERROR, NULL); + + ecs_type_info_t *ti = column->ti; + int32_t size = column->size; + int32_t count = column->data.count; + int32_t src_size = column->data.size; + int32_t dst_count = count + to_add; + bool can_realloc = dst_size != src_size; + void *result = NULL; + + ecs_assert(dst_size >= dst_count, ECS_INTERNAL_ERROR, NULL); + + /* If the array could possibly realloc and the component has a move action + * defined, move old elements manually */ + ecs_move_t move_ctor; + if (count && can_realloc && (move_ctor = ti->hooks.ctor_move_dtor)) { + ecs_xtor_t ctor = ti->hooks.ctor; + ecs_assert(ctor != NULL, ECS_INTERNAL_ERROR, NULL); + ecs_assert(move_ctor != NULL, ECS_INTERNAL_ERROR, NULL); + + /* Create vector */ + ecs_vec_t dst; + ecs_vec_init(&world->allocator, &dst, size, dst_size); + dst.count = dst_count; + + void *src_buffer = column->data.array; + void *dst_buffer = dst.array; + + /* Move (and construct) existing elements to new vector */ + move_ctor(dst_buffer, src_buffer, count, ti); + + if (construct) { + /* Construct new element(s) */ + result = ECS_ELEM(dst_buffer, size, count); + ctor(result, to_add, ti); + } + + /* Free old vector */ + ecs_vec_fini(&world->allocator, &column->data, size); + + column->data = dst; + } else { + /* If array won't realloc or has no move, simply add new elements */ + if (can_realloc) { + ecs_vec_set_size(&world->allocator, &column->data, size, dst_size); + } + + result = ecs_vec_grow(&world->allocator, &column->data, size, to_add); + + ecs_xtor_t ctor; + if (construct && (ctor = ti->hooks.ctor)) { + /* If new elements need to be constructed and component has a + * constructor, construct */ + ctor(result, to_add, ti); + } + } + + ecs_assert(column->data.size == dst_size, ECS_INTERNAL_ERROR, NULL); + + return result; +} + +/* Append entity to table data */ +int32_t flecs_table_data_append( + ecs_world_t *world, + ecs_table_t *table, + ecs_entity_t entity, + ecs_record_t *record, + bool construct, + bool on_add) +{ + ecs_table_data_t *data = table->data; + + /* Get count & size before growing entities array. This tells us whether the + * arrays will realloc */ + int32_t count = data->entities.count; + int32_t column_count = data->column_count; + ecs_column_t *columns = data->columns; + + /* Grow buffer with entity ids, set new element to new entity */ + ecs_entity_t *e = ecs_vec_append_t(&world->allocator, + &data->entities, ecs_entity_t); + ecs_assert(e != NULL, ECS_INTERNAL_ERROR, NULL); + *e = entity; + + /* Add record ptr to array with record ptrs */ + ecs_record_t **r = ecs_vec_append_t(&world->allocator, + &data->records, ecs_record_t*); + ecs_assert(r != NULL, ECS_INTERNAL_ERROR, NULL); + *r = record; + + /* If the table is monitored indicate that there has been a change */ + flecs_table_data_mark_table_dirty(data, 0); + ecs_assert(count >= 0, ECS_INTERNAL_ERROR, NULL); + + /* Fast path: no switch columns, no lifecycle actions */ + if (!(data->flags & EcsTableIsComplex)) { + flecs_table_data_fast_append(world, columns, column_count); + return count; + } + + ecs_entity_t *entities = data->entities.array; + + /* Reobtain size to ensure that the columns have the same size as the + * entities and record vectors. This keeps reasoning about when allocations + * occur easier. */ + int32_t size = data->entities.size; + + /* Grow component arrays with 1 element */ + int32_t i; + for (i = 0; i < column_count; i ++) { + ecs_column_t *column = &columns[i]; + flecs_table_data_column_append(world, column, 1, size, construct); + + ecs_iter_action_t on_add_hook; + if (on_add && (on_add_hook = column->ti->hooks.on_add)) { + flecs_table_data_invoke_hook(world, table, on_add_hook, EcsOnAdd, + column, &entities[count], count, 1); + } + + ecs_assert(columns[i].data.size == + data->entities.size, ECS_INTERNAL_ERROR, NULL); + ecs_assert(columns[i].data.count == + data->entities.count, ECS_INTERNAL_ERROR, NULL); + } + + int32_t bs_count = data->bs_count; + ecs_bitset_column_t *bitsets = data->bitsets; + + /* Add element to each bitset column */ + for (i = 0; i < bs_count; i ++) { + ecs_assert(bitsets != NULL, ECS_INTERNAL_ERROR, NULL); + ecs_bitset_t *bs = &bitsets[i].data; + flecs_bitset_addn(bs, 1); + } + + return count; +} + +/* Grow all data structures in a table */ +int32_t flecs_table_data_appendn( + ecs_world_t *world, + ecs_table_t *table, + int32_t to_add, + const ecs_entity_t *ids) +{ + ecs_table_data_t *data = table->data; + + ecs_assert(data != NULL, ECS_INTERNAL_ERROR, NULL); + + int32_t cur_count = data->entities.count; + int32_t column_count = data->column_count; + int32_t size = to_add + cur_count; + ecs_allocator_t *a = &world->allocator; + + /* Add record to record ptr array */ + ecs_vec_set_size_t(a, &data->records, ecs_record_t*, size); + ecs_record_t **r = ecs_vec_last_t(&data->records, ecs_record_t*) + 1; + data->records.count += to_add; + if (data->records.size > size) { + size = data->records.size; + } + + /* Add entity to column with entity ids */ + ecs_vec_set_size_t(a, &data->entities, ecs_entity_t, size); + ecs_entity_t *e = ecs_vec_last_t(&data->entities, ecs_entity_t) + 1; + data->entities.count += to_add; + ecs_assert(data->entities.size == size, ECS_INTERNAL_ERROR, NULL); + + /* Initialize entity ids and record ptrs */ + int32_t i; + if (ids) { + ecs_os_memcpy_n(e, ids, ecs_entity_t, to_add); + } else { + ecs_os_memset(e, 0, ECS_SIZEOF(ecs_entity_t) * to_add); + } + ecs_os_memset(r, 0, ECS_SIZEOF(ecs_record_t*) * to_add); + + /* Add elements to each column array */ + ecs_column_t *columns = data->columns; + for (i = 0; i < column_count; i ++) { + flecs_table_data_column_append(world, &columns[i], to_add, size, true); + ecs_assert(columns[i].data.size == size, ECS_INTERNAL_ERROR, NULL); + flecs_table_data_invoke_add_hooks(world, table, &columns[i], e, + cur_count, to_add, false); + } + + int32_t bs_count = data->bs_count; + ecs_bitset_column_t *bitsets = data->bitsets; + + /* Add elements to each bitset column */ + for (i = 0; i < bs_count; i ++) { + ecs_assert(bitsets != NULL, ECS_INTERNAL_ERROR, NULL); + ecs_bitset_t *bs = &bitsets[i].data; + flecs_bitset_addn(bs, to_add); + } + + /* If the table is monitored indicate that there has been a change */ + flecs_table_data_mark_table_dirty(data, 0); + + /* Return index of first added entity */ + return cur_count; +} + +/* Move operation for tables that don't have any complex logic */ +static +void flecs_table_fast_move( + ecs_table_data_t *dst_data, + int32_t dst_index, + ecs_table_data_t *src_data, + int32_t src_index) +{ + int32_t i_dst = 0, dst_column_count = dst_data->column_count; + int32_t i_src = 0, src_column_count = src_data->column_count; + + ecs_column_t *dst_columns = dst_data->columns; + ecs_column_t *src_columns = src_data->columns; + + for (; (i_dst < dst_column_count) && (i_src < src_column_count);) { + ecs_column_t *dst_column = &dst_columns[i_dst]; + ecs_column_t *src_column = &src_columns[i_src]; + ecs_id_t dst_id = dst_column->id; + ecs_id_t src_id = src_column->id; + + if (dst_id == src_id) { + int32_t size = dst_column->size; + void *dst = ecs_vec_get(&dst_column->data, size, dst_index); + void *src = ecs_vec_get(&src_column->data, size, src_index); + ecs_os_memcpy(dst, src, size); + } + + i_dst += dst_id <= src_id; + i_src += dst_id >= src_id; + } +} + +/* Table move logic for bitset (toggle component) column */ +static +void flecs_table_data_move_bitset_columns( + ecs_table_t *dst_table, + int32_t dst_index, + ecs_table_t *src_table, + int32_t src_index, + int32_t count, + bool clear) +{ + ecs_table_data_t *dst_data = dst_table->data; + ecs_table_data_t *src_data = src_table->data; + + int32_t i_dst = 0, dst_column_count = dst_data->bs_count; + int32_t i_src = 0, src_column_count = src_data->bs_count; + + if (!src_column_count && !dst_column_count) { + return; + } + + ecs_bitset_column_t *src_columns = src_data->bitsets; + ecs_bitset_column_t *dst_columns = dst_data->bitsets; + + for (; (i_dst < dst_column_count) && (i_src < src_column_count);) { + ecs_bitset_column_t *dst_column = &dst_columns[i_dst]; + ecs_bitset_column_t *src_column = &src_columns[i_src]; + ecs_id_t dst_id = dst_column->id, src_id = src_column->id; + + if (dst_id == src_id) { + ecs_bitset_t *src_bs = &src_columns[i_src].data; + ecs_bitset_t *dst_bs = &dst_columns[i_dst].data; + + flecs_bitset_ensure(dst_bs, dst_index + count); + + int i; + for (i = 0; i < count; i ++) { + uint64_t value = flecs_bitset_get(src_bs, src_index + i); + flecs_bitset_set(dst_bs, dst_index + i, value); + } + + if (clear) { + ecs_assert(count == flecs_bitset_count(src_bs), + ECS_INTERNAL_ERROR, NULL); + flecs_bitset_fini(src_bs); + } + } else if (dst_id > src_id) { + ecs_bitset_t *src_bs = &src_columns[i_src].data; + flecs_bitset_fini(src_bs); + } + + i_dst += dst_id <= src_id; + i_src += dst_id >= src_id; + } + + /* Clear remaining columns */ + if (clear) { + for (; (i_src < src_column_count); i_src ++) { + ecs_bitset_t *src_bs = &src_columns[i_src].data; + ecs_assert(count == flecs_bitset_count(src_bs), + ECS_INTERNAL_ERROR, NULL); + flecs_bitset_fini(src_bs); + } + } +} + +/* Move entity from src to dst table */ +void flecs_table_data_move( + ecs_world_t *world, + ecs_entity_t dst_entity, + ecs_entity_t src_entity, + ecs_table_t *dst_table, + int32_t dst_index, + ecs_table_t *src_table, + int32_t src_index, + bool construct) +{ + ecs_table_data_t *dst_data = dst_table->data; + ecs_table_data_t *src_data = src_table->data; + ecs_assert(dst_data != NULL, ECS_INTERNAL_ERROR, NULL); + ecs_assert(src_data != NULL, ECS_INTERNAL_ERROR, NULL); + + if (!((dst_data->flags | src_data->flags) & EcsTableIsComplex)) { + flecs_table_fast_move(dst_data, dst_index, src_data, src_index); + return; + } + + flecs_table_data_move_bitset_columns( + dst_table, dst_index, src_table, src_index, 1, false); + + /* If the source and destination entities are the same, move component + * between tables. If the entities are not the same (like when cloning) use + * a copy. */ + bool same_entity = dst_entity == src_entity; + + /* Call move_dtor for moved away from storage only if the entity is at the + * last index in the source table. If it isn't the last entity, the last + * entity in the table will be moved to the src storage, which will take + * care of cleaning up resources. */ + bool use_move_dtor = ecs_table_count(src_table) == (src_index + 1); + + int32_t i_dst = 0, dst_column_count = dst_data->column_count; + int32_t i_src = 0, src_column_count = src_data->column_count; + + ecs_column_t *dst_columns = dst_data->columns; + ecs_column_t *src_columns = src_data->columns; + + for (; (i_dst < dst_column_count) && (i_src < src_column_count); ) { + ecs_column_t *dst_column = &dst_columns[i_dst]; + ecs_column_t *src_column = &src_columns[i_src]; + ecs_id_t dst_id = dst_column->id; + ecs_id_t src_id = src_column->id; + + if (dst_id == src_id) { + int32_t size = dst_column->size; + + ecs_assert(size != 0, ECS_INTERNAL_ERROR, NULL); + void *dst = ecs_vec_get(&dst_column->data, size, dst_index); + void *src = ecs_vec_get(&src_column->data, size, src_index); + ecs_type_info_t *ti = dst_column->ti; + + if (same_entity) { + ecs_move_t move = ti->hooks.move_ctor; + if (use_move_dtor || !move) { + /* Also use move_dtor if component doesn't have a move_ctor + * registered, to ensure that the dtor gets called to + * cleanup resources. */ + move = ti->hooks.ctor_move_dtor; + } + + if (move) { + move(dst, src, 1, ti); + } else { + ecs_os_memcpy(dst, src, size); + } + } else { + ecs_copy_t copy = ti->hooks.copy_ctor; + if (copy) { + copy(dst, src, 1, ti); + } else { + ecs_os_memcpy(dst, src, size); + } + } + } else { + if (dst_id < src_id) { + flecs_table_data_invoke_add_hooks(world, dst_table, + dst_column, &dst_entity, dst_index, 1, construct); + } else { + flecs_table_data_invoke_remove_hooks(world, src_table, + src_column, &src_entity, src_index, 1, use_move_dtor); + } + } + + i_dst += dst_id <= src_id; + i_src += dst_id >= src_id; + } + + for (; (i_dst < dst_column_count); i_dst ++) { + flecs_table_data_invoke_add_hooks(world, dst_table, &dst_columns[i_dst], + &dst_entity, dst_index, 1, construct); + } + + for (; (i_src < src_column_count); i_src ++) { + flecs_table_data_invoke_remove_hooks(world, src_table, &src_columns[i_src], + &src_entity, src_index, 1, use_move_dtor); + } +} + + +/* Delete last operation for tables that don't have any complex logic */ +static +void flecs_table_data_fast_delete_last( + ecs_column_t *columns, + int32_t column_count) +{ + int i; + for (i = 0; i < column_count; i ++) { + ecs_vec_remove_last(&columns[i].data); + } +} + +/* Delete operation for tables that don't have any complex logic */ +static +void flecs_table_data_fast_delete( + ecs_column_t *columns, + int32_t column_count, + int32_t index) +{ + int i; + for (i = 0; i < column_count; i ++) { + ecs_column_t *column = &columns[i]; + ecs_vec_remove(&column->data, column->size, index); + } +} + +/* Delete entity from table */ +int32_t flecs_table_data_delete( + ecs_world_t *world, + ecs_table_t *table, + int32_t index, + bool destruct) +{ + ecs_table_data_t *data = table->data; + ecs_assert(data != NULL, ECS_INTERNAL_ERROR, NULL); + int32_t count = data->entities.count; + + ecs_assert(count > 0, ECS_INTERNAL_ERROR, NULL); + count --; + ecs_assert(index <= count, ECS_INTERNAL_ERROR, NULL); + + /* Move last entity id to index */ + ecs_entity_t *entities = data->entities.array; + ecs_entity_t entity_to_move = entities[count]; + ecs_entity_t entity_to_delete = entities[index]; + entities[index] = entity_to_move; + ecs_vec_remove_last(&data->entities); + + /* Move last record ptr to index */ + ecs_assert(count < data->records.count, ECS_INTERNAL_ERROR, NULL); + + ecs_record_t **records = data->records.array; + ecs_record_t *record_to_move = records[count]; + records[index] = record_to_move; + ecs_vec_remove_last(&data->records); + + /* Update record of moved entity in entity index */ + if (index != count) { + if (record_to_move) { + uint32_t row_flags = record_to_move->row & ECS_ROW_FLAGS_MASK; + record_to_move->row = ECS_ROW_TO_RECORD(index, row_flags); + ecs_assert(record_to_move->table != NULL, ECS_INTERNAL_ERROR, NULL); + ecs_assert(record_to_move->table == table, ECS_INTERNAL_ERROR, NULL); + } + } + + /* If the table is monitored indicate that there has been a change */ + flecs_table_data_mark_table_dirty(data, 0); + + /* Destruct component data */ + ecs_column_t *columns = data->columns; + int32_t column_count = data->column_count; + int32_t i; + + /* If this is a table without lifecycle callbacks or special columns, take + * fast path that just remove an element from the array(s) */ + if (!(data->flags & EcsTableIsComplex)) { + if (index == count) { + flecs_table_data_fast_delete_last(columns, column_count); + } else { + flecs_table_data_fast_delete(columns, column_count, index); + } + return count; + } + + /* Last element, destruct & remove */ + if (index == count) { + /* If table has component destructors, invoke */ + if (destruct && (data->flags & EcsTableHasDtors)) { + for (i = 0; i < column_count; i ++) { + flecs_table_data_invoke_remove_hooks(world, table, &columns[i], + &entity_to_delete, index, 1, true); + } + } + + flecs_table_data_fast_delete_last(columns, column_count); + + /* Not last element, move last element to deleted element & destruct */ + } else { + /* If table has component destructors, invoke */ + if ((data->flags & (EcsTableHasDtors | EcsTableHasMove))) { + for (i = 0; i < column_count; i ++) { + ecs_column_t *column = &columns[i]; + ecs_type_info_t *ti = column->ti; + ecs_size_t size = column->size; + void *dst = ecs_vec_get(&column->data, size, index); + void *src = ecs_vec_last(&column->data, size); + + ecs_iter_action_t on_remove = ti->hooks.on_remove; + if (destruct && on_remove) { + flecs_table_data_invoke_hook(world, table, on_remove, EcsOnRemove, + column, &entity_to_delete, index, 1); + } + + ecs_move_t move_dtor = ti->hooks.move_dtor; + if (move_dtor) { + move_dtor(dst, src, 1, ti); + } else { + ecs_os_memcpy(dst, src, size); + } + + ecs_vec_remove_last(&column->data); + } + } else { + flecs_table_data_fast_delete(columns, column_count, index); + } + } + + /* Remove elements from bitset columns */ + ecs_bitset_column_t *bitsets = data->bitsets; + int32_t bs_count = data->bs_count; + for (i = 0; i < bs_count; i ++) { + ecs_assert(bitsets != NULL, ECS_INTERNAL_ERROR, NULL); + flecs_bitset_remove(&bitsets[i].data, index); + } + + return count; +} + +/* Swap operation for bitset (toggle component) columns */ +static +void flecs_table_data_swap_bitset_columns( + ecs_table_data_t *data, + int32_t row_1, + int32_t row_2) +{ + int32_t i = 0, column_count = data->bs_count; + if (!column_count) { + return; + } + + ecs_bitset_column_t *columns = data->bitsets; + for (i = 0; i < column_count; i ++) { + ecs_bitset_t *bs = &columns[i].data; + flecs_bitset_swap(bs, row_1, row_2); + } +} + +/* Swap two rows in a table. Used for table sorting. */ +void flecs_table_data_swap( + ecs_world_t *world, + ecs_table_t *table, + int32_t row_1, + int32_t row_2) +{ + if (row_1 == row_2) { + return; + } + + ecs_table_data_t *data = table->data; + + /* If the table is monitored indicate that there has been a change */ + flecs_table_data_mark_table_dirty(data, 0); + + ecs_entity_t *entities = data->entities.array; + ecs_entity_t e1 = entities[row_1]; + ecs_entity_t e2 = entities[row_2]; + + ecs_record_t **records = data->records.array; + ecs_record_t *record_ptr_1 = records[row_1]; + ecs_record_t *record_ptr_2 = records[row_2]; + + ecs_assert(record_ptr_1 != NULL, ECS_INTERNAL_ERROR, NULL); + ecs_assert(record_ptr_2 != NULL, ECS_INTERNAL_ERROR, NULL); + + /* Keep track of whether entity is watched */ + uint32_t flags_1 = ECS_RECORD_TO_ROW_FLAGS(record_ptr_1->row); + uint32_t flags_2 = ECS_RECORD_TO_ROW_FLAGS(record_ptr_2->row); + + /* Swap entities & records */ + entities[row_1] = e2; + entities[row_2] = e1; + record_ptr_1->row = ECS_ROW_TO_RECORD(row_2, flags_1); + record_ptr_2->row = ECS_ROW_TO_RECORD(row_1, flags_2); + records[row_1] = record_ptr_2; + records[row_2] = record_ptr_1; + + flecs_table_data_swap_bitset_columns(data, row_1, row_2); + + ecs_column_t *columns = data->columns; + + /* Find the maximum size of column elements + * and allocate a temporary buffer for swapping */ + int32_t i, temp_buffer_size = ECS_SIZEOF(uint64_t); + int32_t column_count = table->data->column_count; + for (i = 0; i < column_count; i++) { + temp_buffer_size = ECS_MAX(temp_buffer_size, columns[i].size); + } + + void* tmp = ecs_os_alloca(temp_buffer_size); + + /* Swap columns */ + for (i = 0; i < column_count; i ++) { + int32_t size = columns[i].size; + void *ptr = columns[i].data.array; + + void *el_1 = ECS_ELEM(ptr, size, row_1); + void *el_2 = ECS_ELEM(ptr, size, row_2); + + ecs_os_memcpy(tmp, el_1, size); + ecs_os_memcpy(el_1, el_2, size); + ecs_os_memcpy(el_2, tmp, size); + } +} + +/* Merge data from one table column into other table column */ +static +void flecs_table_data_merge_column( + ecs_world_t *world, + ecs_column_t *dst, + ecs_column_t *src, + int32_t column_size) +{ + ecs_size_t size = dst->size; + int32_t dst_count = dst->data.count; + + if (!dst_count) { + ecs_vec_fini(&world->allocator, &dst->data, size); + *dst = *src; + src->data.array = NULL; + src->data.count = 0; + src->data.size = 0; + + /* If the new table is not empty, copy the contents from the + * src into the dst. */ + } else { + int32_t src_count = src->data.count; + + flecs_table_data_column_append(world, dst, src_count, column_size, true); + void *dst_ptr = ECS_ELEM(dst->data.array, size, dst_count); + void *src_ptr = src->data.array; + + /* Move values into column */ + ecs_type_info_t *ti = dst->ti; + ecs_assert(ti != NULL, ECS_INTERNAL_ERROR, NULL); + ecs_move_t move = ti->hooks.move_dtor; + if (move) { + move(dst_ptr, src_ptr, src_count, ti); + } else { + ecs_os_memcpy(dst_ptr, src_ptr, size * src_count); + } + + ecs_vec_fini(&world->allocator, &src->data, size); + } +} + +/* Merge storage of two tables. */ +static +void flecs_table_data_merge_columns( + ecs_world_t *world, + ecs_table_t *dst_table, + ecs_table_t *src_table, + int32_t src_count, + int32_t dst_count, + ecs_table_data_t *src_data, + ecs_table_data_t *dst_data) +{ + int32_t i_new = 0, dst_column_count = dst_table->data->column_count; + int32_t i_old = 0, src_column_count = src_table->data->column_count; + ecs_column_t *src_columns = src_data->columns; + ecs_column_t *dst_columns = dst_data->columns; + + ecs_assert(!dst_column_count || dst_columns, ECS_INTERNAL_ERROR, NULL); + + if (!src_count) { + return; + } + + /* Merge entities & records vectors */ + ecs_allocator_t *a = &world->allocator; + ecs_vec_merge_t(a, &dst_data->entities, &src_data->entities, ecs_entity_t); + ecs_assert(dst_data->entities.count == src_count + dst_count, + ECS_INTERNAL_ERROR, NULL); + ecs_vec_merge_t(a, &dst_data->records, &src_data->records, ecs_record_t*); + + int32_t column_size = dst_data->entities.size; + for (; (i_new < dst_column_count) && (i_old < src_column_count); ) { + ecs_column_t *dst_column = &dst_columns[i_new]; + ecs_column_t *src_column = &src_columns[i_old]; + ecs_id_t dst_id = dst_column->id; + ecs_id_t src_id = src_column->id; + + if (dst_id == src_id) { + flecs_table_data_merge_column(world, dst_column, src_column, column_size); + flecs_table_data_mark_table_dirty(dst_data, i_new + 1); + i_new ++; + i_old ++; + } else if (dst_id < src_id) { + /* New column, make sure vector is large enough. */ + ecs_size_t size = dst_column->size; + ecs_vec_set_size(a, &dst_column->data, size, column_size); + ecs_vec_set_count(a, &dst_column->data, size, src_count + dst_count); + flecs_table_data_invoke_ctor(dst_column, dst_count, src_count); + i_new ++; + } else if (dst_id > src_id) { + /* Old column does not occur in new table, destruct */ + flecs_table_data_invoke_dtor(src_column, 0, src_count); + ecs_vec_fini(a, &src_column->data, src_column->size); + i_old ++; + } + } + + flecs_table_data_move_bitset_columns( + dst_table, dst_count, src_table, 0, src_count, true); + + /* Initialize remaining columns */ + for (; i_new < dst_column_count; i_new ++) { + ecs_column_t *column = &dst_columns[i_new]; + int32_t size = column->size; + ecs_assert(size != 0, ECS_INTERNAL_ERROR, NULL); + ecs_vec_set_size(a, &column->data, size, column_size); + ecs_vec_set_count(a, &column->data, size, src_count + dst_count); + flecs_table_data_invoke_ctor(column, dst_count, src_count); + } + + /* Destruct remaining columns */ + for (; i_old < src_column_count; i_old ++) { + ecs_column_t *column = &src_columns[i_old]; + flecs_table_data_invoke_dtor(column, 0, src_count); + ecs_vec_fini(a, &column->data, column->size); + } + + /* Mark entity column as dirty */ + flecs_table_data_mark_table_dirty(dst_data, 0); +} + +/* Merge source table into destination table. This typically happens as result + * of a bulk operation, like when a component is removed from all entities in + * the source table (like for the Remove OnDelete policy). */ +void flecs_table_data_merge( + ecs_world_t *world, + ecs_table_t *dst_table, + ecs_table_t *src_table) +{ + ecs_assert(src_table != NULL, ECS_INTERNAL_ERROR, NULL); + ecs_assert(!src_table->_->lock, ECS_LOCKED_STORAGE, NULL); + + ecs_table_data_t *dst_data = dst_table->data; + ecs_table_data_t *src_data = src_table->data; + ecs_assert(dst_data != NULL, ECS_INTERNAL_ERROR, NULL); + ecs_assert(src_data != NULL, ECS_INTERNAL_ERROR, NULL); + + bool move_data = false; + + ecs_entity_t *src_entities = src_data->entities.array; + int32_t src_count = src_data->entities.count; + int32_t dst_count = dst_data->entities.count; + ecs_record_t **src_records = src_data->records.array; + + /* First, update entity index so old entities point to new type */ + int32_t i; + for(i = 0; i < src_count; i ++) { + ecs_record_t *record; + if (dst_table != src_table) { + record = src_records[i]; + ecs_assert(record != NULL, ECS_INTERNAL_ERROR, NULL); + } else { + record = flecs_entities_ensure(world, src_entities[i]); + } + + uint32_t flags = ECS_RECORD_TO_ROW_FLAGS(record->row); + record->row = ECS_ROW_TO_RECORD(dst_count + i, flags); + record->table = dst_table; + } + + /* Merge table columns */ + if (move_data) { + *dst_data = *src_data; + } else { + flecs_table_data_merge_columns(world, dst_table, src_table, + src_count, dst_count, src_data, dst_data); + } +} + +/* Shrink table storage to fit number of entities */ +bool flecs_table_data_shrink( + ecs_world_t *world, + ecs_table_t *table) +{ + ecs_table_data_t *data = table->data; + bool has_payload = data->entities.array != NULL; + ecs_vec_reclaim_t(&world->allocator, &data->entities, ecs_entity_t); + ecs_vec_reclaim_t(&world->allocator, &data->records, ecs_record_t*); + + int32_t i, count = data->column_count; + for (i = 0; i < count; i ++) { + ecs_column_t *column = &data->columns[i]; + ecs_vec_reclaim(&world->allocator, &column->data, column->size); + } + + return has_payload; +} + /** * @file table_graph.c * @brief Data structure to speed up table transitions. diff --git a/flecs.h b/flecs.h index 055a3c163b..4aac758436 100644 --- a/flecs.h +++ b/flecs.h @@ -3116,7 +3116,7 @@ extern "C" { typedef struct ecs_stage_t ecs_stage_t; /** Table data */ -typedef struct ecs_data_t ecs_data_t; +typedef struct ecs_table_data_t ecs_table_data_t; /* Switch list */ typedef struct ecs_switch_t ecs_switch_t; diff --git a/include/flecs/private/api_types.h b/include/flecs/private/api_types.h index d0a5b21eb4..33b6bb79f5 100644 --- a/include/flecs/private/api_types.h +++ b/include/flecs/private/api_types.h @@ -24,7 +24,7 @@ extern "C" { typedef struct ecs_stage_t ecs_stage_t; /** Table data */ -typedef struct ecs_data_t ecs_data_t; +typedef struct ecs_table_data_t ecs_table_data_t; /* Switch list */ typedef struct ecs_switch_t ecs_switch_t; diff --git a/src/bootstrap.c b/src/bootstrap.c index f76c2454cb..2724a117d0 100644 --- a/src/bootstrap.c +++ b/src/bootstrap.c @@ -541,7 +541,7 @@ ecs_table_t* flecs_bootstrap_component_table( }; ecs_table_t *result = flecs_table_find_or_create(world, &array); - ecs_data_t *data = flecs_table_data(result); + ecs_table_data_t *data = flecs_table_data(result); /* Preallocate enough memory for initial components */ ecs_allocator_t *a = &world->allocator; diff --git a/src/entity.c b/src/entity.c index eccf1182cf..ba8eac622c 100644 --- a/src/entity.c +++ b/src/entity.c @@ -260,7 +260,7 @@ void flecs_instantiate_children( } ecs_type_t type = child_table->type; - ecs_data_t *child_data = flecs_table_data(child_table); + ecs_table_data_t *child_data = flecs_table_data(child_table); ecs_entity_t slot_of = 0; ecs_entity_t *ids = type.array; @@ -765,7 +765,7 @@ const ecs_entity_t* flecs_bulk_new( component_array.count = type.count; } - ecs_data_t *data = flecs_table_data(table); + ecs_table_data_t *data = flecs_table_data(table); int32_t row = flecs_table_appendn(world, table, count, entities); /* Update entity index. */ @@ -988,7 +988,7 @@ void flecs_notify_on_set( bool owned) { ecs_assert(ids != NULL, ECS_INTERNAL_ERROR, NULL); - ecs_data_t *data = flecs_table_data(table); + ecs_table_data_t *data = flecs_table_data(table); ecs_entity_t *entities = &flecs_table_entities_array(table)[row]; ecs_assert(entities != NULL, ECS_INTERNAL_ERROR, NULL); @@ -3149,11 +3149,11 @@ void ecs_enable_id( } ecs_assert(table->_ != NULL, ECS_INTERNAL_ERROR, NULL); - index -= flecs_table_data(table)->bs_offset; + index -= table->_->bs_offset; ecs_assert(index >= 0, ECS_INTERNAL_ERROR, NULL); /* Data cannot be NULl, since entity is stored in the table */ - ecs_bitset_t *bs = &flecs_table_data(table)->bs_columns[index]; + ecs_bitset_t *bs = &flecs_table_data(table)->bitsets[index].data; ecs_assert(bs != NULL, ECS_INTERNAL_ERROR, NULL); flecs_bitset_set(bs, ECS_RECORD_TO_ROW(r->row), enable); @@ -3189,9 +3189,9 @@ bool ecs_is_enabled_id( } ecs_assert(table->_ != NULL, ECS_INTERNAL_ERROR, NULL); - index -= flecs_table_data(table)->bs_offset; + index -= table->_->bs_offset; ecs_assert(index >= 0, ECS_INTERNAL_ERROR, NULL); - ecs_bitset_t *bs = &flecs_table_data(table)->bs_columns[index]; + ecs_bitset_t *bs = &flecs_table_data(table)->bitsets[index].data; return flecs_bitset_get(bs, ECS_RECORD_TO_ROW(r->row)); error: diff --git a/src/entity_filter.c b/src/entity_filter.c index 5b39557924..6c156d5638 100644 --- a/src/entity_filter.c +++ b/src/entity_filter.c @@ -66,7 +66,7 @@ int flecs_entity_filter_bitset_next( int32_t i, count = ecs_vec_count(&iter->entity_filter->bs_terms); flecs_bitset_term_t *terms = ecs_vec_first(&iter->entity_filter->bs_terms); - int32_t bs_offset = flecs_table_data(table)->bs_offset; + int32_t bs_offset = table->_->bs_offset; int32_t first = iter->bs_offset; int32_t last = 0; @@ -77,7 +77,7 @@ int flecs_entity_filter_bitset_next( if (!bs) { int32_t index = column->column_index; ecs_assert((index - bs_offset >= 0), ECS_INTERNAL_ERROR, NULL); - bs = &flecs_table_data(table)->bs_columns[index - bs_offset]; + bs = &flecs_table_data(table)->bitsets[index - bs_offset].data; terms[i].bs_column = bs; } @@ -228,7 +228,7 @@ int32_t flecs_get_flattened_target( } if (table->flags & EcsTableHasTarget) { - int32_t col = table->column_map[flecs_table_data(table)->ft_offset]; + int32_t col = table->column_map[table->_->ft_offset]; ecs_assert(col != -1, ECS_INTERNAL_ERROR, NULL); EcsTarget *next = flecs_table_column(table, col)->data.array; next = ECS_ELEM_T(next, EcsTarget, ECS_RECORD_TO_ROW(r->row)); diff --git a/src/filter.c b/src/filter.c index 05157f2d62..fd38108335 100644 --- a/src/filter.c +++ b/src/filter.c @@ -1957,7 +1957,7 @@ bool flecs_term_match_table( if ((column == -1) && (src->flags & EcsUp) && (table->flags & EcsTableHasTarget)) { ecs_assert(table->_ != NULL, ECS_INTERNAL_ERROR, NULL); - ecs_id_t rel = ECS_PAIR_SECOND(table->type.array[flecs_table_data(table)->ft_offset]); + ecs_id_t rel = ECS_PAIR_SECOND(table->type.array[table->_->ft_offset]); if (rel == (uint32_t)src->trav) { result = true; } diff --git a/src/observable.c b/src/observable.c index 1180b33e95..5e302c464e 100644 --- a/src/observable.c +++ b/src/observable.c @@ -1056,7 +1056,7 @@ void flecs_emit( const ecs_event_record_t *er_onset = flecs_event_record_get_if(observable, EcsOnSet); const ecs_event_record_t *er_unset = flecs_event_record_get_if(observable, EcsUnSet); - ecs_data_t *storage = NULL; + ecs_table_data_t *storage = NULL; ecs_column_t *columns = NULL; if (count) { storage = table->data; diff --git a/src/storage/table.c b/src/storage/table.c index 520c1f2fef..041cf654f9 100644 --- a/src/storage/table.c +++ b/src/storage/table.c @@ -31,7 +31,7 @@ #ifdef FLECS_SANITIZE static void flecs_table_check_sanity(ecs_table_t *table) { - ecs_data_t *data = table->data; + ecs_table_data_t *data = table->data; int32_t size = ecs_vec_size(&data->entities); int32_t count = ecs_vec_count(&data->entities); @@ -87,114 +87,7 @@ void flecs_table_check_sanity(ecs_table_t *table) { #define flecs_table_check_sanity(table) #endif -/* Set flags for type hooks so table operations can quickly check whether a - * fast or complex operation that invokes hooks is required. */ -static -ecs_flags32_t flecs_type_info_flags( - const ecs_type_info_t *ti) -{ - ecs_flags32_t flags = 0; - - if (ti->hooks.ctor) { - flags |= EcsTableHasCtors; - } - if (ti->hooks.on_add) { - flags |= EcsTableHasCtors; - } - if (ti->hooks.dtor) { - flags |= EcsTableHasDtors; - } - if (ti->hooks.on_remove) { - flags |= EcsTableHasDtors; - } - if (ti->hooks.copy) { - flags |= EcsTableHasCopy; - } - if (ti->hooks.move) { - flags |= EcsTableHasMove; - } - - return flags; -} - -static -void flecs_table_init_columns( - ecs_world_t *world, - ecs_table_t *table, - int32_t column_count) -{ - if (!column_count) { - return; - } - - int32_t i, cur = 0, ids_count = table->type.count; - ecs_column_t *columns = flecs_wcalloc_n(world, ecs_column_t, column_count); - table->data->columns = columns; - - ecs_id_t *ids = table->type.array; - ecs_table_record_t *records = table->_->records; - int32_t *t2s = table->column_map; - int32_t *s2t = &table->column_map[ids_count]; - - for (i = 0; i < ids_count; i ++) { - ecs_table_record_t *tr = &records[i]; - ecs_id_record_t *idr = (ecs_id_record_t*)tr->hdr.cache; - const ecs_type_info_t *ti = idr->type_info; - if (!ti) { - t2s[i] = -1; - continue; - } - - t2s[i] = cur; - s2t[cur] = i; - tr->column = flecs_ito(int16_t, cur); - - columns[cur].ti = ECS_CONST_CAST(ecs_type_info_t*, ti); - columns[cur].id = ids[i]; - columns[cur].size = ti->size; - - if (ECS_IS_PAIR(ids[i])) { - ecs_table_record_t *wc_tr = flecs_id_record_get_table( - idr->parent, table); - if (wc_tr->index == tr->index) { - wc_tr->column = tr->column; - } - } - -#ifdef FLECS_DEBUG - ecs_vec_init(NULL, &columns[cur].data, ti->size, 0); -#endif - - table->flags |= flecs_type_info_flags(ti); - cur ++; - } -} - -/* Initialize table storage */ -static -void flecs_table_init_data( - ecs_world_t *world, - ecs_table_t *table, - int32_t column_count) -{ - ecs_data_t *data = table->data; - data->column_count = flecs_ito(int16_t, column_count); - ecs_vec_init_t(NULL, &data->entities, ecs_entity_t, 0); - ecs_vec_init_t(NULL, &data->records, ecs_record_t*, 0); - - flecs_table_init_columns(world, table, column_count); - - int32_t i, bs_count = data->bs_count; - - if (bs_count) { - data->bs_columns = flecs_wcalloc_n(world, ecs_bitset_t, bs_count); - for (i = 0; i < bs_count; i ++) { - flecs_bitset_init(&data->bs_columns[i]); - } - } -} - -ecs_data_t* flecs_table_data( +ecs_table_data_t* flecs_table_data( const ecs_table_t *table) { ecs_assert(table != NULL, ECS_INTERNAL_ERROR, NULL); @@ -290,21 +183,17 @@ void flecs_table_init_flags( } else if (id == ecs_pair_t(EcsIdentifier, EcsName)) { table->flags |= EcsTableHasName; } else if (r == ecs_id(EcsTarget)) { - ecs_data_t *data = table->data; table->flags |= EcsTableHasTarget; - data->ft_offset = flecs_ito(int16_t, i); + table->_->ft_offset = flecs_ito(int16_t, i); } else if (r == ecs_id(EcsPoly)) { table->flags |= EcsTableHasBuiltins; } } else { if (ECS_HAS_ID_FLAG(id, TOGGLE)) { - ecs_data_t *data = table->data; - table->flags |= EcsTableHasToggle; - - if (!data->bs_count) { - data->bs_offset = flecs_ito(int16_t, i); + if (!(table->flags & EcsTableHasToggle)) { + table->_->bs_offset = flecs_ito(int16_t, i); } - data->bs_count ++; + table->flags |= EcsTableHasToggle; } if (ECS_HAS_ID_FLAG(id, OVERRIDE)) { table->flags |= EcsTableHasOverrides; @@ -347,7 +236,7 @@ void flecs_table_init( ecs_table_t *table, ecs_table_t *from) { - table->data = ecs_os_calloc_t(ecs_data_t); + table->data = ecs_os_calloc_t(ecs_table_data_t); /* Make sure table->flags is initialized */ flecs_table_init_flags(world, table); @@ -599,7 +488,7 @@ void flecs_table_init( dst_count + column_count); } - flecs_table_init_data(world, table, column_count); + flecs_table_data_init(world, table, column_count); if (table->flags & EcsTableHasName) { ecs_assert(childof_idr != NULL, ECS_INTERNAL_ERROR, NULL); @@ -783,7 +672,7 @@ void flecs_table_dtor_all( /* Can't delete and not update the entity index */ ecs_assert(!is_delete || update_entity_index, ECS_INTERNAL_ERROR, NULL); - ecs_data_t *data = flecs_table_data(table); + ecs_table_data_t *data = flecs_table_data(table); int32_t ids_count = data->column_count; ecs_record_t **records = data->records.array; ecs_entity_t *entities = data->entities.array; @@ -891,7 +780,7 @@ void flecs_table_fini_data( { ecs_assert(!table->_->lock, ECS_LOCKED_STORAGE, NULL); ecs_assert(table != NULL, ECS_INTERNAL_ERROR, NULL); - ecs_data_t *data = table->data; + ecs_table_data_t *data = table->data; ecs_assert(data != NULL, ECS_INTERNAL_ERROR, NULL); int32_t count = ecs_table_count(table); @@ -899,7 +788,7 @@ void flecs_table_fini_data( if (do_on_remove) { flecs_notify_on_remove(world, table, NULL, 0, count, &table->type); } - + flecs_table_dtor_all(world, table, 0, count, update_entity_index, is_delete); } @@ -922,14 +811,14 @@ void flecs_table_fini_data( data->columns = NULL; } - ecs_bitset_t *bs_columns = data->bs_columns; + ecs_bitset_column_t *bs_columns = data->bitsets; if (bs_columns) { int32_t c, column_count = data->bs_count; for (c = 0; c < column_count; c ++) { - flecs_bitset_fini(&bs_columns[c]); + flecs_bitset_fini(&bs_columns[c].data); } flecs_wfree_n(world, ecs_bitset_t, column_count, bs_columns); - data->bs_columns = NULL; + data->bitsets = NULL; } ecs_vec_fini_t(&world->allocator, &data->entities, ecs_entity_t); @@ -1072,20 +961,6 @@ void flecs_table_traversable_add( } } -/* Mark table column dirty. This usually happens as the result of a set - * operation, or iteration of a query with [out] fields. */ -static -void flecs_table_mark_table_dirty( - ecs_world_t *world, - ecs_table_t *table, - int32_t index) -{ - (void)world; - if (flecs_table_data(table)->dirty_state) { - flecs_table_data(table)->dirty_state[index] ++; - } -} - /* Mark table component dirty */ void flecs_table_mark_dirty( ecs_world_t *world, @@ -1129,154 +1004,6 @@ int32_t* flecs_table_get_dirty_state( return flecs_table_data(table)->dirty_state; } -/* Table move logic for bitset (toggle component) column */ -static -void flecs_table_move_bitset_columns( - ecs_table_t *dst_table, - int32_t dst_index, - ecs_table_t *src_table, - int32_t src_index, - int32_t count, - bool clear) -{ - ecs_data_t *dst_data = flecs_table_data(dst_table); - ecs_data_t *src_data = flecs_table_data(src_table); - if (!dst_data && !src_data) { - return; - } - - int32_t i_old = 0, src_column_count = src_data ? src_data->bs_count : 0; - int32_t i_new = 0, dst_column_count = dst_data ? dst_data->bs_count : 0; - - if (!src_column_count && !dst_column_count) { - return; - } - - ecs_bitset_t *src_columns = src_data ? src_data->bs_columns : NULL; - ecs_bitset_t *dst_columns = dst_data ? dst_data->bs_columns : NULL; - - ecs_type_t dst_type = dst_table->type; - ecs_type_t src_type = src_table->type; - - int32_t offset_new = dst_data ? dst_data->bs_offset : 0; - int32_t offset_old = src_data ? src_data->bs_offset : 0; - - ecs_id_t *dst_ids = dst_type.array; - ecs_id_t *src_ids = src_type.array; - - for (; (i_new < dst_column_count) && (i_old < src_column_count);) { - ecs_id_t dst_id = dst_ids[i_new + offset_new]; - ecs_id_t src_id = src_ids[i_old + offset_old]; - - if (dst_id == src_id) { - ecs_bitset_t *src_bs = &src_columns[i_old]; - ecs_bitset_t *dst_bs = &dst_columns[i_new]; - - flecs_bitset_ensure(dst_bs, dst_index + count); - - int i; - for (i = 0; i < count; i ++) { - uint64_t value = flecs_bitset_get(src_bs, src_index + i); - flecs_bitset_set(dst_bs, dst_index + i, value); - } - - if (clear) { - ecs_assert(count == flecs_bitset_count(src_bs), - ECS_INTERNAL_ERROR, NULL); - flecs_bitset_fini(src_bs); - } - } else if (dst_id > src_id) { - ecs_bitset_t *src_bs = &src_columns[i_old]; - flecs_bitset_fini(src_bs); - } - - i_new += dst_id <= src_id; - i_old += dst_id >= src_id; - } - - /* Clear remaining columns */ - if (clear) { - for (; (i_old < src_column_count); i_old ++) { - ecs_bitset_t *src_bs = &src_columns[i_old]; - ecs_assert(count == flecs_bitset_count(src_bs), - ECS_INTERNAL_ERROR, NULL); - flecs_bitset_fini(src_bs); - } - } -} - -/* Grow table column. When a column needs to be reallocated this function takes - * care of correctly invoking ctor/move/dtor hooks. */ -static -void* flecs_table_grow_column( - ecs_world_t *world, - ecs_column_t *column, - int32_t to_add, - int32_t dst_size, - bool construct) -{ - ecs_assert(column != NULL, ECS_INTERNAL_ERROR, NULL); - - ecs_type_info_t *ti = column->ti; - int32_t size = column->size; - int32_t count = column->data.count; - int32_t src_size = column->data.size; - int32_t dst_count = count + to_add; - bool can_realloc = dst_size != src_size; - void *result = NULL; - - ecs_assert(dst_size >= dst_count, ECS_INTERNAL_ERROR, NULL); - - /* If the array could possibly realloc and the component has a move action - * defined, move old elements manually */ - ecs_move_t move_ctor; - if (count && can_realloc && (move_ctor = ti->hooks.ctor_move_dtor)) { - ecs_xtor_t ctor = ti->hooks.ctor; - ecs_assert(ctor != NULL, ECS_INTERNAL_ERROR, NULL); - ecs_assert(move_ctor != NULL, ECS_INTERNAL_ERROR, NULL); - - /* Create vector */ - ecs_vec_t dst; - ecs_vec_init(&world->allocator, &dst, size, dst_size); - dst.count = dst_count; - - void *src_buffer = column->data.array; - void *dst_buffer = dst.array; - - /* Move (and construct) existing elements to new vector */ - move_ctor(dst_buffer, src_buffer, count, ti); - - if (construct) { - /* Construct new element(s) */ - result = ECS_ELEM(dst_buffer, size, count); - ctor(result, to_add, ti); - } - - /* Free old vector */ - ecs_vec_fini(&world->allocator, &column->data, size); - - column->data = dst; - } else { - /* If array won't realloc or has no move, simply add new elements */ - if (can_realloc) { - ecs_vec_set_size(&world->allocator, &column->data, size, dst_size); - } - - result = ecs_vec_grow(&world->allocator, &column->data, size, to_add); - - ecs_xtor_t ctor; - if (construct && (ctor = ti->hooks.ctor)) { - /* If new elements need to be constructed and component has a - * constructor, construct */ - ctor(result, to_add, ti); - } - } - - ecs_assert(column->data.size == dst_size, ECS_INTERNAL_ERROR, NULL); - - return result; -} - /* Grow all data structures in a table */ int32_t flecs_table_appendn( ecs_world_t *world, @@ -1286,59 +1013,12 @@ int32_t flecs_table_appendn( { ecs_assert(table != NULL, ECS_INTERNAL_ERROR, NULL); ecs_assert(!table->_->lock, ECS_LOCKED_STORAGE, NULL); - ecs_data_t *data = table->data; - ecs_assert(data != NULL, ECS_INTERNAL_ERROR, NULL); flecs_table_check_sanity(table); int32_t cur_count = ecs_table_count(table); - int32_t size = to_add + cur_count; - int32_t column_count = table->data->column_count; - - /* Add record to record ptr array */ - ecs_vec_set_size_t(&world->allocator, &data->records, ecs_record_t*, size); - ecs_record_t **r = ecs_vec_last_t(&data->records, ecs_record_t*) + 1; - data->records.count += to_add; - if (data->records.size > size) { - size = data->records.size; - } - - /* Add entity to column with entity ids */ - ecs_vec_set_size_t(&world->allocator, &data->entities, ecs_entity_t, size); - ecs_entity_t *e = ecs_vec_last_t(&data->entities, ecs_entity_t) + 1; - data->entities.count += to_add; - ecs_assert(data->entities.size == size, ECS_INTERNAL_ERROR, NULL); - - /* Initialize entity ids and record ptrs */ - int32_t i; - if (ids) { - ecs_os_memcpy_n(e, ids, ecs_entity_t, to_add); - } else { - ecs_os_memset(e, 0, ECS_SIZEOF(ecs_entity_t) * to_add); - } - ecs_os_memset(r, 0, ECS_SIZEOF(ecs_record_t*) * to_add); - - /* Add elements to each column array */ - ecs_column_t *columns = data->columns; - for (i = 0; i < column_count; i ++) { - flecs_table_grow_column(world, &columns[i], to_add, size, true); - ecs_assert(columns[i].data.size == size, ECS_INTERNAL_ERROR, NULL); - flecs_table_invoke_add_hooks(world, table, &columns[i], e, - cur_count, to_add, false); - } - - int32_t bs_count = data->bs_count; - ecs_bitset_t *bs_columns = data->bs_columns; - - /* Add elements to each bitset column */ - for (i = 0; i < bs_count; i ++) { - ecs_bitset_t *bs = &bs_columns[i]; - flecs_bitset_addn(bs, to_add); - } - - /* If the table is monitored indicate that there has been a change */ - flecs_table_mark_table_dirty(world, table, 0); + flecs_table_data_appendn(world, table, to_add, ids); if (!(world->flags & EcsWorldReadonly) && !cur_count) { flecs_table_set_empty(world, table); } @@ -1349,21 +1029,6 @@ int32_t flecs_table_appendn( return cur_count; } -/* Append operation for tables that don't have any complex logic */ -static -void flecs_table_fast_append( - ecs_world_t *world, - ecs_column_t *columns, - int32_t count) -{ - /* Add elements to each column array */ - int32_t i; - for (i = 0; i < count; i ++) { - ecs_column_t *column = &columns[i]; - ecs_vec_append(&world->allocator, &column->data, column->size); - } -} - /* Append entity to table */ int32_t flecs_table_append( ecs_world_t *world, @@ -1380,108 +1045,17 @@ int32_t flecs_table_append( flecs_table_check_sanity(table); - /* Get count & size before growing entities array. This tells us whether the - * arrays will realloc */ - ecs_data_t *data = table->data; - int32_t count = data->entities.count; - int32_t column_count = table->data->column_count; - ecs_column_t *columns = flecs_table_columns(table); - - /* Grow buffer with entity ids, set new element to new entity */ - ecs_entity_t *e = ecs_vec_append_t(&world->allocator, - &data->entities, ecs_entity_t); - ecs_assert(e != NULL, ECS_INTERNAL_ERROR, NULL); - *e = entity; - - /* Add record ptr to array with record ptrs */ - ecs_record_t **r = ecs_vec_append_t(&world->allocator, - &data->records, ecs_record_t*); - ecs_assert(r != NULL, ECS_INTERNAL_ERROR, NULL); - *r = record; - - /* If the table is monitored indicate that there has been a change */ - flecs_table_mark_table_dirty(world, table, 0); - ecs_assert(count >= 0, ECS_INTERNAL_ERROR, NULL); - - /* Fast path: no switch columns, no lifecycle actions */ - if (!(table->flags & EcsTableIsComplex)) { - flecs_table_fast_append(world, columns, column_count); - if (!count) { - flecs_table_set_empty(world, table); /* See below */ - } - return count; - } - - ecs_entity_t *entities = data->entities.array; - - /* Reobtain size to ensure that the columns have the same size as the - * entities and record vectors. This keeps reasoning about when allocations - * occur easier. */ - int32_t size = data->entities.size; - - /* Grow component arrays with 1 element */ - int32_t i; - for (i = 0; i < column_count; i ++) { - ecs_column_t *column = &columns[i]; - flecs_table_grow_column(world, column, 1, size, construct); - - ecs_iter_action_t on_add_hook; - if (on_add && (on_add_hook = column->ti->hooks.on_add)) { - flecs_table_invoke_hook(world, table, on_add_hook, EcsOnAdd, column, - &entities[count], count, 1); - } - - ecs_assert(columns[i].data.size == - data->entities.size, ECS_INTERNAL_ERROR, NULL); - ecs_assert(columns[i].data.count == - data->entities.count, ECS_INTERNAL_ERROR, NULL); - } - - int32_t bs_count = data->bs_count; - ecs_bitset_t *bs_columns = data->bs_columns; - - /* Add element to each bitset column */ - for (i = 0; i < bs_count; i ++) { - ecs_assert(bs_columns != NULL, ECS_INTERNAL_ERROR, NULL); - ecs_bitset_t *bs = &bs_columns[i]; - flecs_bitset_addn(bs, 1); - } - - /* If this is the first entity in this table, signal queries so that the - * table moves from an inactive table to an active table. */ - if (!count) { + int32_t row = flecs_table_data_append(world, table, + entity, record, construct, on_add); + if (row == 0) { + /* If this is the first entity in this table, signal queries so that the + * table moves from an inactive table to an active table. */ flecs_table_set_empty(world, table); } flecs_table_check_sanity(table); - return count; -} - -/* Delete last operation for tables that don't have any complex logic */ -static -void flecs_table_fast_delete_last( - ecs_column_t *columns, - int32_t column_count) -{ - int i; - for (i = 0; i < column_count; i ++) { - ecs_vec_remove_last(&columns[i].data); - } -} - -/* Delete operation for tables that don't have any complex logic */ -static -void flecs_table_fast_delete( - ecs_column_t *columns, - int32_t column_count, - int32_t index) -{ - int i; - for (i = 0; i < column_count; i ++) { - ecs_column_t *column = &columns[i]; - ecs_vec_remove(&column->data, column->size, index); - } + return row; } /* Delete entity from table */ @@ -1499,149 +1073,13 @@ void flecs_table_delete( flecs_table_check_sanity(table); - ecs_data_t *data = table->data; - int32_t count = data->entities.count; - - ecs_assert(count > 0, ECS_INTERNAL_ERROR, NULL); - count --; - ecs_assert(index <= count, ECS_INTERNAL_ERROR, NULL); - - /* Move last entity id to index */ - ecs_entity_t *entities = data->entities.array; - ecs_entity_t entity_to_move = entities[count]; - ecs_entity_t entity_to_delete = entities[index]; - entities[index] = entity_to_move; - ecs_vec_remove_last(&data->entities); - - /* Move last record ptr to index */ - ecs_assert(count < data->records.count, ECS_INTERNAL_ERROR, NULL); - - ecs_record_t **records = data->records.array; - ecs_record_t *record_to_move = records[count]; - records[index] = record_to_move; - ecs_vec_remove_last(&data->records); - - /* Update record of moved entity in entity index */ - if (index != count) { - if (record_to_move) { - uint32_t row_flags = record_to_move->row & ECS_ROW_FLAGS_MASK; - record_to_move->row = ECS_ROW_TO_RECORD(index, row_flags); - ecs_assert(record_to_move->table != NULL, ECS_INTERNAL_ERROR, NULL); - ecs_assert(record_to_move->table == table, ECS_INTERNAL_ERROR, NULL); - } - } - - /* If the table is monitored indicate that there has been a change */ - flecs_table_mark_table_dirty(world, table, 0); - - /* If table is empty, deactivate it */ - if (!count) { + if (!flecs_table_data_delete(world, table, index, destruct)) { flecs_table_set_empty(world, table); } - /* Destruct component data */ - ecs_column_t *columns = data->columns; - int32_t column_count = table->data->column_count; - int32_t i; - - /* If this is a table without lifecycle callbacks or special columns, take - * fast path that just remove an element from the array(s) */ - if (!(table->flags & EcsTableIsComplex)) { - if (index == count) { - flecs_table_fast_delete_last(columns, column_count); - } else { - flecs_table_fast_delete(columns, column_count, index); - } - - flecs_table_check_sanity(table); - return; - } - - /* Last element, destruct & remove */ - if (index == count) { - /* If table has component destructors, invoke */ - if (destruct && (table->flags & EcsTableHasDtors)) { - for (i = 0; i < column_count; i ++) { - flecs_table_invoke_remove_hooks(world, table, &columns[i], - &entity_to_delete, index, 1, true); - } - } - - flecs_table_fast_delete_last(columns, column_count); - - /* Not last element, move last element to deleted element & destruct */ - } else { - /* If table has component destructors, invoke */ - if ((table->flags & (EcsTableHasDtors | EcsTableHasMove))) { - for (i = 0; i < column_count; i ++) { - ecs_column_t *column = &columns[i]; - ecs_type_info_t *ti = column->ti; - ecs_size_t size = column->size; - void *dst = ecs_vec_get(&column->data, size, index); - void *src = ecs_vec_last(&column->data, size); - - ecs_iter_action_t on_remove = ti->hooks.on_remove; - if (destruct && on_remove) { - flecs_table_invoke_hook(world, table, on_remove, EcsOnRemove, - column, &entity_to_delete, index, 1); - } - - ecs_move_t move_dtor = ti->hooks.move_dtor; - if (move_dtor) { - move_dtor(dst, src, 1, ti); - } else { - ecs_os_memcpy(dst, src, size); - } - - ecs_vec_remove_last(&column->data); - } - } else { - flecs_table_fast_delete(columns, column_count, index); - } - } - - /* Remove elements from bitset columns */ - ecs_bitset_t *bs_columns = data->bs_columns; - int32_t bs_count = data->bs_count; - for (i = 0; i < bs_count; i ++) { - flecs_bitset_remove(&bs_columns[i], index); - } - flecs_table_check_sanity(table); } -/* Move operation for tables that don't have any complex logic */ -static -void flecs_table_fast_move( - ecs_table_t *dst_table, - int32_t dst_index, - ecs_table_t *src_table, - int32_t src_index) -{ - int32_t i_new = 0, dst_column_count = dst_table->data->column_count; - int32_t i_old = 0, src_column_count = src_table->data->column_count; - - ecs_column_t *src_columns = flecs_table_columns(src_table); - ecs_column_t *dst_columns = flecs_table_columns(dst_table); - - for (; (i_new < dst_column_count) && (i_old < src_column_count);) { - ecs_column_t *dst_column = &dst_columns[i_new]; - ecs_column_t *src_column = &src_columns[i_old]; - ecs_id_t dst_id = dst_column->id; - ecs_id_t src_id = src_column->id; - - if (dst_id == src_id) { - int32_t size = dst_column->size; - void *dst = ecs_vec_get(&dst_column->data, size, dst_index); - void *src = ecs_vec_get(&src_column->data, size, src_index); - ecs_os_memcpy(dst, src, size); - } - - i_new += dst_id <= src_id; - i_old += dst_id >= src_id; - } -} - /* Move entity from src to dst table */ void flecs_table_move( ecs_world_t *world, @@ -1664,140 +1102,13 @@ void flecs_table_move( flecs_table_check_sanity(dst_table); flecs_table_check_sanity(src_table); - if (!((dst_table->flags | src_table->flags) & EcsTableIsComplex)) { - flecs_table_fast_move(dst_table, dst_index, src_table, src_index); - flecs_table_check_sanity(dst_table); - flecs_table_check_sanity(src_table); - return; - } - - flecs_table_move_bitset_columns(dst_table, dst_index, src_table, src_index, 1, false); - - /* If the source and destination entities are the same, move component - * between tables. If the entities are not the same (like when cloning) use - * a copy. */ - bool same_entity = dst_entity == src_entity; - - /* Call move_dtor for moved away from storage only if the entity is at the - * last index in the source table. If it isn't the last entity, the last - * entity in the table will be moved to the src storage, which will take - * care of cleaning up resources. */ - bool use_move_dtor = ecs_table_count(src_table) == (src_index + 1); - - int32_t i_new = 0, dst_column_count = dst_table->data->column_count; - int32_t i_old = 0, src_column_count = src_table->data->column_count; - - ecs_column_t *src_columns = flecs_table_columns(src_table); - ecs_column_t *dst_columns = flecs_table_columns(dst_table); - - for (; (i_new < dst_column_count) && (i_old < src_column_count); ) { - ecs_column_t *dst_column = &dst_columns[i_new]; - ecs_column_t *src_column = &src_columns[i_old]; - ecs_id_t dst_id = dst_column->id; - ecs_id_t src_id = src_column->id; - - if (dst_id == src_id) { - int32_t size = dst_column->size; - - ecs_assert(size != 0, ECS_INTERNAL_ERROR, NULL); - void *dst = ecs_vec_get(&dst_column->data, size, dst_index); - void *src = ecs_vec_get(&src_column->data, size, src_index); - ecs_type_info_t *ti = dst_column->ti; - - if (same_entity) { - ecs_move_t move = ti->hooks.move_ctor; - if (use_move_dtor || !move) { - /* Also use move_dtor if component doesn't have a move_ctor - * registered, to ensure that the dtor gets called to - * cleanup resources. */ - move = ti->hooks.ctor_move_dtor; - } - - if (move) { - move(dst, src, 1, ti); - } else { - ecs_os_memcpy(dst, src, size); - } - } else { - ecs_copy_t copy = ti->hooks.copy_ctor; - if (copy) { - copy(dst, src, 1, ti); - } else { - ecs_os_memcpy(dst, src, size); - } - } - } else { - if (dst_id < src_id) { - flecs_table_invoke_add_hooks(world, dst_table, - dst_column, &dst_entity, dst_index, 1, construct); - } else { - flecs_table_invoke_remove_hooks(world, src_table, - src_column, &src_entity, src_index, 1, use_move_dtor); - } - } - - i_new += dst_id <= src_id; - i_old += dst_id >= src_id; - } - - for (; (i_new < dst_column_count); i_new ++) { - flecs_table_invoke_add_hooks(world, dst_table, &dst_columns[i_new], - &dst_entity, dst_index, 1, construct); - } - - for (; (i_old < src_column_count); i_old ++) { - flecs_table_invoke_remove_hooks(world, src_table, &src_columns[i_old], - &src_entity, src_index, 1, use_move_dtor); - } + flecs_table_data_move(world, dst_entity, src_entity, dst_table, dst_index, + src_table, src_index, construct); flecs_table_check_sanity(dst_table); flecs_table_check_sanity(src_table); } -/* Shrink table storage to fit number of entities */ -bool flecs_table_shrink( - ecs_world_t *world, - ecs_table_t *table) -{ - ecs_assert(table != NULL, ECS_LOCKED_STORAGE, NULL); - ecs_assert(!table->_->lock, ECS_LOCKED_STORAGE, NULL); - (void)world; - - flecs_table_check_sanity(table); - - ecs_data_t *data = table->data; - bool has_payload = data->entities.array != NULL; - ecs_vec_reclaim_t(&world->allocator, &data->entities, ecs_entity_t); - ecs_vec_reclaim_t(&world->allocator, &data->records, ecs_record_t*); - - int32_t i, count = table->data->column_count; - for (i = 0; i < count; i ++) { - ecs_column_t *column = &data->columns[i]; - ecs_vec_reclaim(&world->allocator, &column->data, column->size); - } - - return has_payload; -} - -/* Swap operation for bitset (toggle component) columns */ -static -void flecs_table_swap_bitset_columns( - ecs_table_t *table, - int32_t row_1, - int32_t row_2) -{ - int32_t i = 0, column_count = flecs_table_data(table)->bs_count; - if (!column_count) { - return; - } - - ecs_bitset_t *columns = flecs_table_data(table)->bs_columns; - for (i = 0; i < column_count; i ++) { - ecs_bitset_t *bs = &columns[i]; - flecs_bitset_swap(bs, row_1, row_2); - } -} - /* Swap two rows in a table. Used for table sorting. */ void flecs_table_swap( ecs_world_t *world, @@ -1812,192 +1123,12 @@ void flecs_table_swap( ecs_assert(row_2 >= 0, ECS_INTERNAL_ERROR, NULL); flecs_table_check_sanity(table); - - if (row_1 == row_2) { - return; - } - - /* If the table is monitored indicate that there has been a change */ - flecs_table_mark_table_dirty(world, table, 0); - - ecs_entity_t *entities = flecs_table_entities_array(table); - ecs_entity_t e1 = entities[row_1]; - ecs_entity_t e2 = entities[row_2]; - - ecs_record_t **records = flecs_table_records_array(table); - ecs_record_t *record_ptr_1 = records[row_1]; - ecs_record_t *record_ptr_2 = records[row_2]; - - ecs_assert(record_ptr_1 != NULL, ECS_INTERNAL_ERROR, NULL); - ecs_assert(record_ptr_2 != NULL, ECS_INTERNAL_ERROR, NULL); - - /* Keep track of whether entity is watched */ - uint32_t flags_1 = ECS_RECORD_TO_ROW_FLAGS(record_ptr_1->row); - uint32_t flags_2 = ECS_RECORD_TO_ROW_FLAGS(record_ptr_2->row); - /* Swap entities & records */ - entities[row_1] = e2; - entities[row_2] = e1; - record_ptr_1->row = ECS_ROW_TO_RECORD(row_2, flags_1); - record_ptr_2->row = ECS_ROW_TO_RECORD(row_1, flags_2); - records[row_1] = record_ptr_2; - records[row_2] = record_ptr_1; - - flecs_table_swap_bitset_columns(table, row_1, row_2); - - ecs_column_t *columns = flecs_table_columns(table); - if (!columns) { - flecs_table_check_sanity(table); - return; - } - - /* Find the maximum size of column elements - * and allocate a temporary buffer for swapping */ - int32_t i, temp_buffer_size = ECS_SIZEOF(uint64_t); - int32_t column_count = table->data->column_count; - for (i = 0; i < column_count; i++) { - temp_buffer_size = ECS_MAX(temp_buffer_size, columns[i].size); - } - - void* tmp = ecs_os_alloca(temp_buffer_size); - - /* Swap columns */ - for (i = 0; i < column_count; i ++) { - int32_t size = columns[i].size; - void *ptr = columns[i].data.array; - - void *el_1 = ECS_ELEM(ptr, size, row_1); - void *el_2 = ECS_ELEM(ptr, size, row_2); - - ecs_os_memcpy(tmp, el_1, size); - ecs_os_memcpy(el_1, el_2, size); - ecs_os_memcpy(el_2, tmp, size); - } + flecs_table_data_swap(world, table, row_1, row_2); flecs_table_check_sanity(table); } -/* Merge data from one table column into other table column */ -static -void flecs_table_merge_column( - ecs_world_t *world, - ecs_column_t *dst, - ecs_column_t *src, - int32_t column_size) -{ - ecs_size_t size = dst->size; - int32_t dst_count = dst->data.count; - - if (!dst_count) { - ecs_vec_fini(&world->allocator, &dst->data, size); - *dst = *src; - src->data.array = NULL; - src->data.count = 0; - src->data.size = 0; - - /* If the new table is not empty, copy the contents from the - * src into the dst. */ - } else { - int32_t src_count = src->data.count; - - flecs_table_grow_column(world, dst, src_count, column_size, true); - void *dst_ptr = ECS_ELEM(dst->data.array, size, dst_count); - void *src_ptr = src->data.array; - - /* Move values into column */ - ecs_type_info_t *ti = dst->ti; - ecs_assert(ti != NULL, ECS_INTERNAL_ERROR, NULL); - ecs_move_t move = ti->hooks.move_dtor; - if (move) { - move(dst_ptr, src_ptr, src_count, ti); - } else { - ecs_os_memcpy(dst_ptr, src_ptr, size * src_count); - } - - ecs_vec_fini(&world->allocator, &src->data, size); - } -} - -/* Merge storage of two tables. */ -static -void flecs_table_merge_data( - ecs_world_t *world, - ecs_table_t *dst_table, - ecs_table_t *src_table, - int32_t src_count, - int32_t dst_count, - ecs_data_t *src_data, - ecs_data_t *dst_data) -{ - int32_t i_new = 0, dst_column_count = dst_table->data->column_count; - int32_t i_old = 0, src_column_count = src_table->data->column_count; - ecs_column_t *src_columns = src_data->columns; - ecs_column_t *dst_columns = dst_data->columns; - - ecs_assert(!dst_column_count || dst_columns, ECS_INTERNAL_ERROR, NULL); - - if (!src_count) { - return; - } - - /* Merge entities & records vectors */ - ecs_allocator_t *a = &world->allocator; - ecs_vec_merge_t(a, &dst_data->entities, &src_data->entities, ecs_entity_t); - ecs_assert(dst_data->entities.count == src_count + dst_count, - ECS_INTERNAL_ERROR, NULL); - ecs_vec_merge_t(a, &dst_data->records, &src_data->records, ecs_record_t*); - - int32_t column_size = dst_data->entities.size; - for (; (i_new < dst_column_count) && (i_old < src_column_count); ) { - ecs_column_t *dst_column = &dst_columns[i_new]; - ecs_column_t *src_column = &src_columns[i_old]; - ecs_id_t dst_id = dst_column->id; - ecs_id_t src_id = src_column->id; - - if (dst_id == src_id) { - flecs_table_merge_column(world, dst_column, src_column, column_size); - flecs_table_mark_table_dirty(world, dst_table, i_new + 1); - i_new ++; - i_old ++; - } else if (dst_id < src_id) { - /* New column, make sure vector is large enough. */ - ecs_size_t size = dst_column->size; - ecs_vec_set_size(a, &dst_column->data, size, column_size); - ecs_vec_set_count(a, &dst_column->data, size, src_count + dst_count); - flecs_table_invoke_ctor(dst_column, dst_count, src_count); - i_new ++; - } else if (dst_id > src_id) { - /* Old column does not occur in new table, destruct */ - flecs_table_invoke_dtor(src_column, 0, src_count); - ecs_vec_fini(a, &src_column->data, src_column->size); - i_old ++; - } - } - - flecs_table_move_bitset_columns( - dst_table, dst_count, src_table, 0, src_count, true); - - /* Initialize remaining columns */ - for (; i_new < dst_column_count; i_new ++) { - ecs_column_t *column = &dst_columns[i_new]; - int32_t size = column->size; - ecs_assert(size != 0, ECS_INTERNAL_ERROR, NULL); - ecs_vec_set_size(a, &column->data, size, column_size); - ecs_vec_set_count(a, &column->data, size, src_count + dst_count); - flecs_table_invoke_ctor(column, dst_count, src_count); - } - - /* Destruct remaining columns */ - for (; i_old < src_column_count; i_old ++) { - ecs_column_t *column = &src_columns[i_old]; - flecs_table_invoke_dtor(column, 0, src_count); - ecs_vec_fini(a, &column->data, column->size); - } - - /* Mark entity column as dirty */ - flecs_table_mark_table_dirty(world, dst_table, 0); -} - /* Merge source table into destination table. This typically happens as result * of a bulk operation, like when a component is removed from all entities in * the source table (like for the Remove OnDelete policy). */ @@ -2011,11 +1142,9 @@ void flecs_table_merge( flecs_table_check_sanity(src_table); flecs_table_check_sanity(dst_table); - - bool move_data = false; - ecs_data_t *dst_data = dst_table->data; - ecs_data_t *src_data = src_table->data; + ecs_table_data_t *dst_data = dst_table->data; + ecs_table_data_t *src_data = src_table->data; ecs_assert(dst_data != NULL, ECS_INTERNAL_ERROR, NULL); ecs_assert(src_data != NULL, ECS_INTERNAL_ERROR, NULL); @@ -2028,46 +1157,10 @@ void flecs_table_merge( ecs_assert(!dst_table->_->lock, ECS_LOCKED_STORAGE, NULL); } - /* If there is no data to merge, drop out */ - if (!src_data) { - return; - } - - if (!dst_data) { - dst_data = dst_table->data; - if (dst_table == src_table) { - move_data = true; - } - } - - ecs_entity_t *src_entities = src_data->entities.array; int32_t src_count = src_data->entities.count; int32_t dst_count = dst_data->entities.count; - ecs_record_t **src_records = src_data->records.array; - - /* First, update entity index so old entities point to new type */ - int32_t i; - for(i = 0; i < src_count; i ++) { - ecs_record_t *record; - if (dst_table != src_table) { - record = src_records[i]; - ecs_assert(record != NULL, ECS_INTERNAL_ERROR, NULL); - } else { - record = flecs_entities_ensure(world, src_entities[i]); - } - - uint32_t flags = ECS_RECORD_TO_ROW_FLAGS(record->row); - record->row = ECS_ROW_TO_RECORD(dst_count + i, flags); - record->table = dst_table; - } - /* Merge table columns */ - if (move_data) { - *dst_data = *src_data; - } else { - flecs_table_merge_data(world, dst_table, src_table, src_count, dst_count, - src_data, dst_data); - } + flecs_table_data_merge(world, dst_table, src_table); if (src_count) { if (!dst_count) { @@ -2084,6 +1177,24 @@ void flecs_table_merge( flecs_table_check_sanity(dst_table); } +/* Shrink table storage to fit number of entities */ +bool flecs_table_shrink( + ecs_world_t *world, + ecs_table_t *table) +{ + ecs_assert(table != NULL, ECS_LOCKED_STORAGE, NULL); + ecs_assert(!table->_->lock, ECS_LOCKED_STORAGE, NULL); + (void)world; + + flecs_table_check_sanity(table); + + bool has_payload = flecs_table_data_shrink(world, table); + + flecs_table_check_sanity(table); + + return has_payload; +} + /* Internal mechanism for propagating information to tables */ void flecs_table_notify( ecs_world_t *world, diff --git a/src/storage/table.h b/src/storage/table.h index 711446c962..0c7929aab4 100644 --- a/src/storage/table.h +++ b/src/storage/table.h @@ -7,6 +7,7 @@ #define FLECS_TABLE_H #include "table_graph.h" +#include "table_data.h" /* Table event type for notifying tables of world events */ typedef enum ecs_table_eventkind_t { @@ -36,6 +37,8 @@ typedef struct ecs_table__t { uint64_t hash; /* Type hash */ int32_t lock; /* Prevents modifications */ int32_t traversable_count; /* Traversable relationship targets in table */ + int16_t bs_offset; /* First bitset id in type */ + int16_t ft_offset; /* First flattened id in type */ uint16_t generation; /* Used for table cleanup */ int16_t record_count; /* Table record count including wildcards */ @@ -43,26 +46,31 @@ typedef struct ecs_table__t { ecs_hashmap_t *name_index; /* Cached pointer to name index */ } ecs_table__t; -/** Table column */ +/** Component column */ typedef struct ecs_column_t { ecs_vec_t data; /* Vector with component data */ - ecs_id_t id; /* Component id */ + ecs_id_t id; /* Column id */ ecs_type_info_t *ti; /* Component type info */ ecs_size_t size; /* Component size */ } ecs_column_t; +/** Bitset column */ +typedef struct ecs_bitset_column_t { + ecs_bitset_t data; /* Bitset columns */ + ecs_id_t id; /* Column id */ +} ecs_bitset_column_t; + /** Table data */ -struct ecs_data_t { +struct ecs_table_data_t { ecs_vec_t entities; /* Entity ids */ ecs_vec_t records; /* Ptrs to records in entity index */ ecs_column_t *columns; /* Component data */ int16_t column_count; /* Number of components (excluding tags) */ + ecs_flags32_t flags; /* Flags for testing table data properties */ int32_t *dirty_state; /* Keep track of changes in columns */ - ecs_bitset_t *bs_columns; /* Bitset columns */ - int16_t bs_count; - int16_t bs_offset; - int16_t ft_offset; + ecs_bitset_column_t *bitsets; /* Bitset columns */ + int16_t bs_count; /* Number of bitset columns */ }; /** A table is the Flecs equivalent of an archetype. Tables store all entities @@ -74,7 +82,7 @@ struct ecs_table_t { ecs_flags32_t flags; /* Flags for testing table properties */ ecs_type_t type; /* Vector with component ids */ - ecs_data_t *data; /* Component storage */ + ecs_table_data_t *data; /* Component storage */ ecs_graph_node_t node; /* Graph node */ int32_t *column_map; /* Map type index <-> column @@ -86,7 +94,7 @@ struct ecs_table_t { }; /* Get table data */ -ecs_data_t* flecs_table_data( +ecs_table_data_t* flecs_table_data( const ecs_table_t *table); /* Get table columns */ diff --git a/src/storage/table_data.c b/src/storage/table_data.c new file mode 100644 index 0000000000..bd835e23bf --- /dev/null +++ b/src/storage/table_data.c @@ -0,0 +1,1076 @@ +/** + * @file table_data.c + * @brief Table data implementation. + */ + +#include "../private_api.h" + +/* Construct components */ +static +void flecs_table_data_invoke_ctor( + ecs_column_t *column, + int32_t row, + int32_t count) +{ + ecs_type_info_t *ti = column->ti; + ecs_assert(ti != NULL, ECS_INTERNAL_ERROR, NULL); + ecs_xtor_t ctor = ti->hooks.ctor; + if (ctor) { + void *ptr = ecs_vec_get(&column->data, column->size, row); + ctor(ptr, count, ti); + } +} + +/* Destruct components */ +static +void flecs_table_data_invoke_dtor( + ecs_column_t *column, + int32_t row, + int32_t count) +{ + ecs_type_info_t *ti = column->ti; + ecs_assert(ti != NULL, ECS_INTERNAL_ERROR, NULL); + ecs_xtor_t dtor = ti->hooks.dtor; + if (dtor) { + void *ptr = ecs_vec_get(&column->data, column->size, row); + dtor(ptr, count, ti); + } +} + +/* Invoke type hook for entities in table */ +static +void flecs_table_data_invoke_hook( + ecs_world_t *world, + ecs_table_t *table, + ecs_iter_action_t callback, + ecs_entity_t event, + ecs_column_t *column, + ecs_entity_t *entities, + int32_t row, + int32_t count) +{ + void *ptr = ecs_vec_get(&column->data, column->size, row); + flecs_invoke_hook(world, table, count, row, entities, ptr, column->id, + column->ti, event, callback); +} + +/* Run hooks that get invoked when component is added to entity */ +static +void flecs_table_data_invoke_add_hooks( + ecs_world_t *world, + ecs_table_t *table, + ecs_column_t *column, + ecs_entity_t *entities, + int32_t row, + int32_t count, + bool construct) +{ + ecs_type_info_t *ti = column->ti; + ecs_assert(ti != NULL, ECS_INTERNAL_ERROR, NULL); + + if (construct) { + flecs_table_data_invoke_ctor(column, row, count); + } + + ecs_iter_action_t on_add = ti->hooks.on_add; + if (on_add) { + flecs_table_data_invoke_hook(world, table, on_add, EcsOnAdd, column, + entities, row, count); + } +} + +/* Run hooks that get invoked when component is removed from entity */ +static +void flecs_table_data_invoke_remove_hooks( + ecs_world_t *world, + ecs_table_t *table, + ecs_column_t *column, + ecs_entity_t *entities, + int32_t row, + int32_t count, + bool dtor) +{ + ecs_type_info_t *ti = column->ti; + ecs_assert(ti != NULL, ECS_INTERNAL_ERROR, NULL); + + ecs_iter_action_t on_remove = ti->hooks.on_remove; + if (on_remove) { + flecs_table_data_invoke_hook(world, table, on_remove, EcsOnRemove, + column, entities, row, count); + } + + if (dtor) { + flecs_table_data_invoke_dtor(column, row, count); + } +} + +/* Mark table column dirty. This usually happens as the result of a set + * operation, or iteration of a query with [out] fields. */ +static +void flecs_table_data_mark_table_dirty( + ecs_table_data_t *data, + int32_t index) +{ + if (data->dirty_state) { + data->dirty_state[index] ++; + } +} + +/* Set flags for type hooks so table operations can quickly check whether a + * fast or complex operation that invokes hooks is required. */ +static +ecs_flags32_t flecs_type_info_flags( + const ecs_type_info_t *ti) +{ + ecs_flags32_t flags = 0; + + if (ti->hooks.ctor) { + flags |= EcsTableHasCtors; + } + if (ti->hooks.on_add) { + flags |= EcsTableHasCtors; + } + if (ti->hooks.dtor) { + flags |= EcsTableHasDtors; + } + if (ti->hooks.on_remove) { + flags |= EcsTableHasDtors; + } + if (ti->hooks.copy) { + flags |= EcsTableHasCopy; + } + if (ti->hooks.move) { + flags |= EcsTableHasMove; + } + + return flags; +} + +static +void flecs_table_data_init_columns( + ecs_world_t *world, + ecs_table_t *table, + int32_t column_count) +{ + if (!column_count) { + return; + } + + int32_t i, cur = 0, ids_count = table->type.count; + ecs_column_t *columns = flecs_wcalloc_n(world, ecs_column_t, column_count); + table->data->columns = columns; + + ecs_id_t *ids = table->type.array; + ecs_table_record_t *records = table->_->records; + int32_t *t2s = table->column_map; + int32_t *s2t = &table->column_map[ids_count]; + + for (i = 0; i < ids_count; i ++) { + ecs_table_record_t *tr = &records[i]; + ecs_id_record_t *idr = (ecs_id_record_t*)tr->hdr.cache; + const ecs_type_info_t *ti = idr->type_info; + if (!ti) { + t2s[i] = -1; + continue; + } + + t2s[i] = cur; + s2t[cur] = i; + tr->column = flecs_ito(int16_t, cur); + + columns[cur].ti = ECS_CONST_CAST(ecs_type_info_t*, ti); + columns[cur].id = ids[i]; + columns[cur].size = ti->size; + + if (ECS_IS_PAIR(ids[i])) { + ecs_table_record_t *wc_tr = flecs_id_record_get_table( + idr->parent, table); + if (wc_tr->index == tr->index) { + wc_tr->column = tr->column; + } + } + +#ifdef FLECS_DEBUG + ecs_vec_init(NULL, &columns[cur].data, ti->size, 0); +#endif + + table->flags |= flecs_type_info_flags(ti); + cur ++; + } +} + +/* Initialize table storage */ +void flecs_table_data_init( + ecs_world_t *world, + ecs_table_t *table, + int32_t column_count) +{ + ecs_table_data_t *data = table->data = ecs_os_calloc_t(ecs_table_data_t);; + data->column_count = flecs_ito(int16_t, column_count); + ecs_vec_init_t(NULL, &data->entities, ecs_entity_t, 0); + ecs_vec_init_t(NULL, &data->records, ecs_record_t*, 0); + + flecs_table_data_init_columns(world, table, column_count); + + if (table->flags & EcsTableHasToggle) { + int32_t i, bs_count = 0; + for (i = table->_->bs_offset; i < table->type.count; i ++) { + if (ECS_HAS_ID_FLAG(table->type.array[i], TOGGLE)) { + bs_count ++; + } + } + + ecs_assert(bs_count > 0, ECS_INTERNAL_ERROR, NULL); + + data->bitsets = flecs_wcalloc_n(world, ecs_bitset_column_t, bs_count); + for (i = 0; i < bs_count; i ++) { + flecs_bitset_init(&data->bitsets[i].data); + } + + data->bs_count = flecs_ito(int16_t, bs_count); + } + + data->flags = table->flags; // TODO +} + +/* Append operation for tables that don't have any complex logic */ +static +void flecs_table_data_fast_append( + ecs_world_t *world, + ecs_column_t *columns, + int32_t count) +{ + /* Add elements to each column array */ + int32_t i; + for (i = 0; i < count; i ++) { + ecs_column_t *column = &columns[i]; + ecs_vec_append(&world->allocator, &column->data, column->size); + } +} + +/* Grow table column. When a column needs to be reallocated this function takes + * care of correctly invoking ctor/move/dtor hooks. */ +static +void* flecs_table_data_column_append( + ecs_world_t *world, + ecs_column_t *column, + int32_t to_add, + int32_t dst_size, + bool construct) +{ + ecs_assert(column != NULL, ECS_INTERNAL_ERROR, NULL); + + ecs_type_info_t *ti = column->ti; + int32_t size = column->size; + int32_t count = column->data.count; + int32_t src_size = column->data.size; + int32_t dst_count = count + to_add; + bool can_realloc = dst_size != src_size; + void *result = NULL; + + ecs_assert(dst_size >= dst_count, ECS_INTERNAL_ERROR, NULL); + + /* If the array could possibly realloc and the component has a move action + * defined, move old elements manually */ + ecs_move_t move_ctor; + if (count && can_realloc && (move_ctor = ti->hooks.ctor_move_dtor)) { + ecs_xtor_t ctor = ti->hooks.ctor; + ecs_assert(ctor != NULL, ECS_INTERNAL_ERROR, NULL); + ecs_assert(move_ctor != NULL, ECS_INTERNAL_ERROR, NULL); + + /* Create vector */ + ecs_vec_t dst; + ecs_vec_init(&world->allocator, &dst, size, dst_size); + dst.count = dst_count; + + void *src_buffer = column->data.array; + void *dst_buffer = dst.array; + + /* Move (and construct) existing elements to new vector */ + move_ctor(dst_buffer, src_buffer, count, ti); + + if (construct) { + /* Construct new element(s) */ + result = ECS_ELEM(dst_buffer, size, count); + ctor(result, to_add, ti); + } + + /* Free old vector */ + ecs_vec_fini(&world->allocator, &column->data, size); + + column->data = dst; + } else { + /* If array won't realloc or has no move, simply add new elements */ + if (can_realloc) { + ecs_vec_set_size(&world->allocator, &column->data, size, dst_size); + } + + result = ecs_vec_grow(&world->allocator, &column->data, size, to_add); + + ecs_xtor_t ctor; + if (construct && (ctor = ti->hooks.ctor)) { + /* If new elements need to be constructed and component has a + * constructor, construct */ + ctor(result, to_add, ti); + } + } + + ecs_assert(column->data.size == dst_size, ECS_INTERNAL_ERROR, NULL); + + return result; +} + +/* Append entity to table data */ +int32_t flecs_table_data_append( + ecs_world_t *world, + ecs_table_t *table, + ecs_entity_t entity, + ecs_record_t *record, + bool construct, + bool on_add) +{ + ecs_table_data_t *data = table->data; + + /* Get count & size before growing entities array. This tells us whether the + * arrays will realloc */ + int32_t count = data->entities.count; + int32_t column_count = data->column_count; + ecs_column_t *columns = data->columns; + + /* Grow buffer with entity ids, set new element to new entity */ + ecs_entity_t *e = ecs_vec_append_t(&world->allocator, + &data->entities, ecs_entity_t); + ecs_assert(e != NULL, ECS_INTERNAL_ERROR, NULL); + *e = entity; + + /* Add record ptr to array with record ptrs */ + ecs_record_t **r = ecs_vec_append_t(&world->allocator, + &data->records, ecs_record_t*); + ecs_assert(r != NULL, ECS_INTERNAL_ERROR, NULL); + *r = record; + + /* If the table is monitored indicate that there has been a change */ + flecs_table_data_mark_table_dirty(data, 0); + ecs_assert(count >= 0, ECS_INTERNAL_ERROR, NULL); + + /* Fast path: no switch columns, no lifecycle actions */ + if (!(data->flags & EcsTableIsComplex)) { + flecs_table_data_fast_append(world, columns, column_count); + return count; + } + + ecs_entity_t *entities = data->entities.array; + + /* Reobtain size to ensure that the columns have the same size as the + * entities and record vectors. This keeps reasoning about when allocations + * occur easier. */ + int32_t size = data->entities.size; + + /* Grow component arrays with 1 element */ + int32_t i; + for (i = 0; i < column_count; i ++) { + ecs_column_t *column = &columns[i]; + flecs_table_data_column_append(world, column, 1, size, construct); + + ecs_iter_action_t on_add_hook; + if (on_add && (on_add_hook = column->ti->hooks.on_add)) { + flecs_table_data_invoke_hook(world, table, on_add_hook, EcsOnAdd, + column, &entities[count], count, 1); + } + + ecs_assert(columns[i].data.size == + data->entities.size, ECS_INTERNAL_ERROR, NULL); + ecs_assert(columns[i].data.count == + data->entities.count, ECS_INTERNAL_ERROR, NULL); + } + + int32_t bs_count = data->bs_count; + ecs_bitset_column_t *bitsets = data->bitsets; + + /* Add element to each bitset column */ + for (i = 0; i < bs_count; i ++) { + ecs_assert(bitsets != NULL, ECS_INTERNAL_ERROR, NULL); + ecs_bitset_t *bs = &bitsets[i].data; + flecs_bitset_addn(bs, 1); + } + + return count; +} + +/* Grow all data structures in a table */ +int32_t flecs_table_data_appendn( + ecs_world_t *world, + ecs_table_t *table, + int32_t to_add, + const ecs_entity_t *ids) +{ + ecs_table_data_t *data = table->data; + + ecs_assert(data != NULL, ECS_INTERNAL_ERROR, NULL); + + int32_t cur_count = data->entities.count; + int32_t column_count = data->column_count; + int32_t size = to_add + cur_count; + ecs_allocator_t *a = &world->allocator; + + /* Add record to record ptr array */ + ecs_vec_set_size_t(a, &data->records, ecs_record_t*, size); + ecs_record_t **r = ecs_vec_last_t(&data->records, ecs_record_t*) + 1; + data->records.count += to_add; + if (data->records.size > size) { + size = data->records.size; + } + + /* Add entity to column with entity ids */ + ecs_vec_set_size_t(a, &data->entities, ecs_entity_t, size); + ecs_entity_t *e = ecs_vec_last_t(&data->entities, ecs_entity_t) + 1; + data->entities.count += to_add; + ecs_assert(data->entities.size == size, ECS_INTERNAL_ERROR, NULL); + + /* Initialize entity ids and record ptrs */ + int32_t i; + if (ids) { + ecs_os_memcpy_n(e, ids, ecs_entity_t, to_add); + } else { + ecs_os_memset(e, 0, ECS_SIZEOF(ecs_entity_t) * to_add); + } + ecs_os_memset(r, 0, ECS_SIZEOF(ecs_record_t*) * to_add); + + /* Add elements to each column array */ + ecs_column_t *columns = data->columns; + for (i = 0; i < column_count; i ++) { + flecs_table_data_column_append(world, &columns[i], to_add, size, true); + ecs_assert(columns[i].data.size == size, ECS_INTERNAL_ERROR, NULL); + flecs_table_data_invoke_add_hooks(world, table, &columns[i], e, + cur_count, to_add, false); + } + + int32_t bs_count = data->bs_count; + ecs_bitset_column_t *bitsets = data->bitsets; + + /* Add elements to each bitset column */ + for (i = 0; i < bs_count; i ++) { + ecs_assert(bitsets != NULL, ECS_INTERNAL_ERROR, NULL); + ecs_bitset_t *bs = &bitsets[i].data; + flecs_bitset_addn(bs, to_add); + } + + /* If the table is monitored indicate that there has been a change */ + flecs_table_data_mark_table_dirty(data, 0); + + /* Return index of first added entity */ + return cur_count; +} + +/* Move operation for tables that don't have any complex logic */ +static +void flecs_table_fast_move( + ecs_table_data_t *dst_data, + int32_t dst_index, + ecs_table_data_t *src_data, + int32_t src_index) +{ + int32_t i_dst = 0, dst_column_count = dst_data->column_count; + int32_t i_src = 0, src_column_count = src_data->column_count; + + ecs_column_t *dst_columns = dst_data->columns; + ecs_column_t *src_columns = src_data->columns; + + for (; (i_dst < dst_column_count) && (i_src < src_column_count);) { + ecs_column_t *dst_column = &dst_columns[i_dst]; + ecs_column_t *src_column = &src_columns[i_src]; + ecs_id_t dst_id = dst_column->id; + ecs_id_t src_id = src_column->id; + + if (dst_id == src_id) { + int32_t size = dst_column->size; + void *dst = ecs_vec_get(&dst_column->data, size, dst_index); + void *src = ecs_vec_get(&src_column->data, size, src_index); + ecs_os_memcpy(dst, src, size); + } + + i_dst += dst_id <= src_id; + i_src += dst_id >= src_id; + } +} + +/* Table move logic for bitset (toggle component) column */ +static +void flecs_table_data_move_bitset_columns( + ecs_table_t *dst_table, + int32_t dst_index, + ecs_table_t *src_table, + int32_t src_index, + int32_t count, + bool clear) +{ + ecs_table_data_t *dst_data = dst_table->data; + ecs_table_data_t *src_data = src_table->data; + + int32_t i_dst = 0, dst_column_count = dst_data->bs_count; + int32_t i_src = 0, src_column_count = src_data->bs_count; + + if (!src_column_count && !dst_column_count) { + return; + } + + ecs_bitset_column_t *src_columns = src_data->bitsets; + ecs_bitset_column_t *dst_columns = dst_data->bitsets; + + for (; (i_dst < dst_column_count) && (i_src < src_column_count);) { + ecs_bitset_column_t *dst_column = &dst_columns[i_dst]; + ecs_bitset_column_t *src_column = &src_columns[i_src]; + ecs_id_t dst_id = dst_column->id, src_id = src_column->id; + + if (dst_id == src_id) { + ecs_bitset_t *src_bs = &src_columns[i_src].data; + ecs_bitset_t *dst_bs = &dst_columns[i_dst].data; + + flecs_bitset_ensure(dst_bs, dst_index + count); + + int i; + for (i = 0; i < count; i ++) { + uint64_t value = flecs_bitset_get(src_bs, src_index + i); + flecs_bitset_set(dst_bs, dst_index + i, value); + } + + if (clear) { + ecs_assert(count == flecs_bitset_count(src_bs), + ECS_INTERNAL_ERROR, NULL); + flecs_bitset_fini(src_bs); + } + } else if (dst_id > src_id) { + ecs_bitset_t *src_bs = &src_columns[i_src].data; + flecs_bitset_fini(src_bs); + } + + i_dst += dst_id <= src_id; + i_src += dst_id >= src_id; + } + + /* Clear remaining columns */ + if (clear) { + for (; (i_src < src_column_count); i_src ++) { + ecs_bitset_t *src_bs = &src_columns[i_src].data; + ecs_assert(count == flecs_bitset_count(src_bs), + ECS_INTERNAL_ERROR, NULL); + flecs_bitset_fini(src_bs); + } + } +} + +/* Move entity from src to dst table */ +void flecs_table_data_move( + ecs_world_t *world, + ecs_entity_t dst_entity, + ecs_entity_t src_entity, + ecs_table_t *dst_table, + int32_t dst_index, + ecs_table_t *src_table, + int32_t src_index, + bool construct) +{ + ecs_table_data_t *dst_data = dst_table->data; + ecs_table_data_t *src_data = src_table->data; + ecs_assert(dst_data != NULL, ECS_INTERNAL_ERROR, NULL); + ecs_assert(src_data != NULL, ECS_INTERNAL_ERROR, NULL); + + if (!((dst_data->flags | src_data->flags) & EcsTableIsComplex)) { + flecs_table_fast_move(dst_data, dst_index, src_data, src_index); + return; + } + + flecs_table_data_move_bitset_columns( + dst_table, dst_index, src_table, src_index, 1, false); + + /* If the source and destination entities are the same, move component + * between tables. If the entities are not the same (like when cloning) use + * a copy. */ + bool same_entity = dst_entity == src_entity; + + /* Call move_dtor for moved away from storage only if the entity is at the + * last index in the source table. If it isn't the last entity, the last + * entity in the table will be moved to the src storage, which will take + * care of cleaning up resources. */ + bool use_move_dtor = ecs_table_count(src_table) == (src_index + 1); + + int32_t i_dst = 0, dst_column_count = dst_data->column_count; + int32_t i_src = 0, src_column_count = src_data->column_count; + + ecs_column_t *dst_columns = dst_data->columns; + ecs_column_t *src_columns = src_data->columns; + + for (; (i_dst < dst_column_count) && (i_src < src_column_count); ) { + ecs_column_t *dst_column = &dst_columns[i_dst]; + ecs_column_t *src_column = &src_columns[i_src]; + ecs_id_t dst_id = dst_column->id; + ecs_id_t src_id = src_column->id; + + if (dst_id == src_id) { + int32_t size = dst_column->size; + + ecs_assert(size != 0, ECS_INTERNAL_ERROR, NULL); + void *dst = ecs_vec_get(&dst_column->data, size, dst_index); + void *src = ecs_vec_get(&src_column->data, size, src_index); + ecs_type_info_t *ti = dst_column->ti; + + if (same_entity) { + ecs_move_t move = ti->hooks.move_ctor; + if (use_move_dtor || !move) { + /* Also use move_dtor if component doesn't have a move_ctor + * registered, to ensure that the dtor gets called to + * cleanup resources. */ + move = ti->hooks.ctor_move_dtor; + } + + if (move) { + move(dst, src, 1, ti); + } else { + ecs_os_memcpy(dst, src, size); + } + } else { + ecs_copy_t copy = ti->hooks.copy_ctor; + if (copy) { + copy(dst, src, 1, ti); + } else { + ecs_os_memcpy(dst, src, size); + } + } + } else { + if (dst_id < src_id) { + flecs_table_data_invoke_add_hooks(world, dst_table, + dst_column, &dst_entity, dst_index, 1, construct); + } else { + flecs_table_data_invoke_remove_hooks(world, src_table, + src_column, &src_entity, src_index, 1, use_move_dtor); + } + } + + i_dst += dst_id <= src_id; + i_src += dst_id >= src_id; + } + + for (; (i_dst < dst_column_count); i_dst ++) { + flecs_table_data_invoke_add_hooks(world, dst_table, &dst_columns[i_dst], + &dst_entity, dst_index, 1, construct); + } + + for (; (i_src < src_column_count); i_src ++) { + flecs_table_data_invoke_remove_hooks(world, src_table, &src_columns[i_src], + &src_entity, src_index, 1, use_move_dtor); + } +} + + +/* Delete last operation for tables that don't have any complex logic */ +static +void flecs_table_data_fast_delete_last( + ecs_column_t *columns, + int32_t column_count) +{ + int i; + for (i = 0; i < column_count; i ++) { + ecs_vec_remove_last(&columns[i].data); + } +} + +/* Delete operation for tables that don't have any complex logic */ +static +void flecs_table_data_fast_delete( + ecs_column_t *columns, + int32_t column_count, + int32_t index) +{ + int i; + for (i = 0; i < column_count; i ++) { + ecs_column_t *column = &columns[i]; + ecs_vec_remove(&column->data, column->size, index); + } +} + +/* Delete entity from table */ +int32_t flecs_table_data_delete( + ecs_world_t *world, + ecs_table_t *table, + int32_t index, + bool destruct) +{ + ecs_table_data_t *data = table->data; + ecs_assert(data != NULL, ECS_INTERNAL_ERROR, NULL); + int32_t count = data->entities.count; + + ecs_assert(count > 0, ECS_INTERNAL_ERROR, NULL); + count --; + ecs_assert(index <= count, ECS_INTERNAL_ERROR, NULL); + + /* Move last entity id to index */ + ecs_entity_t *entities = data->entities.array; + ecs_entity_t entity_to_move = entities[count]; + ecs_entity_t entity_to_delete = entities[index]; + entities[index] = entity_to_move; + ecs_vec_remove_last(&data->entities); + + /* Move last record ptr to index */ + ecs_assert(count < data->records.count, ECS_INTERNAL_ERROR, NULL); + + ecs_record_t **records = data->records.array; + ecs_record_t *record_to_move = records[count]; + records[index] = record_to_move; + ecs_vec_remove_last(&data->records); + + /* Update record of moved entity in entity index */ + if (index != count) { + if (record_to_move) { + uint32_t row_flags = record_to_move->row & ECS_ROW_FLAGS_MASK; + record_to_move->row = ECS_ROW_TO_RECORD(index, row_flags); + ecs_assert(record_to_move->table != NULL, ECS_INTERNAL_ERROR, NULL); + ecs_assert(record_to_move->table == table, ECS_INTERNAL_ERROR, NULL); + } + } + + /* If the table is monitored indicate that there has been a change */ + flecs_table_data_mark_table_dirty(data, 0); + + /* Destruct component data */ + ecs_column_t *columns = data->columns; + int32_t column_count = data->column_count; + int32_t i; + + /* If this is a table without lifecycle callbacks or special columns, take + * fast path that just remove an element from the array(s) */ + if (!(data->flags & EcsTableIsComplex)) { + if (index == count) { + flecs_table_data_fast_delete_last(columns, column_count); + } else { + flecs_table_data_fast_delete(columns, column_count, index); + } + return count; + } + + /* Last element, destruct & remove */ + if (index == count) { + /* If table has component destructors, invoke */ + if (destruct && (data->flags & EcsTableHasDtors)) { + for (i = 0; i < column_count; i ++) { + flecs_table_data_invoke_remove_hooks(world, table, &columns[i], + &entity_to_delete, index, 1, true); + } + } + + flecs_table_data_fast_delete_last(columns, column_count); + + /* Not last element, move last element to deleted element & destruct */ + } else { + /* If table has component destructors, invoke */ + if ((data->flags & (EcsTableHasDtors | EcsTableHasMove))) { + for (i = 0; i < column_count; i ++) { + ecs_column_t *column = &columns[i]; + ecs_type_info_t *ti = column->ti; + ecs_size_t size = column->size; + void *dst = ecs_vec_get(&column->data, size, index); + void *src = ecs_vec_last(&column->data, size); + + ecs_iter_action_t on_remove = ti->hooks.on_remove; + if (destruct && on_remove) { + flecs_table_data_invoke_hook(world, table, on_remove, EcsOnRemove, + column, &entity_to_delete, index, 1); + } + + ecs_move_t move_dtor = ti->hooks.move_dtor; + if (move_dtor) { + move_dtor(dst, src, 1, ti); + } else { + ecs_os_memcpy(dst, src, size); + } + + ecs_vec_remove_last(&column->data); + } + } else { + flecs_table_data_fast_delete(columns, column_count, index); + } + } + + /* Remove elements from bitset columns */ + ecs_bitset_column_t *bitsets = data->bitsets; + int32_t bs_count = data->bs_count; + for (i = 0; i < bs_count; i ++) { + ecs_assert(bitsets != NULL, ECS_INTERNAL_ERROR, NULL); + flecs_bitset_remove(&bitsets[i].data, index); + } + + return count; +} + +/* Swap operation for bitset (toggle component) columns */ +static +void flecs_table_data_swap_bitset_columns( + ecs_table_data_t *data, + int32_t row_1, + int32_t row_2) +{ + int32_t i = 0, column_count = data->bs_count; + if (!column_count) { + return; + } + + ecs_bitset_column_t *columns = data->bitsets; + for (i = 0; i < column_count; i ++) { + ecs_bitset_t *bs = &columns[i].data; + flecs_bitset_swap(bs, row_1, row_2); + } +} + +/* Swap two rows in a table. Used for table sorting. */ +void flecs_table_data_swap( + ecs_world_t *world, + ecs_table_t *table, + int32_t row_1, + int32_t row_2) +{ + if (row_1 == row_2) { + return; + } + + ecs_table_data_t *data = table->data; + + /* If the table is monitored indicate that there has been a change */ + flecs_table_data_mark_table_dirty(data, 0); + + ecs_entity_t *entities = data->entities.array; + ecs_entity_t e1 = entities[row_1]; + ecs_entity_t e2 = entities[row_2]; + + ecs_record_t **records = data->records.array; + ecs_record_t *record_ptr_1 = records[row_1]; + ecs_record_t *record_ptr_2 = records[row_2]; + + ecs_assert(record_ptr_1 != NULL, ECS_INTERNAL_ERROR, NULL); + ecs_assert(record_ptr_2 != NULL, ECS_INTERNAL_ERROR, NULL); + + /* Keep track of whether entity is watched */ + uint32_t flags_1 = ECS_RECORD_TO_ROW_FLAGS(record_ptr_1->row); + uint32_t flags_2 = ECS_RECORD_TO_ROW_FLAGS(record_ptr_2->row); + + /* Swap entities & records */ + entities[row_1] = e2; + entities[row_2] = e1; + record_ptr_1->row = ECS_ROW_TO_RECORD(row_2, flags_1); + record_ptr_2->row = ECS_ROW_TO_RECORD(row_1, flags_2); + records[row_1] = record_ptr_2; + records[row_2] = record_ptr_1; + + flecs_table_data_swap_bitset_columns(data, row_1, row_2); + + ecs_column_t *columns = data->columns; + + /* Find the maximum size of column elements + * and allocate a temporary buffer for swapping */ + int32_t i, temp_buffer_size = ECS_SIZEOF(uint64_t); + int32_t column_count = table->data->column_count; + for (i = 0; i < column_count; i++) { + temp_buffer_size = ECS_MAX(temp_buffer_size, columns[i].size); + } + + void* tmp = ecs_os_alloca(temp_buffer_size); + + /* Swap columns */ + for (i = 0; i < column_count; i ++) { + int32_t size = columns[i].size; + void *ptr = columns[i].data.array; + + void *el_1 = ECS_ELEM(ptr, size, row_1); + void *el_2 = ECS_ELEM(ptr, size, row_2); + + ecs_os_memcpy(tmp, el_1, size); + ecs_os_memcpy(el_1, el_2, size); + ecs_os_memcpy(el_2, tmp, size); + } +} + +/* Merge data from one table column into other table column */ +static +void flecs_table_data_merge_column( + ecs_world_t *world, + ecs_column_t *dst, + ecs_column_t *src, + int32_t column_size) +{ + ecs_size_t size = dst->size; + int32_t dst_count = dst->data.count; + + if (!dst_count) { + ecs_vec_fini(&world->allocator, &dst->data, size); + *dst = *src; + src->data.array = NULL; + src->data.count = 0; + src->data.size = 0; + + /* If the new table is not empty, copy the contents from the + * src into the dst. */ + } else { + int32_t src_count = src->data.count; + + flecs_table_data_column_append(world, dst, src_count, column_size, true); + void *dst_ptr = ECS_ELEM(dst->data.array, size, dst_count); + void *src_ptr = src->data.array; + + /* Move values into column */ + ecs_type_info_t *ti = dst->ti; + ecs_assert(ti != NULL, ECS_INTERNAL_ERROR, NULL); + ecs_move_t move = ti->hooks.move_dtor; + if (move) { + move(dst_ptr, src_ptr, src_count, ti); + } else { + ecs_os_memcpy(dst_ptr, src_ptr, size * src_count); + } + + ecs_vec_fini(&world->allocator, &src->data, size); + } +} + +/* Merge storage of two tables. */ +static +void flecs_table_data_merge_columns( + ecs_world_t *world, + ecs_table_t *dst_table, + ecs_table_t *src_table, + int32_t src_count, + int32_t dst_count, + ecs_table_data_t *src_data, + ecs_table_data_t *dst_data) +{ + int32_t i_new = 0, dst_column_count = dst_table->data->column_count; + int32_t i_old = 0, src_column_count = src_table->data->column_count; + ecs_column_t *src_columns = src_data->columns; + ecs_column_t *dst_columns = dst_data->columns; + + ecs_assert(!dst_column_count || dst_columns, ECS_INTERNAL_ERROR, NULL); + + if (!src_count) { + return; + } + + /* Merge entities & records vectors */ + ecs_allocator_t *a = &world->allocator; + ecs_vec_merge_t(a, &dst_data->entities, &src_data->entities, ecs_entity_t); + ecs_assert(dst_data->entities.count == src_count + dst_count, + ECS_INTERNAL_ERROR, NULL); + ecs_vec_merge_t(a, &dst_data->records, &src_data->records, ecs_record_t*); + + int32_t column_size = dst_data->entities.size; + for (; (i_new < dst_column_count) && (i_old < src_column_count); ) { + ecs_column_t *dst_column = &dst_columns[i_new]; + ecs_column_t *src_column = &src_columns[i_old]; + ecs_id_t dst_id = dst_column->id; + ecs_id_t src_id = src_column->id; + + if (dst_id == src_id) { + flecs_table_data_merge_column(world, dst_column, src_column, column_size); + flecs_table_data_mark_table_dirty(dst_data, i_new + 1); + i_new ++; + i_old ++; + } else if (dst_id < src_id) { + /* New column, make sure vector is large enough. */ + ecs_size_t size = dst_column->size; + ecs_vec_set_size(a, &dst_column->data, size, column_size); + ecs_vec_set_count(a, &dst_column->data, size, src_count + dst_count); + flecs_table_data_invoke_ctor(dst_column, dst_count, src_count); + i_new ++; + } else if (dst_id > src_id) { + /* Old column does not occur in new table, destruct */ + flecs_table_data_invoke_dtor(src_column, 0, src_count); + ecs_vec_fini(a, &src_column->data, src_column->size); + i_old ++; + } + } + + flecs_table_data_move_bitset_columns( + dst_table, dst_count, src_table, 0, src_count, true); + + /* Initialize remaining columns */ + for (; i_new < dst_column_count; i_new ++) { + ecs_column_t *column = &dst_columns[i_new]; + int32_t size = column->size; + ecs_assert(size != 0, ECS_INTERNAL_ERROR, NULL); + ecs_vec_set_size(a, &column->data, size, column_size); + ecs_vec_set_count(a, &column->data, size, src_count + dst_count); + flecs_table_data_invoke_ctor(column, dst_count, src_count); + } + + /* Destruct remaining columns */ + for (; i_old < src_column_count; i_old ++) { + ecs_column_t *column = &src_columns[i_old]; + flecs_table_data_invoke_dtor(column, 0, src_count); + ecs_vec_fini(a, &column->data, column->size); + } + + /* Mark entity column as dirty */ + flecs_table_data_mark_table_dirty(dst_data, 0); +} + +/* Merge source table into destination table. This typically happens as result + * of a bulk operation, like when a component is removed from all entities in + * the source table (like for the Remove OnDelete policy). */ +void flecs_table_data_merge( + ecs_world_t *world, + ecs_table_t *dst_table, + ecs_table_t *src_table) +{ + ecs_assert(src_table != NULL, ECS_INTERNAL_ERROR, NULL); + ecs_assert(!src_table->_->lock, ECS_LOCKED_STORAGE, NULL); + + ecs_table_data_t *dst_data = dst_table->data; + ecs_table_data_t *src_data = src_table->data; + ecs_assert(dst_data != NULL, ECS_INTERNAL_ERROR, NULL); + ecs_assert(src_data != NULL, ECS_INTERNAL_ERROR, NULL); + + bool move_data = false; + + ecs_entity_t *src_entities = src_data->entities.array; + int32_t src_count = src_data->entities.count; + int32_t dst_count = dst_data->entities.count; + ecs_record_t **src_records = src_data->records.array; + + /* First, update entity index so old entities point to new type */ + int32_t i; + for(i = 0; i < src_count; i ++) { + ecs_record_t *record; + if (dst_table != src_table) { + record = src_records[i]; + ecs_assert(record != NULL, ECS_INTERNAL_ERROR, NULL); + } else { + record = flecs_entities_ensure(world, src_entities[i]); + } + + uint32_t flags = ECS_RECORD_TO_ROW_FLAGS(record->row); + record->row = ECS_ROW_TO_RECORD(dst_count + i, flags); + record->table = dst_table; + } + + /* Merge table columns */ + if (move_data) { + *dst_data = *src_data; + } else { + flecs_table_data_merge_columns(world, dst_table, src_table, + src_count, dst_count, src_data, dst_data); + } +} + +/* Shrink table storage to fit number of entities */ +bool flecs_table_data_shrink( + ecs_world_t *world, + ecs_table_t *table) +{ + ecs_table_data_t *data = table->data; + bool has_payload = data->entities.array != NULL; + ecs_vec_reclaim_t(&world->allocator, &data->entities, ecs_entity_t); + ecs_vec_reclaim_t(&world->allocator, &data->records, ecs_record_t*); + + int32_t i, count = data->column_count; + for (i = 0; i < count; i ++) { + ecs_column_t *column = &data->columns[i]; + ecs_vec_reclaim(&world->allocator, &column->data, column->size); + } + + return has_payload; +} diff --git a/src/storage/table_data.h b/src/storage/table_data.h new file mode 100644 index 0000000000..64bf63c7cb --- /dev/null +++ b/src/storage/table_data.h @@ -0,0 +1,59 @@ +/** + * @file table_data.h + * @brief Table data implementation. + */ + +#ifndef FLECS_TABLE_DATA_H +#define FLECS_TABLE_DATA_H + +void flecs_table_data_init( + ecs_world_t *world, + ecs_table_t *table, + int32_t column_count); + +int32_t flecs_table_data_append( + ecs_world_t *world, + ecs_table_t *table, + ecs_entity_t entity, + ecs_record_t *record, + bool construct, + bool on_add); + +int32_t flecs_table_data_appendn( + ecs_world_t *world, + ecs_table_t *table, + int32_t to_add, + const ecs_entity_t *ids); + +void flecs_table_data_move( + ecs_world_t *world, + ecs_entity_t dst_entity, + ecs_entity_t src_entity, + ecs_table_t *dst_table, + int32_t dst_index, + ecs_table_t *src_table, + int32_t src_index, + bool construct); + +int32_t flecs_table_data_delete( + ecs_world_t *world, + ecs_table_t *table, + int32_t index, + bool destruct); + +void flecs_table_data_swap( + ecs_world_t *world, + ecs_table_t *table, + int32_t row_1, + int32_t row_2); + +void flecs_table_data_merge( + ecs_world_t *world, + ecs_table_t *dst_table, + ecs_table_t *src_table); + +bool flecs_table_data_shrink( + ecs_world_t *world, + ecs_table_t *table); + +#endif