mirror of
https://git.code.sf.net/p/quake/quakeforge
synced 2024-11-23 12:52:46 +00:00
7e338b7e29
Removing a hierarchy from an entity can result in a large number of component removes in the same pool, thus changing index of the lest element of the pool. This *seems* to fix the memory corruption I've been experiencing with the debug UI.
148 lines
4.3 KiB
C
148 lines
4.3 KiB
C
/*
|
|
entity.c
|
|
|
|
ECS entity management
|
|
|
|
Copyright (C) 2022 Bill Currke
|
|
|
|
This program is free software; you can redistribute it and/or
|
|
modify it under the terms of the GNU General Public License
|
|
as published by the Free Software Foundation; either version 2
|
|
of the License, or (at your option) any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
|
|
|
See the GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program; if not, write to:
|
|
|
|
Free Software Foundation, Inc.
|
|
59 Temple Place - Suite 330
|
|
Boston, MA 02111-1307, USA
|
|
|
|
*/
|
|
#ifdef HAVE_CONFIG_H
|
|
# include "config.h"
|
|
#endif
|
|
|
|
#include "QF/heapsort.h"
|
|
#include "QF/sys.h"
|
|
|
|
#define IMPLEMENT_ECS_ENTITY_Funcs
|
|
#include "QF/ecs.h"
|
|
|
|
#include "compat.h"
|
|
|
|
static void swap_inds (uint32_t *a, uint32_t *b)
|
|
{
|
|
uint32_t t = *a;
|
|
*a = *b;
|
|
*b = t;
|
|
}
|
|
|
|
VISIBLE void *
|
|
Ent_AddComponent (uint32_t ent, uint32_t comp, ecs_registry_t *registry)
|
|
{
|
|
ecs_pool_t *pool = ®istry->comp_pools[comp];
|
|
ecs_subpool_t *subpool = ®istry->subpools[comp];
|
|
component_t *c = ®istry->components.a[comp];
|
|
uint32_t id = Ent_Index (ent);
|
|
uint32_t ind = pool->sparse[id];
|
|
if (ind >= pool->count || pool->dense[ind] != ent) {
|
|
if (pool->count == pool->max_count) {
|
|
pool->max_count += COMP_GROW;
|
|
pool->dense = realloc (pool->dense,
|
|
pool->max_count * sizeof (uint32_t));
|
|
Component_ResizeArray (c, &pool->data, pool->max_count);
|
|
}
|
|
uint32_t ind = pool->count++;
|
|
pool->sparse[id] = ind;
|
|
pool->dense[ind] = ent;
|
|
uint32_t rind = subpool->num_ranges - subpool->available;
|
|
if (rind && c->rangeid) {
|
|
uint32_t rangeid = c->rangeid (registry, ent, comp);
|
|
uint32_t rangeind = subpool->sorted[Ent_Index (rangeid)];
|
|
while (rind-- > rangeind) {
|
|
if (subpool->ranges[rind] == ind) {
|
|
subpool->ranges[rind]++;
|
|
continue;
|
|
}
|
|
uint32_t end = subpool->ranges[rind]++;
|
|
Component_MoveElements (c, pool->data, ind, end, 1);
|
|
swap_inds (&pool->sparse[Ent_Index (pool->dense[end])],
|
|
&pool->sparse[Ent_Index (pool->dense[ind])]);
|
|
swap_inds (&pool->dense[ind], &pool->dense[end]);
|
|
ind = end;
|
|
}
|
|
}
|
|
}
|
|
return Ent_GetComponent (ent, comp, registry);
|
|
}
|
|
|
|
static int
|
|
range_cmp (const void *_key, const void *_range, void *_subpool)
|
|
{
|
|
const uint32_t *key = _key;
|
|
const uint32_t *range = _range;
|
|
ecs_subpool_t *subpool = _subpool;
|
|
|
|
// range is the first index for the next subpool range
|
|
if (*key >= *range) {
|
|
return 1;
|
|
}
|
|
if (range - subpool->ranges > 0) {
|
|
return *key >= range[-1] ? 0 : -1;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static uint32_t *
|
|
find_range (ecs_subpool_t *subpool, uint32_t ind)
|
|
{
|
|
return bsearch_r (&ind, subpool->ranges,
|
|
subpool->num_ranges - subpool->available,
|
|
sizeof (uint32_t), range_cmp, subpool);
|
|
}
|
|
|
|
VISIBLE void
|
|
Ent_RemoveComponent (uint32_t ent, uint32_t comp, ecs_registry_t *registry)
|
|
{
|
|
uint32_t id = Ent_Index (ent);
|
|
ecs_pool_t *pool = ®istry->comp_pools[comp];
|
|
ecs_subpool_t *subpool = ®istry->subpools[comp];
|
|
uint32_t ind = pool->sparse[id];
|
|
component_t *c = ®istry->components.a[comp];
|
|
if (ind < pool->count && pool->dense[ind] == ent) {
|
|
// invalidate the entity for this component to prevent the component
|
|
// being double-removed due to deletion of the component resulting
|
|
// in the entity being deleted (happens with hierarchies)
|
|
pool->dense[ind] = -1;
|
|
Component_DestroyElements (c, pool->data, ind, 1);
|
|
uint32_t range_count = subpool->num_ranges - subpool->available;
|
|
// if ind >= the last range, then it is outside the subpools
|
|
if (range_count && ind < subpool->ranges[range_count - 1]) {
|
|
uint32_t *range = find_range (subpool, ind);
|
|
while ((size_t) (range - subpool->ranges) < range_count) {
|
|
uint32_t end = --*range;
|
|
range++;
|
|
if (ind < end) {
|
|
pool->sparse[Ent_Index (pool->dense[end])] = ind;
|
|
pool->dense[ind] = pool->dense[end];
|
|
Component_MoveElements (c, pool->data, ind, end, 1);
|
|
ind = end;
|
|
}
|
|
}
|
|
}
|
|
uint32_t last = pool->count - 1;
|
|
if (last > ind) {
|
|
pool->sparse[Ent_Index (pool->dense[last])] = ind;
|
|
pool->dense[ind] = pool->dense[last];
|
|
Component_MoveElements (c, pool->data, ind, last, 1);
|
|
}
|
|
pool->count--;
|
|
pool->sparse[id] = nullent;
|
|
}
|
|
}
|