Initial code review to keep me honest

Hi all,
Thanks to this forum and a few tutorials around the place I have finally been able to make the beginning of a game in ECS. I am starting to be able to self correct my mistakes and trying a few ideas but what I think would be really helpful is pulling up the covers and getting some feedback on a couple of my systems and approach.

  • I have x amount of locations (currently 50). They are just ‘areas’ 50 in the x and z.
  • in those locations I spawn (currently) 100 entities
  • These entities are basic cubes, no physics. They have 1 component (will split next code changes) called HumanData
  • Will probably split Infection stuff to InfectionData
  • Conceptually for the game, they can either be infected or uninfected. true or false. Keep it simple for now.

public struct HumanData : IComponentData {
    public int Location; // simple int will correspond to a location
    public float3 OffsetPosition; // offset for position
    public float3 StartPos;
    public float3 EndPos;
    public float3 TravelVector;
    public float Distance;
    public float Speed;
    public int MyIndex;

    // infection stuff
    public bool isInfected;
    public bool isAsymptomatic;
}

  • I randomise movement so they can move around to different points in there locations (HumanMoveSystem)
  • They cannot currently move outside that initial location area. There new ‘endPos’ is offset by the locations position so they always stay in that area.

    [UpdateAfter(typeof(RandomSystem))]
    public class HumanMoveSystem : SystemBase
    {

        protected override void OnUpdate()
        {
            float deltaTime = Time.DeltaTime;
            var randomArray = World.GetExistingSystem<RandomSystem>().RandomArray;
            float locationSize = (GameDataManager.instance.locationSize - 1) / 2.0f;

            Entities
                .WithNativeDisableParallelForRestriction(randomArray)
                .WithName("HumanMoveSystem")
                .WithBurst()
                .ForEach((int nativeThreadIndex, ref Translation position, ref Rotation rotation, ref HumanData humanData) =>
                {
                    float3 travelVector = humanData.EndPos - humanData.StartPos;
                    position.Value += math.normalize(travelVector) * deltaTime * humanData.Speed;
                    humanData.TravelVector = math.normalize(travelVector) * deltaTime * humanData.Speed;
                    float distance = math.distance(position.Value, humanData.EndPos);
                    humanData.Distance = distance;
                    if (distance < 1f)
                    {
                        humanData.StartPos = humanData.EndPos;
                        float distanceCheck;

                        do
                        {
                            var random = randomArray[nativeThreadIndex];
                            float x = random.NextFloat(-1 * locationSize, locationSize);
                            float z = random.NextFloat(-1 * locationSize, locationSize);
                            humanData.EndPos = new float3(x, 0, z) + humanData.OffsetPosition;
                            distanceCheck = math.distance(humanData.StartPos, humanData.EndPos);
                            randomArray[nativeThreadIndex] = random;
                        }
                        while (distanceCheck < locationSize/2.0f);
                    }
                })
                .Schedule();
        }
    }

  • Because of this, I have a patient zero in each location (for now, while I work on improvements)
  • as an entity A distance to another entity B reaches close enough (distance < 0.1f) - if A or B are infected, then both become infected.(HumanInfectionDetectionSystem)

    [UpdateAfter(typeof(HumanMoveSystem))]
    public class HumanInfectionDetectionSystem : JobComponentSystem
    {
        EntityQuery m_Group;
        protected override void OnCreate()
        {

            var query = new EntityQueryDesc()
            {
                All = new ComponentType[]
                {
                    typeof(HumanData),
                    ComponentType.ReadOnly<Translation>()
                }
            };

            m_Group = GetEntityQuery(query);

        }

        [BurstCompile]
        public struct DistanceCheckJob : IJobParallelFor
        {
            [NativeDisableParallelForRestriction] public NativeArray<HumanData> HumanData;
            [ReadOnly] public NativeArray<Translation> TranslationData;

            public void Execute(int index)
            {
                var currentData = HumanData[index];

                var currentPos = TranslationData[index];
                for (var i = index + 1; i < HumanData.Length; i++)
                {
                    var humanBPosition = TranslationData[i];
                    var humanBData = HumanData[i];
                    if (currentData.Location == humanBData.Location)
                    {
                        // if we are already infected, ignore
                        if ((!currentData.isInfected && humanBData.isInfected) ||
                            (currentData.isInfected && !humanBData.isInfected))
                        {
                            if (math.distance(currentPos.Value, humanBPosition.Value) < 0.5f)
                            {
                                currentData.isInfected = true;
                                humanBData.isInfected = true;
                                HumanData[index] = currentData;
                                HumanData[i] = humanBData;
                            }
                        }
                    }

                }
            }
        }

        protected override JobHandle OnUpdate(JobHandle inputDependencies)
        {
            var humanData = m_Group.ToComponentDataArray<HumanData>(Allocator.TempJob);
            var translationData = m_Group.ToComponentDataArray<Translation>(Allocator.TempJob);
            var distanceCheckJob = new DistanceCheckJob
            {
                HumanData = humanData,
                TranslationData = translationData,
            };
            var collisionJobHandle = distanceCheckJob.Schedule(translationData.Length, 32);
            collisionJobHandle.Complete();

            // ToComponentDataArray copies the data, so we need to write it back again
            m_Group.CopyFromComponentDataArray(humanData);

            humanData.Dispose();
            translationData.Dispose();
            return collisionJobHandle;

        }
    }

  • If an entity is infected, I change the material on it. This is a bottleneck system due to needing to use EntityManager on a SharedComponent (ShowInfectionMaterialSystem)

[UpdateAfter(typeof(HumanMoveSystem))]
    public class ShowInfectionMaterialSystem : SystemBase
    {
        Material newMaterial;
        
        // this just checks to see if the object is infected, if so, change the material
        protected override void OnUpdate()
        {
            if (newMaterial == null)
                newMaterial = GameDataManager.instance.infectedMaterial;

            Entities
                .WithName("ShowInfectionMaterialSystem")
                .WithoutBurst()
                .WithStructuralChanges()
                .ForEach((Entity entity, ref HumanData infectionData) =>
                {
                    if (infectionData.isInfected && !infectionData.isAsymptomatic)
                    {
                        var render = EntityManager.GetSharedComponentData<RenderMesh>(entity);
                        EntityManager.SetSharedComponentData(entity, new RenderMesh() { mesh = render.mesh, material = newMaterial });
                        infectionData.isAsymptomatic = true;
                    }
                })
                .Run();

        }
    }

  • Finally just for visual purposes, over each location I have a text field I update with the ‘% infected’ (InformationSystem)

    [UpdateAfter(typeof(HumanInfectionDetectionSystem))]
    public class InformationSystem : JobComponentSystem
    {
        EntityQuery m_Group;

        protected override void OnCreate()
        {

            var query = new EntityQueryDesc()
            {
                All = new ComponentType[]
                {
                    ComponentType.ReadOnly<HumanData>()
                }
            };

            m_Group = GetEntityQuery(query);
        }

        [BurstCompile]
        public struct PercentageUpdateJob : IJobFor
        {
            [ReadOnly] public NativeArray<HumanData> HumanData;
            public NativeArray<int> Totals;


            public void Execute(int index)
            {
                var currentData = HumanData[index];
                if (currentData.isInfected && currentData.isAsymptomatic)
                    Totals[currentData.Location]++;
            }
        }

        protected override JobHandle OnUpdate(JobHandle inputDependencies)
        {
            var humanData = m_Group.ToComponentDataArray<HumanData>(Allocator.TempJob);
            NativeArray<int> totals = new NativeArray<int>(GameDataManager.instance.Locations.Length, Allocator.TempJob);

            var percentageUpdateJob = new PercentageUpdateJob
            {
                HumanData = humanData,
                Totals = totals
            };
            var collisionJobHandle = percentageUpdateJob.Schedule(humanData.Length, inputDependencies);
            collisionJobHandle.Complete();
            InfoManager.instance.PercentageTotals = totals.ToArray();
            totals.Dispose();
            humanData.Dispose();
            return collisionJobHandle;

        }
    }

  • I have messed around with chunks, but was unsure how I could chunk some of the ideas going on here.
  • I think my distance check is actually expensive (have not measured). Would an AABB type check be better, that is just checking my position values?
  • The bottleneck is the ShowInfectionMaterialSystem. How can I get an improvement from this idea?
  • Since locations is set at the start, I was hoping to perhaps set my native int array ‘totals’, in the InformationSystem once, rather than every job, as it will never grow or shrink.
  • Any other tips would be very helpful at this stage.

Cheers and Beers

Only looked through briefly, but instead of changing material, which is slow as you point out, you could change a material per instance property. E.g. if all you want to do is update the color of an infected entity, then you can do this very easily and it can be done in a job. HybridRenderer has a few in-built IComponentData types you can use for it that are picked up automatically (see link).

Not sure about the distance check. If it is slow, maybe using DOTS physics with simple AABB colliders triggering this would be faster.

@charleshendry thanks for the tip on the hybrid renderer. Previously I had tried to use this but failed hard. This time, I updated to 2020.1.2f1, followed the instructions, and now gained 25-30fps. I was previously sitting around 55-65 fps, now it’s 85-90…very cool. Super easy change and a great performance boost!

Updated ShowInfectedMaterialSystem:

[UpdateAfter(typeof(HumanMoveSystem))]
public class ShowInfectionMaterialSystem : SystemBase
{       
    // this just checks to see if the object is infected, if so, change the material
    protected override void OnUpdate()
    {
        Entities
            .WithName("ShowInfectionMaterialSystem")
            .WithBurst()
            .ForEach((ref HumanData infectionData, ref InfectedColourData colourData) =>
            {
                if (infectionData.isInfected && !infectionData.isAsymptomatic)
                {
                    colourData.Value = new float4(1, 0, 0, 1);
                    infectionData.isAsymptomatic = true;
                }
            })
            .Schedule();
    }
}

I’ll go back to the dots physics way I was doing it previously. It felt like a waste because the only thing I wanted was a trigger at the time, but I will give it another shot.

One tip that might be interesting to comment on is that the data-driven pattern has the use of simple components, instead of complex components, as it will prioritize code reuse as well as processing.

1 Like

A couple of things:

  1. Your Entities.ForEach is using Schedule instead of ScheduleParallel, so it is running single-threaded (the behavior is different from the old JobComponentSystem). I’m guessing from the rest of the code you did not intend for this to be single-threaded.
  2. Typically, what you are doing in lines 39-47 of DistanceCheckJob is a race condition, as multiple threads could be writing to the same i-th value at the same time. However, you almost lucked out a little. Due to the rest of the logic, you are only ever writing one value. If you also never read that value within the same job, you’d actually be fine as the order that those threads write the identical value would not matter. (Fun fact, this is how ComponentDataFromEntity change filter updates get away with not modifying the version number atomically.) One solution would be to write the new infection values to a seperate boolean array, and then chain that into a job which copies those booleans back into the original humanData array.

@ScriptsEngineer Thanks for that, I will be breaking down that ‘HumanData’ component today, at least into 2 for now so one can carry infection information, the other can carry location data.

@DreamingImLatios Excellent reminder about the parrallel job stuff. Just from that tip a bunch of light bulbs are going off in regards to stuff I have read or experimented with. And see below on getting rid of my distance check.

@charleshendry Thanks so much for the reminder about the Dots Physics.
I had a trigger system before and dropped it at the time (experimenting, learning). Of course I should be using the DOTS physics. Another simple change to re-enable the physics collisions almost doubled my fps.

[UpdateAfter(typeof(EndFramePhysicsSystem))]
public class InfectionTriggerDetectionSystem : JobComponentSystem
{
    // we need a couple of extra worlds that have finished simulating
    BuildPhysicsWorld physicsWorld;
    StepPhysicsWorld stepWorld;

    protected override void OnCreate()
    {
        physicsWorld = World.GetOrCreateSystem<BuildPhysicsWorld>();
        stepWorld = World.GetOrCreateSystem<StepPhysicsWorld>();
    }


    struct BoxTriggerEventJob : ITriggerEventsJob
    {
        public ComponentDataFromEntity<HumanData> HumanDataGroup;

        public void Execute(TriggerEvent triggerEvent)
        {
            Entity entityA = triggerEvent.Entities.EntityA;
            Entity entityB = triggerEvent.Entities.EntityB;

            var componentA = HumanDataGroup[entityA];
            var componentB = HumanDataGroup[entityB];

            bool infectedA = componentA.isInfected && componentA.isAsymptomatic;
            bool infectedB = componentB.isInfected && componentB.isAsymptomatic;
            if (infectedA || infectedB)
            {
                componentA.isInfected = true;
                componentB.isInfected = true;
                HumanDataGroup[entityA] = componentA;
                HumanDataGroup[entityB] = componentB;
            }
        }
    }

    protected override JobHandle OnUpdate(JobHandle inputDeps)
    {
        JobHandle jobHandle = new BoxTriggerEventJob
        {
            HumanDataGroup = GetComponentDataFromEntity<HumanData>()
        }.Schedule(stepWorld.Simulation, ref physicsWorld.PhysicsWorld, inputDeps);

        return jobHandle;
    }
}

I am not that familiar yet with the physics system, so I might be running this in a single thread again but based on the performance increase, I think I am getting the benefits. I’ll do some more reading and checking.

After all the changes above (Hybrid Materials, ParallelJobs and DOTS physics) I am now hitting 155-165fp with 6000 entities.

As a test I bumped it to 25000 and was still hitting 50-60 fps. Now my bottleneck is probably displaying and updating 60 UI panels, even though they are all on their own canvas, they all get called at the same time.

Any reason you are using JobComponentSystem instead of SystemBase? As far as I’m aware, JobComponentSystem is deprecated and SystemBase should be used in all cases. You should be able to use the Dependency property from SystemBase instead of inputDeps.

Dependency = new BoxTriggerEventJob
{
     HumanDataGroup = GetComponentDataFromEntity<HumanData>()

}.Schedule(stepWorld.Simulation, ref physicsWorld.PhysicsWorld, Dependency);

@Bivens32 legend, cheers. The only reason is that’s from the tutorial I followed sprinkled with some misunderstanding about Execute v ForEach. I’ll take that lead and refer to the documentation. My information system is also using the JobComponentSystem. I will update that and maybe try the idea of chaining this systems work as suggested above.

@ScriptsEngineer I have updated the components to be a little more particular about their concerns:

[GenerateAuthoringComponent]
public struct HumanMovement : IComponentData
{
    public float3 StartPos;
    public float3 EndPos;
    public float3 TravelVector;
    public float Distance;
    public float Speed;
}

[GenerateAuthoringComponent]
public struct HumanLocation : IComponentData
{
    public int Location; // simple int will correspond to a location
    public float3 OffsetPosition; // offset for position
}

[GenerateAuthoringComponent]
public struct HumanInfection : IComponentData
{
    public bool isInfected;
    public bool isAsymptomatic;
}

I have now refactored the systems to use the right component data.
I might be scraping the sides of optimism but I would swear this has given me an increase in frames (5-10).
I am now hitting 190 FPS on my pc, and will test later on the mac.

Hi all, now I have added in all these changes, my final part consists of how I can get the concept of locations to work.

A location is somewhere a human can be. This can be linked to one or more locations.
In the above HumanLocation the ‘int Location’ should really be ‘int index’ as all it is is my location index when I initially spawn my entities.

 void Start()
        {
            store = new BlobAssetStore();
            manager = World.DefaultGameObjectInjectionWorld.EntityManager;
            var settings = GameObjectConversionSettings.FromWorld(World.DefaultGameObjectInjectionWorld, store);
            var human = GameObjectConversionUtility.ConvertGameObjectHierarchy(HumanPrefab, settings);
            float locationSize = (GameDataManager.instance.locationSize - 1) / 2.0f;
            int totalLocations = GameDataManager.instance.Locations.Length;
            int patientZero = UnityEngine.Random.Range(0, totalLocations * EntitiesPerLocation);

            for (int x = 0; x < totalLocations; x++)
            {
                for (int y = 0; y < EntitiesPerLocation; y++)
                {
                    var instance = manager.Instantiate(human);
                    float startX = Random.Range(-1 * locationSize, locationSize);
                    float startZ = Random.Range(-1 * locationSize, locationSize);
                    float endX = Random.Range(-1 * locationSize, locationSize);
                    float endZ = Random.Range(-1 * locationSize, locationSize);
                    float speed = Random.Range(2.0f, 5.0f);
                    var startPos = new Vector3(startX, 0, startZ);
                    var endPos = new Vector3(endX, 0, endZ);

                    Vector3 offset = GameDataManager.instance.Locations[x].position;
                    manager.SetComponentData(instance, new HumanMovement
                    {
                        StartPos = startPos + offset,
                        EndPos = endPos + offset,
                        Speed = speed
                    });

                    manager.SetComponentData(instance, new HumanLocation
                    {
                        OffsetPosition = offset,
                        Location = x
                    });

                    manager.SetComponentData(instance, new HumanInfection
                    {
                        isAsymptomatic = false,
                        isInfected = ((x * 100) + y == patientZero)
                    });

                    manager.SetComponentData(instance, new Translation { Value = startPos + offset });

                }
            }
        }

So spawning entities in the one location is not a problem, getting them to travel to different locations is where I am stuck.

Some initial ideas:

  • I tried creating a normal ‘location’ game object that existed on the object and retained the location index and an array of indexes you could get to from this one.
  • eg index 1, linked to indexes 2,3,5
  • That meant I had to copy all that location data to a native container at some point so it could be used in a system somewhere. If I wanted to change locations, I had to know where I could go. NativeList of NativeArrays?
  • Got stuck, crashed and burned.

Another Idea I have not tried yet:

  • The move system could flag on a new component that the entity needs to exit the location
  • Another system checks purely if this flag is set
  • If location is an entity, then it could contain a new Location component that contains the indexes you can traverse to
  • Here is where I am getting a little stuck in my thoughts - some system still needs to know locations and their exit nodes. They are always two different component groups that need information about the other and I am unsure which way to go.

Can anyone make any suggestions here?

Cheers

I’m not sure I fully understand what you are struggling with, but there’s a few ideas I will throw at the wall for you.

First, you could have location entities which contain dynamic buffers of their connected locations in the form of an entity reference.

Second, you could instead build a blob asset containing all locations and all connections between them, and then give each human a BlobAssetReference pointing to that blob.

For getting the randomized movements, you will likely want both timer components and a random number generators (either Random as a component or as part of a system or singleton).

Hi @DreamingImLatios , the blob assets actually sound perfect. After some reading they really suit the idea of these locations as the location data wont ever change, but a reference to which location the human is currently in does change.

Cheers

Hi again, thanks to everyone for all the hints and tips so far. I feel like I am starting to make some progress in ECS.

In regards to this project, I am now trying to work with the concept of my humans being able to move from one location to another.

What I settled on was creating a new Location Entity.
The location entity has an index, an offset and a Dynamic buffer:

    public struct LocationData : IComponentData
    {
        public int index;
        public float3 offset;
        public float scale;
    }

    [InternalBufferCapacity(8)]
    public struct DestinationBuffer : IBufferElementData
    {
        public int Value;
    }

The dynamic buffer retains an array of other location indexes.

When I create my entities now in my ECS manager, I loop over the location entities, then create an amount of humans per location. The humans retain the index of that location.

    public struct HumanLocationData : IComponentData
    {
        public int Location; // simple int will correspond to a location
        public float3 OffsetPosition; // offset for position
        public bool changeDestination;
    }

At some point now in my HumanMoveSystem, I pretty much roll a dice and decide whether or not that human needs to move to a new location or stay in the same location and move to a new position:

Updated system:

Entities
    .WithNativeDisableParallelForRestriction(randomArray)
    .WithName("HumanMoveSystem")
    .WithBurst()
    .ForEach((int nativeThreadIndex, ref Translation position, ref HumanMovement movementData, ref HumanLocationData locationData) =>
    {
        if (locationData.changeDestination)
            return;

        float3 travelVector = movementData.EndPos - movementData.StartPos;
        position.Value += math.normalize(travelVector) * deltaTime * movementData.Speed;
        movementData.TravelVector = math.normalize(travelVector) * deltaTime * movementData.Speed;
        float distance = math.distance(position.Value, movementData.EndPos);
        movementData.Distance = distance;
        var random = randomArray[nativeThreadIndex];
        if (distance < 1f)
        {
            movementData.StartPos = movementData.EndPos;
            float distanceCheck;
            float change = random.NextFloat(0, 100);

            if (change > 75.0f)
                locationData.changeDestination = true;
            else
            {
                do
                {
                    float x = random.NextFloat(-1 * locationSize, locationSize);
                    float z = random.NextFloat(-1 * locationSize, locationSize);
                    movementData.EndPos = new float3(x, 0, z) + locationData.OffsetPosition;
                    distanceCheck = math.distance(movementData.StartPos, movementData.EndPos);
                    randomArray[nativeThreadIndex] = random;
                }
                while (distanceCheck < locationSize / 2.0f);
            }
        }

        randomArray[nativeThreadIndex] = random;

    })
    .ScheduleParallel();

This all works well and I get an indication the human wants to move locations.

My plan of attack at this point has been to create 2 new systems:

HumanGetNewLocationSystem

  • Loop over the locations
  • Foreach location, loop over all the humans
  • If the human currently belongs to this location && they want to change location
  • Go through the dynamic buffer of this location, pick a random new location index
  • Write back the human data array

HumanSetNewLocationSystem

  • Update after HumanGetLocationSystem
  • Loop over the humans
  • if the human wants to change
  • loop over the locations
  • when I find the right location, update the humans offset, to the same as the new location.

I have made the first system:

using Unity.Entities;
using Unity.Jobs;
using OutbreakAgent.Component;
using Unity.Collections;

// https://docs.unity3d.com/Packages/com.unity.entities@0.13/manual/chunk_iteration_job.html
namespace OutbreakAgent.Systems
{
    [UpdateAfter(typeof(ShowInfectionMaterialSystem))]
    public class HumanGetNewLocationSystem : SystemBase
    {
        EntityQueryDesc infectedQuery;
        EntityQuery newDestinationGroup;

        protected override void OnCreate()
        {

            infectedQuery = new EntityQueryDesc()
            {
                All = new ComponentType[]
                {
                    ComponentType.ReadOnly<HumanLocationData>()
                }
            };

            newDestinationGroup = GetEntityQuery(infectedQuery);
        }
        
        protected override void OnUpdate()
        {
            var randomArray = World.GetExistingSystem<RandomSystem>().RandomArray;
            
            NativeArray<HumanLocationData> humanLocationData = newDestinationGroup.ToComponentDataArray<HumanLocationData>(Allocator.TempJob);
            
            JobHandle pickDestination = Entities
                .WithName("HumanGetNewLocationSystem")
                .WithBurst()
                .ForEach((int nativeThreadIndex, DynamicBuffer<DestinationBuffer> buffer, ref LocationData locationData) =>
            {
                for (int x = 0; x < humanLocationData.Length; x++)
                {
                    if (locationData.index == humanLocationData[x].Location)
                    {
                        if (humanLocationData[x].changeDestination)
                        {
                            var random = randomArray[nativeThreadIndex];
                            int newLocation = random.NextInt(0, buffer.Length - 1);
                            if (buffer[newLocation].Value > 0)
                            {
                                HumanLocationData hld = humanLocationData[x];
                                hld.Location = buffer[newLocation].Value;
                                //hld.changeDestination = false;  // only set this for now so it returns back to the human move system, otherwise leave on for HumanSetLocationSystem
                                humanLocationData[x] = hld;
                                randomArray[nativeThreadIndex] = random;
                            }
                            else UnityEngine.Debug.Log("BAD BUFFER!");
                        }
                    } 
                }
            }).Schedule(this.Dependency);

            pickDestination.Complete();
            newDestinationGroup.CopyFromComponentDataArray(humanLocationData);
            humanLocationData.Dispose();
        }
    }
}

The issue I have is it just feels wrong in some way.
I am certain I am running the last system currently in a single Job, which is ok because I need to write to the human locations. But I am certain I gradually get a slow down in performance and I cant yet figure out why.

I feel like perhaps to speed this up I could:

  • When rolling the dice to decide to move, attach a new tag component?
  • SetNewLocation would then remove the component, and it could then become part of the move system archetype again.
  • I would lose some speed in the move system initially because no entities would be tagged to change destination.
  • I would need to use EntityManager to add a new component (which forces me back to main thread?)
  • In both the move and Get/Set new location systems, I would gain a little speed from more defined archetypes - groups with or without the tag component.
  • eg:
infectedQuery = new EntityQueryDesc()
            {
                All = new ComponentType[]
                {
                    ComponentType.ReadOnly<Tag_ChangeDestination>()
                }
            };

            newDestinationGroup = GetEntityQuery(infectedQuery);

Anyone got some help for me above? Is my approach good, bad, in between?

Cheers

Probably your biggest issue is the fact that you are looping through all humans for each location in a single-threaded job. You might want to consider building up a NativeHashMap<int, Entity> locationToLocationEntity and then iterating humans instead. That will also get rid of your need to use ToComponentDataArray which might help you avoid calling Complete in the system.

@DreamingImLatios

Thanks for the reply. That’s right about the single thread, I felt I was forced to because of the loop over the humans and the write back their entries which I guessed would cause a race condition otherwise. That was where it felt wrong.

NativeHashMap<int, Entity> locationToLocationEntity

For each human that wants to change location, I then loop over the hash map of locations
If I have the Location entity in the loop and want the dynamic buffer on it, I have to use EntityManager.GetBuffer

https://docs.unity3d.com/Packages/com.unity.entities@0.14/api/Unity.Entities.EntityManager.html#Unity_Entities_EntityManager_GetBuffer__1_Unity_Entities_Entity_

to get the buffer - I can probably assume this wont force main threadness (by using EntityManager) because it’s a read operation and is not altering the structure of the location entity

So if I ForEach over the humans, I can see where I could get the win because I can now run in parrallel this way.


For the HumanSetNewLocationSystem I could do the same thing. I don’t need to write out at all to the location data, I just need to read it, update the HumanLocationData and I am fine.

Cheers and beers

That assumption was correct, because every thread is checking Location which you are also writing to. If you you had a second HumanLocationData array that you wrote to separate from the one you are reading from, then yes this would be thread-safe (barely).

SystemBase.GetBuffer or GetBufferFromEntity are actually what you want here. You also don’t need to iterate over the hashmap. It is a hashmap, so you can do direct lookups.

@DreamingImLatios Thanks for the help, I have managed to get something going and it’s been an excellent few days of ECS lessons. Of course, there is 1 final thing.
I cannot for the life of me seem to set the position of a given entity. I just cant see what I have missed.

    [UpdateAfter(typeof(ShowInfectionMaterialSystem))]
    public class HumanGetNewLocationSystem : SystemBase
    {
        EntityQueryDesc locationQuery;
        EntityQuery locationGroup;
        protected override void OnCreate()
        {

            locationQuery = new EntityQueryDesc()
            {
                All = new ComponentType[]
                {
                    ComponentType.ReadOnly<LocationData>(),
                    ComponentType.ReadOnly<DestinationBuffer>()
                }
            };

            locationGroup = GetEntityQuery(locationQuery);

        }
        
        protected override void OnUpdate()
        {
            var randomArray = World.GetExistingSystem<RandomSystem>().RandomArray;

            NativeArray<Entity> locationEntities = locationGroup.ToEntityArray(Allocator.TempJob);
            NativeArray<LocationData> locationDatas = locationGroup.ToComponentDataArray<LocationData>(Allocator.TempJob);
            NativeHashMap<int, Entity> locationsHashMap = new NativeHashMap<int, Entity>(locationEntities.Length, Allocator.TempJob);

            for (int i = 0; i < locationEntities.Length; i++)
            {
                locationsHashMap.Add(locationDatas[i].index, locationEntities[i]);
            }

            locationEntities.Dispose();
            locationDatas.Dispose();
            BufferFromEntity<DestinationBuffer> bufferLookup = GetBufferFromEntity<DestinationBuffer>();
            ComponentDataFromEntity<LocationData> locationDataLookup = GetComponentDataFromEntity<LocationData>();

            for (int x = 0; x < locationEntities.Length; x++)
            {
                UnityEngine.Debug.Log(locationsHashMap[x]);
            }

            Entities
                .WithName("HumanGetNewLocationSystem")
                .WithNativeDisableParallelForRestriction(randomArray)
                .WithReadOnly(locationsHashMap)
                .WithReadOnly(bufferLookup)
                .WithReadOnly(locationDataLookup)
                .WithDisposeOnCompletion(locationsHashMap)
                .WithBurst()
                .ForEach((int nativeThreadIndex, ref Translation position, ref HumanLocationData humanLocationData, ref HumanMovement humanMovement) =>
            {
                if (humanLocationData.changeDestination)
                {
                    // get a new location index
                    var destinations = bufferLookup[locationsHashMap[humanLocationData.Location]];
                    var random = randomArray[nativeThreadIndex];
                    int index = random.NextInt(0, destinations.Length - 1);
                    int newLocationIndex = destinations[index].Value;

                    //UnityEngine.Debug.Log($"{position.Value} {humanLocationData.Location} {humanMovement.EndPos}");

                    // set it on the human data
                    humanLocationData.Location = newLocationIndex;

                    // remove the previous offset first
                    humanMovement.StartPos -= humanLocationData.OffsetPosition;
                    humanMovement.EndPos -= humanLocationData.OffsetPosition;

                    // get the new locations data
                    var newLocationData = locationDataLookup[locationsHashMap[newLocationIndex]];

                    // reset our offsets back on this human
                    humanLocationData.OffsetPosition = newLocationData.offset;
                    humanMovement.StartPos += newLocationData.offset;
                    humanMovement.EndPos += newLocationData.offset;

                    // set our position 
                    position.Value = humanMovement.StartPos;

                    // finish up so we can move back in the move system;
                    randomArray[nativeThreadIndex] = random;
                    humanLocationData.changeDestination = false;

                    //UnityEngine.Debug.Log($"{position.Value} {humanLocationData.Location} {humanMovement.EndPos}");
                }
            }).ScheduleParallel();
        }

    }

I cannot figure out why position.Value = humanMovement.StartPos; simply does not work. It tells me if I Debug.Log that the position.Value is set to what I expect, but by the time my movement system comes about, it’s as if it was ignored. From everything I am reading this should work? What have I missed?

Cheers

I see nothing wrong right now. This looks like it might be a system execution order issue or a job issue or something. Will need to see more of what is going on.

I have combined the HumanMoveSystem and HumanGetNewLocationSystem and finally got what I expected.
I noticed the archetype I was using in the for each was exactly the same and wondered if that was the reason I had the issue?

With that said, It’s not that bad at all:

[UpdateAfter(typeof(RandomSystem))]
public class HumanMoveSystem : SystemBase
{
    EntityQueryDesc locationQuery;
    EntityQuery locationGroup;
    protected override void OnCreate()
    {

        locationQuery = new EntityQueryDesc()
        {
            All = new ComponentType[]
            {
                ComponentType.ReadOnly<LocationData>(),
                ComponentType.ReadOnly<DestinationBuffer>()
            }
        };

        locationGroup = GetEntityQuery(locationQuery);

    }

    protected override void OnUpdate()
    {
        float deltaTime = Time.DeltaTime;
        var randomArray = World.GetExistingSystem<RandomSystem>().RandomArray;
        float locationSize = 98 / 2.0f;

        NativeArray<Entity> locationEntities = locationGroup.ToEntityArray(Allocator.TempJob);
        NativeArray<LocationData> locationDatas = locationGroup.ToComponentDataArray<LocationData>(Allocator.TempJob);
        NativeHashMap<int, Entity> locationsHashMap = new NativeHashMap<int, Entity>(locationEntities.Length, Allocator.TempJob);

        for (int i = 0; i < locationEntities.Length; i++)
        {
            locationsHashMap.Add(locationDatas[i].index, locationEntities[i]);
        }

        locationEntities.Dispose();
        locationDatas.Dispose();
        BufferFromEntity<DestinationBuffer> bufferLookup = GetBufferFromEntity<DestinationBuffer>();
        ComponentDataFromEntity<LocationData> locationDataLookup = GetComponentDataFromEntity<LocationData>();

        for (int x = 0; x < locationEntities.Length; x++)
        {
            UnityEngine.Debug.Log(locationsHashMap[x]);
        }

        Entities
            .WithName("HumanGetNewLocationSystem")
            .WithNativeDisableParallelForRestriction(randomArray)
            .WithReadOnly(locationsHashMap)
            .WithReadOnly(bufferLookup)
            .WithReadOnly(locationDataLookup)
            .WithDisposeOnCompletion(locationsHashMap)
            .WithBurst()
            .ForEach((int nativeThreadIndex, ref Translation position, ref HumanMovement humanMovementData, ref HumanLocationData humanLocationData) =>
            {
     
                position.Value += humanMovementData.TravelVector * deltaTime;
                float distance = math.distance(position.Value, humanMovementData.EndPos);
                humanMovementData.Distance = distance;
                var random = randomArray[nativeThreadIndex];
                if (distance < 1f)
                {
                    float change = random.NextFloat(0, 100);
                    if (change > 90.0f)
                    {
                           
                        // get a new location index
                        var destinations = bufferLookup[locationsHashMap[humanLocationData.Location]];
                        int index = random.NextInt(0, destinations.Length - 1);
                        int newLocationIndex = destinations[index].Value;

                        // set it on the human data
                        humanLocationData.Location = newLocationIndex;

                        // remove the previous offset first
                        humanMovementData.StartPos -= humanLocationData.OffsetPosition;
                        humanMovementData.EndPos -= humanLocationData.OffsetPosition;

                        // get the new locations data
                        var newLocationData = locationDataLookup[locationsHashMap[newLocationIndex]];

                        // reset our offsets back on this human
                        humanLocationData.OffsetPosition = newLocationData.offset;
                        humanMovementData.StartPos += newLocationData.offset;
                        humanMovementData.EndPos += newLocationData.offset;

                        // set our position 
                        position.Value = humanMovementData.StartPos;
                    }
                    else
                    {
                        humanMovementData.StartPos = humanMovementData.EndPos;
                        float distanceCheck;

                        do
                        {
                            float x = random.NextFloat(-1 * locationSize, locationSize);
                            float z = random.NextFloat(-1 * locationSize, locationSize);
                            humanMovementData.EndPos = new float3(x, 0, z) + humanLocationData.OffsetPosition;
                            distanceCheck = math.distance(humanMovementData.StartPos, humanMovementData.EndPos);
                            float3 travelVector = humanMovementData.EndPos - humanMovementData.StartPos;
                            humanMovementData.TravelVector = math.normalize(travelVector) * humanMovementData.Speed;
                        }
                        while (distanceCheck < locationSize / 2.0f);
                    }

                }

                randomArray[nativeThreadIndex] = random;

            })
            .ScheduleParallel();
    }
}

Thanks for all your help on this. I have learnt a bunch about ECS getting this from what I had to now. There is some improvements to make as I am only getting 90-100 fps with 10k entities? which does not seem like close to what I should be getting.

Thanks again.