Using HybridECS, getting this: The NativeArray has been deallocated, it is not allowed to access it.

I’m building my game using default monobehaviour and hybrid ECS where it makes sense (where the components is used on multiple objects). However, I’m now getting this error in two of my scripts: The NativeArray has been deallocated, it is not allowed to access it.

The enemy spawner gets the error on line 57, and the fireball script gets it on line 89. So it’s when I’m trying to access the entitiy in the entities array, makes sense that it doesn’t work if the “NativeArray has been deallocated”. However, why is it deallocated, what am I doing wrong?

using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using Unity.Entities;

public class EnemySpawner : MonoBehaviour
{
    public float spawnDelay = 5f;

    [HideInInspector]
    public PoolManager.PoolType enemyToSpawn;
    [HideInInspector]
    public float elapsed = 0f;
}

class EnemySpawnerSystem : ComponentSystem
{
    private PoolManager poolManager;
    private GameManager gameManager;
    private List<PoolManager.PoolType> possibleEnemies;

    struct Components
    {
        public Transform transform;
        public EnemySpawner spawner;
    }

    protected override void OnStartRunning()
    {
        //base.OnStartRunning();

        poolManager = PoolManager.instance;
        gameManager = GameManager.instance;
        possibleEnemies = new List<PoolManager.PoolType>();

        var entities = GetEntities<Components>();
        for (int entityIndex = 0; entityIndex < entities.Length; ++entityIndex)
        {
            var e = entities[entityIndex];
            e.spawner.elapsed = Random.Range(0f, e.spawner.spawnDelay - 2f);
        }

        possibleEnemies.Add(PoolManager.PoolType.POOL_ORC_WARRIOR_A);
        possibleEnemies.Add(PoolManager.PoolType.POOL_ORC_WARRIOR_B);
        possibleEnemies.Add(PoolManager.PoolType.POOL_ORC_WARRIOR_C);
    }

    protected override void OnUpdate()
    {
        float deltaTime = Time.deltaTime;
        var entities = GetEntities<Components>();

        if (gameManager.numEnemiesToSpawn > 0)
        {
            for (int entityIndex = 0; entityIndex < entities.Length; ++entityIndex)
            {
                var e = entities[entityIndex];

                if (e.spawner.elapsed >= e.spawner.spawnDelay)
                {
                    SpawnEnemy(e.spawner);
                }

                e.spawner.elapsed += deltaTime;
            }
        }
    }

    private void SpawnEnemy(EnemySpawner spawner)
    {
        spawner.enemyToSpawn = possibleEnemies[Random.Range(0, possibleEnemies.Count)];
        poolManager.SpawnFromPool(spawner.enemyToSpawn, spawner.transform.position + Random.insideUnitSphere, spawner.transform.rotation);
        spawner.elapsed = 0f;
        gameManager.numEnemiesToSpawn--;
    }
}
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using UnityEngine.Events;
using Unity.Entities;

public class Fireball : ProjectileBase
{
    [Header("Settings")]
    public float destroyDelay = 0.35f;
    public float damage = 30f;
   
    [Header("Link Settings")]
    public ParticleSystem fireballParticles;
    public ParticleSystem smokeParticles;

    [HideInInspector]
    public bool wantsRemove = false;
    [HideInInspector]
    public float destroyCounter = 0f;

    private PoolManager poolManager;
    private EventManager eventManager;

    private void Start()
    {
        poolManager = PoolManager.instance;
        eventManager = EventManager.instance;
    }

    public void OnDisable()
    {
        destroyCounter = 0f;
        wantsRemove = false;
        movedUnits = 0f;
    }

    private void OnTriggerEnter(Collider other)
    {
        if (((1 << other.gameObject.layer) & hitLayer) != 0)
        {
            Health otherHealth = other.GetComponent<Health>();
            otherHealth.TakeDamage(damage);
            Explode();
        }
    }

    public void Explode()
    {
        if (wantsRemove)
        {
            return;
        }

        eventManager.TriggerEvent("FireballExplosion");
        poolManager.SpawnFromPool(PoolManager.PoolType.POOL_EXPLOSION, transform.position, transform.rotation);
        fireballParticles.Stop();
        smokeParticles.Stop();
        wantsRemove = true;
    }
}

class FireballSystem : ComponentSystem
{
    private EventManager eventManager;
    private PoolManager poolManager;

    struct Components
    {
        public Transform transform;
        public Fireball projectile;
    }

    protected override void OnStartRunning()
    {
        base.OnStartRunning();

        eventManager = EventManager.instance;
        poolManager = PoolManager.instance;
    }

    protected override void OnUpdate()
    {
        float deltaTime = Time.deltaTime;
        var entities = GetEntities<Components>();

        for (int entityIndex = 0; entityIndex < entities.Length; ++entityIndex)
        {
            var e = entities[entityIndex];
            if (e.projectile.movedUnits >= e.projectile.range)
            {
                e.projectile.Explode();

                if (e.projectile.destroyCounter >= e.projectile.destroyDelay)
                {
                    e.projectile.gameObject.SetActive(false);
                }
                e.projectile.destroyCounter += deltaTime;
               
                continue;
            }

            Vector3 movement = e.projectile.direction * e.projectile.movementSpeed * deltaTime;
            Vector3 deltaMovement = e.transform.position + movement - e.transform.position;

            e.transform.position += movement;
            e.projectile.movedUnits += Mathf.Abs(deltaMovement.x) + Mathf.Abs(deltaMovement.y) + Mathf.Abs(deltaMovement.z);
        }
    }
}

In your fireballSystem and possibly poolManager.SpawnFromPool, you are activating/disabling game objects. If those game objects have a GameObjectEntity attached, then doing this is creating and destroying entities, which in turn is invalidating your ComponentGroup arrays.

The way around this is to make a sort of command list from your loops, then spawn/SetActive everything once you’re done looping over everything. Much like the EntityCommandBuffer.

1 Like