From 8bb5e652440f8d59cbf74e7b804520ce8fc7d8a0 Mon Sep 17 00:00:00 2001 From: Unity Technologies <@unity.com> Date: Fri, 5 Apr 2019 13:00:55 +0000 Subject: [PATCH] com.unity.entities@0.0.12-preview.30 --- Documentation~/TableOfContents.md | 46 +- Documentation~/chunk_iteration.md | 8 +- Documentation~/chunk_iteration_job.md | 54 +- Documentation~/component_group.md | 80 +- Documentation~/component_system.md | 2 +- Documentation~/ecs_components.md | 2 +- Documentation~/ecs_in_detail.md | 6 +- Documentation~/ecs_job_overview.md | 2 +- Documentation~/ecs_project_status.md | 2 +- Documentation~/entity_command_buffer.md | 2 +- Documentation~/entity_iteration_job.md | 36 +- Documentation~/entity_manager.md | 2 +- Documentation~/filter.yml | 37 + Documentation~/images/sec1-1.png | Bin 0 -> 10519 bytes Documentation~/images/sec1-2.png | Bin 0 -> 12922 bytes Documentation~/images/sec2-1.png | Bin 0 -> 17431 bytes Documentation~/images/sec2-2.png | Bin 0 -> 24023 bytes Documentation~/images/sec2-3.png | Bin 0 -> 28299 bytes Documentation~/images/sec4-1.png | Bin 0 -> 11551 bytes Documentation~/images/sec4-2.png | Bin 0 -> 11571 bytes Documentation~/images/sec4-3.png | Bin 0 -> 22917 bytes Documentation~/images/sec4-4.png | Bin 0 -> 22115 bytes Documentation~/images/sec4-5.png | Bin 0 -> 34326 bytes Documentation~/images/sec5-1.png | Bin 0 -> 25089 bytes Documentation~/images/sec5-2.png | Bin 0 -> 26003 bytes Documentation~/images/sec5-3.png | Bin 0 -> 25264 bytes Documentation~/images/sec5-4.png | Bin 0 -> 44037 bytes Documentation~/images/sec5-5.png | Bin 0 -> 47287 bytes Documentation~/images/sec5-6.png | Bin 0 -> 11378 bytes Documentation~/images/sec5-7.png | Bin 0 -> 45985 bytes Documentation~/job_component_system.md | 6 +- Documentation~/manual_iteration.md | 10 +- Documentation~/shared_component_data.md | 4 +- Documentation~/system_update_order.md | 2 +- Documentation~/toc.yml | 2 +- Documentation~/transform_system.md | 500 +-- Documentation~/version_numbers.md | 2 +- Unity.Entities.BuildUtils/MonoExtensions.cs | 42 + Unity.Entities.BuildUtils/TypeUtils.cs | 31 +- .../ComponentGroupGUITests.cs | 4 +- .../ComponentTypeFilterUITests.cs | 6 +- .../EntityArrayListAdapterTests.cs | 4 +- .../EntityDebuggerTests.cs | 42 +- Unity.Entities.Editor.Tests/ListViewTests.cs | 57 +- ...ComponentGroupGUI.cs => EntityQueryGUI.cs} | 6 +- ...Control.cs.meta => EntityQueryGUI.cs.meta} | 2 +- ...GUIControl.cs => EntityQueryGUIControl.cs} | 14 +- .../EntityQueryGUIControl.cs.meta} | 2 +- .../CommonGUI/SystemInclusionList.cs | 8 +- .../EntityDebugger/ChunkInfoListView.cs | 4 +- .../EntityDebugger/ComponentTypeFilterUI.cs | 18 +- .../EntityDebugger/ComponentTypeListView.cs | 2 +- .../EntityDebugger/EntityDebugger.cs | 87 +- .../EntityDebugger/EntityListQuery.cs | 10 +- .../EntityDebugger/EntityListView.cs | 36 +- ...roupListView.cs => EntityQueryListView.cs} | 64 +- .../EntityQueryListView.cs.meta} | 2 +- .../EntityDebugger/SystemListView.cs | 12 +- .../EntityInspector/EntitySelectionProxy.cs | 2 +- .../EntitySelectionProxyEditor.cs | 8 +- Unity.Entities.Editor/ExtraTypesProvider.cs | 12 +- .../IJobProcessComponentDataGenerator.cs | 358 -- .../ScriptTemplates.meta | 0 .../ScriptTemplates/AuthoringComponent.txt | 36 + .../AuthoringComponent.txt.meta | 7 + .../ScriptTemplates/RunTimeComponent.txt | 8 +- .../ScriptTemplates/RunTimeComponent.txt.meta | 0 .../ScriptTemplates/ScriptTemplates.cs | 37 + .../ScriptTemplates.cs.meta} | 2 +- .../ScriptTemplates/System.txt | 10 +- .../ScriptTemplates/System.txt.meta | 0 ...ataProxy_EntityManager_IntegrationTests.cs | 2 +- ...ComponentGroupTransformAccessArrayTests.cs | 114 +- ...onentSystem_GameObject_IntegrationTests.cs | 122 - ...System_GameObject_IntegrationTests.cs.meta | 11 - .../ConversionMappingTests.cs | 2 +- .../Runtime/GameObjectEntityTests.cs | 22 - .../InjectComponentGroupTestsHybrid.cs | 58 - .../InjectComponentGroupTestsHybrid.cs.meta | 11 - .../Runtime/SharedComponentSerializeTests.cs | 4 +- .../Components/GameObjectEntity.cs | 2 +- .../GameObjectConversionMappingSystem.cs | 9 +- .../GameObjectConversionSystem.cs | 8 +- .../GameObjectConversionUtility.cs | 26 +- .../Injection/DefaultWorldInitialization.cs | 17 +- .../Iterators/ComponentArray.cs | 134 +- .../Iterators/GameObjectArray.cs | 130 - .../Iterators/GameObjectArray.cs.meta | 11 - .../Iterators/TransformAccessArrayIterator.cs | 52 +- .../EntityCommandBufferPerformanceTests.cs | 2 +- .../EntityManagerPerformanceTests.cs | 6 +- .../EntitySerializationPerformanceTests.cs | 4 +- ...JobForEachPrefilteringPerformanceTests.cs} | 102 +- ...rEachPrefilteringPerformanceTests.cs.meta} | 0 .../SharedComponentPerformanceTests.cs | 2 +- .../EntitySerializationTests.cs | 2 +- .../StaticTypeRegistry.cs | 22 + .../ArchetypeChunkArrayTests.cs | 31 +- Unity.Entities.Tests/BlobificationTests.cs | 7 +- .../BufferElementDataInstantiateTests.cs | 2 - ...rElementDataSystemStateInstantiateTests.cs | 1 - .../BufferElementDataSystemStateTests.cs | 189 +- .../BufferElementDataTests.cs | 192 +- Unity.Entities.Tests/ChangeVersionTests.cs | 150 +- .../ChunkChangeVersionTests.cs | 3 - Unity.Entities.Tests/ChunkComponentTests.cs | 14 +- .../ComponentGroupArrayTests.cs | 130 - .../ComponentGroupArrayTests.cs.meta | 3 - Unity.Entities.Tests/ComponentGroupDelta.cs | 134 +- Unity.Entities.Tests/ComponentGroupTests.cs | 43 +- .../ComponentOrderVersionTests.cs | 21 +- .../ComponentSystemGroupTests.cs | 48 +- .../ComponentSystemInjectionTests.cs | 71 - .../ComponentSystemInjectionTests.cs.meta | 3 - .../ComponentSystemStartStopRunningTests.cs | 14 +- Unity.Entities.Tests/ComponentSystemTests.cs | 174 +- Unity.Entities.Tests/CreateAndDestroyTests.cs | 80 +- Unity.Entities.Tests/DisableComponentTests.cs | 24 +- Unity.Entities.Tests/ECSTestsFixture.cs | 45 +- .../EntityArchetypeQueryTests.cs | 112 + .../EntityArchetypeQueryTests.cs.meta | 11 + .../EntityCommandBufferTests.cs | 88 +- Unity.Entities.Tests/EntityManagerBugTests.cs | 18 +- ...ityManagerComponentGroupOperationsTests.cs | 134 +- .../EntityManagerPrefabTests.cs | 4 +- Unity.Entities.Tests/EntityManagerTests.cs | 53 +- .../EntityQueryBuilderTests.cs | 24 +- Unity.Entities.Tests/EntityQueryCacheTests.cs | 44 +- .../EntityRemapUtilityTests.cs | 2 + .../EntityTransactionTests.cs | 5 +- .../ForEach.meta | 2 +- .../ForEach/ForEachBasicMemoryTests.cs | 41 + .../ForEach/ForEachBasicMemoryTests.cs.meta | 11 + .../ForEach/ForEachDelegateMemoryTests.gen.cs | 729 ++++ .../ForEachDelegateMemoryTests.gen.cs.meta | 11 + .../ForEach/ForEachDelegateMemoryTests.tt | 214 ++ .../ForEachDelegateMemoryTests.tt.meta | 2 +- .../ForEachGeneralTests.cs} | 103 +- .../ForEach/ForEachGeneralTests.cs.meta | 11 + .../ForEach/ForEachMemoryTestFixtureBase.cs | 62 + .../ForEachMemoryTestFixtureBase.cs.meta | 11 + Unity.Entities.Tests/ForEachTests.cs.meta | 11 - Unity.Entities.Tests/IJobChunkTests.cs | 82 +- ...sts.cs => IJobForEachCombinationsTests.cs} | 65 +- ...a => IJobForEachCombinationsTests.cs.meta} | 0 ...ponentDataTests.cs => IJobForEachTests.cs} | 62 +- ...Tests.cs.meta => IJobForEachTests.cs.meta} | 0 .../IJobProcessComponentInjection.cs | 14 +- .../InjectComponentGroupTests.cs | 339 -- Unity.Entities.Tests/IterationTests.cs | 86 +- .../JobComponentSystemDependencyTests.cs | 70 +- Unity.Entities.Tests/JobSafetyTests.cs | 36 +- Unity.Entities.Tests/LockedEntityTests.cs | 18 +- Unity.Entities.Tests/MoveEntitiesFromTests.cs | 75 +- Unity.Entities.Tests/PrefabComponentTests.cs | 14 +- Unity.Entities.Tests/SafetyTests.cs | 82 - Unity.Entities.Tests/SerializeTests.cs | 44 +- .../SharedComponentDataTests.cs | 122 +- Unity.Entities.Tests/SingletonTests.cs | 8 +- Unity.Entities.Tests/SizeTests.cs | 13 +- Unity.Entities.Tests/StandaloneFixme.cs | 2 +- .../SystemStateBufferElementTests.cs | 32 +- .../SystemStateComponentTests.cs | 24 +- Unity.Entities.Tests/TestComponentSystems.cs | 6 +- Unity.Entities.Tests/TypeIndexOrderTests.cs | 1 - .../WorldDebuggingToolsTests.cs | 48 +- Unity.Entities.Tests/WorldDiffTests.cs | 10 +- Unity.Entities.Tests/WorldTests.cs | 74 +- Unity.Entities.Tests/WriteGroupTests.cs | 40 +- Unity.Entities.Tests/ZeroPlayerTests.cs | 56 + Unity.Entities.Tests/ZeroPlayerTests.cs.meta | 11 + Unity.Entities/ArchetypeManager.cs | 103 +- Unity.Entities/AssemblyInfo.cs | 1 + Unity.Entities/BinarySerialization.cs | 2 +- Unity.Entities/ComponentJobManager.cs | 12 +- Unity.Entities/ComponentSystem.cs | 495 +-- Unity.Entities/ComponentSystemGroup.cs | 16 +- Unity.Entities/CopyUtility.cs | 34 - Unity.Entities/DebugView.cs | 6 +- .../DefaultTinyWorldInitialization.cs | 20 +- .../DefaultTinyWorldInitialization.cs.meta | 0 Unity.Entities/DefaultWorld.cs | 26 +- Unity.Entities/DeprecatedAPIStubs.cs | 54 + .../DeprecatedAPIStubs.cs.meta | 4 +- Unity.Entities/EntityCommandBuffer.cs | 90 +- Unity.Entities/EntityDataManager.cs | 28 +- Unity.Entities/EntityManager.cs | 300 +- Unity.Entities/EntityQueryBuilder.cs | 46 +- Unity.Entities/EntityQueryBuilder.gen.cs.meta | 11 - ...n.cs => EntityQueryBuilder_ForEach.gen.cs} | 3261 ++++++----------- .../EntityQueryBuilder_ForEach.gen.cs.meta | 11 + ...ilder.tt => EntityQueryBuilder_ForEach.tt} | 134 +- .../EntityQueryBuilder_ForEach.tt.meta | 7 + Unity.Entities/EntityQueryBuilder_With.gen.cs | 156 + .../EntityQueryBuilder_With.gen.cs.meta | 11 + Unity.Entities/EntityQueryBuilder_With.tt | 49 + .../EntityQueryBuilder_With.tt.meta | 7 + Unity.Entities/EntityQueryCache.cs | 37 +- Unity.Entities/EntityRemapUtility.cs | 2 +- Unity.Entities/ExclusiveEntityTransaction.cs | 2 +- Unity.Entities/IJobChunk.cs | 32 +- ...ProcessComponentData.cs => IJobForEach.cs} | 128 +- ...ponentData.cs.meta => IJobForEach.cs.meta} | 0 Unity.Entities/IJobForEach.gen.cs | 1649 +++++++++ Unity.Entities/IJobForEach.gen.cs.meta | 11 + Unity.Entities/IJobForEach.tt | 337 ++ Unity.Entities/IJobForEach.tt.meta | 7 + .../IJobProcessComponentData.generated.cs | 1598 -------- ...IJobProcessComponentData.generated.cs.meta | 11 - .../Injection/ComponentSystemInjection.cs | 115 - .../ComponentSystemInjection.cs.meta | 3 - Unity.Entities/Injection/InjectAttribute.cs | 10 - .../Injection/InjectAttribute.cs.meta | 3 - .../Injection/InjectComponentFromEntity.cs | 87 - .../InjectComponentFromEntity.cs.meta | 3 - .../Injection/InjectComponentGroup.cs | 264 -- .../Injection/InjectComponentGroup.cs.meta | 12 - Unity.Entities/Injection/InjectionData.cs | 26 - .../Injection/InjectionData.cs.meta | 3 - Unity.Entities/Injection/InjectionHook.cs | 134 - .../Injection/InjectionHook.cs.meta | 11 - Unity.Entities/Injection/World.cs | 543 --- .../Injection/WorldDebuggingTools.cs | 54 - Unity.Entities/Iterators/BufferArray.cs | 92 - Unity.Entities/Iterators/BufferArray.cs.meta | 13 - .../Iterators/ChunkDataGatherJobs.cs | 36 +- .../Iterators/ComponentChunkIterator.cs | 94 +- .../Iterators/ComponentDataArray.cs | 163 - .../Iterators/ComponentDataArray.cs.meta | 13 - .../Iterators/ComponentGroup.cs.meta | 13 - .../Iterators/ComponentGroup.obsolete.cs | 275 ++ .../Iterators/ComponentGroup.obsolete.cs.meta | 11 + .../Iterators/ComponentGroupArray.cs | 417 --- .../Iterators/ComponentGroupArray.cs.meta | 11 - Unity.Entities/Iterators/DynamicBuffer.cs | 39 +- Unity.Entities/Iterators/EntityArray.cs | 109 - Unity.Entities/Iterators/EntityArray.cs.meta | 13 - .../Iterators/EntityGroupManager.cs | 160 +- .../{ComponentGroup.cs => EntityQuery.cs} | 492 ++- Unity.Entities/Iterators/EntityQuery.cs.meta | 11 + .../Iterators/SharedComponentDataArray.cs | 63 - .../SharedComponentDataArray.cs.meta | 11 - Unity.Entities/ScriptBehaviourManager.cs | 72 - Unity.Entities/ScriptBehaviourUpdateOrder.cs | 18 +- Unity.Entities/SerializeUtility.cs | 2 +- Unity.Entities/SharedComponentManager.cs | 83 +- Unity.Entities/Types/BufferHeader.cs | 12 +- Unity.Entities/Types/ComponentType.cs | 4 +- Unity.Entities/Types/FastEquality.cs | 4 +- Unity.Entities/Types/Hash128.cs | 2 +- Unity.Entities/Types/TypeManager.cs | 603 ++- Unity.Entities/World.cs | 562 +++ Unity.Entities/{Injection => }/World.cs.meta | 0 Unity.Entities/WorldDebuggingTools.cs | 53 + .../WorldDebuggingTools.cs.meta | 0 Unity.Entities/WorldDiff.cs | 90 +- .../ConversionWarningsEditor.cs | 2 +- Unity.Scenes.Editor/EditorEntityScenes.cs | 78 +- Unity.Scenes.Editor/SubSceneInspector.cs | 2 +- Unity.Scenes.Editor/SubSceneLiveLinkSystem.cs | 90 +- .../SaveAndLoadEndToEnd.cs | 10 +- .../EntitySceneOptimization.cs | 12 +- Unity.Scenes.Hybrid/SubScene.cs | 2 +- .../SubSceneStreamingSystem.cs | 61 +- ...opyInitialTransformFromGameObjectSystem.cs | 12 +- .../CopyTransformFromGameObjectSystem.cs | 10 +- .../CopyTransformToGameObjectSystem.cs | 24 +- Unity.Transforms.Tests/TransformTests.cs | 28 +- Unity.Transforms/CompositeRotation.cs | 8 +- Unity.Transforms/CompositeScale.cs | 8 +- Unity.Transforms/LocalToParentSystem.cs | 8 +- Unity.Transforms/ParentScaleInverse.cs | 8 +- Unity.Transforms/ParentSystem.cs | 26 +- Unity.Transforms/PostRotationEuler.cs | 8 +- Unity.Transforms/RotationEuler.cs | 8 +- Unity.Transforms/TRSToLocalToParentSystem.cs | 8 +- Unity.Transforms/TRSToLocalToWorldSystem.cs | 8 +- Unity.Transforms/WorldToLocal.cs | 8 +- package.json | 9 +- 279 files changed, 9772 insertions(+), 11302 deletions(-) create mode 100644 Documentation~/filter.yml create mode 100644 Documentation~/images/sec1-1.png create mode 100644 Documentation~/images/sec1-2.png create mode 100644 Documentation~/images/sec2-1.png create mode 100644 Documentation~/images/sec2-2.png create mode 100644 Documentation~/images/sec2-3.png create mode 100644 Documentation~/images/sec4-1.png create mode 100644 Documentation~/images/sec4-2.png create mode 100644 Documentation~/images/sec4-3.png create mode 100644 Documentation~/images/sec4-4.png create mode 100644 Documentation~/images/sec4-5.png create mode 100644 Documentation~/images/sec5-1.png create mode 100644 Documentation~/images/sec5-2.png create mode 100644 Documentation~/images/sec5-3.png create mode 100644 Documentation~/images/sec5-4.png create mode 100644 Documentation~/images/sec5-5.png create mode 100644 Documentation~/images/sec5-6.png create mode 100644 Documentation~/images/sec5-7.png rename Unity.Entities.Editor/CommonGUI/{ComponentGroupGUI.cs => EntityQueryGUI.cs} (96%) rename Unity.Entities.Editor/CommonGUI/{ComponentGroupGUIControl.cs.meta => EntityQueryGUI.cs.meta} (83%) rename Unity.Entities.Editor/CommonGUI/{ComponentGroupGUIControl.cs => EntityQueryGUIControl.cs} (82%) rename Unity.Entities.Editor/{EntityDebugger/ComponentGroupListView.cs.meta => CommonGUI/EntityQueryGUIControl.cs.meta} (83%) rename Unity.Entities.Editor/EntityDebugger/{ComponentGroupListView.cs => EntityQueryListView.cs} (76%) rename Unity.Entities.Editor/{IJobProcessComponentDataGenerator.cs.meta => EntityDebugger/EntityQueryListView.cs.meta} (83%) delete mode 100644 Unity.Entities.Editor/IJobProcessComponentDataGenerator.cs rename ScriptTemplates.meta => Unity.Entities.Editor/ScriptTemplates.meta (100%) create mode 100644 Unity.Entities.Editor/ScriptTemplates/AuthoringComponent.txt create mode 100644 Unity.Entities.Editor/ScriptTemplates/AuthoringComponent.txt.meta rename ScriptTemplates/ECS__Component Type-NewComponentType.cs.txt => Unity.Entities.Editor/ScriptTemplates/RunTimeComponent.txt (67%) rename ScriptTemplates/ECS__Component Type-NewComponentType.cs.txt.meta => Unity.Entities.Editor/ScriptTemplates/RunTimeComponent.txt.meta (100%) create mode 100644 Unity.Entities.Editor/ScriptTemplates/ScriptTemplates.cs rename Unity.Entities.Editor/{CommonGUI/ComponentGroupGUI.cs.meta => ScriptTemplates/ScriptTemplates.cs.meta} (83%) rename ScriptTemplates/ECS__System-NewSystem.cs.txt => Unity.Entities.Editor/ScriptTemplates/System.txt (81%) rename ScriptTemplates/ECS__System-NewSystem.cs.txt.meta => Unity.Entities.Editor/ScriptTemplates/System.txt.meta (100%) delete mode 100644 Unity.Entities.Hybrid.Tests/ComponentSystem_GameObject_IntegrationTests.cs delete mode 100644 Unity.Entities.Hybrid.Tests/ComponentSystem_GameObject_IntegrationTests.cs.meta delete mode 100644 Unity.Entities.Hybrid.Tests/Runtime/InjectComponentGroupTestsHybrid.cs delete mode 100644 Unity.Entities.Hybrid.Tests/Runtime/InjectComponentGroupTestsHybrid.cs.meta delete mode 100644 Unity.Entities.Hybrid/Iterators/GameObjectArray.cs delete mode 100644 Unity.Entities.Hybrid/Iterators/GameObjectArray.cs.meta rename Unity.Entities.PerformanceTests/{JobProcessComponentDataPrefilteringPerformanceTests.cs => JobForEachPrefilteringPerformanceTests.cs} (78%) rename Unity.Entities.PerformanceTests/{JobProcessComponentDataPrefilteringPerformanceTests.cs.meta => JobForEachPrefilteringPerformanceTests.cs.meta} (100%) delete mode 100644 Unity.Entities.Tests/ComponentGroupArrayTests.cs delete mode 100644 Unity.Entities.Tests/ComponentGroupArrayTests.cs.meta delete mode 100644 Unity.Entities.Tests/ComponentSystemInjectionTests.cs delete mode 100644 Unity.Entities.Tests/ComponentSystemInjectionTests.cs.meta create mode 100644 Unity.Entities.Tests/EntityArchetypeQueryTests.cs create mode 100644 Unity.Entities.Tests/EntityArchetypeQueryTests.cs.meta rename Unity.Entities/Injection.meta => Unity.Entities.Tests/ForEach.meta (77%) create mode 100644 Unity.Entities.Tests/ForEach/ForEachBasicMemoryTests.cs create mode 100644 Unity.Entities.Tests/ForEach/ForEachBasicMemoryTests.cs.meta create mode 100644 Unity.Entities.Tests/ForEach/ForEachDelegateMemoryTests.gen.cs create mode 100644 Unity.Entities.Tests/ForEach/ForEachDelegateMemoryTests.gen.cs.meta create mode 100644 Unity.Entities.Tests/ForEach/ForEachDelegateMemoryTests.tt rename Unity.Entities/EntityQueryBuilder.tt.meta => Unity.Entities.Tests/ForEach/ForEachDelegateMemoryTests.tt.meta (74%) rename Unity.Entities.Tests/{ForEachTests.cs => ForEach/ForEachGeneralTests.cs} (55%) create mode 100644 Unity.Entities.Tests/ForEach/ForEachGeneralTests.cs.meta create mode 100644 Unity.Entities.Tests/ForEach/ForEachMemoryTestFixtureBase.cs create mode 100644 Unity.Entities.Tests/ForEach/ForEachMemoryTestFixtureBase.cs.meta delete mode 100644 Unity.Entities.Tests/ForEachTests.cs.meta rename Unity.Entities.Tests/{IJobProcessComponentDataCombinationsTests.cs => IJobForEachCombinationsTests.cs} (83%) rename Unity.Entities.Tests/{IJobProcessComponentDataCombinationsTests.cs.meta => IJobForEachCombinationsTests.cs.meta} (100%) rename Unity.Entities.Tests/{IJobProcessComponentDataTests.cs => IJobForEachTests.cs} (78%) rename Unity.Entities.Tests/{IJobProcessComponentDataTests.cs.meta => IJobForEachTests.cs.meta} (100%) delete mode 100644 Unity.Entities.Tests/InjectComponentGroupTests.cs create mode 100644 Unity.Entities.Tests/ZeroPlayerTests.cs create mode 100644 Unity.Entities.Tests/ZeroPlayerTests.cs.meta rename Unity.Entities/{Injection => }/DefaultTinyWorldInitialization.cs (84%) rename Unity.Entities/{Injection => }/DefaultTinyWorldInitialization.cs.meta (100%) create mode 100644 Unity.Entities/DeprecatedAPIStubs.cs rename Unity.Entities.Tests/InjectComponentGroupTests.cs.meta => Unity.Entities/DeprecatedAPIStubs.cs.meta (77%) delete mode 100644 Unity.Entities/EntityQueryBuilder.gen.cs.meta rename Unity.Entities/{EntityQueryBuilder.gen.cs => EntityQueryBuilder_ForEach.gen.cs} (73%) create mode 100644 Unity.Entities/EntityQueryBuilder_ForEach.gen.cs.meta rename Unity.Entities/{EntityQueryBuilder.tt => EntityQueryBuilder_ForEach.tt} (69%) create mode 100644 Unity.Entities/EntityQueryBuilder_ForEach.tt.meta create mode 100644 Unity.Entities/EntityQueryBuilder_With.gen.cs create mode 100644 Unity.Entities/EntityQueryBuilder_With.gen.cs.meta create mode 100644 Unity.Entities/EntityQueryBuilder_With.tt create mode 100644 Unity.Entities/EntityQueryBuilder_With.tt.meta rename Unity.Entities/{IJobProcessComponentData.cs => IJobForEach.cs} (79%) rename Unity.Entities/{IJobProcessComponentData.cs.meta => IJobForEach.cs.meta} (100%) create mode 100644 Unity.Entities/IJobForEach.gen.cs create mode 100644 Unity.Entities/IJobForEach.gen.cs.meta create mode 100644 Unity.Entities/IJobForEach.tt create mode 100644 Unity.Entities/IJobForEach.tt.meta delete mode 100644 Unity.Entities/IJobProcessComponentData.generated.cs delete mode 100644 Unity.Entities/IJobProcessComponentData.generated.cs.meta delete mode 100644 Unity.Entities/Injection/ComponentSystemInjection.cs delete mode 100644 Unity.Entities/Injection/ComponentSystemInjection.cs.meta delete mode 100644 Unity.Entities/Injection/InjectAttribute.cs delete mode 100644 Unity.Entities/Injection/InjectAttribute.cs.meta delete mode 100644 Unity.Entities/Injection/InjectComponentFromEntity.cs delete mode 100644 Unity.Entities/Injection/InjectComponentFromEntity.cs.meta delete mode 100644 Unity.Entities/Injection/InjectComponentGroup.cs delete mode 100644 Unity.Entities/Injection/InjectComponentGroup.cs.meta delete mode 100644 Unity.Entities/Injection/InjectionData.cs delete mode 100644 Unity.Entities/Injection/InjectionData.cs.meta delete mode 100644 Unity.Entities/Injection/InjectionHook.cs delete mode 100644 Unity.Entities/Injection/InjectionHook.cs.meta delete mode 100644 Unity.Entities/Injection/World.cs delete mode 100644 Unity.Entities/Injection/WorldDebuggingTools.cs delete mode 100644 Unity.Entities/Iterators/BufferArray.cs delete mode 100644 Unity.Entities/Iterators/BufferArray.cs.meta delete mode 100644 Unity.Entities/Iterators/ComponentDataArray.cs delete mode 100644 Unity.Entities/Iterators/ComponentDataArray.cs.meta delete mode 100644 Unity.Entities/Iterators/ComponentGroup.cs.meta create mode 100644 Unity.Entities/Iterators/ComponentGroup.obsolete.cs create mode 100644 Unity.Entities/Iterators/ComponentGroup.obsolete.cs.meta delete mode 100644 Unity.Entities/Iterators/ComponentGroupArray.cs delete mode 100644 Unity.Entities/Iterators/ComponentGroupArray.cs.meta delete mode 100644 Unity.Entities/Iterators/EntityArray.cs delete mode 100644 Unity.Entities/Iterators/EntityArray.cs.meta rename Unity.Entities/Iterators/{ComponentGroup.cs => EntityQuery.cs} (65%) create mode 100644 Unity.Entities/Iterators/EntityQuery.cs.meta delete mode 100644 Unity.Entities/Iterators/SharedComponentDataArray.cs delete mode 100644 Unity.Entities/Iterators/SharedComponentDataArray.cs.meta create mode 100644 Unity.Entities/World.cs rename Unity.Entities/{Injection => }/World.cs.meta (100%) create mode 100644 Unity.Entities/WorldDebuggingTools.cs rename Unity.Entities/{Injection => }/WorldDebuggingTools.cs.meta (100%) diff --git a/Documentation~/TableOfContents.md b/Documentation~/TableOfContents.md index 6fae25dd..ca3e83e2 100644 --- a/Documentation~/TableOfContents.md +++ b/Documentation~/TableOfContents.md @@ -1,27 +1,27 @@ * [Overview](index.md) * Core ECS -** [Entities](ecs_entities.md) -*** [Worlds](world.md) -** [Components](ecs_components.md) -*** [General Purpose Components](component_data.md) -*** [Shared Components](shared_component_data.md) -*** [System State Components](system_state_components.md) -*** [Dynamic Buffer Components](dynamic_buffers.md) -** [System](ecs_systems.md) -*** [Component Systems](component_system.md) -*** [Job Component Systems](job_component_system.md) -*** [Entity Command Buffers](entity_command_buffer.md) -*** [System Update Order](system_update_order.md) -** [Accessing Entity Data](chunk_iteration.md) -*** [Using IJobProcessComponentData](entity_iteration_job.md) -*** [Using IJobChunk](chunk_iteration_job.md) -*** [Using ComponentSystem and ForEach](entity_iteration_foreach.md) -*** [Manual iteration](manual_iteration.md) -*** [Component Groups](component_group.md) -** [Versions and Generations](version_numbers.md) -** [Jobs in ECS](ecs_job_overview.md) -*** [ECS Job System extensions](ecs_job_extensions.md) + * [Entities](ecs_entities.md) + * [Worlds](world.md) + * [Components](ecs_components.md) + * [General Purpose Components](component_data.md) + * [Shared Components](shared_component_data.md) + * [System State Components](system_state_components.md) + * [Dynamic Buffer Components](dynamic_buffers.md) + * [System](ecs_systems.md) + * [Component Systems](component_system.md) + * [Job Component Systems](job_component_system.md) + * [Entity Command Buffers](entity_command_buffer.md) + * [System Update Order](system_update_order.md) + * [Accessing Entity Data](chunk_iteration.md) + * [Using IJobForEach](entity_iteration_job.md) + * [Using IJobChunk](chunk_iteration_job.md) + * [Using ComponentSystem and ForEach](entity_iteration_foreach.md) + * [Manual iteration](manual_iteration.md) + * [Component Groups](component_group.md) + * [Versions and Generations](version_numbers.md) + * [Jobs in ECS](ecs_job_overview.md) + * [ECS Job System extensions](ecs_job_extensions.md) * Creating Gameplay -** [Transforms](transform_system.md) -** [Rendering](gp_rendering.md) + * [Transforms](transform_system.md) + * [Rendering](gp_rendering.md) diff --git a/Documentation~/chunk_iteration.md b/Documentation~/chunk_iteration.md index b43e186e..f9ed9db5 100644 --- a/Documentation~/chunk_iteration.md +++ b/Documentation~/chunk_iteration.md @@ -9,14 +9,14 @@ In general, the most efficient way to iterate over your entities and components The ECS API provides a number of ways to accomplish iteration, each with its own performance implications and restrictions. You can iterate over ECS data in the following ways: -* [IJobProcessComponentData](entity_iteration_job.md) — the simplest efficient way to process component data entity by entity. +* [IJobForEach](entity_iteration_job.md) — the simplest efficient way to process component data entity by entity. -* [IJobProcessComponentDataWithEntity](entity_iteration_job.md#with-entity) — slightly more complex than IJobProcessComponentData, giving you access to the entity handle and array index of the entity you are processing. +* [IJobForEachWithEntity](entity_iteration_job.md#with-entity) — slightly more complex than IJobForEach, giving you access to the entity handle and array index of the entity you are processing. -* [IJobChunk](chunk_iteration_job.md) — iterates over the eligible blocks of memory (called a *Chunk*) containing matching entities. Your Job Execute() function can iterate over the Elements inside each chunk using a for loop. You can use IJobChunk for more complex situations than supported by IJobProcessComponentData, while maintaining maximum efficiency. +* [IJobChunk](chunk_iteration_job.md) — iterates over the eligible blocks of memory (called a *Chunk*) containing matching entities. Your Job Execute() function can iterate over the Elements inside each chunk using a for loop. You can use IJobChunk for more complex situations than supported by IJobForEach, while maintaining maximum efficiency. * [ComponentSystem](entity_iteration_foreach.md) — the ComponentSystem offers the Entities.ForEach delegate functions to help iterate over your entities. However, ForEach runs on the main thread, so typically, you should only use ComponentSystem implementations for tasks that must be carried out on the main thread anyway. * [Manual iteration](manual_iteration.md) — if the previous methods are insufficient, you can manually iterate over entities or chunks. For example, you can get a NativeArray containing entities or the chunks of the entities that you want to process and iterate over them using a Job, such as IJobParallelFor. -The [ComponentGroup](component_group.md) class provides a way to construct a view of your data that contains only the specific data you need for a given algorithm or process. Many of the iteration methods in the list above use a ComponentGroup, either explicitly or internally. \ No newline at end of file +The [EntityQuery](component_group.md) class provides a way to construct a view of your data that contains only the specific data you need for a given algorithm or process. Many of the iteration methods in the list above use a EntityQuery, either explicitly or internally. \ No newline at end of file diff --git a/Documentation~/chunk_iteration_job.md b/Documentation~/chunk_iteration_job.md index 08d7e417..5622fe75 100644 --- a/Documentation~/chunk_iteration_job.md +++ b/Documentation~/chunk_iteration_job.md @@ -2,38 +2,38 @@ You can implement IJobChunk inside a JobComponentSystem to iterate through your data by chunk. The JobComponentSystem calls your Execute() function once for each chunk that contains the entities that you want the system to process. You can then process the data inside each chunk, entity by entity. -Iterating with IJobChunk requires more code setup than does IJobProcessComponentData, but is also more explicit and represents the most direct access to the data, as it is actually stored. +Iterating with IJobChunk requires more code setup than does IJobForEach, but is also more explicit and represents the most direct access to the data, as it is actually stored. Another benefit of using iterating by chunks is that you can check whether an optional component is present in each chunk (with Archetype.Has) and process all the entities in the chunk accordingly. The steps involved in implementing an IJobChunk Job include: -1. Identify the entities that you want to process by creating a ComponentGroup. +1. Identify the entities that you want to process by creating a EntityQuery. 2. Defining the Job struct, including fields for ArchetypeChunkComponentType objects to identifying the types of components the Job directly accesses, specifying whether the Job reads or writes to those components. 3. Instantiating the Job struct and scheduling the Job in the system OnUpdate() function. 4. In the Execute() function, getting the NativeArray instances for the components the Job reads or writes and, finally, iterating over the current chunk to perform the desired work. The [ECS samples repository](https://github.com/Unity-Technologies/EntityComponentSystemSamples) contains a simple example, HelloCube_03_IJobChunk, that uses IJobChunk. -## Query for data with a ComponentGroup +## Query for data with a EntityQuery -A ComponentGroup defines the set of component types that an archetype must contain for the system to process its associated chunks and entities. An archetype can have additional components as well, but it must have at least those defined by the ComponentGroup. You can also exclude archetypes that contain specific types of components. +A EntityQuery defines the set of component types that an archetype must contain for the system to process its associated chunks and entities. An archetype can have additional components as well, but it must have at least those defined by the EntityQuery. You can also exclude archetypes that contain specific types of components. -For simple queries, you can use the JobComponentSystem.GetComponentGroup() function, passing in the component types: +For simple queries, you can use the JobComponentSystem.GetEntityQuery() function, passing in the component types: ``` c# public class RotationSpeedSystem : JobComponentSystem { - private ComponentGroup m_Group; - protected override void OnCreateManager() + private EntityQuery m_Group; + protected override void OnCreate() { - m_Group = GetComponentGroup(typeof(RotationQuaternion), ComponentType.ReadOnly()); + m_Group = GetEntityQuery(typeof(RotationQuaternion), ComponentType.ReadOnly()); } //… } ```` -For more complex situations, you can use an EntityArchetypeQuery. An EntityArchetypeQuery provides a flexible query mechanism to specify the component types: +For more complex situations, you can use an EntityQueryDesc. An EntityQueryDesc provides a flexible query mechanism to specify the component types: * `All` = All component types in this array must exist in the archetype * `Any` = At least one of the component types in this array must exist in the archetype @@ -42,48 +42,48 @@ For more complex situations, you can use an EntityArchetypeQuery. An EntityArche For example, the following query includes archetypes containing the RotationQuaternion and RotationSpeed components, but excludes any archetypes containing the Frozen component: ``` c# -protected override void OnCreateManager() +protected override void OnCreate() { - var query = new EntityArchetypeQuery + var query = new EntityQueryDesc { None = new ComponentType[]{ typeof(Frozen) }, All = new ComponentType[]{ typeof(RotationQuaternion), ComponentType.ReadOnly() } } }; - m_Group = GetComponentGroup(query); + m_Group = GetEntityQuery(query); } ``` The query uses `ComponentType.ReadOnly` instead of the simpler `typeof` expression to designate that the system does not write to RotationSpeed. -You can also combine multiple queries by passing an array of EntityArchetypeQuery objects rather than a single instance. Each query is combined using a logical OR operation. The following example selects an archetypes that contain a RotationQuaternion component or a RotationSpeed component (or both): +You can also combine multiple queries by passing an array of EntityQueryDesc objects rather than a single instance. Each query is combined using a logical OR operation. The following example selects an archetypes that contain a RotationQuaternion component or a RotationSpeed component (or both): ``` c# -protected override void OnCreateManager() +protected override void OnCreate() { - var query0 = new EntityArchetypeQuery + var query0 = new EntityQueryDesc { All = new ComponentType[] {typeof(RotationQuaternion)} }; - var query1 = new EntityArchetypeQuery + var query1 = new EntityQueryDesc { All = new ComponentType[] {typeof(RotationSpeed)} }; - m_Group = GetComponentGroup(new EntityArchetypeQuery[] {query0, query1}); + m_Group = GetEntityQuery(new EntityQueryDesc[] {query0, query1}); } ``` -**Note:** Do not include completely optional components in the EntityArchetypeQuery. To handle optional components, use the `chunk.Has()` method inside `IJobChunk.Execute()` to determine whether the current ArchetypeChunk has the optional component or not. Since all entities within the same chunk have the same components, you only need to check whether an optional component exists once per chunk -- not once per entity. +**Note:** Do not include completely optional components in the EntityQueryDesc. To handle optional components, use the `chunk.Has()` method inside `IJobChunk.Execute()` to determine whether the current ArchetypeChunk has the optional component or not. Since all entities within the same chunk have the same components, you only need to check whether an optional component exists once per chunk -- not once per entity. -For efficiency and to avoid needless creation of garbage-collected reference types, you should create the ComponentGroups for a system in the system’s OnCreateManager() function and store the result in an instance variable. (In the above examples, the `m_Group` variable is used for this purpose.) +For efficiency and to avoid needless creation of garbage-collected reference types, you should create the EntityQueries for a system in the system’s OnCreate() function and store the result in an instance variable. (In the above examples, the `m_Group` variable is used for this purpose.) ### ## Define the IJobChunk struct The IJobChunk struct defines fields for the data the Job needs when it runs, as well as the Job’s Execute() method. -In order to access the component arrays inside the chunks that the system passes to your Execute() method, you must create an ArchetypeChunkComponentType object for each type of component that the Job reads or writes. These objects allow you to get instances of the NativeArrays providing access to the components of an entity. Include all the components referenced in the Job’s ComponentGroup that the Execute method reads or writes. You can also provide ArchetypeChunkComponentType variables for optional component types that you do not include in the ComponentGroup. (You must check to make sure that the current chunk has an optional component before trying to access it.) +In order to access the component arrays inside the chunks that the system passes to your Execute() method, you must create an ArchetypeChunkComponentType object for each type of component that the Job reads or writes. These objects allow you to get instances of the NativeArrays providing access to the components of an entity. Include all the components referenced in the Job’s EntityQuery that the Execute method reads or writes. You can also provide ArchetypeChunkComponentType variables for optional component types that you do not include in the EntityQuery. (You must check to make sure that the current chunk has an optional component before trying to access it.) For example, the HelloCube_03_IJobChunk example declares a Job struct that defines ArchetypeChunkComponentType variables for two components, RotationQuaternion and RotationSpeed: @@ -140,7 +140,7 @@ for (var i = 0; i < chunk.Count; i++) } ``` -If you the `Any` filter in your EntityArchetypeQuery or have completely optional components that don’t appear in the query at all, you can use the `ArchetypeChunk.Has` function to test whether the current chunk contains the one of those components before using it: +If you the `Any` filter in your EntityQueryDesc or have completely optional components that don’t appear in the query at all, you can use the `ArchetypeChunk.Has` function to test whether the current chunk contains the one of those components before using it: if (chunk.Has(OptionalCompType)) {//...} @@ -149,20 +149,20 @@ __Note:__ If you use a concurrent entity command buffer, pass the chunkIndex arg ## Skipping chunks with unchanged entities -If you only need to update entities when a component value has changed, you can add that component type to the change filter of the ComponentGroup used to select the entities and chunks for the job. For example, if you have a system that reads two components and only needs to update a third when one of the first two has changed, you could use a ComponentGroup as follows: +If you only need to update entities when a component value has changed, you can add that component type to the change filter of the EntityQuery used to select the entities and chunks for the job. For example, if you have a system that reads two components and only needs to update a third when one of the first two has changed, you could use a EntityQuery as follows: ``` c# -ComponentGroup m_Group; -protected override void OnCreateManager() +EntityQuery m_Group; +protected override void OnCreate() { - m_Group = GetComponentGroup(typeof(Output), + m_Group = GetEntityQuery(typeof(Output), ComponentType.ReadOnly(), ComponentType.ReadOnly()); m_Group.SetFilterChanged(new ComponentType{ typeof(InputA), typeof(InputB)}); } ``` -The ComponentGroup change filter supports up to two components. If you want to check more or aren't using a ComponentGroup, you can make the check manually. To make this check, compare the chunk’s change version for the component to the system’s LastSystemVersion using the `ArchetypeChunk.DidChange()` function. If this function returns false, you can skip the current chunk altogether since none of the components of that type have changed since the last time the system ran. +The EntityQuery change filter supports up to two components. If you want to check more or aren't using a EntityQuery, you can make the check manually. To make this check, compare the chunk’s change version for the component to the system’s LastSystemVersion using the `ArchetypeChunk.DidChange()` function. If this function returns false, you can skip the current chunk altogether since none of the components of that type have changed since the last time the system ran. The LastSystemVersion from the system must be passed into the Job using a struct field: @@ -214,6 +214,6 @@ protected override JobHandle OnUpdate(JobHandle inputDependencies) } ``` -When you call the GetArchetypeChunkComponentType function to set your component type variables, make sure that you set the isReadOnly to true for components that the Job reads, but doesn’t write. Setting these parameters correctly can have a significant impact on how efficiently the ECS framework can schedule your Jobs. These access mode settings must match their equivalents in both the struct definition, and the ComponentGroup. +When you call the GetArchetypeChunkComponentType function to set your component type variables, make sure that you set the isReadOnly to true for components that the Job reads, but doesn’t write. Setting these parameters correctly can have a significant impact on how efficiently the ECS framework can schedule your Jobs. These access mode settings must match their equivalents in both the struct definition, and the EntityQuery. Do not cache the return value of GetArchetypeChunkComponentType in a system class variable. The function must be called every time the system runs and the updated value passed to the Job. diff --git a/Documentation~/component_group.md b/Documentation~/component_group.md index 53bc3088..24649ad3 100644 --- a/Documentation~/component_group.md +++ b/Documentation~/component_group.md @@ -1,37 +1,37 @@ --- uid: ecs-component-group --- -# Querying for data using a ComponentGroup +# Querying for data using a EntityQuery -The first step to reading or writing data is finding that data. Data in the ECS framework is stored in components, which are grouped together in memory according to the archetype of the entity to which they belong. To define a view into your ECS data that contains only the specific data you need for a given algorithm or process, you can construct a ComponentGroup. +The first step to reading or writing data is finding that data. Data in the ECS framework is stored in components, which are grouped together in memory according to the archetype of the entity to which they belong. To define a view into your ECS data that contains only the specific data you need for a given algorithm or process, you can construct a EntityQuery. -After creating a ComponentGroup, you can +After creating a EntityQuery, you can * Run a Job to process the entities and components selected for the view * Get a NativeArray containing all the selected entities * Get NativeArrays of the selected components (by component type) -The entity and component arrays returned by a ComponentGroup are guaranteed to be "parallel", that is, the same index value always applies to the same entity in any array. +The entity and component arrays returned by a EntityQuery are guaranteed to be "parallel", that is, the same index value always applies to the same entity in any array. -**Note:** that the `ComponentSystem.Entites.ForEach` delegates and IJobProcessComponentData create internal ComponentGroups based on the component types and attributes you specify for these APIs. +**Note:** that the `ComponentSystem.Entites.ForEach` delegates and IJobForEach create internal EntityQueries based on the component types and attributes you specify for these APIs. - + ## Defining a query -A ComponentGroup query defines the set of component types that an archetype must contain in order for its chunks and entities to be included in the view. You can also exclude archetypes that contain specific types of components. +A EntityQuery query defines the set of component types that an archetype must contain in order for its chunks and entities to be included in the view. You can also exclude archetypes that contain specific types of components. -For simple queries, you can create a ComponentGroup based on an array of component types. The following example defines a ComponentGroup that finds all entities with both RotationQuaternion and RotationSpeed components. +For simple queries, you can create a EntityQuery based on an array of component types. The following example defines a EntityQuery that finds all entities with both RotationQuaternion and RotationSpeed components. ``` c# -ComponentGroup m_Group = GetComponentGroup(typeof(RotationQuaternion), ComponentType.ReadOnly()); +EntityQuery m_Group = GetEntityQuery(typeof(RotationQuaternion), ComponentType.ReadOnly()); ```` The query uses `ComponentType.ReadOnly` instead of the simpler `typeof` expression to designate that the system does not write to RotationSpeed. Always specify read only when possible, since there are fewer constraints on read access to data, which can help the Job scheduler execute your Jobs more efficiently. -### EntityArchetypeQuery +### EntityQueryDesc -For more complex queries, you can use an EntityArchetypeQuery to create the ComponentGroup. An EntityArchetypeQuery provides a flexible query mechanism to specify which archetypes to select based on the following sets of components: +For more complex queries, you can use an EntityQueryDesc to create the EntityQuery. An EntityQueryDesc provides a flexible query mechanism to specify which archetypes to select based on the following sets of components: * `All` = All component types in this array must exist in the archetype * `Any` = At least one of the component types in this array must exist in the archetype @@ -40,19 +40,19 @@ For more complex queries, you can use an EntityArchetypeQuery to create the Comp For example, the following query includes archetypes containing the RotationQuaternion and RotationSpeed components, but excludes any archetypes containing the Frozen component: ``` c# -var query = new EntityArchetypeQuery +var query = new EntityQueryDesc { None = new ComponentType[]{ typeof(Frozen) }, All = new ComponentType[]{ typeof(RotationQuaternion), ComponentType.ReadOnly() } } -ComponentGroup m_Group = GetComponentGroup(query); +EntityQuery m_Group = GetEntityQuery(query); ``` -**Note:** Do not include completely optional components in the EntityArchetypeQuery. To handle optional components, use the `ArchetypeChunk.Has()` method to determine whether a chunk contains the optional component or not. Since all entities within the same chunk have the same components, you only need to check whether an optional component exists once per chunk -- not once per entity. +**Note:** Do not include completely optional components in the EntityQueryDesc. To handle optional components, use the `ArchetypeChunk.Has()` method to determine whether a chunk contains the optional component or not. Since all entities within the same chunk have the same components, you only need to check whether an optional component exists once per chunk -- not once per entity. ### Query options -When you create an EntityArchetypeQuery, you can set its `Options` variable. The options allow for specialized queries (normally you do not need to set them): +When you create an EntityQueryDesc, you can set its `Options` variable. The options allow for specialized queries (normally you do not need to set them): * Default — no options set; the query behaves normally. * IncludePrefab — includes archetypes containing the special Prefab tag component. @@ -73,11 +73,11 @@ public struct C2: IComponentData{} public struct C3: IComponentData{} // ... In a system: -var query = new EntityArchetypeQuery{ +var query = new EntityQueryDesc{ All = new ComponentType{typeof(C1), ComponentType.ReadOnly()}, - Options = EntityArchetypeQueryOptions.FilterWriteGroup + Options = EntityQueryDescOptions.FilterWriteGroup }; -var m_group = GetComponentGroup(query); +var m_group = GetEntityQuery(query); ``` This query excludes any entities with both C2 and C3 because C2 is not explicitly included in the query. While you could design this into the query using `None`, doing it through a Write Group provides an important benefit: you don't need to alter the queries used by other systems (as long as these systems also use Write Groups). @@ -88,45 +88,45 @@ See [Write Groups](ecs_write_groups.md) for more information. ### Combining queries -You can combine multiple queries by passing an array of EntityArchetypeQuery objects rather than a single instance. Each query is combined using a logical OR operation. The following example selects an archetypes that contain a RotationQuaternion component or a RotationSpeed component (or both): +You can combine multiple queries by passing an array of EntityQueryDesc objects rather than a single instance. Each query is combined using a logical OR operation. The following example selects an archetypes that contain a RotationQuaternion component or a RotationSpeed component (or both): ``` c# -var query0 = new EntityArchetypeQuery +var query0 = new EntityQueryDesc { All = new ComponentType[] {typeof(RotationQuaternion)} }; -var query1 = new EntityArchetypeQuery +var query1 = new EntityQueryDesc { All = new ComponentType[] {typeof(RotationSpeed)} }; -ComponentGroup m_Group = GetComponentGroup(new EntityArchetypeQuery[] {query0, query1}); +EntityQuery m_Group = GetEntityQuery(new EntityQueryDesc[] {query0, query1}); ``` -## Creating a ComponentGroup +## Creating a EntityQuery -Outside a system class, you can create a ComponentGroup with the `EntityManager.CreateComponentGroup()` function: +Outside a system class, you can create a EntityQuery with the `EntityManager.CreateEntityQuery()` function: ``` c# -ComponentGroup m_Group = CreateComponentGroup(typeof(RotationQuaternion), ComponentType.ReadOnly()); +EntityQuery m_Group = CreateEntityQuery(typeof(RotationQuaternion), ComponentType.ReadOnly()); ``` -However, in a system class, you must use the `GetComponentGroup()` function: +However, in a system class, you must use the `GetEntityQuery()` function: ``` c# public class RotationSpeedSystem : JobComponentSystem { - private ComponentGroup m_Group; - protected override void OnCreateManager() + private EntityQuery m_Group; + protected override void OnCreate() { - m_Group = GetComponentGroup(typeof(RotationQuaternion), ComponentType.ReadOnly()); + m_Group = GetEntityQuery(typeof(RotationQuaternion), ComponentType.ReadOnly()); } //… } ``` -When you plan to reuse the same view, you should cache the ComponentGroup instance, if possible, instead of creating a new one for each use. For example, in a system, you can create the ComponentGroup in the system’s `OnCreateManager()` function and store the result in an instance variable. The `m_Group` variable in the above example is used for this purpose. +When you plan to reuse the same view, you should cache the EntityQuery instance, if possible, instead of creating a new one for each use. For example, in a system, you can create the EntityQuery in the system’s `OnCreate()` function and store the result in an instance variable. The `m_Group` variable in the above example is used for this purpose. ## Defining filters @@ -137,7 +137,7 @@ In addition to defining which components must be included or excluded from the q ### Shared component filters -To use a shared component filter, first include the shared component in the ComponentGroup (along with other needed components), and then call the `SetFilter()` function, passing in a struct of the same ISharedComponent type that contains the values to select. All values must match. You can add up to two different shared components to the filter. +To use a shared component filter, first include the shared component in the EntityQuery (along with other needed components), and then call the `SetFilter()` function, passing in a struct of the same ISharedComponent type that contains the values to select. All values must match. You can add up to two different shared components to the filter. You can change the filter at any time, but changing the filter does not change any existing arrays of entities or components that you received from the group `ToComponentDataArray()` or `ToEntityArray()` functions. You must recreate these arrays. @@ -151,11 +151,11 @@ struct SharedGrouping : ISharedComponentData class ImpulseSystem : ComponentSystem { - ComponentGroup m_Group; + EntityQuery m_Group; - protected override void OnCreateManager(int capacity) + protected override void OnCreate(int capacity) { - m_Group = GetComponentGroup(typeof(Position), typeof(Displacement), typeof(SharedGrouping)); + m_Group = GetEntityQuery(typeof(Position), typeof(Displacement), typeof(SharedGrouping)); } protected override void OnUpdate() @@ -174,12 +174,12 @@ class ImpulseSystem : ComponentSystem ### Change filters -If you only need to update entities when a component value has changed, you can add that component to the ComponentGroup filter using the `SetFilterChanged()` function. For example, the following ComponentGroup only includes entities from chunks in which another system has already written to the Translation component: +If you only need to update entities when a component value has changed, you can add that component to the EntityQuery filter using the `SetFilterChanged()` function. For example, the following EntityQuery only includes entities from chunks in which another system has already written to the Translation component: ``` c# -protected override void OnCreateManager(int capacity) +protected override void OnCreate(int capacity) { - m_Group = GetComponentGroup(typeof(LocalToWorld), ComponentType.ReadOnly()); + m_Group = GetEntityQuery(typeof(LocalToWorld), ComponentType.ReadOnly()); m_Group.SetFilterChanged(typeof(Translation)); } @@ -189,7 +189,7 @@ Note that for efficiency, the change filter applies to whole chunks not individu ## Executing the query -A ComponentGroup executes its query when you use the ComponentGroup in a Job or you call one of the ComponentGroup methods that returns arrays of entities, components, or chunks in the view: +A EntityQuery executes its query when you use the EntityQuery in a Job or you call one of the EntityQuery methods that returns arrays of entities, components, or chunks in the view: * `ToEntityArray()` returns an array of the selected entities. * `ToComponentDataArray` returns an array of the components of type T for the selected entities. @@ -199,7 +199,7 @@ A ComponentGroup executes its query when you use the ComponentGroup in a Job or ### In Jobs -In a JobComponentSystem, pass the ComponentGroup object to the system's `Schedule()` method. In the following example, from the HelloCube_03_IJobChunk sample, the `m_Group` argument is the ComponentGroup object +In a JobComponentSystem, pass the EntityQuery object to the system's `Schedule()` method. In the following example, from the HelloCube_03_IJobChunk sample, the `m_Group` argument is the EntityQuery object ``` c# // OnUpdate runs on the main thread. @@ -219,5 +219,5 @@ protected override JobHandle OnUpdate(JobHandle inputDependencies) } ``` -A ComponentGroup uses Jobs internally to create the required arrays. When you pass the group to the `Schedule()` method, the ComponentGroup Jobs are scheduled along with the system's own Jobs and can take advantage of parallel processing. +A EntityQuery uses Jobs internally to create the required arrays. When you pass the group to the `Schedule()` method, the EntityQuery Jobs are scheduled along with the system's own Jobs and can take advantage of parallel processing. diff --git a/Documentation~/component_system.md b/Documentation~/component_system.md index 3201680e..48602d23 100644 --- a/Documentation~/component_system.md +++ b/Documentation~/component_system.md @@ -3,7 +3,7 @@ uid: ecs-component-system --- # ComponentSystem -A `ComponentSystem` in Unity (also known as a system in standard ECS terms) performs operations on [entities](entity.md). A `ComponentSystem` cannot contain instance data. To put this in terms of the old Unity system, this is somewhat similar to an old [Component](https://docs.unity3d.com/Manual/Components.html) class, but one that **only contains methods**. One `ComponentSystem` is responsible for updating all entities with a matching set of components (that is defined within a struct called a [ComponentGroup](component_group.md)). +A `ComponentSystem` in Unity (also known as a system in standard ECS terms) performs operations on [entities](entity.md). A `ComponentSystem` cannot contain instance data. To put this in terms of the old Unity system, this is somewhat similar to an old [Component](https://docs.unity3d.com/Manual/Components.html) class, but one that **only contains methods**. One `ComponentSystem` is responsible for updating all entities with a matching set of components (that is defined within a struct called a [EntityQuery](component_group.md)). Unity ECS provides an abstract class called `ComponentSystem` that you can extend in your code. diff --git a/Documentation~/ecs_components.md b/Documentation~/ecs_components.md index 17f37556..f7951929 100644 --- a/Documentation~/ecs_components.md +++ b/Documentation~/ecs_components.md @@ -13,7 +13,7 @@ uid: ecs-components > BufferComponent > ChunkComponent > Prefab and Disabled IComponentData -> ComponentGroup and filtering +> EntityQuery and filtering --> Components are one of the three principle elements of an Entity Component System architecture. They represent the data of your game or program. [Entities](ecs_entities.md) are essentially identifiers that index your collections of components [Systems](ecs_systems.md) provide the behavior. diff --git a/Documentation~/ecs_in_detail.md b/Documentation~/ecs_in_detail.md index fd52bae7..49cc67e1 100644 --- a/Documentation~/ecs_in_detail.md +++ b/Documentation~/ecs_in_detail.md @@ -23,11 +23,11 @@ If you need to access `ComponentData` on another entity, the only stable way of For more information, see the [ComponentDataFromEntity](component_data_from_entity.md) reference page. -## ComponentGroup +## EntityQuery -The `ComponentGroup` is the foundation class on top of which all iteration methods are built `foreach`, `IJobProcessComponentData`, etc.). Essentially a `ComponentGroup` is constructed with a set of required components and or subtractive components. `ComponentGroup` lets you extract individual arrays of entities based on their components. +The `EntityQuery` is the foundation class on top of which all iteration methods are built `foreach`, `IJobForEach`, etc.). Essentially a `EntityQuery` is constructed with a set of required components and or subtractive components. `EntityQuery` lets you extract individual arrays of entities based on their components. -For more information, see the [ComponentGroup](component_group.md) reference page. +For more information, see the [EntityQuery](component_group.md) reference page. ## Entity diff --git a/Documentation~/ecs_job_overview.md b/Documentation~/ecs_job_overview.md index 14c7543c..bbfa4610 100644 --- a/Documentation~/ecs_job_overview.md +++ b/Documentation~/ecs_job_overview.md @@ -17,7 +17,7 @@ For example, the following system updates positions: public class MovementSpeedSystem : JobComponentSystem { [BurstCompile] - struct MovementSpeedJob : IJobProcessComponentData + struct MovementSpeedJob : IJobForEach { public float dT; diff --git a/Documentation~/ecs_project_status.md b/Documentation~/ecs_project_status.md index 02743642..97736a8f 100644 --- a/Documentation~/ecs_project_status.md +++ b/Documentation~/ecs_project_status.md @@ -8,7 +8,7 @@ uid: ecs-project-status As the ECS APIs change, so too will these best practices. For the time being, these are our best recommendations when using the ECS APIs. ### Do's -* Use IJobProcessComponentData for processing many entities' components in a job. +* Use IJobForEach for processing many entities' components in a job. * Use chunk iteration if you need finer-grained control. ### Don'ts diff --git a/Documentation~/entity_command_buffer.md b/Documentation~/entity_command_buffer.md index 50abb3d3..cc41a0e7 100644 --- a/Documentation~/entity_command_buffer.md +++ b/Documentation~/entity_command_buffer.md @@ -6,7 +6,7 @@ uid: ecs-entity-command-buffer The `EntityCommandBuffer` class solves two important problems: 1. When you're in a job, you can't access the `EntityManager`. -2. When you access the `EntityManager` (to say, create an entity) you invalidate all injected arrays and `ComponentGroup` objects. +2. When you access the `EntityManager` (to say, create an entity) you invalidate all injected arrays and `EntityQuery` objects. The `EntityCommandBuffer` abstraction allows you to queue up changes (from either a job or from the main thread) so that they can take effect later on the main thread. There are two ways to use a `EntityCommandBuffer`: diff --git a/Documentation~/entity_iteration_job.md b/Documentation~/entity_iteration_job.md index bd9b0944..c40290fa 100644 --- a/Documentation~/entity_iteration_job.md +++ b/Documentation~/entity_iteration_job.md @@ -1,17 +1,17 @@ -# Using IJobProcessComponentData +# Using IJobForEach -You can define an `IJobProcessComponentData` Job in a JobComponentSystem to read and write component data. +You can define an `IJobForEach` Job in a JobComponentSystem to read and write component data. -When the Job runs, the ECS framework finds all of the entities that have the required components and calls the Job’s Execute() function for each of them. The data is processed in the order it is laid out in memory and the Job runs in parallel, so a `IJobProcessComponentData` combines simplicity and efficiency. +When the Job runs, the ECS framework finds all of the entities that have the required components and calls the Job’s Execute() function for each of them. The data is processed in the order it is laid out in memory and the Job runs in parallel, so a `IJobForEach` combines simplicity and efficiency. -The following example illustrates a simple system that uses `IJobProcessComponentData`. The Job reads a `RotationSpeed` component and writes to a `RotationQuaternion` component. +The following example illustrates a simple system that uses `IJobForEach`. The Job reads a `RotationSpeed` component and writes to a `RotationQuaternion` component. ```c# public class RotationSpeedSystem : JobComponentSystem { // Use the [BurstCompile] attribute to compile a job with Burst. [BurstCompile] - struct RotationSpeedJob : IJobProcessComponentData + struct RotationSpeedJob : IJobForEach { public float DeltaTime; // The [ReadOnly] attribute tells the job scheduler that this job will not write to rotSpeed @@ -39,11 +39,11 @@ protected override JobHandle OnUpdate(JobHandle inputDependencies) Note: this system is part of the HelloCubes_02 example in the [ECS samples repository](https://github.com/Unity-Technologies/EntityComponentSystemSamples). -## Defining the IJobProcessComponentData signature +## Defining the IJobForEach signature -The `IJobProcessComponentData` struct signature identifies which components your system operates on: +The `IJobForEach` struct signature identifies which components your system operates on: - struct RotationSpeedJob : IJobProcessComponentData + struct RotationSpeedJob : IJobForEach You can also use the following attributes to modify which entities the Job selects: @@ -55,13 +55,13 @@ For example, the following Job definition selects entities that have archetypes [ExcludeComponent(typeof(Frozen))] [RequireComponentTag(typeof(Gravity))] [BurstCompile] - struct RotationSpeedJob : IJobProcessComponentData + struct RotationSpeedJob : IJobForEach -If you need a more complex query to select the entities to operate upon, you can use an IJobChunk Job instead of IJobProcessComponentData. +If you need a more complex query to select the entities to operate upon, you can use an IJobChunk Job instead of IJobForEach. ## Writing the Execute() method -The JobComponentSystem calls your Execute() method for each eligible entity, passing in the components identified by the IJobProcessComponentData signature. Thus, the parameters of your Execute() function must match the generic arguments you defined for the struct. +The JobComponentSystem calls your Execute() method for each eligible entity, passing in the components identified by the IJobForEach signature. Thus, the parameters of your Execute() function must match the generic arguments you defined for the struct. For example, the following Execute() method reads a RotationSpeed component and reads and writes a RotationQuaternion component. (Read/write is the default, so no attribute is needed.) @@ -78,9 +78,9 @@ Identifying read-only and write-only components allows the Job scheduler to sche Note that for efficiency, the change filter works on whole chunks of entities; it does not track individual entities. If a chunk has been accessed by another Job which had the ability to write to that type of component, then the ECS framework considers that chunk to have changed and includes all of the entities in the Job. Otherwise, the ECS framework excludes the entities in that chunk entirely. -## Using IJobProcessComponentDataWithEntity +## Using IJobForEachWithEntity -The Jobs implementing the IJobProcessComponentDataWithEntity interface behave much the same as those implementing IJobProcessComponentData. The difference is that the Execute() function signature in IJobProcessComponentDataWithEntity provides you with the Entity object for the current entity and the index into the extended, parallel arrays of components. +The Jobs implementing the IJobForEachWithEntity interface behave much the same as those implementing IJobForEach. The difference is that the Execute() function signature in IJobForEachWithEntity provides you with the Entity object for the current entity and the index into the extended, parallel arrays of components. ### Using the Entity parameter @@ -94,12 +94,12 @@ public class HelloSpawnerSystem : JobComponentSystem // EndFrameBarrier provides the CommandBuffer EndFrameBarrier m_EndFrameBarrier; - protected override void OnCreateManager() + protected override void OnCreate() { // Cache the EndFrameBarrier in a field, so we don't have to get it every frame - m_EndFrameBarrier = World.GetOrCreateManager(); + m_EndFrameBarrier = World.GetOrCreateSystem(); } - struct SpawnJob : IJobProcessComponentDataWithEntity + struct SpawnJob : IJobForEachWithEntity { public EntityCommandBuffer CommandBuffer; public void Execute(Entity entity, int index, [ReadOnly] ref HelloSpawner spawner, @@ -136,12 +136,12 @@ public class HelloSpawnerSystem : JobComponentSystem } ``` -__Note:__ this example uses IJobProcessComponentData.ScheduleSingle(), which performs the Job on a single thread. If you used the Schedule() method instead, the system uses parallel jobs to process the entities. In the parallel case, you must use a concurrent entity command buffer (EntityCommandBuffer.Concurrent). +__Note:__ this example uses IJobForEach.ScheduleSingle(), which performs the Job on a single thread. If you used the Schedule() method instead, the system uses parallel jobs to process the entities. In the parallel case, you must use a concurrent entity command buffer (EntityCommandBuffer.Concurrent). See the [ECS samples repository](https://github.com/Unity-Technologies/EntityComponentSystemSamples) for the full example source code. ### Using the index parameter -You can use the index when adding a command to a concurrent command buffer. You use concurrent command buffers when running Jobs that process entities in parallel. In an IJobProcessComponentDataWithEntity Job, the Job System process entities in parallel when you use the Schedule() method rather than the ScheduleSingle() method used in the example above. Concurrent command buffers should always be used for parallel Jobs to guarantee thread safety and deterministic execution of the buffer commands. +You can use the index when adding a command to a concurrent command buffer. You use concurrent command buffers when running Jobs that process entities in parallel. In an IJobForEachWithEntity Job, the Job System process entities in parallel when you use the Schedule() method rather than the ScheduleSingle() method used in the example above. Concurrent command buffers should always be used for parallel Jobs to guarantee thread safety and deterministic execution of the buffer commands. You can also use the index to reference the same entities across Jobs within the same system. For example, if you need to process a set of entities in multiple passes and collect temporary data along the way, you can use the index to insert the temporary data into a NativeArray in one Job and then use the index to access that data in a subsequent Job. (Naturally, you have to pass the same NativeArray to both Jobs.) diff --git a/Documentation~/entity_manager.md b/Documentation~/entity_manager.md index 0ae63575..b642796c 100644 --- a/Documentation~/entity_manager.md +++ b/Documentation~/entity_manager.md @@ -3,7 +3,7 @@ uid: ecs-entity-manager --- # EntityManager -The `EntityManager` owns `EntityData`, [EntityArchetypes](entity_archetype.md), [SharedComponentData](shared_component_data.md) and [ComponentGroup](component_group.md). +The `EntityManager` owns `EntityData`, [EntityArchetypes](entity_archetype.md), [SharedComponentData](shared_component_data.md) and [EntityQuery](component_group.md). `EntityManager` is where you find APIs to create entities, check if an entity is still alive, instantiate entities and add or remove components. diff --git a/Documentation~/filter.yml b/Documentation~/filter.yml new file mode 100644 index 00000000..d17fb0bd --- /dev/null +++ b/Documentation~/filter.yml @@ -0,0 +1,37 @@ +apiRules: + - exclude: + # inherited Object methods + uidRegex: ^System\.Object\..*$ + type: Method + - exclude: + # mentioning types from System.* namespace + uidRegex: ^System\..*$ + type: Type + - exclude: + hasAttribute: + uid: System.ObsoleteAttribute + type: Member + - exclude: + hasAttribute: + uid: System.ObsoleteAttribute + type: Type + - include: + uidRegex: IJobForEach`6 + type: Interface + - include: + uidRegex: IJobForEachWithEntity`6 + type: Interface + - exclude: + uidRegex: IJobForEach + type: Interface + - exclude: + uidRegex: Unity\.Entities\.JobForEachExtensions\.IBaseJobForEach_ + type: Interface + - exclude: + uidRegex: Unity\.Entities.EntityQueryBuilder + type: Delegate + - exclude: + uidRegex: Unity\.Entities.EntityQueryBuilder\.ForEach``\d + - exclude: + uidRegex: Tests$ + type: Namespace diff --git a/Documentation~/images/sec1-1.png b/Documentation~/images/sec1-1.png new file mode 100644 index 0000000000000000000000000000000000000000..594d1c4c25abfb563311413b65f0a1746e904c18 GIT binary patch literal 10519 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 AZvX%Q literal 0 HcmV?d00001 diff --git a/Documentation~/images/sec1-2.png b/Documentation~/images/sec1-2.png new file mode 100644 index 0000000000000000000000000000000000000000..a9c48e285d0678abf70818e877da9b5899be9a86 GIT binary patch literal 12922 zcmdUVc{G)8*zSuAA#;*>h$M5S$UIA?%(Dz3QwW)bP$a2PndfGyzOL(jBQ(?$FX2<;BM5RySxHU{K`=Am z-}!hr@UtZPl@2^zbWt*JLl8<<^f!jG*0l`;AvCj-mDSL&b#`}lvvqc1RF;)xba8dI zv2(CS5U+_sJx}fX`^Wx&m1Hn2p0neTS>dO3zG1qhDlH`>B1^^?Ny2lH*(p=EzosoH z{Z&ngEnVPHgyBOSm(fjmu2(8te}lg6M_;gC+fSakjoSa@v+k4ByDhO^y+1#X9XX<; zohLIu#jcf?kdecysbKRyd>gqS(ieQ?fXps}Lx7Ku2&qXvuMi%ULK>}wBUE(4&p#$~ z&a-L-DqVQZbu)$%qg5iX=ye{IVB;RPxv+Gh7S^gJazRzWhO$u?gJj3Q6 z3{+Mo!&*%YObU`~p2vKC36aYjFcviT!V0X_%F@78@J60xJSDqoPI1XxN>j6xkXH-K z<@=e;^?im9E&0o;iTX_2Gz=FpBSw~2->NXb!Pvpc!TG{3P-&!S`i69L>ZK|@ed=0b%{9HOKS!B{ zI)6&8nXD4%SfssD-<}WER7Q?4Ds+_X%DCSXY2CU}<0;pXKNN9eLUei6x2r_XQQZCN zFGd9sxjv~72MQL37X)j&Yb48R%uE=;#+|cSvNbFNOc+?K*OU})pM{XQzTIgJmBz$z zeJc~8E}paZ_c^UnV+w`q+Z2SjDFv^TJ%pB;_qFVJEY~GeaL_Na@l19FrWZ|8o026O z;iTEJ0#h~2=}v57&*;9tvQaEM9(?akLMPSHFYS&;C+E}GP+{g-8L(|YZ1An%xFEkk zcjQ~?i-aD%({Nte!oDRNR>i$I@xfOWxNG;q`h(m_)^JI=GPq_HeEJ*Q+38eq`=%=x zT5G-v9#ri~dLKOaBT7G-c-PhEv268sGezneD)0TA&eW5Rkcvv1{T`hkZd*_MH?PFL zPo*j~yOn!;$n+1zlSolnx{N9tr;*!lcRz9DJEcou&PA_^s~Ty zSlxsrfhUE>%u9GtNt&%)>ct(3pr#9|?Swt;cQ~6}3);OMnuSin#dA8OXu2#+m^ra#C z_Z%2MlS`7~hFi8?O-ra&{=t$&!V^w>O`iHS=D-K{Yq&8DR_s^rX1z}DGIqUVpr-qp zp!40lg6nkyMS7Or7`-mtts93QFf-LJ-Z1(awUybbjrmS~L4#26U0z0BPLwv+uu@_g z_x(a`vE3w_L7R$jk94dhThVv#2j-S@cYM7C>!XU&)%)U>ecW+R6WSgclo}hH6;SEo z+>8?p68xysHqkuME811_)!wq+qI_KDGr13**oVZf?+?`<^gb-MUVp$~EnLn}V&*17 zOYrLE_1CeDF)l-&#y{zNO8ms~iBgSbF)^tjDJm(APv7`WA04@8})T zFYU&-MUuinH|Nymx#yjJ3%T;RPF5%0Ax|Rbc^%i>z4+~C;r#HH+LqI{>7SLA{lWb~ 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 MifPlmjhTgi0Hp<+p#T5? literal 0 HcmV?d00001 diff --git a/Documentation~/images/sec2-3.png b/Documentation~/images/sec2-3.png new file mode 100644 index 0000000000000000000000000000000000000000..86df61780eedbf499e76fb3c7fa162533bd85d04 GIT binary patch literal 28299 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%0Aa;>JQPbNlr($(HW3bI7RVdb=W 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 literal 0 HcmV?d00001 diff --git a/Documentation~/images/sec4-2.png b/Documentation~/images/sec4-2.png new file mode 100644 index 0000000000000000000000000000000000000000..96e3a2d057180e1aa24d1a1b9932c86a8337bbf8 GIT binary patch literal 11571 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 literal 0 HcmV?d00001 diff --git a/Documentation~/images/sec4-3.png b/Documentation~/images/sec4-3.png new file mode 100644 index 0000000000000000000000000000000000000000..46d70a21abd5dccb445555ab24699a1e6f36c795 GIT binary patch literal 22917 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 literal 0 HcmV?d00001 diff --git a/Documentation~/images/sec4-4.png b/Documentation~/images/sec4-4.png new file mode 100644 index 0000000000000000000000000000000000000000..0f5d78dbd8dab13ae1524df04ef6e9c46c53a3c3 GIT binary patch literal 22115 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 literal 0 HcmV?d00001 diff --git a/Documentation~/images/sec4-5.png b/Documentation~/images/sec4-5.png new file mode 100644 index 0000000000000000000000000000000000000000..aa61d9234663a3cf33b27f06b1f700bb25a1ac0b GIT binary patch literal 34326 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 literal 0 HcmV?d00001 diff --git a/Documentation~/images/sec5-1.png b/Documentation~/images/sec5-1.png new file mode 100644 index 0000000000000000000000000000000000000000..dc3a9ab15f111dcb4b4b2915012c683b1a52ff77 GIT binary patch literal 25089 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 Y(PJ&wXGala0W~$3-S6$H*cp`gZ=G5a;Q#;t literal 0 HcmV?d00001 diff --git a/Documentation~/images/sec5-4.png b/Documentation~/images/sec5-4.png new file mode 100644 index 0000000000000000000000000000000000000000..248c011336ed66af98a152fc2679ceb8b986ebf6 GIT binary patch literal 44037 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 Ld~!QR$WjL3(JU83F0e5s;G3p@yO7 zJoCPL@3Z#$&Ue;X>zuV1W0?8J^W67!|E}x*Nlitb_&U{f2n0e5eJ%roK=4rD-y?)q z!Jm0@$tB>gYfjJgTp$o~X50%83S-)VK!{ANrKQ!>tQ=h(U923P?n9-e?>jj=T3Fke zLm-}$a2*d#4a}+k_H!xxmmeMx-Zs1bwKJN5N9ma)zo7K(`=K{EuRU=1rae^G_9iR5 zuGES)aP*VDrIr(NSC&0ofqnZ;Hzwk;%@!toRtSMf@j>~d4eX1fYA}n7S3<|1YZgci zQ#^tdBxdJ5RhP4f|F{piEI1f+@94JmrzgBzTqKaXbZn&nQWEmRT;P*})<njZDiT{s&@+%(r1fz89XC0_$NpYWQ;Akhj@hw<%xTA$}pLu3kp;6n4p} z|3c~^hCXRREa^7XA);jK3MAuR@gw!S3M_xHQ=hN%O$*WBQ7m!|K_QnRud^})`L6}y zNmjB}ERaFI;QthPPc(SK#HmV_TCfzOJ$(B$IfOy7u7_#93{nK)hUrnz=eAb3+==3X zgyNMeWrjnC+WMmsWSBO1kx$#8*QWhR7)i$3#uX?eJ0vn6KPMhro9&G0aNR&W8Bl~G z$CY*!?pp6ci0~A2VUaY2OKOKpUVn}pNV1>`S6+~+T{xR-e$h^+ifNT$Svyivhk77J zQ$%Hz8v0G({feK)*EVAn9z^3ET+O>$%gtM5Aa5Lf3ppL4L`Rpom0HK7le2#EOS9AS`?Ij)v-Kr1V4#J24{Y%^#DAZ;#{{J~vD5N|dU7 zyp=1>J6-o6%YjAYEp30eg?#zxNWAM!TFH)~XRd^_GCqS%l_nll0sDHSdVNKwMTJG$ z<9%hH6MJ>eKR%^_53X7;%O56)jf7VcY$*p1y>Y#{MR1Efn|)r+XQ;{b5v>xz;7ld` z&$@2Dqv}Hm@1s}i!gR<~MQ5Ma(lz}i@>F#c-k7}3%(ISnl~op)Uaj9Qdv5-__hRBR zDauTE@`Xl?*YCK63QN;wuQPlNS0f(vTED4g>}9ky>XKSG((0UKk!NAZvMdcdr4qf z(|k#sP?GS$Q=)6npRu${eips+runi`J5g`DC|iqjQMn#>+iD&|CHbs0K~>Zxc) z-sp^5lyiQlCr`&X5T(0d>vNwI6tNPqA<@sPSwy^7ZxDT_qT#Et-iaPq5!sVI((~mDzzYz;3Sv31 zshNrMlC2cdiSxXzHm$;~*6-I?Miz2Ndmr(L@RTKv+Qm;;RK!@= zLFIz{Xx?MQQO7-u#v zHZC6)Y>f?X*rUW7rM-K6bPi|6PHj-6X>{Lxs~0^dmVs z91@LUjDj1(&Ye!X4*3odLQb^4w0kt#f+^0P8>@?7wvXVZjSHr^Gb&svODcRS)v3q* z&HW3h0>UP11KX%6Lqb4x);BmU) z#_Cr1x?Vs{@TtRG?;G@`bmDUQSU#)W42>Z1dY}4xb&l=`zF)38rOiA90wn(P^M4r7 zd51NBx-g*!Da#Xc$G>~V6t}DTH$GG+Vw_NpjmZrX$3(j?eQo^Gh|1`>9XgX3kbYO1 zJ~i$8sh*adwVdS~(;oPsPzxo8vkO?AKm*BVKf(h$q*Fh12RqQ6+%GlBU~-avVH^3# z%5V_#4Bq^`IeIRv86&+ZWyWA2Mfd)Yw2yR}@`SBdEcDtG{h{_IXW9dCDwX&0?{z+y ze0&=AJWQANwLrOvnS+_KnS89rXm(!KP|@nI)wUs|X_09**OaO72VzQo^IxBIgMKak z8Y{yr82}kr^RhL81VM|sSrS^JDk-JZGW<^Dlgg(BvH-efx+uY~-alKjo1rl= zD$L2u5y~@BXDrNY<}bV9nk$;hHX2FllX;a}){Y~sBMI5n7DU}&qF-~2aBD>jMiX2jt4|18IW4O&Hjz4jbt*H2F(&=5aGI>Y4!bZ%dx_DS- zlWz-~w`#t!IULQgb{~0?Q>7e{)MS6vQLZ`E#IkO{kb8fu=s+**?)cpiRytoYv~r(a z>qPJ7#_Uz!dVdlzd9IZf!-3kuW&h$>t+dn?@e-Svj^p8DO9b`VbT_Jwe37lDbkK6X zPIgnaTNX)9z&Z2%ry2A(nm{p_6K=qwuU)%2wpZ3Opy{OrH)%G=)Elbfb}(;!a;g=g z?Pr);ePZLLi!SEVwp}jvs@MMh-Q1|NZW2Adu>!~LH2yiIrvA!5YVZ16x9U@4zUxS9 zpLMe8`P^yiA$z>Rvi)(TR&AA=!SHvosbIU=rdFGd7OHYNjeWC~ywn7LB5YaaINZIJ zV;&s3%%I{lBE9|G)n0WoEvK=`#}*^BX;3k17hq1SE>KpSC0^20#ZZxX{G;@eau{CN zX#m~P{&>)Lf6E}7eZB14E1&N%q^N$tgd z-~R2q>3p%75QqN7|wW-{xiageibza$dk zF*S4AB!Ol)Z_VnzMlVFKMtg62c`9Y9(CGxVQ@10+Mcbrm4c5*L$g21FCy}b_&g;~3H4+d>0 zpF%c_bdZX+fg^;zuj=ZAk?wAOlSFC)Z?YDuPMIO4VQ)}kcp0*~{O22DD1_ygfD*lD za18Jt_f4O}lpzp%HweTf1A<2g-ttd}K+Iwwc-Mdx;qrk%Zu!`9ifKY1^kGn$XPTaq zo6|l{noH*$y8{mV6HN8Bx404-56a=o-E5I=n-y(~4Qjb|WPkQ-b zb7I=1-=%5gc+=%@Q;6KO_a*c-J_N%4W{TypdUVit2qZ(cRFC=Cng9aPCql&WgE!0> z8t@?Z!3#1^41o|j{{52lyaa)?Q2+h9IRk+N?p*!r1$id`7T|yMmAMBKR&PV9otG4r zmX^8}b|0~@$X&WhH2AB`)VxeLsgJwPkmK&1J9#=K#@BD$FwMe)Frs>Ppy^WNBVOk8AS zMspAWD>pYRdU4mY(t3m?E8(wTwuMpkc z(^F;IL9(>G3{Ij%?{^UOH7hGFC8cO|bac)qHa`ApLZ-_~znvynC+Xz2SL$wVZaXM( zzmrFzqJ|Gu(_)##c7KxTS6GA%kBlt2*Yi2fh?<+5=NnWzG$Tb%RbbGqq4Qb@jwdVRnSXh65WdPY7#~K$E#Q;s>vyCq(EWArkU-I?q zgRRLbMO$R?*pD}t+|G_(v+I@1$cNKJbu0`iayBxBw0%|2k{i2@VmJ71wb&U&CGK0p zZ};nm+!aX6X#hBimb&^-5j7Jy1z4mN+L^exxIItNy<=bRleJv1Z=;Pq9?;=~wGs4U zjB2+-wPN&k!@dgcA}1y$nlYvNaQ$kB)bSFN61+=S^BNiq#nV;C#>VPa(tKj0qB6fJ zCMZOca~c;V@>s6-a3VB4J!`;%YH4fdj1}sz^7CuA29qN9mU{S{<|N&=f9n8I%fxyg=AdQ;!+ZO4h7&(X+@R#FL~tM(d#sVa^8 z{UVcfZiU_5-Aqu&hTRXsuB-Q<3W|y&b#5D#<~`(4*vpqM8@G6TkDSacEEJ+|-@eTR zB_N?ub8sjYcHhhsa#`-A$Xe)#V3m+CAt0sG)Yne}>%-@LusUaZg5JpOOBO($IK8Ym zchs1&`|ho(dpa@TpEDqVHGKX0wVerW_tTLZW04Hfyq5hmLa8clo4?daf~BMH5MJiB zpXBx1ZM#P$eap^_&0xNxihv;s^eSEHT6ZEI4iC!Sry z*Vp%HpYqP>LC;8Z>Lv(bhJ0{m<&^sxZSODD?Bg zYQ)$%X10Xdq2?LbS>Mzs7dyKm{VJQbt5*Uo`d@`Uj{51e_8>MPft8J|NITG9=h?G2 zza6SUXM;V}GB!5esGgyVkO!sKJnM#BnQIPu!p5elpzs+6gS`!&=;`UXOGig1%4Dpk z_ZTs2f+F}lJPbSb6_=Hh8&5*aq7ph5^7qe$AQ=;{$$$)3mhLQ<3IRzc!m0H?6&t74 z|59w6TK`kA|5EFJD)wI=|DS)ws@;wkkK>+kiH??*7OxCp$j6Urnwocr@mi<}7Hu4ZQ@*BkX;5=0~Rh5)t zTjw*lr1BXnCJ5~fwQ&b0nf0nr2G{YQH;Nlif8DGVzW5iK5>zX;;s-7LscosDJP4S-C;V)BOez2G| zx}@Sq8RYEx*~X3D)^k7JOxAmpTpSMs7?)${CHx!s9cPMqdzENK+(WMtNI`=8-p~&L z#jXV74sP72FEHy{!L%lTI1Xztc1&H_o@q3gs&>>dG)%clOqKoXmo5}$ZvJg=Z!b1I zy(BxE$=u8g%tw|Z;reL~@RK{@)xY8IBT5Gb5%k-)Z{5AUn#uh3zXZW1W{CSEOu)bZ zf~E%VK=|ksO;lLgjumLFj}<1Uq>IDBxi$3k^vd<&-hVzlthDH(`j(s9v*m6()8K_` zylRJOn?`wMW#z`6QXh=NG#D#j*!m>)j0Z4ae9L6-!; zF*81@tsjU-?5n49a5|6<6c%N*HvDa}(mDnwhy?{tmR43+ZxVLeRN0&yta$^00bRfh z^_cO>C1=yk*xcNdTQM;*g8OXO%2QHO>H}?HvflmEmL%ko1J>fsty|4<_<%tyD|Fes z(PKJ%o;xpas~K{Sa~yPX<*T;qxQQQqFmB*z-TQ@W9a)mj^cd)ln9m_mlRQx9akxP) z7;Vp^0a7H3Vj1Pb9|;KPfL$6*5pppL9RivX2Uh$N0m&mVvDC23ywWM9S9vcX&Xzdi zQd8mQrx;c+{xP!w2%z=MP*6S!P-s5J4G3GzLsi@Yff;z`-o1zQ)5a(UXVcK4@yYtL zu9%1BRr+pAaXQ^=uHdHt3}F55c8NHl_<$=tDW@2uf){+odAUZgW!~ErqO4^ z?;=w=_+Yyp_2l70`RRI(1g6@5XtRfVvU09!dNEqp+1Yt#@U#kP+;bIzJ6h<}*7(~i zH!5wB9AG}@mzJvaByb{C{rAYgn1K$D=4+sv3FxiR>tg_-2Jkj0sF%R;aM|LEfJ~N%CDjjmBYAdGW7Jyt!ixe@fuqLZ^ zk(#b9#ne5k56r5bK|2>wl%@xL0<){DLs_3AB6Rfi^&2zS-0Q0Xu>L|cB;Rs!x-Cwz zj*U1mbFYs}O9Sm!3~Esyffi2& z)yw}5pKXkOU$)c8kR2CQ4($2FCD+#0ws(5YD@XXceww>|w2p=is1C!;Nl&!0b1+%;ib&h^vHw^o;ydmDQP z45w%bkCRp=mun0?8)qUL`<4=uqo2e4{V&XY>`@;-5`v;u-z0Q%AW2>*zjgQS-E!Lc zK8qW4>d*hu+805W7(!c@B?$#jeH>KUFX8U#U&H0U)ZG8a4&Xliudn=H&7*&Ll>hGt z`|mu;eVGZsXjszwG8nHUh?N&$&Lzw#s4ga`L$yIW1r+xBb=jFef~kvwF}aT=cLbV+ z_f8;?e&ho&wJT=Y1B`dJqc6yfI41P!5aW^V{hNcI@3OkCqEd#*s*c{A z&;WJh_Vyfy(x4NqnF6B(0++MFA(D8lVW}Mtm6K$gE`>JW@#>snP{3+60;9tO1sH1L zyZaudp^-0DIAlt)F&Sk2#Sc9P5bS-#hefr23l{-Xe|U zdR|pb0DXFBKhsd#wgt&F4j>Pa1s);Vta>U4DU>UouUdI_lIOU19`xP=3jI`ExBdjU z1Hf&*0*u_M)}+Rp%0Y*Mvbrfu*_d}4C)7? z^k}Fbj6QQt{y3jvcCAu0v$8KwVm-z!Q~bZ|iHvJ@yv=Qed3li?3sHR*8bGN)%OVnv zfiDDrjKh!+K{XSG`ZI95e0+QiE|iLc0Y>rjjlRduV8fK7?~svYB_$=z!ATwK)>zK} z+(R4y&+UzI+5- zRSu3da9d61e>CqhzJRF;U^8yey>jIW7-M6Y@f^S*Cmj+OnFfxHS^&1&wl8rcC;}>O z0M)Cdp^+6EduRG}2krV~)dn*vU=@*+U=i@^_p3611Kk!CYpMY!G1?`@>;P{H$0qH> z8^h=&i~%xgTUg|TgHCydDs$V=4x<(*ZnlLz1mJOYvaYgeRpMn zWchLKA8(jo3nZ||1AXO992VG8T{e9%U&Mib79J_Dr~sGzL=Qer?~8S+H57jR=O<=3 z9tfW|FkNHSjwZ?}f@N*9go~gQPo`1+xsHf)?Wm}zlcR-*F+@Os-`Q?QLIXAv`)bk8 z*rGp$@(c@tg1G4Dud~<-j9Xf@w{SBmHkK)&=UYk&4H{Va{)b4w6E+L&VSC<<>qA+=z*aIOZw2&OR!`yF zpsR3@K=9Ev;fmotCnqNotWRuma$$Knx8K2#JXjTN91^7Uf?3 z<)#B$4k)H#{CVZ$4dm_A+LT2A8t6f3(zUjGCZph%`OLfTxUCM(NB4EiJMm8D^Nn6u zZvGph-tkp2MNg>t@ws919FI5M>w#+~C(=wGcnFN?&W;nUguh{<4~E1aVr6$uHHoE~ zl3-c`kwbS^mpa1#{0=QG?eA)iIAG>C&`1+pRN;2C^AZXJr6^0UWknK19o(APK_4LM zsrc(z(ABm4+_YhU7}W5AoCApilXn=|{nPW6v+T3gtOy)Xfu!Z3O3L8cNN&}OW_+-F z>Z+o<@5YPtvJMVhYo{H68CM0&yFLz2X0irv7Zew31A_?`1P4iQqk3(Ce6dXSvrR_0 zJ`wiVW&;;sneL|@aMViwc4QzA*iMfLsw6DVb((eErWJCQ0g%Mmcn}Q?iCh@fT{eG^ zH!v&4uY&6ocA5)x-~OGSot-^*y0W-v)$;C|&1_SnuXy_MKza!P-MI~F1H?WDmBId_ zWR&sV&dz+RevMORJi9^8H8K_}pyPFbim)CZyH3t&&1{O!nXL~uCqSpgfu8<0lB>D_ zphZ(#JHANUHVZwW>&N#!fVa{&Rm5W)+yE;S7u)~`kbyJ_Fv77?>C4un27mNc0Py}b zz#@;X?v;S{#|3EKhnx8VGoI2Q2FkD4UU3uLEwTrD=YYd9;0Fp^%-Ua-@fPYgQ2HyFT`H_fQ)3xlP);d zhR&h{q=QNL$XJvJRnqzGlfVwE?l>m&m;)zcWJ$@Qnr7D?L2t6M+rDg~p<#M)fk7l6 zZcQaeN8et+oB|W049D!8o-YF>i1-u{Apn#Nj`mC!_v?LkjZEqO&8Ohp7Qo{#f1h%& z-8)GgI1C97=MN^O=VxYSmN-Fx3QY!fj4^*EfF!t8DT%jhadGhr@Q4#U4_D=q@UPk}Gl0{1hiY432a^OG(*XP5rKFr#1xvm8$_B0v~DG3d)M;-L^G?L z0mmUgAT_nLnCqwgyAUp-I;X%s0Fhn;LErEfF4NHx6A36xM&=!;;EBbPqo())iYyMJ zxC4F{2rzyPlP?HqD(jtUU)&i-0dRHQfNEGygQHLKhU<`qSj*ZwljR?hnFhvE7; znBKo;Xe+?tly4mSP62z@Mrzq(#vg&0K?z-Y69{Uw=y-nxV8tU4L%G*Cj)&Vm=-@ zq~3zeVCSZd=|=DkWb7g5lP53`LpPQ<+W4guf?0PAmKNmHz%e`f9Zp!T&ms{1CnM=L zz)^k<4ILfRF@XbC;H&c|-UNXi!vZe`fmAZ7# zY>qPZ@wq?nxrKnanlXq-vM&5Lfe#)$IJh`P*{<}b;*2{0h#-m7)8a2*D7Jpr0%CRc z_CC2Xd;SCj)NuW(Va5p2uedy9f4NV?-Tga=yr;9bJc=(K#-4DtoXUfQw}0X|%~l`O zE;zWei^~{5UO%wrz;S=8b6taNR9OZ@7lN6CHU_y447^6mOy#TWr?jtKyM{Kl`U+6p z8$bvytia{KySux}NeHGY$64c!2zq3D81-1LDn00xE1a$8ivUA# zj|jklaHWyvwM;qf)JhC`50|$9Ka=zE<3|?w>@i)AEeD2L94Y1v9z7VXysIiNFTZh7 z=@1&#vow5hej0>&8LAE{FK8L=v^qT-5fatBQz1;>Z=lV=zJy;+3+qd|@uJ8yB+W=sg5}?+<08sOUon0CC z{KnY-*nDfcUWc0BZV()u!;@(pbBg0M^wyM4h2?<5&bz8*;~wRFWe}3G9^B(F%wv%L zUDMtRmz}flzuCYsnYzsV{)=lKWFdl0o}D ztT|7H(WA$Z;6zE40leRR!Sf^DAOceUb_xw6Bm3du;a}75>(=BL z?kd6Ls%$3H;H(rUr4=^AT>&m-+yINO*>ew;ZW3@_1yxbuPX{Ruyd#)|)&`WttLR65_wHj>Rz+Ms z_vR9T>z|+3fHNu3qv1zHo)qQhs{?4=v5n*b5CM_`I>YLm(Usa0`jZliuIaBW)a)fL z$Kt62OX~R&>-+T#BPkpa3Vy>)EA$cn&HD-b)Y{h3 zL|f`=-hY0mN_^cYmwtV6IePs!gpOSmfWwXjK0EQ>r;=BMN1&Xr8Ms`BL%ac)?f>QT zjU;K$DJ$?iK%XdC8CJ$5VUGuq44Jz?;6ct>*8_75FG#@?dk9okMW$Hl#oPY?)t8hF literal 0 HcmV?d00001 diff --git a/Documentation~/images/sec5-7.png b/Documentation~/images/sec5-7.png new file mode 100644 index 0000000000000000000000000000000000000000..306c06574e62868c2763c9324b32214c4e80b0a9 GIT binary patch literal 45985 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 literal 0 HcmV?d00001 diff --git a/Documentation~/job_component_system.md b/Documentation~/job_component_system.md index 4b1651d1..4b32a22c 100644 --- a/Documentation~/job_component_system.md +++ b/Documentation~/job_component_system.md @@ -11,7 +11,7 @@ Managing dependencies is hard. This is why in `JobComponentSystem` we are doing public class RotationSpeedSystem : JobComponentSystem { [BurstCompile] - struct RotationSpeedRotation : IJobProcessComponentData + struct RotationSpeedRotation : IJobForEach { public float dt; @@ -41,10 +41,10 @@ Thus if a system writes to component `A`, and another system later on reads from ## Dependency management is conservative & deterministic -Dependency management is conservative. `ComponentSystem` simply tracks all `ComponentGroup`objects ever used and stores which types are being written or read based on that. +Dependency management is conservative. `ComponentSystem` simply tracks all `EntityQuery`objects ever used and stores which types are being written or read based on that. Also when scheduling multiple jobs in a single system, dependencies must be passed to all jobs even though different jobs may need less dependencies. If that proves to be a performance issue the best solution is to split a system into two. diff --git a/Documentation~/manual_iteration.md b/Documentation~/manual_iteration.md index a2f06f5b..87e7bc8c 100644 --- a/Documentation~/manual_iteration.md +++ b/Documentation~/manual_iteration.md @@ -1,6 +1,6 @@ # Manual iteration -You can also request all the chunks explicitly in a NativeArray and process them with a Job such as `IJobParallelFor`. This method is recommended if you need to manage chunks in some way that is not appropriate for the simplified model of simply iterating over all the Chunks in a ComponentGroup. As in: +You can also request all the chunks explicitly in a NativeArray and process them with a Job such as `IJobParallelFor`. This method is recommended if you need to manage chunks in some way that is not appropriate for the simplified model of simply iterating over all the Chunks in a EntityQuery. As in: ```c# public class RotationSpeedSystem : JobComponentSystem @@ -30,16 +30,16 @@ public class RotationSpeedSystem : JobComponentSystem } } - ComponentGroup m_group; + EntityQuery m_group; - protected override void OnCreateManager() + protected override void OnCreate() { - var query = new EntityArchetypeQuery + var query = new EntityQueryDesc { All = new ComponentType[]{ typeof(RotationQuaternion), ComponentType.ReadOnly() } }; - m_group = GetComponentGroup(query); + m_group = GetEntityQuery(query); } protected override JobHandle OnUpdate(JobHandle inputDeps) diff --git a/Documentation~/shared_component_data.md b/Documentation~/shared_component_data.md index d80a250c..d403ecfb 100644 --- a/Documentation~/shared_component_data.md +++ b/Documentation~/shared_component_data.md @@ -26,8 +26,8 @@ We use `ISharedComponentData` to group all entities using the same `InstanceRend ## Some important notes about SharedComponentData: - Entities with the same `SharedComponentData` are grouped together in the same [Chunks](chunk_iteration.md). The index to the `SharedComponentData` is stored once per `Chunk`, not per entity. As a result `SharedComponentData` have zero memory overhead on a per entity basis. -- Using `ComponentGroup` we can iterate over all entities with the same type. -- Additionally we can use `ComponentGroup.SetFilter()` to iterate specifically over entities that have a specific `SharedComponentData` value. Due to the data layout this iteration has low overhead. +- Using `EntityQuery` we can iterate over all entities with the same type. +- Additionally we can use `EntityQuery.SetFilter()` to iterate specifically over entities that have a specific `SharedComponentData` value. Due to the data layout this iteration has low overhead. - Using `EntityManager.GetAllUniqueSharedComponents` we can retrieve all unique `SharedComponentData` that is added to any alive entities. - `SharedComponentData` are automatically [reference counted](https://en.wikipedia.org/wiki/Reference_counting). - `SharedComponentData` should change rarely. Changing a `SharedComponentData` involves using [memcpy](https://msdn.microsoft.com/en-us/library/aa246468(v=vs.60).aspx) to copy all `ComponentData` for that entity into a different `Chunk`. diff --git a/Documentation~/system_update_order.md b/Documentation~/system_update_order.md index 580d7b5c..9ef8e5d0 100644 --- a/Documentation~/system_update_order.md +++ b/Documentation~/system_update_order.md @@ -89,6 +89,6 @@ For example, here’s the typical procedure of a custom `MyCustomBootstrap.Initi ## Tips and Best Practices * __Use [UpdateInGroup] to specify the system group for each system you write.__ If not specified, the implicit default group is SimulationSystemGroup. -* __Use manually-ticked ComponentSystemGroups to update systems elsewhere in the Unity player loop.__ Adding the [DisableAutoCreation] attribute to a component system (or system group) prevents it from being created or added to the default system groups. You can still manually create the system with World.GetOrCreateManager() and update it by calling manually calling MySystem.Update() from the main thread. This is an easy way to insert systems elsewhere in the Unity player loop (for example, if you have a system that should run later or earlier in the frame). +* __Use manually-ticked ComponentSystemGroups to update systems elsewhere in the Unity player loop.__ Adding the [DisableAutoCreation] attribute to a component system (or system group) prevents it from being created or added to the default system groups. You can still manually create the system with World.GetOrCreateSystem() and update it by calling manually calling MySystem.Update() from the main thread. This is an easy way to insert systems elsewhere in the Unity player loop (for example, if you have a system that should run later or earlier in the frame). * __Use the existing `EntityCommandBufferSystem`s instead of adding new ones, if possible.__ An `EntityCommandBufferSystem` represents a sync point where the main thread waits for worker threads to complete before processing any outstanding `EntityCommandBuffer`s. Reusing one of the predefined Begin/End systems in each root-level system group is less likely to introduce a new "bubble" into the frame pipeline than creating a new one. * __Avoid putting custom logic in `ComponentSystemGroup.OnUpdate()`__. Since `ComponentSystemGroup` is functionally a component system itself, it may be tempting to add custom processing to its OnUpdate() method, to perform some work, spawn some jobs, etc. We advise against this in general, as it’s not immediately clear from the outside whether the custom logic is executed before or after the group’s members are updated. It’s preferable to keep system groups limited to a grouping mechanism, and to implement the desired logic in a separate component system, explicitly ordered relative to the group. diff --git a/Documentation~/toc.yml b/Documentation~/toc.yml index 5f5d675b..90207383 100644 --- a/Documentation~/toc.yml +++ b/Documentation~/toc.yml @@ -41,7 +41,7 @@ - name: Accessing Entity Data href: chunk_iteration.md items: - - name: Using IJobProcessComponentData + - name: Using IJobForEach href: entity_iteration_job.md - name: Using IJobChunk href: chunk_iteration_job.md diff --git a/Documentation~/transform_system.md b/Documentation~/transform_system.md index 8258b483..61a9bb68 100644 --- a/Documentation~/transform_system.md +++ b/Documentation~/transform_system.md @@ -41,11 +41,8 @@ e.g. If the following components are present... - [TRSToLocalToWorldSystem] Write LocalToWorld <= Translation * Rotation -``` -┌─────────────┐ ┌──────────────┐ ┌──────────┐ -│ Translation │ ──> │ LocalToWorld │ <── │ Rotation │ -└─────────────┘ └──────────────┘ └──────────┘ -``` +![](images/sec1-1.png) + Or, if the following components are present... | (Entity) | @@ -59,17 +56,8 @@ Or, if the following components are present... - [TRSToLocalToWorldSystem] Write LocalToWorld <= Translation * Rotation * Scale -``` - ┌──────────────┐ - │ Scale │ - └──────────────┘ - │ - │ - ∨ -┌─────────────┐ ┌──────────────┐ ┌──────────┐ -│ Translation │ ──> │ LocalToWorld │ <── │ Rotation │ -└─────────────┘ └──────────────┘ └──────────┘ -``` +![](images/sec1-2.png) + ------------------------------------------ Section 2: Hierarchical Transforms (Basic) ------------------------------------------ @@ -94,25 +82,7 @@ e.g. If the following components are present... 1. [TRSToLocalToWorldSystem] Parent: Write LocalToWorld as defined above in "Non-hierarchical Transforms (Basic)" 2. [LocalToParentSystem] Child: Write LocalToWorld <= LocalToWorld[Parent] * LocalToParent -``` ------- Parent: ------ - - ┌──────────────┐ - │ Scale │ - └──────────────┘ - │ - │ - ∨ -┌─────────────┐ ┌──────────────┐ ┌──────────┐ -│ Translation │ ──> │ LocalToWorld │ <── │ Rotation │ -└─────────────┘ └──────────────┘ └──────────┘ - ------- Child: ------ - -┌──────────────────────┐ ┌──────────────┐ ┌───────────────┐ -│ LocalToWorld[Parent] │ ──> │ LocalToWorld │ <── │ LocalToParent │ -└──────────────────────┘ └──────────────┘ └───────────────┘ -``` +![](images/sec2-1.png) LocalToWorld components associated with Parent Entity IDs are guaranteed to be computed before multiplies with LocalToParent associated with Child Entity ID. @@ -164,37 +134,7 @@ e.g. If the following components are present... 2. [TRSToLocalToParentSystem] Child: Write LocalToParent <= Translation * Rotation * Scale 3. [LocalToParentSystem] Child: Write LocalToWorld <= LocalToWorld[Parent] * LocalToParent -``` ------- Parent: ------ - - ┌──────────────┐ - │ Scale │ - └──────────────┘ - │ - │ - ∨ -┌─────────────┐ ┌──────────────┐ ┌──────────┐ -│ Translation │ ──> │ LocalToWorld │ <── │ Rotation │ -└─────────────┘ └──────────────┘ └──────────┘ - ------- Child: ------ - - ┌───────────────┐ ┌──────────────────────┐ - │ LocalToWorld │ <── │ LocalToWorld[Parent] │ - └───────────────┘ └──────────────────────┘ - ∧ - │ - │ -┌──────────┐ ┌───────────────┐ ┌──────────────────────┐ -│ Rotation │ ──> │ LocalToParent │ <── │ Translation │ -└──────────┘ └───────────────┘ └──────────────────────┘ - ∧ - │ - │ - ┌───────────────┐ - │ Scale │ - └───────────────┘ -``` +![](images/sec2-2.png) Parents may of course themselves be children of other LocalToWorld components. @@ -218,43 +158,7 @@ e.g. If the following components are present... 3. [LocalToParentSystem] Parent: Write LocalToWorld <= LocalToWorld[Parent] * LocalToParent 4. [LocalToParentSystem] Child: Write LocalToWorld <= LocalToWorld[Parent] * LocalToParent -``` ------- Parent: ------ - - ┌───────────────┐ ┌──────────────────────┐ - │ LocalToWorld │ <── │ LocalToWorld[Parent] │ - └───────────────┘ └──────────────────────┘ - ∧ - │ - │ -┌──────────┐ ┌───────────────┐ ┌──────────────────────┐ -│ Rotation │ ──> │ LocalToParent │ <── │ Translation │ -└──────────┘ └───────────────┘ └──────────────────────┘ - ∧ - │ - │ - ┌───────────────┐ - │ Scale │ - └───────────────┘ - ------- Child: ------ - - ┌───────────────┐ ┌──────────────────────┐ - │ LocalToWorld │ <── │ LocalToWorld[Parent] │ - └───────────────┘ └──────────────────────┘ - ∧ - │ - │ -┌──────────┐ ┌───────────────┐ ┌──────────────────────┐ -│ Rotation │ ──> │ LocalToParent │ <── │ Translation │ -└──────────┘ └───────────────┘ └──────────────────────┘ - ∧ - │ - │ - ┌───────────────┐ - │ Scale │ - └───────────────┘ -``` +![](images/sec2-3.png) ------------------------------------- Section 3: Default Conversion (Basic) @@ -304,17 +208,7 @@ e.g. If the following components are present... - [TRSToLocalToWorldSystem] Write LocalToWorld <= Translation * Rotation * NonUniformScale -``` - ┌──────────────┐ - │ Rotation │ - └──────────────┘ - │ - │ - ∨ -┌─────────────┐ ┌──────────────┐ ┌─────────────────┐ -│ Translation │ ──> │ LocalToWorld │ <── │ NonUniformScale │ -└─────────────┘ └──────────────┘ └─────────────────┘ -``` +![](images/sec4-1.png) The Rotation component may be written to directly as a quaternion by user code. However, if an Euler interface is preferred, components are available for each rotation order which will cause a write to the Rotation component if present. @@ -339,17 +233,7 @@ e.g. If the following components are present... 1. [RotationEulerSystem] Write Rotation <= RotationEulerXYZ 2. [TRSToLocalToWorldSystem] Write LocalToWorld <= Translation * Rotation * Scale -``` - ┌──────────────┐ - │ Scale │ - └──────────────┘ - │ - │ - ∨ -┌─────────────┐ ┌──────────────┐ ┌──────────┐ ┌──────────────────┐ -│ Translation │ ──> │ LocalToWorld │ <── │ Rotation │ <── │ RotationEulerXYZ │ -└─────────────┘ └──────────────┘ └──────────┘ └──────────────────┘ -``` +![](images/sec4-2.png) It is a setup error to have more than one RotationEuler*** component is associated with the same Entity, however the result is defined. The first to be found in the order of precedence will be applied. That order is: @@ -417,26 +301,7 @@ e.g. If the following components are present... 1. [CompositeRotationSystem] Write CompositeRotation <= RotationPivotTranslation * RotationPivot * Rotation * PostRotation * RotationPivot^-1 2. [TRSToLocalToWorldSystem] Write LocalToWorld <= Translation * CompositeRotation * Scale -``` - ┌──────────────────────────┐ ┌───────────────────┐ - │ Scale │ │ RotationPivot │ - └──────────────────────────┘ └───────────────────┘ - │ │ - │ │ - ∨ ∨ -┌─────────────┐ ┌──────────────────────────┐ ┌───────────────────┐ ┌──────────────────┐ -│ Translation │ ──> │ LocalToWorld │ <── │ │ <── │ PostRotation │ -└─────────────┘ └──────────────────────────┘ │ CompositeRotation │ └──────────────────┘ - ┌──────────────────────────┐ │ │ ┌──────────────────┐ - │ RotationPivotTranslation │ ──> │ │ │ RotationEulerXYZ │ - └──────────────────────────┘ └───────────────────┘ └──────────────────┘ - ∧ │ - │ │ - │ ∨ - │ ┌──────────────────┐ - └────────────────────── │ Rotation │ - └──────────────────┘ -``` +![](images/sec4-3.png) The PostRotation component may be written to directly as a quaternion by user code. However, if an Euler interface is preferred, components are available for each rotation order which will cause a write to the PostRotation component if present. @@ -469,20 +334,7 @@ e.g. If the following components are present... 3. [CompositeRotationSystem] Write CompositeRotation <= RotationPivotTranslation * RotationPivot * Rotation * PostRotation * RotationPivot^-1 4. [TRSToLocalToWorldSystem] Write LocalToWorld <= Translation * CompositeRotation * Scale -``` - ┌──────────────────────────┐ ┌───────────────────┐ - │ Scale │ │ RotationPivot │ - └──────────────────────────┘ └───────────────────┘ - │ │ - │ │ - ∨ ∨ -┌─────────────┐ ┌──────────────────────────┐ ┌───────────────────┐ ┌──────────────┐ ┌──────────────────────┐ -│ Translation │ ──> │ LocalToWorld │ <── │ │ <── │ PostRotation │ <── │ PostRotationEulerXYZ │ -└─────────────┘ └──────────────────────────┘ │ CompositeRotation │ └──────────────┘ └──────────────────────┘ - ┌──────────────────────────┐ │ │ ┌──────────────┐ ┌──────────────────────┐ - │ RotationPivotTranslation │ ──> │ │ <── │ Rotation │ <── │ RotationEulerXYZ │ - └──────────────────────────┘ └───────────────────┘ └──────────────┘ └──────────────────────┘ -``` +![](images/sec4-4.png) For more complex Scale requirements, a CompositeScale (float4x4) component may be used as an alternative to Scale (or NonUniformScale). @@ -543,38 +395,7 @@ e.g. If the following components are present... 4. [CompositeRotationSystem] Write CompositeRotation <= RotationPivotTranslation * RotationPivot * Rotation * PostRotation * RotationPivot^-1 5. [TRSToLocalToWorldSystem] Write LocalToWorld <= Translation * CompositeRotation * CompositeScale -``` - ┌───────────────────┐ - │ RotationPivot │ - └───────────────────┘ - │ - │ - ∨ -┌─────────────┐ ┌──────────────────────────┐ ┌───────────────────┐ ┌───────────────────────┐ ┌──────────────────────┐ -│ Translation │ ──> │ LocalToWorld │ <── │ │ <── │ PostRotation │ <── │ PostRotationEulerXYZ │ -└─────────────┘ └──────────────────────────┘ │ │ └───────────────────────┘ └──────────────────────┘ - ∧ │ │ - ┌────┘ │ CompositeRotation │ - │ │ │ - │ ┌──────────────────────────┐ │ │ ┌───────────────────────┐ ┌──────────────────────┐ - │ │ RotationPivotTranslation │ ──> │ │ <── │ Rotation │ <── │ RotationEulerXYZ │ - │ └──────────────────────────┘ └───────────────────┘ └───────────────────────┘ └──────────────────────┘ - │ - │ ┌─────────────────────────────────────────────────────┐ - │ ∨ │ - │ ┌───────────────────────┐ ┌──────────────────────┐ │ - └──────────────────────────────────────────────────────────── │ CompositeScale │ <── │ Scale │ │ - └───────────────────────┘ └──────────────────────┘ │ - ∧ │ - │ │ - │ │ - ┌───────────────────────┐ │ - │ ScalePivotTranslation │ │ - └───────────────────────┘ │ - ┌───────────────────────┐ │ - │ ScalePivot │ ──────────────────────────────┘ - └───────────────────────┘ -``` +![](images/sec4-5.png) --------------------------------------------- Section 5: Hierarchical Transforms (Advanced) @@ -613,37 +434,7 @@ e.g. If the following components are present... 2. [TRSToLocalToParentSystem] Child: Write LocalToParent <= Translation * Rotation * NonUniformScale 3. [LocalToParentSystem] Child: Write LocalToWorld <= LocalToWorld[Parent] * LocalToParent -``` ------- Parent: ------ - - ┌──────────────┐ - │ Scale │ - └──────────────┘ - │ - │ - ∨ -┌─────────────┐ ┌──────────────┐ ┌──────────┐ -│ Translation │ ──> │ LocalToWorld │ <── │ Rotation │ -└─────────────┘ └──────────────┘ └──────────┘ - ------- Child: ------ - - ┌───────────────┐ ┌──────────────────────┐ - │ LocalToWorld │ <── │ LocalToWorld[Parent] │ - └───────────────┘ └──────────────────────┘ - ∧ - │ - │ -┌─────────────────┐ ┌───────────────┐ ┌──────────────────────┐ -│ NonUniformScale │ ──> │ LocalToParent │ <── │ Translation │ -└─────────────────┘ └───────────────┘ └──────────────────────┘ - ∧ - │ - │ - ┌───────────────┐ - │ Rotation │ - └───────────────┘ -``` +![](images/sec5-1.png) Parent LocalToWorld is multiplied with the Child LocalToWorld, which includes any scaling. However, if removing Parent scale is preferred (AKA Scale Compensate), ParentScaleInverse is available for that purpose. @@ -693,37 +484,7 @@ e.g. If the following components are present... 3. [TRSToLocalToParentSystem] Child: Write LocalToParent <= Translation * ParentScaleInverse * Rotation 4. [LocalToParentSystem] Child: Write LocalToWorld <= LocalToWorld[Parent] * LocalToParent -``` ------- Parent: ------ - - ┌──────────────┐ - │ Scale │ - └──────────────┘ - │ - │ - ∨ -┌─────────────┐ ┌──────────────┐ ┌──────────┐ -│ Translation │ ──> │ LocalToWorld │ <── │ Rotation │ -└─────────────┘ └──────────────┘ └──────────┘ - ------- Child: ------ - - ┌────────────────────┐ ┌──────────────────────┐ - │ LocalToWorld │ <── │ LocalToWorld[Parent] │ - └────────────────────┘ └──────────────────────┘ - ∧ - │ - │ -┌──────────┐ ┌────────────────────┐ ┌──────────────────────┐ -│ Rotation │ ──> │ LocalToParent │ <── │ Translation │ -└──────────┘ └────────────────────┘ └──────────────────────┘ - ∧ - │ - │ - ┌────────────────────┐ ┌──────────────────────┐ - │ ParentScaleInverse │ <── │ Scale[Parent] │ - └────────────────────┘ └──────────────────────┘ -``` +![](images/sec5-2.png) The Rotation component may be written to directly as a quaternion by user code. However, if an Euler interface is preferred, components are available for each rotation order which will cause a write to the Rotation component if present. @@ -753,37 +514,7 @@ e.g. If the following components are present... 3. [TRSToLocalToParentSystem] Child: Write LocalToParent <= Translation * Rotation 4. [LocalToParentSystem] Child: Write LocalToWorld <= LocalToWorld[Parent] * LocalToParent -``` ------- Parent: ------ - - ┌──────────────┐ - │ Scale │ - └──────────────┘ - │ - │ - ∨ -┌─────────────┐ ┌──────────────┐ ┌──────────┐ -│ Translation │ ──> │ LocalToWorld │ <── │ Rotation │ -└─────────────┘ └──────────────┘ └──────────┘ - ------- Child: ------ - -┌───────────────┐ ┌──────────────────────┐ -│ LocalToWorld │ <── │ LocalToWorld[Parent] │ -└───────────────┘ └──────────────────────┘ - ∧ - │ - │ -┌───────────────┐ ┌──────────────────────┐ -│ LocalToParent │ <── │ Translation │ -└───────────────┘ └──────────────────────┘ - ∧ - │ - │ -┌───────────────┐ ┌──────────────────────┐ -│ Rotation │ <── │ RotationEulerXYZ │ -└───────────────┘ └──────────────────────┘ -``` +![](images/sec5-3.png) For more complex Rotation requirements, a CompositeRotation (float4x4) component may be used as an alternative to Rotation. @@ -856,51 +587,7 @@ e.g. If the following components are present... 4. [TRSToLocalToParentSystem] Child: Write LocalToParent <= Translation * CompositeRotation * Scale 5. [LocalToParentSystem] Child: Write LocalToWorld <= LocalToWorld[Parent] * LocalToParent -``` ------- Parent: ------ - - ┌──────────────┐ - │ Scale │ - └──────────────┘ - │ - │ - ∨ -┌─────────────┐ ┌──────────────┐ ┌──────────┐ -│ Translation │ ──> │ LocalToWorld │ <── │ Rotation │ -└─────────────┘ └──────────────┘ └──────────┘ - ------- Child: ------ - ┌───────────────────┐ ┌──────────────────────┐ - │ LocalToWorld │ <── │ LocalToWorld[Parent] │ - └───────────────────┘ └──────────────────────┘ - ∧ - │ - │ -┌──────────────────────────┐ ┌───────────────────┐ ┌──────────────────────┐ -│ Scale │ ──> │ │ <── │ Translation │ -└──────────────────────────┘ │ │ └──────────────────────┘ - │ │ - │ LocalToParent │ <─────────────────────────────┐ - │ │ │ - │ │ ┌──────────────────────┐ │ - │ │ │ RotationEulerXYZ │ │ - └───────────────────┘ └──────────────────────┘ │ - ∧ │ │ - │ │ │ - │ ∨ │ -┌──────────────────────────┐ ┌───────────────────┐ ┌──────────────────────┐ │ -│ RotationPivotTranslation │ ──> │ │ <── │ Rotation │ │ -└──────────────────────────┘ │ CompositeRotation │ └──────────────────────┘ │ -┌──────────────────────────┐ │ │ ┌──────────────────────┐ │ -│ PostRotation │ ──> │ │ │ Scale[Parent] │ │ -└──────────────────────────┘ └───────────────────┘ └──────────────────────┘ │ - ∧ │ │ - │ │ │ - │ ∨ │ - ┌───────────────────┐ ┌──────────────────────┐ │ - │ RotationPivot │ │ ParentScaleInverse │ ─┘ - └───────────────────┘ └──────────────────────┘ -``` +![](images/sec5-4.png) The PostRotation component may be written to directly as a quaternion by user code. However, if an Euler interface is preferred, components are available for each rotation order which will cause a write to the PostRotation component if present. @@ -938,58 +625,7 @@ e.g. If the following components are present... 5. [TRSToLocalToParentSystem] Child: Write LocalToParent <= Translation * CompositeRotation * Scale 6. [LocalToParentSystem] Child: Write LocalToWorld <= LocalToWorld[Parent] * LocalToParent -``` ------- Parent: ------ - - ┌──────────────┐ - │ Scale │ - └──────────────┘ - │ - │ - ∨ -┌─────────────┐ ┌──────────────┐ ┌──────────┐ -│ Translation │ ──> │ LocalToWorld │ <── │ Rotation │ -└─────────────┘ └──────────────┘ └──────────┘ - ------- Child: ------ - - ┌────────────────────┐ ┌──────────────────────┐ - │ LocalToWorld │ <── │ LocalToWorld[Parent] │ - └────────────────────┘ └──────────────────────┘ - ∧ - │ - │ -┌──────────────────────────┐ ┌────────────────────┐ ┌──────────────────────┐ -│ Scale │ ──> │ │ <── │ Translation │ -└──────────────────────────┘ │ │ └──────────────────────┘ - │ │ - │ LocalToParent │ <─────────────────────────────┐ - │ │ │ - │ │ ┌──────────────────────┐ │ - │ │ │ PostRotationEulerXYZ │ │ - └────────────────────┘ └──────────────────────┘ │ - ∧ │ │ - │ │ │ - │ ∨ │ -┌──────────────────────────┐ ┌────────────────────┐ ┌──────────────────────┐ │ -│ RotationPivotTranslation │ ──> │ │ <── │ PostRotation │ │ -└──────────────────────────┘ │ CompositeRotation │ └──────────────────────┘ │ -┌──────────────────────────┐ │ │ ┌──────────────────────┐ │ -│ RotationPivot │ ──> │ │ │ RotationEulerXYZ │ │ -└──────────────────────────┘ └────────────────────┘ └──────────────────────┘ │ - ∧ │ │ - │ │ │ - │ ∨ │ - │ ┌──────────────────────┐ │ - └─────────────────────── │ Rotation │ │ - └──────────────────────┘ │ - │ - ┌──────────────────────────────────────────────────┘ - │ - ┌────────────────────┐ ┌──────────────────────┐ - │ ParentScaleInverse │ <── │ Scale[Parent] │ - └────────────────────┘ └──────────────────────┘ -``` +![](images/sec5-5.png) It is a setup error to have more than one PostRotationEuler*** component is associated with the same Entity, however the result is defined. The first to be found in the order of precedence will be applied. That order is: @@ -1068,19 +704,8 @@ e.g. If the following components are present... 4. [TRSToLocalToParentSystem] Child: Write LocalToParent <= Translation * CompositeRotation * Scale 5. [LocalToParentSystem] Child: Write LocalToWorld <= LocalToWorld[Parent] * LocalToParent -``` ------- Parent: ------ - - ┌──────────────┐ - │ Scale │ - └──────────────┘ - │ - │ - ∨ -┌─────────────┐ ┌──────────────┐ ┌──────────┐ -│ Translation │ ──> │ LocalToWorld │ <── │ Rotation │ -└─────────────┘ └──────────────┘ └──────────┘ -``` +![](images/sec5-6.png) + ...then the transform system will: 1. [TRSToLocalToWorldSystem] Parent: Write LocalToWorld as defined above in "Non-hierarchical Transforms (Basic)" @@ -1091,56 +716,7 @@ e.g. If the following components are present... 6. [TRSToLocalToParentSystem] Child: Write LocalToParent <= Translation * CompositeRotation * Scale 7. [LocalToParentSystem] Child: Write LocalToWorld <= LocalToWorld[Parent] * LocalToParent -``` - ┌───────────────────┐ ┌──────────────────────┐ - │ LocalToWorld │ <── │ LocalToWorld[Parent] │ - └───────────────────┘ └──────────────────────┘ - ∧ - │ - │ - ┌───────────────────┐ ┌──────────────────────┐ - │ │ <── │ Translation │ - │ │ └──────────────────────┘ - │ │ - │ LocalToParent │ <─────────────────────────────┐ - │ │ │ - │ │ ┌──────────────────────┐ │ - ┌─────────────────────────────────> │ │ │ PostRotationEulerXYZ │ │ - │ └───────────────────┘ └──────────────────────┘ │ - │ ∧ │ │ - │ │ │ │ - │ │ ∨ │ - │ ┌──────────────────────────┐ ┌───────────────────┐ ┌──────────────────────┐ │ - │ │ RotationPivotTranslation │ ──> │ │ <── │ PostRotation │ │ - │ └──────────────────────────┘ │ CompositeRotation │ └──────────────────────┘ │ - │ ┌──────────────────────────┐ │ │ ┌──────────────────────┐ │ - │ │ RotationPivot │ ──> │ │ │ RotationEulerXYZ │ │ - │ └──────────────────────────┘ └───────────────────┘ └──────────────────────┘ │ - │ ∧ │ │ - │ │ │ │ - │ │ ∨ │ - │ │ ┌──────────────────────┐ │ - │ └────────────────────── │ Rotation │ │ - │ └──────────────────────┘ │ - │ │ - │ ┌─────────────────────────────────────────────────┘ - │ │ - │ ┌──────────────────────────┐ ┌───────────────────┐ ┌──────────────────────┐ - │ │ ScalePivotTranslation │ ──> │ CompositeScale │ <── │ Scale │ - │ └──────────────────────────┘ └───────────────────┘ └──────────────────────┘ - │ ∧ - │ │ - │ │ - │ ┌───────────────────┐ ┌──────────────────────┐ - │ │ ScalePivot │ │ Scale[Parent] │ - │ └───────────────────┘ └──────────────────────┘ - │ │ - │ │ - │ ∨ - │ ┌──────────────────────┐ - └──────────────────────────────────────────────────────────── │ ParentScaleInverse │ - └──────────────────────┘ -``` +![](images/sec5-7.png) --------------------------------------- Section 6: Custom Transforms (Advanced) @@ -1171,7 +747,7 @@ e.g. public class UserTransformSystem : JobComponentSystem { [BurstCompile] - struct UserTransform : IJobProcessComponentData + struct UserTransform : IJobForEach { public void Execute(ref LocalToWorld localToWorld, [ReadOnly] ref UserComponent userComponent) { @@ -1221,7 +797,7 @@ And the equivalent system: public class UserTransformSystem2 : JobComponentSystem { [BurstCompile] - struct UserTransform2 : IJobProcessComponentData + struct UserTransform2 : IJobForEach { public void Execute(ref LocalToWorld localToWorld, [ReadOnly] ref UserComponent2 userComponent2) { @@ -1269,23 +845,23 @@ And a system which filters based on the WriteGroup of LocalToWorld: public class UserTransformSystem : JobComponentSystem { - private ComponentGroup m_Group; + private EntityQuery m_Group; - protected override void OnCreateManager() + protected override void OnCreate() { - m_Group = GetComponentGroup(new EntityArchetypeQuery() + m_Group = GetEntityQuery(new EntityQueryDesc() { All = new ComponentType[] { ComponentType.ReadWrite(), ComponentType.ReadOnly(), }, - Options = EntityArchetypeQueryOptions.FilterWriteGroup + Options = EntityQueryDescOptions.FilterWriteGroup }); } [BurstCompile] - struct UserTransform : IJobProcessComponentData + struct UserTransform : IJobForEach { public void Execute(ref LocalToWorld localToWorld, [ReadOnly] ref UserComponent userComponent) { @@ -1304,7 +880,7 @@ And a system which filters based on the WriteGroup of LocalToWorld: m_Group in UserTransformSystem will only match the explicitly mentioned components. -For instance, the following with match and be included in the ComponentGroup: +For instance, the following with match and be included in the EntityQuery: | (Entity) | | -------------- | @@ -1327,11 +903,11 @@ However, they may be explicitly supported by UserComponent systems by adding to public class UserTransformExtensionSystem : JobComponentSystem { - private ComponentGroup m_Group; + private EntityQuery m_Group; - protected override void OnCreateManager() + protected override void OnCreate() { - m_Group = GetComponentGroup(new EntityArchetypeQuery() + m_Group = GetEntityQuery(new EntityQueryDesc() { All = new ComponentType[] { @@ -1341,12 +917,12 @@ However, they may be explicitly supported by UserComponent systems by adding to ComponentType.ReadOnly(), ComponentType.ReadOnly(), }, - Options = EntityArchetypeQueryOptions.FilterWriteGroup + Options = EntityQueryDescOptions.FilterWriteGroup }); } [BurstCompile] - struct UserTransform : IJobProcessComponentData + struct UserTransform : IJobForEach { public void Execute(ref LocalToWorld localToWorld, [ReadOnly] ref UserComponent userComponent, [ReadOnly] ref Translation translation, @@ -1388,11 +964,11 @@ However, an explicit query can be created which can resolve the case and ensure public class UserTransformComboSystem : JobComponentSystem { - private ComponentGroup m_Group; + private EntityQuery m_Group; - protected override void OnCreateManager() + protected override void OnCreate() { - m_Group = GetComponentGroup(new EntityArchetypeQuery() + m_Group = GetEntityQuery(new EntityQueryDesc() { All = new ComponentType[] { @@ -1400,12 +976,12 @@ However, an explicit query can be created which can resolve the case and ensure ComponentType.ReadOnly(), ComponentType.ReadOnly(), }, - Options = EntityArchetypeQueryOptions.FilterWriteGroup + Options = EntityQueryDescOptions.FilterWriteGroup }); } [BurstCompile] - struct UserTransform : IJobProcessComponentData + struct UserTransform : IJobForEach { public void Execute(ref LocalToWorld localToWorld, [ReadOnly] ref UserComponent userComponent, diff --git a/Documentation~/version_numbers.md b/Documentation~/version_numbers.md index 927981c9..108e7813 100644 --- a/Documentation~/version_numbers.md +++ b/Documentation~/version_numbers.md @@ -48,7 +48,7 @@ For each component type in the archetype, this array contains the value of `Enti Shared components can never be accessed as writeable, even if there is technically a version number stored for those too, it serves no purpose. -When using the `[ChangedFilter]` attribute in an `IJobProcessComponentData`, the `Chunk.ChangeVersion` for that specific component is compared to `System.LastSystemVersion`, so only chunks whose component arrays have been accessed as writeable since after the system last started running will be processed. +When using the `[ChangedFilter]` attribute in an `IJobForEach`, the `Chunk.ChangeVersion` for that specific component is compared to `System.LastSystemVersion`, so only chunks whose component arrays have been accessed as writeable since after the system last started running will be processed. > If the amount of health points of a group of units is guaranteed not to have changed since the previous frame, checking if those units should update their damage model can be skipped altogether. diff --git a/Unity.Entities.BuildUtils/MonoExtensions.cs b/Unity.Entities.BuildUtils/MonoExtensions.cs index 54f13b09..e7fd5d48 100644 --- a/Unity.Entities.BuildUtils/MonoExtensions.cs +++ b/Unity.Entities.BuildUtils/MonoExtensions.cs @@ -91,6 +91,48 @@ public static bool IsEntityType(this TypeReference typeRef) return (typeRef.FullName == "Unity.Entities.Entity"); } + public static bool IsManagedType(this TypeReference typeRef) + { + // We must check this before calling Resolve() as cecil loses this property otherwise + if (typeRef.IsPointer) + return false; + + if (typeRef.IsArray) + return true; + + var type = typeRef.Resolve(); + + if (type.IsDynamicArray()) + return true; + + TypeDefinition fixedSpecialType = type.FixedSpecialType(); + if (fixedSpecialType != null) + { + if (fixedSpecialType.MetadataType == MetadataType.String) + return true; + return false; + } + + if (type.IsEnum) + return false; + + if (type.IsValueType) + { + // if none of the above check the type's fields + foreach (var field in type.Fields) + { + if (field.IsStatic) + continue; + + if (field.FieldType.IsManagedType()) + return true; + } + + return false; + } + + return true; + } public static bool IsComplex(this TypeReference typeRef) diff --git a/Unity.Entities.BuildUtils/TypeUtils.cs b/Unity.Entities.BuildUtils/TypeUtils.cs index 7825d14d..6231fc89 100644 --- a/Unity.Entities.BuildUtils/TypeUtils.cs +++ b/Unity.Entities.BuildUtils/TypeUtils.cs @@ -267,7 +267,7 @@ public static void PreprocessTypeFields(TypeDefinition valuetype, int bits) ValueTypeIsComplex[bits].Add(valuetype, isComplex); } - internal static void GetEntityFieldOffsets(int offset, TypeDefinition type, List l, int bits) + internal static void GetFieldOffsetsOfRecurse(string queryFullTypeName, int offset, TypeDefinition type, List l, int bits) { int in_type_offset = 0; foreach (var f in type.Fields) @@ -284,34 +284,51 @@ internal static void GetEntityFieldOffsets(int offset, TypeDefinition type, List var tinfo = resize(TypeUtils.AlignAndSizeOfType(f.FieldType, bits)); in_type_offset = (int)alignUp((uint)in_type_offset, (uint)tinfo.align); - if (f.FieldType.IsDynamicArray() && f.FieldType.DynamicArrayElementType().IsEntityType()) + if (f.FieldType.IsDynamicArray() && f.FieldType.DynamicArrayElementType().Resolve().FullName == queryFullTypeName) { // +1 so that we have a way to indicate an Array at position 0 // fixup code subtracts 1 l.Add(-(offset + in_type_offset + 1)); } - else if (f.FieldType.IsEntityType()) + else if (f.FieldType.Resolve().FullName == queryFullTypeName) { l.Add(offset + in_type_offset); } else if (f.FieldType.IsValueType && !f.FieldType.IsPrimitive) { - GetEntityFieldOffsets(offset + in_type_offset, f.FieldType.Resolve(), l, bits); + GetFieldOffsetsOfRecurse(queryFullTypeName, offset + in_type_offset, f.FieldType.Resolve(), l, bits); } in_type_offset += tinfo.size; } } - public static List GetEntityFieldOffsets(TypeDefinition type, int archBits) + public static List GetFieldOffsetsOf(TypeReference typeToFind, TypeDefinition typeToLookIn, int archBits) + { + var offsets = new List(); + + if (typeToLookIn != null) + { + GetFieldOffsetsOfRecurse(typeToFind.FullName, 0, typeToLookIn, offsets, archBits); + } + + return offsets; + } + + public static List GetFieldOffsetsOf(string queryFullTypeName, TypeDefinition typeToLookIn, int archBits) { var offsets = new List(); - if (type != null) + if (typeToLookIn != null) { - GetEntityFieldOffsets(0, type, offsets, archBits); + GetFieldOffsetsOfRecurse(queryFullTypeName, 0, typeToLookIn, offsets, archBits); } return offsets; } + + public static List GetEntityFieldOffsets(TypeDefinition type, int archBits) + { + return GetFieldOffsetsOf("Unity.Entities.Entity", type, archBits); + } } } diff --git a/Unity.Entities.Editor.Tests/ComponentGroupGUITests.cs b/Unity.Entities.Editor.Tests/ComponentGroupGUITests.cs index 5dd9c3d5..2ea12550 100644 --- a/Unity.Entities.Editor.Tests/ComponentGroupGUITests.cs +++ b/Unity.Entities.Editor.Tests/ComponentGroupGUITests.cs @@ -15,14 +15,14 @@ public class ComponentGroupGUITests [Test] public void ComponentGroupGUI_SpecifiedTypeName_NestedTypeInGeneric() { - var typeName = ComponentGroupGUI.SpecifiedTypeName(typeof(GenericClassTest.InternalClass)); + var typeName = EntityQueryGUI.SpecifiedTypeName(typeof(GenericClassTest.InternalClass)); Assert.AreEqual("GenericClassTest.InternalClass", typeName); } [Test] public void ComponentGroupGUI_SpecifiedTypeName_NestedGenericTypeInGeneric() { - var typeName = ComponentGroupGUI.SpecifiedTypeName(typeof(GenericClassTest.InternalGenericClass)); + var typeName = EntityQueryGUI.SpecifiedTypeName(typeof(GenericClassTest.InternalGenericClass)); Assert.AreEqual("GenericClassTest.InternalGenericClass", typeName); } } diff --git a/Unity.Entities.Editor.Tests/ComponentTypeFilterUITests.cs b/Unity.Entities.Editor.Tests/ComponentTypeFilterUITests.cs index d4e8af4c..e3763850 100644 --- a/Unity.Entities.Editor.Tests/ComponentTypeFilterUITests.cs +++ b/Unity.Entities.Editor.Tests/ComponentTypeFilterUITests.cs @@ -39,9 +39,9 @@ public void ComponentTypeFilterUI_ComponentGroupCaches() var filterUI = new ComponentTypeFilterUI(SetFilterDummy, WorldSelectionGetter); var types = new ComponentType[] {ComponentType.ReadWrite(), ComponentType.ReadOnly()}; - Assert.IsNull(filterUI.GetExistingGroup(types)); - var group = filterUI.GetComponentGroup(types); - Assert.AreEqual(group, filterUI.GetExistingGroup(types)); + Assert.IsNull(filterUI.GetExistingQuery(types)); + var group = filterUI.GetEntityQuery(types); + Assert.AreEqual(group, filterUI.GetExistingQuery(types)); } } } diff --git a/Unity.Entities.Editor.Tests/EntityArrayListAdapterTests.cs b/Unity.Entities.Editor.Tests/EntityArrayListAdapterTests.cs index f8363348..bd5fb09b 100644 --- a/Unity.Entities.Editor.Tests/EntityArrayListAdapterTests.cs +++ b/Unity.Entities.Editor.Tests/EntityArrayListAdapterTests.cs @@ -19,14 +19,14 @@ public override void Setup() m_Manager.CreateEntity(archetype, entities); } - var query = new EntityArchetypeQuery() + var query = new EntityQueryDesc() { Any = new ComponentType[0], All = new ComponentType[0], None = new ComponentType[0] }; - var group = m_Manager.CreateComponentGroup(query); + var group = m_Manager.CreateEntityQuery(query); m_ChunkArray = group.CreateArchetypeChunkArray(Allocator.TempJob); } diff --git a/Unity.Entities.Editor.Tests/EntityDebuggerTests.cs b/Unity.Entities.Editor.Tests/EntityDebuggerTests.cs index 462c87e8..721bd799 100644 --- a/Unity.Entities.Editor.Tests/EntityDebuggerTests.cs +++ b/Unity.Entities.Editor.Tests/EntityDebuggerTests.cs @@ -11,7 +11,7 @@ class EntityDebuggerTests : ECSTestsFixture private EntityDebugger m_Window; private ComponentSystem m_System; - private ComponentGroup m_ComponentGroup; + private EntityQuery entityQuery; private Entity m_Entity; [DisableAutoCreation] @@ -23,9 +23,9 @@ protected override void OnUpdate() throw new NotImplementedException(); } - protected override void OnCreateManager() + protected override void OnCreate() { - GetComponentGroup(typeof(EcsTestData)); + GetEntityQuery(typeof(EcsTestData)); } } @@ -48,18 +48,17 @@ public override void Setup() m_Window = EditorWindow.GetWindow(); - m_System = World.Active.GetOrCreateManager(); - World.Active.GetOrCreateManager().AddSystemToUpdateList(m_System); + m_System = World.Active.GetOrCreateSystem(); + World.Active.GetOrCreateSystem().AddSystemToUpdateList(m_System); ScriptBehaviourUpdateOrder.UpdatePlayerLoop(World.Active); World2 = new World(World2Name); - World2.GetOrCreateManager(); - var emptySys = World2.GetOrCreateManager(); - World.Active.GetOrCreateManager().AddSystemToUpdateList(emptySys); - World.Active.GetOrCreateManager().SortSystemUpdateList(); + var emptySys = World2.GetOrCreateSystem(); + World.Active.GetOrCreateSystem().AddSystemToUpdateList(emptySys); + World.Active.GetOrCreateSystem().SortSystemUpdateList(); - m_ComponentGroup = m_System.ComponentGroups[0]; + entityQuery = m_System.EntityQueries[0]; m_Entity = m_Manager.CreateEntity(typeof(EcsTestData)); } @@ -93,11 +92,14 @@ public void WorldPopup_RestorePreviousSelection() [Test] public void EntityDebugger_SetSystemSelection() { + // TODO EntityManager is no longer a system + /* m_Window.SetSystemSelection(m_Manager, World.Active, true, true); Assert.AreEqual(World.Active, m_Window.SystemSelectionWorld); Assert.Throws(() => m_Window.SetSystemSelection(m_Manager, null, true, true)); + */ } [Test] @@ -115,27 +117,27 @@ public void EntityDebugger_DestroySystem() { m_Window.SetWorldSelection(World2, true); Assert.IsFalse(m_Window.systemListView.NeedsReload); - var emptySystem = World2.GetExistingManager(); - World2.DestroyManager(emptySystem); + var emptySystem = World2.GetExistingSystem(); + World2.DestroySystem(emptySystem); Assert.IsTrue(m_Window.systemListView.NeedsReload); } [Test] public void EntityDebugger_SetAllSelections() { - var entityListQuery = new EntityListQuery(m_ComponentGroup); + var entityListQuery = new EntityListQuery(entityQuery); EntityDebugger.SetAllSelections(World.Active, m_System, entityListQuery, m_Entity); Assert.AreEqual(World.Active, m_Window.WorldSelection); Assert.AreEqual(m_System, m_Window.SystemSelection); - Assert.AreEqual(m_ComponentGroup, m_Window.EntityListQuerySelection.Group); + Assert.AreEqual(entityQuery, m_Window.EntityListQuerySelection.Group); Assert.AreEqual(m_Entity, m_Window.EntitySelection); } [Test] public void EntityDebugger_RememberSelections() { - var entityListQuery = new EntityListQuery(m_ComponentGroup); + var entityListQuery = new EntityListQuery(entityQuery); EntityDebugger.SetAllSelections(World.Active, m_System, entityListQuery, m_Entity); m_Window.SetWorldSelection(null, true); @@ -144,14 +146,14 @@ public void EntityDebugger_RememberSelections() Assert.AreEqual(World.Active, m_Window.WorldSelection); Assert.AreEqual(m_System, m_Window.SystemSelection); - Assert.AreEqual(m_ComponentGroup, m_Window.EntityListQuerySelection.Group); + Assert.AreEqual(entityQuery, m_Window.EntityListQuerySelection.Group); Assert.AreEqual(m_Entity, m_Window.EntitySelection); } [Test] public void EntityDebugger_SetAllEntitiesFilter() { - var query = new EntityArchetypeQuery() + var query = new EntityQueryDesc() { All = new ComponentType[] {ComponentType.ReadWrite() }, Any = new ComponentType[0], @@ -162,12 +164,12 @@ public void EntityDebugger_SetAllEntitiesFilter() m_Window.SetWorldSelection(World.Active, true); m_Window.SetSystemSelection(null, null, true, true); m_Window.SetAllEntitiesFilter(listQuery); - Assert.AreEqual(query, m_Window.EntityListQuerySelection.Query); + Assert.AreEqual(query, m_Window.EntityListQuerySelection.QueryDesc); m_Window.SetEntityListSelection(null, true, true); - m_Window.SetSystemSelection(World.Active.GetExistingManager(), World.Active, true, true); + m_Window.SetSystemSelection(null, World.Active, true, true); m_Window.SetAllEntitiesFilter(listQuery); - Assert.AreEqual(query, m_Window.EntityListQuerySelection.Query); + Assert.AreEqual(query, m_Window.EntityListQuerySelection.QueryDesc); m_Window.SetSystemSelection(m_System, World.Active, true, true); m_Window.SetAllEntitiesFilter(listQuery); diff --git a/Unity.Entities.Editor.Tests/ListViewTests.cs b/Unity.Entities.Editor.Tests/ListViewTests.cs index d70753f7..7fa111f8 100644 --- a/Unity.Entities.Editor.Tests/ListViewTests.cs +++ b/Unity.Entities.Editor.Tests/ListViewTests.cs @@ -21,11 +21,11 @@ private static void SetComponentGroupSelection(EntityListQuery query) { } - private static void SetSystemSelection(ScriptBehaviourManager system, World world) + private static void SetSystemSelection(ComponentSystemBase system, World world) { } - private EntityListQuery AllQuery => new EntityListQuery(new EntityArchetypeQuery(){All = new ComponentType[0], Any = new ComponentType[0], None = new ComponentType[0]}); + private EntityListQuery AllQuery => new EntityListQuery(new EntityQueryDesc(){All = new ComponentType[0], Any = new ComponentType[0], None = new ComponentType[0]}); private World World2; @@ -36,9 +36,9 @@ public override void Setup() ScriptBehaviourUpdateOrder.UpdatePlayerLoop(World.Active); World2 = new World("Test World 2"); - var emptySys = World2.GetOrCreateManager(); - World.Active.GetOrCreateManager().AddSystemToUpdateList(emptySys); - World.Active.GetOrCreateManager().SortSystemUpdateList(); + var emptySys = World2.GetOrCreateSystem(); + World.Active.GetOrCreateSystem().AddSystemToUpdateList(emptySys); + World.Active.GetOrCreateSystem().SortSystemUpdateList(); } public override void TearDown() @@ -55,16 +55,12 @@ public override void TearDown() public void EntityListView_ShowNothingWithoutWorld() { m_Manager.CreateEntity(); - var emptySystem = World.Active.GetOrCreateManager(); - ScriptBehaviourManager currentSystem = null; + var emptySystem = World.Active.GetOrCreateSystem(); + ComponentSystemBase currentSystem = null; using (var listView = new EntityListView(new TreeViewState(), null, SetEntitySelection, () => null, () => currentSystem, x => {})) { - currentSystem = World.Active.GetExistingManager(); - listView.SelectedEntityQuery = null; - Assert.IsFalse(listView.ShowingSomething); - currentSystem = emptySystem; listView.SelectedEntityQuery = null; Assert.IsFalse(listView.ShowingSomething); @@ -83,22 +79,25 @@ public void EntityListView_ShowNothingWithoutWorld() public void EntityListView_ShowEntitiesFromWorld() { m_Manager.CreateEntity(); - var emptySystem = World.Active.GetOrCreateManager(); + var emptySystem = World.Active.GetOrCreateSystem(); var selectedWorld = World.Active; - ScriptBehaviourManager currentSystem = null; + ComponentSystemBase currentSystem = null; using (var listView = new EntityListView(new TreeViewState(), null, SetEntitySelection, () => selectedWorld, () => currentSystem, x => {})) { - currentSystem = World.Active.GetExistingManager(); + // TODO EntityManager is no longer a system + /* + currentSystem = World.Active.EntityManager; listView.SelectedEntityQuery = AllQuery; Assert.IsTrue(listView.ShowingSomething); Assert.AreEqual(1, listView.GetRows().Count); - currentSystem = World.Active.GetExistingManager(); + currentSystem = World.Active.EntityManager; listView.SelectedEntityQuery = null; Assert.IsTrue(listView.ShowingSomething); Assert.AreEqual(1, listView.GetRows().Count); + */ currentSystem = emptySystem; listView.SelectedEntityQuery = null; @@ -106,10 +105,24 @@ public void EntityListView_ShowEntitiesFromWorld() } } + [Test] + public void EntityListView_ShowNothingWithNoEntityManager() + { + using (var incompleteWorld = new World("test 2")) + { + using (var listView = new EntityListView(new TreeViewState(), null, SetEntitySelection, () => incompleteWorld, + () => null, x => {})) + { + listView.SelectedEntityQuery = null; + Assert.AreEqual(0, listView.GetRows().Count); + } + } + } + [Test] public void ComponentGroupListView_CanSetNullSystem() { - var listView = new ComponentGroupListView(new TreeViewState(), EmptySystem, SetComponentGroupSelection, GetWorldSelection); + var listView = new EntityQueryListView(new TreeViewState(), EmptySystem, SetComponentGroupSelection, GetWorldSelection); Assert.DoesNotThrow(() => listView.SelectedSystem = null); } @@ -125,7 +138,7 @@ public void ComponentGroupListView_SortOrderExpected() typeList.Add(subtractive); typeList.Add(readOnly); typeList.Add(readWrite); - typeList.Sort(ComponentGroupGUI.CompareTypes); + typeList.Sort(EntityQueryGUI.CompareTypes); Assert.AreEqual(readOnly, typeList[0]); Assert.AreEqual(readWrite, typeList[1]); @@ -156,7 +169,7 @@ public void SystemListView_ShowExactlyWorldSystems() () => true); var managerItems = listView.GetRows().Where(x => listView.managersById.ContainsKey(x.id)).Select(x => listView.managersById[x.id]); var managerList = managerItems.ToList(); - Assert.AreEqual(World2.BehaviourManagers.Count(x => !(x is EntityManager)), managerList.Intersect(World2.BehaviourManagers).Count()); + Assert.AreEqual(World2.Systems.Count(), managerList.Intersect(World2.Systems).Count()); } [Test] @@ -169,11 +182,11 @@ public void SystemListView_NullWorldShowsAllSystems() () => null, () => true); var managerItems = listView.GetRows().Where(x => listView.managersById.ContainsKey(x.id)).Select(x => listView.managersById[x.id]); - var allManagers = new List(); - allManagers.AddRange(World.Active.BehaviourManagers); - allManagers.AddRange(World2.BehaviourManagers); + var allManagers = new List(); + allManagers.AddRange(World.Active.Systems); + allManagers.AddRange(World2.Systems); var managerList = managerItems.ToList(); - Assert.AreEqual(allManagers.Count(x => !(x is EntityManager || x is ComponentSystemGroup) ), allManagers.Intersect(managerList).Count()); + Assert.AreEqual(allManagers.Count(x => !(x is ComponentSystemGroup) ), allManagers.Intersect(managerList).Count()); } } diff --git a/Unity.Entities.Editor/CommonGUI/ComponentGroupGUI.cs b/Unity.Entities.Editor/CommonGUI/EntityQueryGUI.cs similarity index 96% rename from Unity.Entities.Editor/CommonGUI/ComponentGroupGUI.cs rename to Unity.Entities.Editor/CommonGUI/EntityQueryGUI.cs index c5e670df..e5ff40ae 100644 --- a/Unity.Entities.Editor/CommonGUI/ComponentGroupGUI.cs +++ b/Unity.Entities.Editor/CommonGUI/EntityQueryGUI.cs @@ -6,7 +6,7 @@ namespace Unity.Entities.Editor { - internal static class ComponentGroupGUI + internal static class EntityQueryGUI { internal static int CompareTypes(ComponentType x, ComponentType y) @@ -42,7 +42,7 @@ static string SpecifiedTypeName(Type type, Queue args) if (type.IsGenericParameter) { return name; - } + } if (type.IsNested) { name = $"{SpecifiedTypeName(type.DeclaringType, args)}.{name}"; @@ -62,7 +62,7 @@ static string SpecifiedTypeName(Type type, Queue args) genericTypeNames.Append(SpecifiedTypeName(args.Dequeue())); } - if (genericTypeNames.Length > 0) + if (genericTypeNames.Length > 0) { name = $"{name}<{genericTypeNames}>"; } diff --git a/Unity.Entities.Editor/CommonGUI/ComponentGroupGUIControl.cs.meta b/Unity.Entities.Editor/CommonGUI/EntityQueryGUI.cs.meta similarity index 83% rename from Unity.Entities.Editor/CommonGUI/ComponentGroupGUIControl.cs.meta rename to Unity.Entities.Editor/CommonGUI/EntityQueryGUI.cs.meta index 0dc703dd..aa066e65 100644 --- a/Unity.Entities.Editor/CommonGUI/ComponentGroupGUIControl.cs.meta +++ b/Unity.Entities.Editor/CommonGUI/EntityQueryGUI.cs.meta @@ -1,5 +1,5 @@ fileFormatVersion: 2 -guid: 910d2d0d5f8554bc5a32518b8fdcfee6 +guid: a2599509af17e8a4881f190635eea7a9 MonoImporter: externalObjects: {} serializedVersion: 2 diff --git a/Unity.Entities.Editor/CommonGUI/ComponentGroupGUIControl.cs b/Unity.Entities.Editor/CommonGUI/EntityQueryGUIControl.cs similarity index 82% rename from Unity.Entities.Editor/CommonGUI/ComponentGroupGUIControl.cs rename to Unity.Entities.Editor/CommonGUI/EntityQueryGUIControl.cs index 60690f82..c0c30c1c 100644 --- a/Unity.Entities.Editor/CommonGUI/ComponentGroupGUIControl.cs +++ b/Unity.Entities.Editor/CommonGUI/EntityQueryGUIControl.cs @@ -6,7 +6,7 @@ namespace Unity.Entities.Editor { - internal class ComponentGroupGUIControl + internal class EntityQueryGUIControl { private List styles; private List names; @@ -19,12 +19,12 @@ public float Height get { return height; } } - public ComponentGroupGUIControl(IEnumerable types, bool archetypeQueryMode) + public EntityQueryGUIControl(IEnumerable types, bool archetypeQueryMode) { CalculateDrawingParts(types, null, archetypeQueryMode); } - public ComponentGroupGUIControl(IEnumerable types, IEnumerable readWriteTypes, bool archetypeQueryMode) + public EntityQueryGUIControl(IEnumerable types, IEnumerable readWriteTypes, bool archetypeQueryMode) { CalculateDrawingParts(types, readWriteTypes, archetypeQueryMode); } @@ -32,7 +32,7 @@ public ComponentGroupGUIControl(IEnumerable types, IEnumerable types, IEnumerable readWriteTypes, bool archetypeQueryMode) { var typeList = types.ToList(); - typeList.Sort((Comparison) ComponentGroupGUI.CompareTypes); + typeList.Sort((Comparison) EntityQueryGUI.CompareTypes); styles = new List(typeList.Count); names = new List(typeList.Count); rects = new List(typeList.Count); @@ -45,7 +45,7 @@ void CalculateDrawingParts(IEnumerable types, IEnumerable types, IEnumerable>> cachedMatches = new List>>(); - private readonly Dictionary cachedControls = new Dictionary(); + private readonly List>> cachedMatches = new List>>(); + private readonly Dictionary cachedControls = new Dictionary(); private bool repainted = true; [SerializeField] private bool showSystems; @@ -26,14 +26,14 @@ public void OnGUI(World world, Entity entity) if (repainted == true) { cachedMatches.Clear(); - WorldDebuggingTools.MatchEntityInComponentGroups(world, entity, cachedMatches); + WorldDebuggingTools.MatchEntityInEntityQueries(world, entity, cachedMatches); foreach (var pair in cachedMatches) { foreach (var componentGroup in pair.Item2) { if (!cachedControls.ContainsKey(componentGroup)) { - cachedControls.Add(componentGroup, new ComponentGroupGUIControl(componentGroup.GetQueryTypes(), false)); + cachedControls.Add(componentGroup, new EntityQueryGUIControl(componentGroup.GetQueryTypes(), false)); } } } diff --git a/Unity.Entities.Editor/EntityDebugger/ChunkInfoListView.cs b/Unity.Entities.Editor/EntityDebugger/ChunkInfoListView.cs index 9b4d6a94..3a6aef97 100644 --- a/Unity.Entities.Editor/EntityDebugger/ChunkInfoListView.cs +++ b/Unity.Entities.Editor/EntityDebugger/ChunkInfoListView.cs @@ -77,7 +77,7 @@ static Styles() } public EntityArchetype archetype; - public ComponentGroupGUIControl control; + public EntityQueryGUIControl control; public int[] counts; public int maxCount; public int firstChunkIndex; @@ -225,7 +225,7 @@ protected override TreeViewItem BuildRoot() var stats = new ArchetypeInfo() { archetype = currentArchetype, - control = new ComponentGroupGUIControl(currentArchetype.GetComponentTypes(), true), + control = new EntityQueryGUIControl(currentArchetype.GetComponentTypes(), true), counts = new int[currentArchetype.ChunkCapacity], firstChunkIndex = currentChunkIndex }; diff --git a/Unity.Entities.Editor/EntityDebugger/ComponentTypeFilterUI.cs b/Unity.Entities.Editor/EntityDebugger/ComponentTypeFilterUI.cs index 9bef2cdb..1ceb6288 100644 --- a/Unity.Entities.Editor/EntityDebugger/ComponentTypeFilterUI.cs +++ b/Unity.Entities.Editor/EntityDebugger/ComponentTypeFilterUI.cs @@ -13,7 +13,7 @@ internal class ComponentTypeFilterUI private readonly List selectedFilterTypes = new List(); private readonly List filterTypes = new List(); - private readonly List entityQueries = new List(); + private readonly List entityQueries = new List(); public ComponentTypeFilterUI(SetFilterAction setFilter, WorldSelectionGetter worldSelectionGetter) { @@ -53,8 +53,8 @@ internal void GetTypes() filterTypes.AddRange(requiredTypes); filterTypes.AddRange(subtractiveTypes); - - filterTypes.Sort(ComponentGroupGUI.CompareTypes); + + filterTypes.Sort(EntityQueryGUI.CompareTypes); } } @@ -68,7 +68,7 @@ public void OnGUI() { ++filterCount; var style = filterTypes[i].AccessModeType == ComponentType.AccessMode.Exclude ? EntityDebuggerStyles.ComponentExclude : EntityDebuggerStyles.ComponentRequired; - GUILayout.Label(ComponentGroupGUI.SpecifiedTypeName(filterTypes[i].GetManagedType()), style); + GUILayout.Label(EntityQueryGUI.SpecifiedTypeName(filterTypes[i].GetManagedType()), style); } } if (filterCount == 0) @@ -90,7 +90,7 @@ public void OnGUI() } } - internal ComponentGroup GetExistingGroup(ComponentType[] components) + internal EntityQuery GetExistingQuery(ComponentType[] components) { foreach (var existingGroup in entityQueries) { @@ -101,12 +101,12 @@ internal ComponentGroup GetExistingGroup(ComponentType[] components) return null; } - internal ComponentGroup GetComponentGroup(ComponentType[] components) + internal EntityQuery GetEntityQuery(ComponentType[] components) { - var group = GetExistingGroup(components); + var group = GetExistingQuery(components); if (group != null) return group; - group = getWorldSelection().GetExistingManager().CreateComponentGroup(components); + group = getWorldSelection().EntityManager.CreateEntityQuery(components); entityQueries.Add(group); return group; @@ -120,7 +120,7 @@ private void ComponentFilterChanged() if (selectedFilterTypes[i]) selectedTypes.Add(filterTypes[i]); } - var group = GetComponentGroup(selectedTypes.ToArray()); + var group = GetEntityQuery(selectedTypes.ToArray()); setFilter(new EntityListQuery(group)); } } diff --git a/Unity.Entities.Editor/EntityDebugger/ComponentTypeListView.cs b/Unity.Entities.Editor/EntityDebugger/ComponentTypeListView.cs index 0b23ee6e..8f347aca 100644 --- a/Unity.Entities.Editor/EntityDebugger/ComponentTypeListView.cs +++ b/Unity.Entities.Editor/EntityDebugger/ComponentTypeListView.cs @@ -21,7 +21,7 @@ public ComponentTypeListView(TreeViewState state, List types, Lis this.typeSelections = typeSelections; typeNames = new List(types.Count); for (var i = 0; i < types.Count; ++i) - typeNames.Add(new GUIContent(ComponentGroupGUI.SpecifiedTypeName(types[i].GetManagedType()))); + typeNames.Add(new GUIContent(EntityQueryGUI.SpecifiedTypeName(types[i].GetManagedType()))); Reload(); } diff --git a/Unity.Entities.Editor/EntityDebugger/EntityDebugger.cs b/Unity.Entities.Editor/EntityDebugger/EntityDebugger.cs index 957caa14..ead0d351 100644 --- a/Unity.Entities.Editor/EntityDebugger/EntityDebugger.cs +++ b/Unity.Entities.Editor/EntityDebugger/EntityDebugger.cs @@ -3,6 +3,7 @@ using UnityEditor; using UnityEngine; using UnityEditor.IMGUI.Controls; +using UnityEngine.Serialization; namespace Unity.Entities.Editor { @@ -66,7 +67,7 @@ private static GUIStyle BoxStyle private static GUIStyle boxStyle; - public ScriptBehaviourManager SystemSelection { get; private set; } + public ComponentSystemBase SystemSelection { get; private set; } public World SystemSelectionWorld { @@ -74,7 +75,7 @@ public World SystemSelectionWorld private set { systemSelectionWorld = value; } } - public void SetSystemSelection(ScriptBehaviourManager manager, World world, bool updateList, bool propagate) + public void SetSystemSelection(ComponentSystemBase manager, World world, bool updateList, bool propagate) { if (manager != null && world == null) throw new ArgumentNullException("System cannot have null world"); @@ -82,11 +83,11 @@ public void SetSystemSelection(ScriptBehaviourManager manager, World world, bool SystemSelectionWorld = world; if (updateList) systemListView.SetSystemSelection(manager, world); - CreateComponentGroupListView(); + CreateEntityQueryListView(); if (propagate) { if (SystemSelection is ComponentSystemBase) - componentGroupListView.TouchSelection(); + entityQueryListView.TouchSelection(); else ApplyAllEntitiesFilter(); } @@ -99,7 +100,7 @@ public void SetEntityListSelection(EntityListQuery newSelection, bool updateList chunkInfoListView.ClearSelection(); EntityListQuerySelection = newSelection; if (updateList) - componentGroupListView.SetEntityListSelection(newSelection); + entityQueryListView.SetEntityListSelection(newSelection); entityListView.SelectedEntityQuery = newSelection; if (propagate) entityListView.TouchSelection(); @@ -112,7 +113,7 @@ internal void SetEntitySelection(Entity newSelection, bool updateList) if (updateList) entityListView.SetEntitySelection(newSelection); - var world = WorldSelection ?? (SystemSelection as ComponentSystemBase)?.World ?? (SystemSelection as EntityManager)?.World; + var world = WorldSelection ?? (SystemSelection as ComponentSystemBase)?.World; if (world != null && newSelection != Entity.Null) { selectionProxy.SetEntity(world, newSelection); @@ -139,25 +140,27 @@ internal static void SetAllSelections(World world, ComponentSystemBase system, E private static EntityDebugger Instance { get; set; } private EntitySelectionProxy selectionProxy; - - [SerializeField] private List componentGroupListStates = new List(); - [SerializeField] private List componentGroupListStateNames = new List(); - private ComponentGroupListView componentGroupListView; - + + [FormerlySerializedAs("componentGroupListStates")] + [SerializeField] private List entityQueryListStates = new List(); + [FormerlySerializedAs("componentGroupListStateNames")] + [SerializeField] private List entityQueryListStateNames = new List(); + private EntityQueryListView entityQueryListView; + [SerializeField] private List systemListStates = new List(); [SerializeField] private List systemListStateNames = new List(); internal SystemListView systemListView; [SerializeField] private TreeViewState entityListState = new TreeViewState(); private EntityListView entityListView; - + [SerializeField] private ChunkInfoListView.State chunkInfoListState = new ChunkInfoListView.State(); private ChunkInfoListView chunkInfoListView; internal WorldPopup m_WorldPopup; - + private ComponentTypeFilterUI filterUI; - + public World WorldSelection { get @@ -167,7 +170,7 @@ public World WorldSelection return null; } } - + [SerializeField] private string lastEditModeWorldSelection = WorldPopup.kNoWorldName; [SerializeField] private string lastPlayModeWorldSelection = WorldPopup.kNoWorldName; [SerializeField] private bool showingPlayerLoop; @@ -185,7 +188,7 @@ public void SetWorldSelection(World selection, bool propagate) else lastEditModeWorldSelection = worldSelection.Name; } - + CreateSystemListView(); if (propagate) systemListView.TouchSelection(); @@ -217,9 +220,9 @@ private void CreateSystemListView() systemListView.multiColumnHeader.ResizeToFit(); } - private void CreateComponentGroupListView() + private void CreateEntityQueryListView() { - componentGroupListView = ComponentGroupListView.CreateList(SystemSelection as ComponentSystemBase, componentGroupListStates, componentGroupListStateNames, x => SetEntityListSelection(x, false, true), () => SystemSelectionWorld); + entityQueryListView = EntityQueryListView.CreateList(SystemSelection as ComponentSystemBase, entityQueryListStates, entityQueryListStateNames, x => SetEntityListSelection(x, false, true), () => SystemSelectionWorld); } [SerializeField] private bool ShowInactiveSystems; @@ -253,7 +256,7 @@ private void OnEnable() CreateEntitySelectionProxy(); CreateWorldPopup(); CreateSystemListView(); - CreateComponentGroupListView(); + CreateEntityQueryListView(); CreateEntityListView(); CreateChunkInfoListView(); systemListView.TouchSelection(); @@ -285,7 +288,7 @@ private void OnDisable() Instance = null; if (selectionProxy) DestroyImmediate(selectionProxy); - + EditorApplication.playModeStateChanged -= OnPlayModeStateChange; } @@ -302,8 +305,8 @@ private void OnPlayModeStateChange(PlayModeStateChange change) private void Update() { systemListView.UpdateTimings(); - - + + if (repaintLimiter.SimulationAdvanced()) { @@ -311,7 +314,7 @@ private void Update() } else if (!Application.isPlaying) { - if (systemListView.NeedsReload || componentGroupListView.NeedsReload || entityListView.NeedsReload || !filterUI.TypeListValid()) + if (systemListView.NeedsReload || entityQueryListView.NeedsReload || entityListView.NeedsReload || !filterUI.TypeListValid()) Repaint(); } } @@ -342,7 +345,7 @@ private void SystemHeader() ShowWorldPopup(); GUILayout.EndHorizontal(); } - + const float kChunkInfoButtonWidth = 60f; private void EntityHeader() @@ -371,15 +374,15 @@ private void ChunkInfoToggle(Rect rect) ShowingChunkInfoView = GUI.Toggle(rect, ShowingChunkInfoView, "Chunk Info", EditorStyles.miniButton); } - private void ComponentGroupList() + private void EntityQueryList() { - if (SystemSelection is ComponentSystemBase) + if (SystemSelection != null) { - componentGroupListView.SetWidth(CurrentEntityViewWidth); - var height = Mathf.Min(componentGroupListView.Height + BoxStyle.padding.vertical, position.height*0.5f); + entityQueryListView.SetWidth(CurrentEntityViewWidth); + var height = Mathf.Min(entityQueryListView.Height + BoxStyle.padding.vertical, position.height*0.5f); GUILayout.BeginVertical(BoxStyle, GUILayout.Height(height)); - - componentGroupListView.OnGUI(GUIHelpers.GetExpandingRect()); + + entityQueryListView.OnGUI(GUIHelpers.GetExpandingRect()); GUILayout.EndVertical(); } else if (WorldSelection != null) @@ -399,11 +402,11 @@ private void ComponentGroupList() public void SetAllEntitiesFilter(EntityListQuery entityQuery) { filterQuery = entityQuery; - if (WorldSelection == null || SystemSelection is ComponentSystemBase) + if (WorldSelection == null || SystemSelection != null) return; ApplyAllEntitiesFilter(); } - + private void ApplyAllEntitiesFilter() { SetEntityListSelection(filterQuery, false, true); @@ -415,7 +418,7 @@ void EntityList() entityListView.OnGUI(GUIHelpers.GetExpandingRect()); GUILayout.EndVertical(); } - + private void ChunkInfoView() { GUILayout.BeginHorizontal(BoxStyle); @@ -459,10 +462,10 @@ private void OnGUI() { systemListView.ReloadIfNecessary(); filterUI.GetTypes(); - componentGroupListView.ReloadIfNecessary(); + entityQueryListView.ReloadIfNecessary(); entityListView.ReloadIfNecessary(); } - + if (Selection.activeObject == selectionProxy) { if (!selectionProxy.Exists) @@ -473,17 +476,17 @@ private void OnGUI() } GUILayout.BeginArea(new Rect(0f, 0f, kSystemListWidth, position.height)); // begin System side SystemHeader(); - + GUILayout.BeginVertical(BoxStyle); SystemList(); GUILayout.EndVertical(); - + GUILayout.EndArea(); // end System side - + EntityHeader(); - + GUILayout.BeginArea(new Rect(kSystemListWidth, kLineHeight, CurrentEntityViewWidth, position.height - kLineHeight)); - ComponentGroupList(); + EntityQueryList(); EntityList(); GUILayout.EndArea(); @@ -493,8 +496,8 @@ private void OnGUI() ChunkInfoView(); GUILayout.EndArea(); } - + repaintLimiter.RecordRepaint(); } } -} \ No newline at end of file +} diff --git a/Unity.Entities.Editor/EntityDebugger/EntityListQuery.cs b/Unity.Entities.Editor/EntityDebugger/EntityListQuery.cs index f5f439d0..d7728d43 100644 --- a/Unity.Entities.Editor/EntityDebugger/EntityListQuery.cs +++ b/Unity.Entities.Editor/EntityDebugger/EntityListQuery.cs @@ -5,18 +5,18 @@ namespace Unity.Entities.Editor internal class EntityListQuery { - public ComponentGroup Group { get; } + public EntityQuery Group { get; } - public EntityArchetypeQuery Query { get; } + public EntityQueryDesc QueryDesc { get; } - public EntityListQuery(ComponentGroup group) + public EntityListQuery(EntityQuery group) { this.Group = group; } - public EntityListQuery(EntityArchetypeQuery query) + public EntityListQuery(EntityQueryDesc queryDesc) { - this.Query = query; + this.QueryDesc = queryDesc; } } diff --git a/Unity.Entities.Editor/EntityDebugger/EntityListView.cs b/Unity.Entities.Editor/EntityDebugger/EntityListView.cs index 1054c787..5707beac 100644 --- a/Unity.Entities.Editor/EntityDebugger/EntityListView.cs +++ b/Unity.Entities.Editor/EntityDebugger/EntityListView.cs @@ -6,12 +6,12 @@ namespace Unity.Entities.Editor { - + internal delegate void EntitySelectionCallback(Entity selection); internal delegate World WorldSelectionGetter(); - internal delegate ScriptBehaviourManager SystemSelectionGetter(); + internal delegate ComponentSystemBase SystemSelectionGetter(); internal delegate void ChunkArrayAssignmentCallback(NativeArray chunkArray); - + internal class EntityListView : TreeView, IDisposable { public EntityListQuery SelectedEntityQuery @@ -41,7 +41,7 @@ public void SetFilter(ChunkFilter filter) private readonly WorldSelectionGetter getWorldSelection; private readonly SystemSelectionGetter getSystemSelection; private readonly ChunkArrayAssignmentCallback setChunkArray; - + private readonly EntityArrayListAdapter rows; public NativeArray ChunkArray => chunkArray; @@ -64,7 +64,7 @@ public EntityListView(TreeViewState state, EntityListQuery entityQuery, EntitySe private int lastVersion = -1; - public bool NeedsReload => ShowingSomething && getWorldSelection().GetExistingManager().Version != lastVersion; + public bool NeedsReload => ShowingSomething && getWorldSelection().EntityManager.Version != lastVersion; public void ReloadIfNecessary() { @@ -77,42 +77,42 @@ public void ReloadIfNecessary() protected override TreeViewItem BuildRoot() { var root = new TreeViewItem { id = 0, depth = -1, displayName = "Root" }; - + return root; } - + protected override IList BuildRows(TreeViewItem root) { if (!ShowingSomething) return new List(); - var entityManager = getWorldSelection().GetExistingManager(); + var entityManager = getWorldSelection().EntityManager; if (chunkArray.IsCreated) chunkArray.Dispose(); - + entityManager.CompleteAllJobs(); var group = SelectedEntityQuery?.Group; - + if (group == null) { - var query = SelectedEntityQuery?.Query; + var query = SelectedEntityQuery?.QueryDesc; if (query == null) - group = entityManager.UniversalGroup; + group = entityManager.UniversalQuery; else { - group = entityManager.CreateComponentGroup(query); + group = entityManager.CreateEntityQuery(query); } } - + chunkArray = group.CreateArchetypeChunkArray(Allocator.Persistent); rows.SetSource(chunkArray, entityManager, chunkFilter); setChunkArray(chunkArray); lastVersion = entityManager.Version; - + return rows; } @@ -128,7 +128,7 @@ protected override IList GetDescendantsThatHaveChildren(int id) public override void OnGUI(Rect rect) { - if (getWorldSelection()?.GetExistingManager()?.IsCreated == true) + if (getWorldSelection()?.EntityManager.IsCreated == true) base.OnGUI(rect); } @@ -165,7 +165,7 @@ protected override void RenameEnded(RenameEndedArgs args) { if (args.acceptedRename) { - var manager = getWorldSelection()?.GetExistingManager(); + var manager = getWorldSelection()?.EntityManager; if (manager != null) { Entity entity; @@ -184,7 +184,7 @@ public void SelectNothing() public void SetEntitySelection(Entity entitySelection) { - if (entitySelection != Entity.Null && getWorldSelection().GetExistingManager().Exists(entitySelection)) + if (entitySelection != Entity.Null && getWorldSelection().EntityManager.Exists(entitySelection)) SetSelection(new List{EntityArrayListAdapter.IndexToItemId(entitySelection.Index)}); } diff --git a/Unity.Entities.Editor/EntityDebugger/ComponentGroupListView.cs b/Unity.Entities.Editor/EntityDebugger/EntityQueryListView.cs similarity index 76% rename from Unity.Entities.Editor/EntityDebugger/ComponentGroupListView.cs rename to Unity.Entities.Editor/EntityDebugger/EntityQueryListView.cs index 157ce81a..3a24ee06 100644 --- a/Unity.Entities.Editor/EntityDebugger/ComponentGroupListView.cs +++ b/Unity.Entities.Editor/EntityDebugger/EntityQueryListView.cs @@ -9,15 +9,15 @@ namespace Unity.Entities.Editor { internal delegate void SetEntityListSelection(EntityListQuery query); - internal class ComponentGroupListView : TreeView { - private static Dictionary> queriesBySystem = new Dictionary>(); - private static readonly Dictionary queriesByGroup = new Dictionary(); + internal class EntityQueryListView : TreeView { + private static Dictionary> queriesBySystem = new Dictionary>(); + private static readonly Dictionary queriesByGroup = new Dictionary(); - private static EntityArchetypeQuery GetQueryForGroup(ComponentGroup group) + private static EntityQueryDesc GetQueryForGroup(EntityQuery group) { if (!queriesByGroup.ContainsKey(group)) { - var query = new EntityArchetypeQuery() + var query = new EntityQueryDesc() { All = group.GetQueryTypes().Where(x => x.AccessModeType != ComponentType.AccessMode.Exclude).ToArray(), Any = new ComponentType[0], @@ -29,9 +29,9 @@ private static EntityArchetypeQuery GetQueryForGroup(ComponentGroup group) return queriesByGroup[group]; } - private readonly Dictionary componentGroupsById = new Dictionary(); - private readonly Dictionary queriesById = new Dictionary(); - private readonly Dictionary controlsById = new Dictionary(); + private readonly Dictionary componentGroupsById = new Dictionary(); + private readonly Dictionary queriesById = new Dictionary(); + private readonly Dictionary controlsById = new Dictionary(); public ComponentSystemBase SelectedSystem { @@ -62,21 +62,21 @@ private static TreeViewState GetStateForSystem(ComponentSystemBase system, List< return stateForCurrentSystem; stateForCurrentSystem = new TreeViewState(); - if (system.ComponentGroups != null && system.ComponentGroups.Length > 0) + if (system.EntityQueries != null && system.EntityQueries.Length > 0) stateForCurrentSystem.expandedIDs = new List {1}; states.Add(stateForCurrentSystem); stateNames.Add(currentSystemName); return stateForCurrentSystem; } - public static ComponentGroupListView CreateList(ComponentSystemBase system, List states, List stateNames, + public static EntityQueryListView CreateList(ComponentSystemBase system, List states, List stateNames, SetEntityListSelection entityQuerySelectionCallback, WorldSelectionGetter worldSelectionGetter) { var state = GetStateForSystem(system, states, stateNames); - return new ComponentGroupListView(state, system, entityQuerySelectionCallback, worldSelectionGetter); + return new EntityQueryListView(state, system, entityQuerySelectionCallback, worldSelectionGetter); } - public ComponentGroupListView(TreeViewState state, ComponentSystemBase system, SetEntityListSelection entityListSelectionCallback, WorldSelectionGetter worldSelectionGetter) : base(state) + public EntityQueryListView(TreeViewState state, ComponentSystemBase system, SetEntityListSelection entityListSelectionCallback, WorldSelectionGetter worldSelectionGetter) : base(state) { this.getWorldSelection = worldSelectionGetter; this.entityListSelectionCallback = entityListSelectionCallback; @@ -93,13 +93,13 @@ protected override float GetCustomRowHeight(int row, TreeViewItem item) return controlsById.ContainsKey(item.id) ? controlsById[item.id].Height + 2 : rowHeight; } - private static List GetQueriesForSystem(ComponentSystemBase system) + private static List GetQueriesForSystem(ComponentSystemBase system) { - List queries; + List queries; if (queriesBySystem.TryGetValue(system, out queries)) return queries; - queries = new List(); + queries = new List(); var currentType = system.GetType(); @@ -107,8 +107,8 @@ private static List GetQueriesForSystem(ComponentSystemBas { foreach (var field in currentType.GetFields(BindingFlags.Instance | BindingFlags.NonPublic | BindingFlags.Public)) { - if (field.FieldType == typeof(EntityArchetypeQuery)) - queries.Add(field.GetValue(system) as EntityArchetypeQuery); + if (field.FieldType == typeof(EntityQueryDesc)) + queries.Add(field.GetValue(system) as EntityQueryDesc); } currentType = currentType.BaseType; @@ -135,20 +135,20 @@ protected override TreeViewItem BuildRoot() else { var queries = GetQueriesForSystem(SelectedSystem); - var entityManager = getWorldSelection().GetExistingManager(); + var entityManager = getWorldSelection().EntityManager; foreach (var query in queries) { - var group = entityManager.CreateComponentGroup(query); + var group = entityManager.CreateEntityQuery(query); queriesById.Add(currentId, query); componentGroupsById.Add(currentId, group); var groupItem = new TreeViewItem { id = currentId++ }; root.AddChild(groupItem); } - if (SelectedSystem.ComponentGroups != null) + if (SelectedSystem.EntityQueries != null) { - foreach (var group in SelectedSystem.ComponentGroups) + foreach (var group in SelectedSystem.EntityQueries) { componentGroupsById.Add(currentId, group); @@ -166,7 +166,7 @@ protected override TreeViewItem BuildRoot() foreach (var idGroupPair in componentGroupsById) { - var newControl = new ComponentGroupGUIControl(idGroupPair.Value.GetQueryTypes(), idGroupPair.Value.GetReadAndWriteTypes(), true); + var newControl = new EntityQueryGUIControl(idGroupPair.Value.GetQueryTypes(), idGroupPair.Value.GetReadAndWriteTypes(), true); controlsById.Add(idGroupPair.Key, newControl); } } @@ -195,7 +195,7 @@ public void SetWidth(float newWidth) public override void OnGUI(Rect rect) { - if (getWorldSelection()?.GetExistingManager()?.IsCreated == true) + if (getWorldSelection()?.EntityManager.IsCreated == true) { if (Event.current.type == EventType.Repaint) { @@ -207,10 +207,10 @@ public override void OnGUI(Rect rect) protected void DrawCount(RowGUIArgs args) { - ComponentGroup componentGroup; - if (componentGroupsById.TryGetValue(args.item.id, out componentGroup)) + EntityQuery entityQuery; + if (componentGroupsById.TryGetValue(args.item.id, out entityQuery)) { - var countString = componentGroup.CalculateLength().ToString(); + var countString = entityQuery.CalculateLength().ToString(); DefaultGUI.LabelRightAligned(args.rowRect, countString, args.selected, args.focused); } } @@ -234,9 +234,9 @@ protected override void SelectionChanged(IList selectedIds) { if (selectedIds.Count > 0) { - ComponentGroup componentGroup; - if (componentGroupsById.TryGetValue(selectedIds[0], out componentGroup)) - entityListSelectionCallback(new EntityListQuery(componentGroup)); + EntityQuery entityQuery; + if (componentGroupsById.TryGetValue(selectedIds[0], out entityQuery)) + entityListSelectionCallback(new EntityListQuery(entityQuery)); } else { @@ -271,7 +271,7 @@ public void SetEntityListSelection(EntityListQuery newListQuery) { foreach (var pair in queriesById) { - if (pair.Value == newListQuery.Query) + if (pair.Value == newListQuery.QueryDesc) { SetSelection(new List {pair.Key}); return; @@ -281,7 +281,7 @@ public void SetEntityListSelection(EntityListQuery newListQuery) SetSelection(new List()); } - public void SetComponentGroupSelection(ComponentGroup group) + public void SetEntityQuerySelection(EntityQuery group) { SetSelection(new List()); } @@ -295,7 +295,7 @@ public bool NeedsReload { get { - var expectedGroupCount = SelectedSystem?.ComponentGroups?.Length ?? 0; + var expectedGroupCount = SelectedSystem?.EntityQueries?.Length ?? 0; return expectedGroupCount != componentGroupsById.Count; } diff --git a/Unity.Entities.Editor/IJobProcessComponentDataGenerator.cs.meta b/Unity.Entities.Editor/EntityDebugger/EntityQueryListView.cs.meta similarity index 83% rename from Unity.Entities.Editor/IJobProcessComponentDataGenerator.cs.meta rename to Unity.Entities.Editor/EntityDebugger/EntityQueryListView.cs.meta index 216c4e00..ce66e690 100644 --- a/Unity.Entities.Editor/IJobProcessComponentDataGenerator.cs.meta +++ b/Unity.Entities.Editor/EntityDebugger/EntityQueryListView.cs.meta @@ -1,5 +1,5 @@ fileFormatVersion: 2 -guid: 102f0b713c27d488c87e97a3506bd2d9 +guid: ef7ed95af4a49594e977be31384c6908 MonoImporter: externalObjects: {} serializedVersion: 2 diff --git a/Unity.Entities.Editor/EntityDebugger/SystemListView.cs b/Unity.Entities.Editor/EntityDebugger/SystemListView.cs index 13809355..8a0b250c 100644 --- a/Unity.Entities.Editor/EntityDebugger/SystemListView.cs +++ b/Unity.Entities.Editor/EntityDebugger/SystemListView.cs @@ -8,7 +8,7 @@ namespace Unity.Entities.Editor { - internal delegate void SystemSelectionCallback(ScriptBehaviourManager manager, World world); + internal delegate void SystemSelectionCallback(ComponentSystemBase manager, World world); internal class SystemListView : TreeView { @@ -43,7 +43,7 @@ public float ReadMilliseconds() } internal readonly Dictionary managersById = new Dictionary(); private readonly Dictionary worldsById = new Dictionary(); - private readonly Dictionary recordersByManager = new Dictionary(); + private readonly Dictionary recordersByManager = new Dictionary(); private readonly Dictionary hideNodesById = new Dictionary(); private const float kToggleWidth = 22f; @@ -249,7 +249,7 @@ private HideNode BuildNodesForPlayerLoopSystem(PlayerLoopSystem system, ref int if (executionDelegate != null && (dummy = executionDelegate.Target as ScriptBehaviourUpdateOrder.DummyDelegateWrapper) != null) { - var rootSystem = dummy.Manager; + var rootSystem = dummy.System; return BuildNodesForComponentSystem(rootSystem, ref currentId); } } @@ -264,7 +264,7 @@ private HideNode BuildNodesForPlayerLoopSystem(PlayerLoopSystem system, ref int return null; } - private HideNode BuildNodesForComponentSystem(ScriptBehaviourManager manager, ref int currentId) + private HideNode BuildNodesForComponentSystem(ComponentSystemBase manager, ref int currentId) { switch (manager) { @@ -497,7 +497,7 @@ public bool NeedsReload { if (manager is ComponentSystemBase system) { - if (system.World == null || !system.World.BehaviourManagers.Contains(manager)) + if (system.World == null || !system.World.Systems.Contains(manager)) return true; } } @@ -527,7 +527,7 @@ public void UpdateTimings() lastTimedFrame = Time.frameCount; } - public void SetSystemSelection(ScriptBehaviourManager manager, World world) + public void SetSystemSelection(ComponentSystemBase manager, World world) { foreach (var pair in managersById) { diff --git a/Unity.Entities.Editor/EntityInspector/EntitySelectionProxy.cs b/Unity.Entities.Editor/EntityInspector/EntitySelectionProxy.cs index 1ed2f124..5aece152 100644 --- a/Unity.Entities.Editor/EntityInspector/EntitySelectionProxy.cs +++ b/Unity.Entities.Editor/EntityInspector/EntitySelectionProxy.cs @@ -35,7 +35,7 @@ public void SetEntity(World world, Entity entity) { this.World = world; this.Entity = entity; - this.EntityManager = world.GetExistingManager(); + this.EntityManager = world.EntityManager; this.Container = new EntityContainer(EntityManager, Entity); EditorUtility.SetDirty(this); } diff --git a/Unity.Entities.Editor/EntityInspector/EntitySelectionProxyEditor.cs b/Unity.Entities.Editor/EntityInspector/EntitySelectionProxyEditor.cs index 0e0ddcf8..252f3b66 100644 --- a/Unity.Entities.Editor/EntityInspector/EntitySelectionProxyEditor.cs +++ b/Unity.Entities.Editor/EntityInspector/EntitySelectionProxyEditor.cs @@ -13,7 +13,7 @@ internal class EntitySelectionProxyEditor : UnityEditor.Editor private readonly RepaintLimiter repaintLimiter = new RepaintLimiter(); [SerializeField] private SystemInclusionList inclusionList; - + void OnEnable() { visitor = new EntityIMGUIVisitor((entity) => @@ -32,7 +32,7 @@ void OnEnable() private uint GetVersion() { var container = target as EntitySelectionProxy; - return container.World.GetExistingManager().GetChunkVersionHash(container.Entity); + return container.World.EntityManager.GetChunkVersionHash(container.Entity); } public override void OnInspectorGUI() @@ -46,9 +46,9 @@ public override void OnInspectorGUI() (targetProxy.Container.PropertyBag as StructPropertyBag)?.Visit(ref container, visitor); GUI.enabled = true; - + inclusionList.OnGUI(targetProxy.World, targetProxy.Entity); - + repaintLimiter.RecordRepaint(); lastVersion = GetVersion(); } diff --git a/Unity.Entities.Editor/ExtraTypesProvider.cs b/Unity.Entities.Editor/ExtraTypesProvider.cs index 7b6951f6..770c6775 100644 --- a/Unity.Entities.Editor/ExtraTypesProvider.cs +++ b/Unity.Entities.Editor/ExtraTypesProvider.cs @@ -11,11 +11,11 @@ namespace Unity.Entities.Editor [InitializeOnLoad] public sealed class ExtraTypesProvider { - static void AddIJobProcessComponentData(Type type, HashSet extraTypes) + static void AddIJobForEach(Type type, HashSet extraTypes) { foreach (var typeInterface in type.GetInterfaces()) { - if (typeInterface.Name.StartsWith("IJobProcessComponentData")) + if (typeInterface.Name.StartsWith("IJobForEach")) { var genericArgumentList = new List { type }; genericArgumentList.AddRange(typeInterface.GetGenericArguments()); @@ -23,7 +23,7 @@ static void AddIJobProcessComponentData(Type type, HashSet extraTypes) var producerAttribute = (JobProducerTypeAttribute) typeInterface.GetCustomAttribute(typeof(JobProducerTypeAttribute), true); if (producerAttribute == null) - throw new System.ArgumentException("IJobProcessComponentData interface must have [JobProducerType]"); + throw new System.ArgumentException("IJobForEach interface must have [JobProducerType]"); var generatedType = producerAttribute.ProducerType.MakeGenericType(genericArgumentList.ToArray()); extraTypes.Add(generatedType.ToString()); @@ -35,7 +35,7 @@ static void AddIJobProcessComponentData(Type type, HashSet extraTypes) static ExtraTypesProvider() { - //@TODO: Only produce JobProcessComponentDataExtensions.JobStruct_Process1 + //@TODO: Only produce JobForEachExtensions.JobStruct_Process1 // if there is any use of that specific type in deployed code. PlayerBuildInterface.ExtraTypesProvider += () => @@ -49,9 +49,9 @@ static ExtraTypesProvider() foreach (var type in assembly.GetTypes()) { - if (typeof(JobProcessComponentDataExtensions.IBaseJobProcessComponentData).IsAssignableFrom(type) && !type.IsAbstract) + if (typeof(JobForEachExtensions.IBaseJobForEach).IsAssignableFrom(type) && !type.IsAbstract) { - AddIJobProcessComponentData(type, extraTypes); + AddIJobForEach(type, extraTypes); } } } diff --git a/Unity.Entities.Editor/IJobProcessComponentDataGenerator.cs b/Unity.Entities.Editor/IJobProcessComponentDataGenerator.cs deleted file mode 100644 index 82d27b69..00000000 --- a/Unity.Entities.Editor/IJobProcessComponentDataGenerator.cs +++ /dev/null @@ -1,358 +0,0 @@ -using System; -using System.Collections.Generic; -using System.IO; -using System.Text; -using UnityEditor; - -#if false -namespace Unity.Entities -{ - public static class IJobProcessComponentDataGenerator - { - enum Combination - { - D - } - - [MenuItem("Tools/Gen IJobProcessComponentData")] - static void GenTest() - { - var combinations = new List(); - for (int count = 1; count <= 6; count++) - { - var array = new Combination[count]; - for (var c = 0; c != array.Length; c++) - array[c] = Combination.D; - - combinations.Add(array); - } - - var res = GenerateFile(combinations); - - File.WriteAllText("Packages/com.unity.entities/Unity.Entities/IJobProcessComponentData.generated.cs", res); - AssetDatabase.Refresh(); - - } - - static string GetComboString(bool withEntity, Combination[] combinations) - { - var baseType = new StringBuilder(); - - if (withEntity) - baseType.Append("E"); - foreach (var c in combinations) - baseType.Append(Enum.GetName(typeof(Combination), c)); - return baseType.ToString(); - } - - static void GenerateScheduleFunc(List combinations, string funcName, int forEach, - string scheduleMode, StringBuilder gen) - { - gen.Append - ( - $@" - public static JobHandle {funcName}(this T jobData, ComponentSystemBase system, JobHandle dependsOn = default(JobHandle)) - where T : struct, IBaseJobProcessComponentData - {{ - var typeT = typeof(T);" - ); - - foreach (var combination in combinations) - { - string comboString; - comboString = GetComboString(false, combination); - gen.Append( - $@" - if (typeof(IBaseJobProcessComponentData_{comboString}).IsAssignableFrom(typeT)) - return ScheduleInternal_{comboString}(ref jobData, system, null, {forEach}, dependsOn, {scheduleMode});" - ); - - comboString = GetComboString(true, combination); - gen.Append( - $@" - if (typeof(IBaseJobProcessComponentData_{comboString}).IsAssignableFrom(typeT)) - return ScheduleInternal_{comboString}(ref jobData, system, null, {forEach}, dependsOn, {scheduleMode});" - ); - } - - gen.Append - ( - $@" - throw new System.ArgumentException({'"'}Not supported{'"'}); - }} -" - ); - } - - static void GenerateScheduleGroupFunc(List combinations, string funcName, int forEach, - string scheduleMode, StringBuilder gen) - { - gen.Append - ( - $@" - public static JobHandle {funcName}(this T jobData, ComponentGroup componentGroup, JobHandle dependsOn = default(JobHandle)) - where T : struct, IBaseJobProcessComponentData - {{ - var typeT = typeof(T);" - ); - - foreach (var combination in combinations) - { - string comboString; - comboString = GetComboString(false, combination); - gen.Append( - $@" - if (typeof(IBaseJobProcessComponentData_{comboString}).IsAssignableFrom(typeT)) - return ScheduleInternal_{comboString}(ref jobData, null, componentGroup, {forEach}, dependsOn, {scheduleMode});" - ); - - comboString = GetComboString(true, combination); - gen.Append( - $@" - if (typeof(IBaseJobProcessComponentData_{comboString}).IsAssignableFrom(typeT)) - return ScheduleInternal_{comboString}(ref jobData, null, componentGroup, {forEach}, dependsOn, {scheduleMode});" - ); - } - - gen.Append - ( - $@" - throw new System.ArgumentException({'"'}Not supported{'"'}); - }} -" - ); - } - - static string GenerateFile(List combinations) - { - var gen = new StringBuilder(); - var igen = new StringBuilder(); - - GenerateScheduleFunc(combinations, "Schedule", 1, "ScheduleMode.Batched", gen); - GenerateScheduleFunc(combinations, "ScheduleSingle", -1, "ScheduleMode.Batched", gen); - GenerateScheduleFunc(combinations, "Run", -1, "ScheduleMode.Run", gen); - - GenerateScheduleGroupFunc(combinations, "ScheduleGroup", 1, "ScheduleMode.Batched", gen); - GenerateScheduleGroupFunc(combinations, "ScheduleGroupSingle", -1, "ScheduleMode.Batched", gen); - GenerateScheduleGroupFunc(combinations, "RunGroup", -1, "ScheduleMode.Run", gen); - - foreach (var combination in combinations) - { - Generate(false, combination, gen, igen); - Generate(true, combination, gen, igen); - } - - var res = - ( - $@"// Generated by IJobProcessComponentDataGenerator.cs -//------------------------------------------------------------------------------ -// -// This code was generated by a tool. -// -// Changes to this file may cause incorrect behavior and will be lost if -// the code is regenerated. -// -//------------------------------------------------------------------------------ -#if !UNITY_ZEROPLAYER - -using Unity.Collections; -using Unity.Collections.LowLevel.Unsafe; -using Unity.Jobs; -using Unity.Jobs.LowLevel.Unsafe; -using System.Runtime.InteropServices; -using UnityEngine.Scripting; -using System; - - -namespace Unity.Entities -{{ - -{igen} - - public static partial class JobProcessComponentDataExtensions - {{ -{gen} - }} -}} - -#endif -" - ); - - return res; - } - - - static string Generate(bool withEntity, Combination[] combination, StringBuilder gen, StringBuilder igen) - { - var comboString = GetComboString(withEntity, combination); - var untypedGenericParams = new StringBuilder(); - var genericParams = new StringBuilder(); - var genericConstraints = new StringBuilder(); - - var executeParams = new StringBuilder(); - var executeCallParams = new StringBuilder(); - var ptrs = new StringBuilder(); - var typeLookupCache = new StringBuilder(); - - var interfaceName = withEntity ? "IJobProcessComponentDataWithEntity" : "IJobProcessComponentData"; - - if (withEntity) - { - executeCallParams.Append("ptrE[i], i + beginIndex, "); - executeParams.Append("Entity entity, int index, "); - ptrs.AppendLine - ( -$" var ptrE = (Entity*)UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, false, 0, jobData.Iterator.GlobalSystemVersion));" - ); - } - - for (int i = 0; i != combination.Length; i++) - { - ptrs.AppendLine - ( - -$@" ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex{i}, ref typeLookupCache{i}); - var ptr{i} = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly{i} == 0, typeLookupCache{i}, jobData.Iterator.GlobalSystemVersion));" - ); - - typeLookupCache.AppendLine - ( - -$@" var typeLookupCache{i} = 0; " - ); - - genericConstraints.Append($" where U{i} : struct, IComponentData"); - if (i != combination.Length - 1) - genericConstraints.AppendLine(); - untypedGenericParams.Append(","); - - genericParams.Append($"U{i}"); - if (i != combination.Length - 1) - genericParams.Append(", "); - - executeCallParams.Append($"ref UnsafeUtilityEx.ArrayElementAsRef(ptr{i}, i)"); - if (i != combination.Length - 1) - executeCallParams.Append(", "); - - executeParams.Append($"ref U{i} c{i}"); - if (i != combination.Length - 1) - executeParams.Append(", "); - } - - - igen.Append - ( - $@" - - [JobProducerType(typeof(JobProcessComponentDataExtensions.JobStruct_Process_{comboString}<{untypedGenericParams}>))] - public interface {interfaceName}<{genericParams}> : JobProcessComponentDataExtensions.IBaseJobProcessComponentData_{comboString} -{genericConstraints} - {{ - void Execute({executeParams}); - }}" - ); - - - gen.Append - ( - $@" - internal static unsafe JobHandle ScheduleInternal_{comboString}(ref T jobData, ComponentSystemBase system, ComponentGroup componentGroup, int innerloopBatchCount, JobHandle dependsOn, ScheduleMode mode) - where T : struct - {{ - JobStruct_ProcessInfer_{comboString} fullData; - fullData.Data = jobData; - - var isParallelFor = innerloopBatchCount != -1; - Initialize(system, componentGroup, typeof(T), typeof(JobStruct_Process_{comboString}<{untypedGenericParams}>), isParallelFor, ref JobStruct_ProcessInfer_{comboString}.Cache, out fullData.Iterator); - - var unfilteredChunkCount = fullData.Iterator.m_Length; - var iterator = JobStruct_ProcessInfer_{comboString}.Cache.ComponentGroup.GetComponentChunkIterator(); - - var prefilterHandle = ComponentChunkIterator.PreparePrefilteredChunkLists(unfilteredChunkCount, iterator.m_MatchingArchetypeList, iterator.m_Filter, dependsOn, mode, out fullData.PrefilterData, out var deferredCountData); - - return Schedule(UnsafeUtility.AddressOf(ref fullData), fullData.PrefilterData, fullData.Iterator.m_Length, innerloopBatchCount, isParallelFor, iterator.RequiresFilter(), ref JobStruct_ProcessInfer_{comboString}.Cache, deferredCountData, prefilterHandle, mode); - }} - - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public interface IBaseJobProcessComponentData_{comboString} : IBaseJobProcessComponentData {{ }} - - [StructLayout(LayoutKind.Sequential)] - private struct JobStruct_ProcessInfer_{comboString} where T : struct - {{ - public static JobProcessComponentDataCache Cache; - - public ProcessIterationData Iterator; - public T Data; - - [DeallocateOnJobCompletion] - [NativeDisableContainerSafetyRestriction] - public NativeArray PrefilterData; - }} - - [StructLayout(LayoutKind.Sequential)] - internal struct JobStruct_Process_{comboString} - where T : struct, {interfaceName}<{genericParams}> -{genericConstraints} - {{ - public ProcessIterationData Iterator; - public T Data; - - [DeallocateOnJobCompletion] - [NativeDisableContainerSafetyRestriction] - public NativeArray PrefilterData; - - [Preserve] - public static IntPtr Initialize(JobType jobType) - {{ - return JobsUtility.CreateJobReflectionData(typeof(JobStruct_Process_{comboString}), typeof(T), jobType, (ExecuteJobFunction) Execute); - }} - - delegate void ExecuteJobFunction(ref JobStruct_Process_{comboString} data, IntPtr additionalPtr, IntPtr bufferRangePatchData, ref JobRanges ranges, int jobIndex); - - public static unsafe void Execute(ref JobStruct_Process_{comboString} jobData, IntPtr additionalPtr, IntPtr bufferRangePatchData, ref JobRanges ranges, int jobIndex) - {{ - var chunks = (ArchetypeChunk*)NativeArrayUnsafeUtility.GetUnsafePtr(jobData.PrefilterData); - var entityIndices = (int*) (chunks + jobData.Iterator.m_Length); - var chunkCount = *(entityIndices + jobData.Iterator.m_Length); - - if (jobData.Iterator.m_IsParallelFor) - {{ - int begin, end; - while (JobsUtility.GetWorkStealingRange(ref ranges, jobIndex, out begin, out end)) - ExecuteChunk(ref jobData, bufferRangePatchData, begin, end, chunks, entityIndices); - }} - else - {{ - ExecuteChunk(ref jobData, bufferRangePatchData, 0, chunkCount, chunks, entityIndices); - }} - }} - - static unsafe void ExecuteChunk(ref JobStruct_Process_{comboString} jobData, IntPtr bufferRangePatchData, int begin, int end, ArchetypeChunk* chunks, int* entityIndices) - {{ -{typeLookupCache} - for (var blockIndex = begin; blockIndex != end; ++blockIndex) - {{ - var chunk = chunks[blockIndex]; - int beginIndex = entityIndices[blockIndex]; - var count = chunk.Count; - #if ENABLE_UNITY_COLLECTIONS_CHECKS - JobsUtility.PatchBufferMinMaxRanges(bufferRangePatchData, UnsafeUtility.AddressOf(ref jobData), beginIndex, beginIndex + count); - #endif -{ptrs} - - for (var i = 0; i != count; i++) - {{ - jobData.Data.Execute({executeCallParams}); - }} - }} - }} - }} -" - ); - - return gen.ToString(); - } - } -} -#endif diff --git a/ScriptTemplates.meta b/Unity.Entities.Editor/ScriptTemplates.meta similarity index 100% rename from ScriptTemplates.meta rename to Unity.Entities.Editor/ScriptTemplates.meta diff --git a/Unity.Entities.Editor/ScriptTemplates/AuthoringComponent.txt b/Unity.Entities.Editor/ScriptTemplates/AuthoringComponent.txt new file mode 100644 index 00000000..9eb2e86b --- /dev/null +++ b/Unity.Entities.Editor/ScriptTemplates/AuthoringComponent.txt @@ -0,0 +1,36 @@ +using Unity.Entities; +using Unity.Mathematics; +using UnityEngine; + +[DisallowMultipleComponent] +[RequiresEntityConversion] +public class #SCRIPTNAME# : MonoBehaviour, IConvertGameObjectToEntity +{ + // Add fields to your component here. Remember that: + // + // * The purpose of this class is to store data for authoring purposes - it is not for use while the game is + // running. + // + // * Traditional Unity serialization rules apply: fields must be public or marked with [SerializeField], and + // must be one of the supported types. + // + // For example, + // public float scale; + #NOTRIM# + #NOTRIM# + + public void Convert(Entity entity, EntityManager dstManager, GameObjectConversionSystem conversionSystem) + { + // Call methods on 'dstManager' to create runtime components on 'entity' here. Remember that: + // + // * You can add more than one component to the entity. It's also OK to not add any at all. + // + // * If you want to create more than one entity from the data in this class, use the 'conversionSystem' + // to do it, instead of adding entities through 'dstManager' directly. + // + // For example, + // dstManager.AddComponentData(entity, new Unity.Transforms.Scale { Value = scale }); + #NOTRIM# + #NOTRIM# + } +} diff --git a/Unity.Entities.Editor/ScriptTemplates/AuthoringComponent.txt.meta b/Unity.Entities.Editor/ScriptTemplates/AuthoringComponent.txt.meta new file mode 100644 index 00000000..305e310e --- /dev/null +++ b/Unity.Entities.Editor/ScriptTemplates/AuthoringComponent.txt.meta @@ -0,0 +1,7 @@ +fileFormatVersion: 2 +guid: d47db0d26b3754ceca18d8e1a04f577a +TextScriptImporter: + externalObjects: {} + userData: + assetBundleName: + assetBundleVariant: diff --git a/ScriptTemplates/ECS__Component Type-NewComponentType.cs.txt b/Unity.Entities.Editor/ScriptTemplates/RunTimeComponent.txt similarity index 67% rename from ScriptTemplates/ECS__Component Type-NewComponentType.cs.txt rename to Unity.Entities.Editor/ScriptTemplates/RunTimeComponent.txt index fa08bab0..a96e461a 100644 --- a/ScriptTemplates/ECS__Component Type-NewComponentType.cs.txt +++ b/Unity.Entities.Editor/ScriptTemplates/RunTimeComponent.txt @@ -8,13 +8,17 @@ public struct #SCRIPTNAME# : IComponentData { // Add fields to your component here. Remember that: // - // * A component itself is for storing data and doesn't 'do' anything + // * A component itself is for storing data and doesn't 'do' anything. // - // * To act on the data, you will need a System + // * To act on the data, you will need a System. // // * Data in a component must be blittable, which means a component can // only contain fields which are primitive types or other blittable // structs; they cannot contain references to classes. + // + // * You should focus on the data structure that makes the most sense + // for runtime use here. Authoring Components will be used for + // authoring the data in the Editor. #NOTRIM# #NOTRIM# } diff --git a/ScriptTemplates/ECS__Component Type-NewComponentType.cs.txt.meta b/Unity.Entities.Editor/ScriptTemplates/RunTimeComponent.txt.meta similarity index 100% rename from ScriptTemplates/ECS__Component Type-NewComponentType.cs.txt.meta rename to Unity.Entities.Editor/ScriptTemplates/RunTimeComponent.txt.meta diff --git a/Unity.Entities.Editor/ScriptTemplates/ScriptTemplates.cs b/Unity.Entities.Editor/ScriptTemplates/ScriptTemplates.cs new file mode 100644 index 00000000..dbb875c3 --- /dev/null +++ b/Unity.Entities.Editor/ScriptTemplates/ScriptTemplates.cs @@ -0,0 +1,37 @@ +using System.Collections; +using System.Collections.Generic; +using UnityEditor; + +namespace Unity.Entities.Editor +{ + + internal class ScriptTemplates + { + public const string TemplatesRoot = "Packages/com.unity.entities/Unity.Entities.Editor/ScriptTemplates"; + + [MenuItem("Assets/Create/ECS/Runtime Component Type")] + public static void CreateRuntimeComponentType() + { + ProjectWindowUtil.CreateScriptAssetFromTemplateFile( + $"{TemplatesRoot}/RunTimeComponent.txt", + "NewComponent.cs"); + } + + [MenuItem("Assets/Create/ECS/Authoring Component Type")] + public static void CreateAuthoringComponentType() + { + ProjectWindowUtil.CreateScriptAssetFromTemplateFile( + $"{TemplatesRoot}/AuthoringComponent.txt", + "NewComponent.cs"); + } + + [MenuItem("Assets/Create/ECS/System")] + public static void CreateSystem() + { + ProjectWindowUtil.CreateScriptAssetFromTemplateFile( + $"{TemplatesRoot}/System.txt", + "NewSystem.cs"); + } + } + +} \ No newline at end of file diff --git a/Unity.Entities.Editor/CommonGUI/ComponentGroupGUI.cs.meta b/Unity.Entities.Editor/ScriptTemplates/ScriptTemplates.cs.meta similarity index 83% rename from Unity.Entities.Editor/CommonGUI/ComponentGroupGUI.cs.meta rename to Unity.Entities.Editor/ScriptTemplates/ScriptTemplates.cs.meta index 79798205..b803356e 100644 --- a/Unity.Entities.Editor/CommonGUI/ComponentGroupGUI.cs.meta +++ b/Unity.Entities.Editor/ScriptTemplates/ScriptTemplates.cs.meta @@ -1,5 +1,5 @@ fileFormatVersion: 2 -guid: 01aaabedd6a3443f1b0702e31df8e517 +guid: 3db8275d04ea14fd9a69331827c45292 MonoImporter: externalObjects: {} serializedVersion: 2 diff --git a/ScriptTemplates/ECS__System-NewSystem.cs.txt b/Unity.Entities.Editor/ScriptTemplates/System.txt similarity index 81% rename from ScriptTemplates/ECS__System-NewSystem.cs.txt rename to Unity.Entities.Editor/ScriptTemplates/System.txt index 78bc761f..63199072 100644 --- a/ScriptTemplates/ECS__System-NewSystem.cs.txt +++ b/Unity.Entities.Editor/ScriptTemplates/System.txt @@ -9,15 +9,15 @@ using static Unity.Mathematics.math; public class #SCRIPTNAME# : JobComponentSystem { // This declares a new kind of job, which is a unit of work to do. - // The job is declared as an IJobProcessComponentData, + // The job is declared as an IJobForEach, // meaning it will process all entities in the world that have both - // Position and Rotation components. Change it to process the component + // Translation and Rotation components. Change it to process the component // types you want. // // The job is also tagged with the BurstCompile attribute, which means // that the Burst compiler will optimize it for the best performance. [BurstCompile] - struct #SCRIPTNAME#Job : IJobProcessComponentData + struct #SCRIPTNAME#Job : IJobForEach { // Add fields here that your job needs to do its work. // For example, @@ -25,7 +25,7 @@ public class #SCRIPTNAME# : JobComponentSystem #NOTRIM# #NOTRIM# #NOTRIM# - public void Execute(ref Position position, [ReadOnly] ref Rotation rotation) + public void Execute(ref Translation translation, [ReadOnly] ref Rotation rotation) { // Implement the work to perform for each entity here. // You should only access data that is local or that is a @@ -34,7 +34,7 @@ public class #SCRIPTNAME# : JobComponentSystem // but allows this job to run in parallel with other jobs // that want to read Rotation component data. // For example, - // position.Value += mul(rotation.Value, new float3(0, 0, 1)) * deltaTime; + // translation.Value += mul(rotation.Value, new float3(0, 0, 1)) * deltaTime; #NOTRIM# #NOTRIM# } diff --git a/ScriptTemplates/ECS__System-NewSystem.cs.txt.meta b/Unity.Entities.Editor/ScriptTemplates/System.txt.meta similarity index 100% rename from ScriptTemplates/ECS__System-NewSystem.cs.txt.meta rename to Unity.Entities.Editor/ScriptTemplates/System.txt.meta diff --git a/Unity.Entities.Hybrid.Tests/ComponentDataProxy_EntityManager_IntegrationTests.cs b/Unity.Entities.Hybrid.Tests/ComponentDataProxy_EntityManager_IntegrationTests.cs index 573cf071..30f41fa9 100644 --- a/Unity.Entities.Hybrid.Tests/ComponentDataProxy_EntityManager_IntegrationTests.cs +++ b/Unity.Entities.Hybrid.Tests/ComponentDataProxy_EntityManager_IntegrationTests.cs @@ -125,7 +125,7 @@ public void ComponentDataProxy_WhenEntityManagerIsInvalid_SynchronizesWithEntity Type proxyType, object expectedValue, MockSetter setValue, MockGetter getValue ) { - var entityManager = new EntityManager(); + var entityManager = EntityManager.CreateEntityManagerInUninitializedState(); k_EntityManagerBackingField.SetValue(m_GameObjectEntity, entityManager); Assume.That(entityManager.IsCreated, Is.False, "EntityManager was not in correct state in test arrangement"); var proxy = m_GameObjectEntity.gameObject.AddComponent(proxyType) as ComponentDataProxyBase; diff --git a/Unity.Entities.Hybrid.Tests/ComponentGroupTransformAccessArrayTests.cs b/Unity.Entities.Hybrid.Tests/ComponentGroupTransformAccessArrayTests.cs index fec5c1da..7cf2d6c0 100644 --- a/Unity.Entities.Hybrid.Tests/ComponentGroupTransformAccessArrayTests.cs +++ b/Unity.Entities.Hybrid.Tests/ComponentGroupTransformAccessArrayTests.cs @@ -7,24 +7,6 @@ namespace Unity.Entities.Tests { class ComponentGroupTransformAccessArrayTests : ECSTestsFixture { -#pragma warning disable 618 - TransformAccessArrayInjectionHook m_TransformAccessArrayInjectionHook = new TransformAccessArrayInjectionHook(); - ComponentArrayInjectionHook m_ComponentArrayInjectionHook = new ComponentArrayInjectionHook(); -#pragma warning restore 618 - [OneTimeSetUp] - public void Init() - { - InjectionHookSupport.RegisterHook(m_ComponentArrayInjectionHook); - InjectionHookSupport.RegisterHook(m_TransformAccessArrayInjectionHook); - } - - [OneTimeTearDown] - public void Cleanup() - { - InjectionHookSupport.RegisterHook(m_TransformAccessArrayInjectionHook); - InjectionHookSupport.UnregisterHook(m_ComponentArrayInjectionHook); - } - public ComponentGroupTransformAccessArrayTests() { Assert.IsTrue(Unity.Jobs.LowLevel.Unsafe.JobsUtility.JobDebuggerEnabled, "JobDebugger must be enabled for these tests"); @@ -40,7 +22,7 @@ public class TransformAccessArrayTestTagProxy : ComponentDataProxy(); - var group = EmptySystem.GetComponentGroup(typeof(Transform), typeof(TransformAccessArrayTestTag)); + var group = EmptySystem.GetEntityQuery(typeof(Transform), typeof(TransformAccessArrayTestTag)); var ta = group.GetTransformAccessArray(); Assert.AreEqual(1, ta.length); @@ -60,7 +42,7 @@ public void AddAndGetNewTransformAccessArrayUpdatesContent() { var go = new GameObject(); go.AddComponent(); - var group = EmptySystem.GetComponentGroup(typeof(Transform), typeof(TransformAccessArrayTestTag)); + var group = EmptySystem.GetEntityQuery(typeof(Transform), typeof(TransformAccessArrayTestTag)); var ta = group.GetTransformAccessArray(); Assert.AreEqual(1, ta.length); @@ -78,7 +60,7 @@ public void AddAndUseOldTransformAccessArrayDoesNotUpdateContent() { var go = new GameObject(); go.AddComponent(); - var group = EmptySystem.GetComponentGroup(typeof(Transform), typeof(TransformAccessArrayTestTag)); + var group = EmptySystem.GetEntityQuery(typeof(Transform), typeof(TransformAccessArrayTestTag)); var ta = group.GetTransformAccessArray(); Assert.AreEqual(1, ta.length); @@ -97,7 +79,7 @@ public void DestroyAndGetNewTransformAccessArrayUpdatesContent() var go2 = new GameObject(); go2.AddComponent(); - var group = EmptySystem.GetComponentGroup(typeof(Transform), typeof(TransformAccessArrayTestTag)); + var group = EmptySystem.GetEntityQuery(typeof(Transform), typeof(TransformAccessArrayTestTag)); var ta = group.GetTransformAccessArray(); Assert.AreEqual(2, ta.length); @@ -117,7 +99,7 @@ public void DestroyAndUseOldTransformAccessArrayDoesNotUpdateContent() var go2 = new GameObject(); go2.AddComponent(); - var group = EmptySystem.GetComponentGroup(typeof(Transform), typeof(TransformAccessArrayTestTag)); + var group = EmptySystem.GetEntityQuery(typeof(Transform), typeof(TransformAccessArrayTestTag)); var ta = group.GetTransformAccessArray(); Assert.AreEqual(2, ta.length); @@ -127,89 +109,5 @@ public void DestroyAndUseOldTransformAccessArrayDoesNotUpdateContent() Object.DestroyImmediate(go2); } - - [DisableAutoCreation] - public class GameObjectArrayWithTransformAccessSystem : ComponentSystem - { - public ComponentGroup group; - protected override void OnUpdate() - { - } - - public new void UpdateInjectedComponentGroups() - { - base.UpdateInjectedComponentGroups(); - } - - protected override void OnCreateManager() - { - group = GetComponentGroup(typeof(Transform)); - } - } - - #pragma warning disable 618 - [Test] - public void GameObjectArrayWorksWithTransformAccessArray() - { - var hook = new GameObjectArrayInjectionHook(); - InjectionHookSupport.RegisterHook(hook); - - var go = new GameObject("test"); - GameObjectEntity.AddToEntityManager(m_Manager, go); - - var manager = World.GetOrCreateManager(); - - manager.UpdateInjectedComponentGroups(); - - Assert.AreEqual(1, manager.group.CalculateLength()); - Assert.AreEqual(go, manager.group.GetGameObjectArray()[0]); - Assert.AreEqual(go, manager.group.GetTransformAccessArray()[0].gameObject); - - Object.DestroyImmediate (go); - - InjectionHookSupport.UnregisterHook(hook); - - TearDown(); - } - #pragma warning restore 618 - - [DisableAutoCreation] - public class TransformWithTransformAccessSystem : ComponentSystem - { - public ComponentGroup group; - - protected override void OnUpdate() - { - } - - public new void UpdateInjectedComponentGroups() - { - base.UpdateInjectedComponentGroups(); - } - - protected override void OnCreateManager() - { - group = GetComponentGroup(typeof(Transform)); - } - } - - #pragma warning disable 618 - [Test] - public void TransformArrayWorksWithTransformAccessArray() - { - var go = new GameObject("test"); - GameObjectEntity.AddToEntityManager(m_Manager, go); - - var manager = World.GetOrCreateManager(); - - manager.UpdateInjectedComponentGroups(); - - Assert.AreEqual(1, manager.group.CalculateLength()); - Assert.AreEqual(manager.group.GetComponentArray()[0].gameObject, manager.group.GetTransformAccessArray()[0].gameObject); - - Object.DestroyImmediate (go); - TearDown(); - } - #pragma warning restore 618 } } diff --git a/Unity.Entities.Hybrid.Tests/ComponentSystem_GameObject_IntegrationTests.cs b/Unity.Entities.Hybrid.Tests/ComponentSystem_GameObject_IntegrationTests.cs deleted file mode 100644 index 5fba114d..00000000 --- a/Unity.Entities.Hybrid.Tests/ComponentSystem_GameObject_IntegrationTests.cs +++ /dev/null @@ -1,122 +0,0 @@ -using System; -using NUnit.Framework; -using Unity.Entities; -using Unity.Entities.Tests; -#pragma warning disable 649 -#pragma warning disable 618 - -namespace UnityEngine.Entities.Tests -{ - class ComponentSystem_GameObject_IntegrationTests : ECSTestsFixture - { - ComponentArrayInjectionHook m_ComponentArrayInjectionHook = new ComponentArrayInjectionHook(); - GameObjectArrayInjectionHook m_GameObjectArrayInjectionHook = new GameObjectArrayInjectionHook(); - - [OneTimeSetUp] - public void Init() - { - InjectionHookSupport.RegisterHook(m_ComponentArrayInjectionHook); - InjectionHookSupport.RegisterHook(m_GameObjectArrayInjectionHook); - } - - [OneTimeTearDown] - public void Cleanup() - { - InjectionHookSupport.UnregisterHook(m_GameObjectArrayInjectionHook); - InjectionHookSupport.RegisterHook(m_ComponentArrayInjectionHook); - } - - GameObject m_GameObject; - - [SetUp] - public override void Setup() - { - base.Setup(); - m_GameObject = new GameObject(TestContext.CurrentContext.Test.Name); - } - - [TearDown] - public override void TearDown() - { - base.TearDown(); - if (m_GameObject != null) - GameObject.DestroyImmediate(m_GameObject); - } - - [DisableAutoCreation] - public class GameObjectArraySystem : ComponentSystem - { - public struct Group - { - public readonly int Length; - public GameObjectArray gameObjects; - - public ComponentArray colliders; - } - - [Inject] - public Group group; - - protected override void OnUpdate() - { - } - - public new void UpdateInjectedComponentGroups() - { - base.UpdateInjectedComponentGroups(); - } - } - - [Test] - public void UpdateInjectedComponentGroups_WhenObjectWithMatchingComponentExists_GameObjectArrayIsPopulated() - { - m_GameObject.AddComponent(); - GameObjectEntity.AddToEntityManager(m_Manager, m_GameObject); - var manager = World.GetOrCreateManager(); - - manager.UpdateInjectedComponentGroups(); - - Assert.That(manager.group.gameObjects.ToArray(), Is.EqualTo(new[] { m_GameObject })); - } - - [Test] - public void UpdateInjectedComponentGroups_WhenObjectWithMatchingComponentExists_ComponentArrayIsPopulated() - { - var c = m_GameObject.AddComponent(); - GameObjectEntity.AddToEntityManager(m_Manager, m_GameObject); - var manager = World.GetOrCreateManager(); - - manager.UpdateInjectedComponentGroups(); - - Assert.That(manager.group.colliders.ToArray(), Is.EqualTo(new[] { c })); - } - - [Test] - public void GetComponentGroup_WhenObjectWithMatchingComponentExists_ComponentArrayIsPopulated() - { - var rb = m_GameObject.AddComponent(); - GameObjectEntity.AddToEntityManager(m_Manager, m_GameObject); - - var grp = EmptySystem.GetComponentGroup(typeof(Rigidbody)); - - var array = grp.GetComponentArray(); - Assert.That(array.ToArray(), Is.EqualTo(new[] { rb })); - } - - [Test] - public void GetComponentGroup_WhenObjectWithMatchingComponentAndECSDataExists_ComponentArraysArePopulated() - { - var goe = m_GameObject.AddComponent().GetComponent(); - var expectedTestData = new EcsTestData(5); - m_Manager.SetComponentData(goe.Entity, expectedTestData); - - var grp = EmptySystem.GetComponentGroup(typeof(Transform), typeof(EcsTestData)); - - var transformArray = grp.GetComponentArray(); - Assert.That(transformArray.ToArray(), Is.EqualTo(new[] { goe.transform })); - var ecsDataArray = grp.GetComponentDataArray(); - Assert.That(ecsDataArray.Length, Is.EqualTo(1)); - Assert.That(ecsDataArray[0], Is.EqualTo(expectedTestData)); - } - } -} diff --git a/Unity.Entities.Hybrid.Tests/ComponentSystem_GameObject_IntegrationTests.cs.meta b/Unity.Entities.Hybrid.Tests/ComponentSystem_GameObject_IntegrationTests.cs.meta deleted file mode 100644 index ce578ad6..00000000 --- a/Unity.Entities.Hybrid.Tests/ComponentSystem_GameObject_IntegrationTests.cs.meta +++ /dev/null @@ -1,11 +0,0 @@ -fileFormatVersion: 2 -guid: f12b2ef99b3ca499b9c97e13baf3c0a3 -MonoImporter: - externalObjects: {} - serializedVersion: 2 - defaultReferences: [] - executionOrder: 0 - icon: {instanceID: 0} - userData: - assetBundleName: - assetBundleVariant: diff --git a/Unity.Entities.Hybrid.Tests/ConversionMappingTests.cs b/Unity.Entities.Hybrid.Tests/ConversionMappingTests.cs index f132c2b7..db0fc478 100644 --- a/Unity.Entities.Hybrid.Tests/ConversionMappingTests.cs +++ b/Unity.Entities.Hybrid.Tests/ConversionMappingTests.cs @@ -15,7 +15,7 @@ public override void Setup() { base.Setup(); dstWorld = new World("Test dst world"); - mappingSystem = World.CreateManager(dstWorld, default(Unity.Entities.Hash128), GameObjectConversionUtility.ConversionFlags.None); + mappingSystem = World.CreateSystem(dstWorld, default(Unity.Entities.Hash128), GameObjectConversionUtility.ConversionFlags.None); } [TearDown] diff --git a/Unity.Entities.Hybrid.Tests/Runtime/GameObjectEntityTests.cs b/Unity.Entities.Hybrid.Tests/Runtime/GameObjectEntityTests.cs index 83a8d215..ba2ff279 100644 --- a/Unity.Entities.Hybrid.Tests/Runtime/GameObjectEntityTests.cs +++ b/Unity.Entities.Hybrid.Tests/Runtime/GameObjectEntityTests.cs @@ -52,28 +52,6 @@ public void AddComponentDuringForeachProtection() { //* Check for string in MyEntity and other illegal constructs... } - [Test] - unsafe public void ComponentEnumerator() - { - var go = new GameObject("test", typeof(Rigidbody), typeof(Light)); - var entity = GameObjectEntity.AddToEntityManager(m_Manager, go); - - m_Manager.AddComponentData(entity, new EcsTestData(5)); - m_Manager.AddComponentData(entity, new EcsTestData2(6)); - - int iterations = 0; - foreach (var e in EmptySystem.GetEntities() ) - { - Assert.AreEqual(5, e.testData->value); - Assert.AreEqual(6, e.testData2->value0); - Assert.AreEqual(go.GetComponent(), e.light); - Assert.AreEqual(go.GetComponent(), e.rigidbody); - iterations++; - } - Assert.AreEqual(1, iterations); - - Object.DestroyImmediate(go); - } [Test] public void AddRemoveGetComponentObject() diff --git a/Unity.Entities.Hybrid.Tests/Runtime/InjectComponentGroupTestsHybrid.cs b/Unity.Entities.Hybrid.Tests/Runtime/InjectComponentGroupTestsHybrid.cs deleted file mode 100644 index 337bcb3a..00000000 --- a/Unity.Entities.Hybrid.Tests/Runtime/InjectComponentGroupTestsHybrid.cs +++ /dev/null @@ -1,58 +0,0 @@ -using NUnit.Framework; -using UnityEngine; -#pragma warning disable 649 -#pragma warning disable 618 - -namespace Unity.Entities.Tests -{ - class InjectComponentGroupTestsHybrid : ECSTestsFixture - { - [DisableAutoCreation] - [AlwaysUpdateSystem] - public class ExcludeSystem : ComponentSystem - { - public struct Datas - { - public ComponentDataArray Data; - public ExcludeComponent Data2; - public ExcludeComponent Rigidbody; - } - - [Inject] - public Datas Group; - - protected override void OnUpdate() - { - } - } - - [Test] - public void ExcludeComponent() - { - var subtractiveSystem = World.GetOrCreateManager (); - - var entity = m_Manager.CreateEntity (typeof(EcsTestData)); - - var go = new GameObject("Test", typeof(EcsTestProxy)); - - // Ensure entities without the subtractive components are present - subtractiveSystem.Update (); - Assert.AreEqual (2, subtractiveSystem.Group.Data.Length); - Assert.AreEqual (0, subtractiveSystem.Group.Data[0].value); - Assert.AreEqual (0, subtractiveSystem.Group.Data[1].value); - - // Ensure adding the subtractive components, removes them from the injection - m_Manager.AddComponentData (entity, new EcsTestData2()); - - // TODO: This should be automatic... - go.AddComponent(); - go.GetComponent().enabled = false; - go.GetComponent().enabled = true; - - subtractiveSystem.Update (); - Assert.AreEqual (0, subtractiveSystem.Group.Data.Length); - - Object.DestroyImmediate(go); - } - } -} diff --git a/Unity.Entities.Hybrid.Tests/Runtime/InjectComponentGroupTestsHybrid.cs.meta b/Unity.Entities.Hybrid.Tests/Runtime/InjectComponentGroupTestsHybrid.cs.meta deleted file mode 100644 index e9f18fe8..00000000 --- a/Unity.Entities.Hybrid.Tests/Runtime/InjectComponentGroupTestsHybrid.cs.meta +++ /dev/null @@ -1,11 +0,0 @@ -fileFormatVersion: 2 -guid: 241a4da0c00ab408d84c63e4dd20a864 -MonoImporter: - externalObjects: {} - serializedVersion: 2 - defaultReferences: [] - executionOrder: 0 - icon: {instanceID: 0} - userData: - assetBundleName: - assetBundleVariant: diff --git a/Unity.Entities.Hybrid.Tests/Runtime/SharedComponentSerializeTests.cs b/Unity.Entities.Hybrid.Tests/Runtime/SharedComponentSerializeTests.cs index 95075ddd..1b3a2db7 100644 --- a/Unity.Entities.Hybrid.Tests/Runtime/SharedComponentSerializeTests.cs +++ b/Unity.Entities.Hybrid.Tests/Runtime/SharedComponentSerializeTests.cs @@ -61,9 +61,9 @@ public void SharedComponentSerialize() var reader = new TestBinaryReader(writer); var world = new World("temp"); - SerializeUtilityHybrid.Deserialize (world.GetOrCreateManager(), reader, sharedComponents); + SerializeUtilityHybrid.Deserialize (world.EntityManager, reader, sharedComponents); - var newWorldEntities = world.GetOrCreateManager(); + var newWorldEntities = world.EntityManager; { var entities = newWorldEntities.GetAllEntities(); diff --git a/Unity.Entities.Hybrid/Components/GameObjectEntity.cs b/Unity.Entities.Hybrid/Components/GameObjectEntity.cs index b79d4742..b40f64ec 100644 --- a/Unity.Entities.Hybrid/Components/GameObjectEntity.cs +++ b/Unity.Entities.Hybrid/Components/GameObjectEntity.cs @@ -154,7 +154,7 @@ void Initialize() DefaultWorldInitialization.DefaultLazyEditModeInitialize(); if (World.Active != null) { - m_EntityManager = World.Active.GetOrCreateManager(); + m_EntityManager = World.Active.EntityManager; m_Entity = AddToEntityManager(m_EntityManager, gameObject); } } diff --git a/Unity.Entities.Hybrid/GameObjectConversion/GameObjectConversionMappingSystem.cs b/Unity.Entities.Hybrid/GameObjectConversion/GameObjectConversionMappingSystem.cs index c84085b7..4e8a6370 100644 --- a/Unity.Entities.Hybrid/GameObjectConversion/GameObjectConversionMappingSystem.cs +++ b/Unity.Entities.Hybrid/GameObjectConversion/GameObjectConversionMappingSystem.cs @@ -76,7 +76,6 @@ public void Dispose() } [DisableAutoCreation] -[Preserve] class GameObjectConversionMappingSystem : ComponentSystem { NativeHashMap m_GameObjectToEntity = new NativeHashMap(100 * 1000, Allocator.Persistent); @@ -103,14 +102,14 @@ public GameObjectConversionMappingSystem(World DstWorld, Hash128 sceneGUID, Conv m_DstWorld = DstWorld; m_SceneGUID = sceneGUID; m_ConversionFlags = conversionFlags; - m_DstManager = DstWorld.GetOrCreateManager(); + m_DstManager = DstWorld.EntityManager; m_Entities = new Entity[128]; m_Next = new int[128]; m_EntitiesCount = 0; } - protected override void OnDestroyManager() + protected override void OnDestroy() { m_GameObjectToEntity.Dispose(); } @@ -354,7 +353,7 @@ internal static void CreateEntitiesForGameObjectsRecurse(Transform transform, En internal static void CreateEntitiesForGameObjects(Scene scene, World gameObjectWorld) { - var entityManager = gameObjectWorld.GetOrCreateManager(); + var entityManager = gameObjectWorld.EntityManager; var gameObjects = scene.GetRootGameObjects(); foreach (var go in gameObjects) @@ -365,4 +364,4 @@ protected override void OnUpdate() { } -} \ No newline at end of file +} diff --git a/Unity.Entities.Hybrid/GameObjectConversion/GameObjectConversionSystem.cs b/Unity.Entities.Hybrid/GameObjectConversion/GameObjectConversionSystem.cs index 06d8fc48..d9a2d815 100644 --- a/Unity.Entities.Hybrid/GameObjectConversion/GameObjectConversionSystem.cs +++ b/Unity.Entities.Hybrid/GameObjectConversion/GameObjectConversionSystem.cs @@ -11,13 +11,13 @@ public abstract class GameObjectConversionSystem : ComponentSystem GameObjectConversionMappingSystem m_MappingSystem; - protected override void OnCreateManager() + protected override void OnCreate() { - base.OnCreateManager(); + base.OnCreate(); - m_MappingSystem = World.GetOrCreateManager(); + m_MappingSystem = World.GetOrCreateSystem(); DstWorld = m_MappingSystem.DstWorld; - DstEntityManager = DstWorld.GetOrCreateManager(); + DstEntityManager = DstWorld.EntityManager; } public Entity GetPrimaryEntity(Component component) diff --git a/Unity.Entities.Hybrid/GameObjectConversion/GameObjectConversionUtility.cs b/Unity.Entities.Hybrid/GameObjectConversion/GameObjectConversionUtility.cs index 15e1c5e2..bb28ae6b 100644 --- a/Unity.Entities.Hybrid/GameObjectConversion/GameObjectConversionUtility.cs +++ b/Unity.Entities.Hybrid/GameObjectConversion/GameObjectConversionUtility.cs @@ -56,7 +56,7 @@ internal static World CreateConversionWorld(World dstEntityWorld, Hash128 scene m_CreateConversionWorld.Begin(); var gameObjectWorld = new World("GameObject World"); - gameObjectWorld.CreateManager(dstEntityWorld, sceneGUID, conversionFlags); + gameObjectWorld.CreateSystem(dstEntityWorld, sceneGUID, conversionFlags); AddConversionSystems(gameObjectWorld); @@ -68,14 +68,14 @@ internal static World CreateConversionWorld(World dstEntityWorld, Hash128 scene internal static void Convert(World gameObjectWorld, World dstEntityWorld) { - var mappingSystem = gameObjectWorld.GetExistingManager(); + var mappingSystem = gameObjectWorld.GetExistingSystem(); using (m_UpdateSystems.Auto()) { // Convert all the data into dstEntityWorld - gameObjectWorld.GetExistingManager().Update(); - gameObjectWorld.GetExistingManager().Update(); - gameObjectWorld.GetExistingManager().Update(); + gameObjectWorld.GetExistingSystem().Update(); + gameObjectWorld.GetExistingSystem().Update(); + gameObjectWorld.GetExistingSystem().Update(); } using (m_AddPrefabComponentDataTag.Auto()) @@ -86,7 +86,7 @@ internal static void Convert(World gameObjectWorld, World dstEntityWorld) internal static Entity GameObjectToConvertedEntity(World gameObjectWorld, GameObject gameObject) { - var mappingSystem = gameObjectWorld.GetExistingManager(); + var mappingSystem = gameObjectWorld.GetExistingSystem(); return mappingSystem.GetPrimaryEntity(gameObject); } @@ -98,7 +98,7 @@ public static Entity ConvertGameObjectHierarchy(GameObject root, World dstEntity Entity convertedEntity; using (var gameObjectWorld = CreateConversionWorld(dstEntityWorld, default(Hash128), 0)) { - var mappingSystem = gameObjectWorld.GetExistingManager(); + var mappingSystem = gameObjectWorld.GetExistingSystem(); using (m_CreateEntitiesForGameObjects.Auto()) { @@ -139,13 +139,13 @@ public static void ConvertScene(Scene scene, Hash128 sceneHash, World dstEntityW static void AddConversionSystems(World gameObjectWorld) { - var init = gameObjectWorld.GetOrCreateManager(); - var convert = gameObjectWorld.GetOrCreateManager(); - var afterConvert = gameObjectWorld.GetOrCreateManager(); + var init = gameObjectWorld.GetOrCreateSystem(); + var convert = gameObjectWorld.GetOrCreateSystem(); + var afterConvert = gameObjectWorld.GetOrCreateSystem(); // Ensure the following systems run first in this order... - init.AddSystemToUpdateList(gameObjectWorld.GetOrCreateManager()); - init.AddSystemToUpdateList(gameObjectWorld.GetOrCreateManager()); + init.AddSystemToUpdateList(gameObjectWorld.GetOrCreateSystem()); + init.AddSystemToUpdateList(gameObjectWorld.GetOrCreateSystem()); var systems = DefaultWorldInitialization.GetAllSystems(WorldSystemFilterFlags.GameObjectConversion); foreach (var system in systems) @@ -185,7 +185,7 @@ static void AddSystemAndLogException(World world, ComponentSystemGroup group, Ty { try { - group.AddSystemToUpdateList(world.GetOrCreateManager(type) as ComponentSystemBase); + group.AddSystemToUpdateList(world.GetOrCreateSystem(type) as ComponentSystemBase); } catch (Exception e) { diff --git a/Unity.Entities.Hybrid/Injection/DefaultWorldInitialization.cs b/Unity.Entities.Hybrid/Injection/DefaultWorldInitialization.cs index 7235162c..9cba197e 100644 --- a/Unity.Entities.Hybrid/Injection/DefaultWorldInitialization.cs +++ b/Unity.Entities.Hybrid/Injection/DefaultWorldInitialization.cs @@ -26,11 +26,11 @@ static void DomainUnloadShutdown() ScriptBehaviourUpdateOrder.UpdatePlayerLoop(null); } - static ScriptBehaviourManager GetOrCreateManagerAndLogException(World world, Type type) + static ComponentSystemBase GetOrCreateManagerAndLogException(World world, Type type) { try { - return world.GetOrCreateManager(type); + return world.GetOrCreateSystem(type); } catch (Exception e) { @@ -41,13 +41,6 @@ static ScriptBehaviourManager GetOrCreateManagerAndLogException(World world, Typ public static void Initialize(string worldName, bool editorWorld) { - // Register hybrid injection hooks - #pragma warning disable 0618 - InjectionHookSupport.RegisterHook(new GameObjectArrayInjectionHook()); - InjectionHookSupport.RegisterHook(new TransformAccessArrayInjectionHook()); - InjectionHookSupport.RegisterHook(new ComponentArrayInjectionHook()); - #pragma warning restore 0618 - PlayerLoopManager.RegisterDomainUnload(DomainUnloadShutdown, 10000); var world = new World(worldName); @@ -64,9 +57,9 @@ public static void Initialize(string worldName, bool editorWorld) } // create presentation system and simulation system - InitializationSystemGroup initializationSystemGroup = world.GetOrCreateManager(); - SimulationSystemGroup simulationSystemGroup = world.GetOrCreateManager(); - PresentationSystemGroup presentationSystemGroup = world.GetOrCreateManager(); + InitializationSystemGroup initializationSystemGroup = world.GetOrCreateSystem(); + SimulationSystemGroup simulationSystemGroup = world.GetOrCreateSystem(); + PresentationSystemGroup presentationSystemGroup = world.GetOrCreateSystem(); // Add systems to their groups, based on the [UpdateInGroup] attribute. foreach (var type in systems) { diff --git a/Unity.Entities.Hybrid/Iterators/ComponentArray.cs b/Unity.Entities.Hybrid/Iterators/ComponentArray.cs index 8602e784..0663b66c 100644 --- a/Unity.Entities.Hybrid/Iterators/ComponentArray.cs +++ b/Unity.Entities.Hybrid/Iterators/ComponentArray.cs @@ -7,26 +7,15 @@ namespace Unity.Entities { - public static class ComponentGroupExtensionsForComponentArray + public static class EntityQueryExtensionsForComponentArray { - [Obsolete("GetComponentArray has been deprecated. Use ComponentSystem.ForEach to access managed components.")] - public static ComponentArray GetComponentArray(this ComponentGroup group) where T : Component + public static T[] ToComponentArray(this EntityQuery group) where T : Component { int length = group.CalculateLength(); ComponentChunkIterator iterator = group.GetComponentChunkIterator(); - var indexInComponentGroup = group.GetIndexInComponentGroup(TypeManager.GetTypeIndex()); + var indexInComponentGroup = group.GetIndexInEntityQuery(TypeManager.GetTypeIndex()); - iterator.IndexInComponentGroup = indexInComponentGroup; - return new ComponentArray(iterator, length, group.ArchetypeManager); - } - - public static T[] ToComponentArray(this ComponentGroup group) where T : Component - { - int length = group.CalculateLength(); - ComponentChunkIterator iterator = group.GetComponentChunkIterator(); - var indexInComponentGroup = group.GetIndexInComponentGroup(TypeManager.GetTypeIndex()); - - iterator.IndexInComponentGroup = indexInComponentGroup; + iterator.IndexInEntityQuery = indexInComponentGroup; var arr = new T[length]; var cache = default(ComponentChunkCache); @@ -34,7 +23,7 @@ public static T[] ToComponentArray(this ComponentGroup group) where T : Compo { if (i < cache.CachedBeginIndex || i >= cache.CachedEndIndex) iterator.MoveToEntityIndexAndUpdateCache(i, out cache, true); - + arr[i] = (T)iterator.GetManagedObject(group.ArchetypeManager, cache.CachedBeginIndex, i); } @@ -42,116 +31,3 @@ public static T[] ToComponentArray(this ComponentGroup group) where T : Compo } } } - -namespace Unity.Entities -{ - [Obsolete("ComponentArray has been deprecated. Use ComponentSystem.ForEach to access managed components.")] - public struct ComponentArray where T: Component - { - ComponentChunkIterator m_Iterator; - ComponentChunkCache m_Cache; - readonly int m_Length; - readonly ArchetypeManager m_ArchetypeManager; - - internal ComponentArray(ComponentChunkIterator iterator, int length, ArchetypeManager typeMan) - { - m_Length = length; - m_Cache = default(ComponentChunkCache); - m_Iterator = iterator; - m_ArchetypeManager = typeMan; - } - - public T this[int index] - { - get - { - //@TODO: Unnecessary.. integrate into cache instead... - if ((uint)index >= (uint)m_Length) - FailOutOfRangeError(index); - - if (index < m_Cache.CachedBeginIndex || index >= m_Cache.CachedEndIndex) - m_Iterator.MoveToEntityIndexAndUpdateCache(index, out m_Cache, true); - - return (T)m_Iterator.GetManagedObject(m_ArchetypeManager, m_Cache.CachedBeginIndex, index); - } - } - - public T[] ToArray() - { - var arr = new T[m_Length]; - var i = 0; - while (i < m_Length) - { - m_Iterator.MoveToEntityIndexAndUpdateCache(i, out m_Cache, true); - int start, length; - var objs = m_Iterator.GetManagedObjectRange(m_ArchetypeManager, m_Cache.CachedBeginIndex, i, out start, out length); - for (var obj = 0; obj < length; ++obj) - arr[i+obj] = (T)objs[start+obj]; - i += length; - } - return arr; - } - - void FailOutOfRangeError(int index) - { - throw new IndexOutOfRangeException($"Index {index} is out of range of '{Length}' Length."); - } - - public int Length => m_Length; - } - - [Preserve] - [CustomInjectionHook] - [Obsolete("ComponentArray and injection have been deprecated. Use ComponentSystem.ForEach to access managed components.")] - sealed class ComponentArrayInjectionHook : InjectionHook - { - public override Type FieldTypeOfInterest => typeof(ComponentArray<>); - - public override bool IsInterestedInField(FieldInfo fieldInfo) - { - return fieldInfo.FieldType.IsGenericType && fieldInfo.FieldType.GetGenericTypeDefinition() == typeof(ComponentArray<>); - } - - public override string ValidateField(FieldInfo field, bool isReadOnly, InjectionContext injectionInfo) - { - if (field.FieldType != typeof(ComponentArray<>)) - return null; - - if (isReadOnly) - return "[ReadOnly] may not be used on ComponentArray<>, it can only be used on ComponentDataArray<>"; - - return null; - } - - public override InjectionContext.Entry CreateInjectionInfoFor(FieldInfo field, bool isReadOnly) - { - var componentType = field.FieldType.GetGenericArguments()[0]; - var accessMode = isReadOnly ? ComponentType.AccessMode.ReadOnly : ComponentType.AccessMode.ReadWrite; - return new InjectionContext.Entry - { - Hook = this, - FieldInfo = field, - IsReadOnly = false /* isReadOnly */, - AccessMode = accessMode, - IndexInComponentGroup = -1, - FieldOffset = UnsafeUtility.GetFieldOffset(field), - ComponentType = new ComponentType(componentType, accessMode), - ComponentRequirements = componentType == typeof(Transform) - ? new[] { typeof(Transform), componentType } - : new []{ componentType } - }; - } - - public override void PrepareEntry(ref InjectionContext.Entry entry, ComponentGroup entityGroup) - { - entry.IndexInComponentGroup = entityGroup.GetIndexInComponentGroup(entry.ComponentType.TypeIndex); - } - - internal override unsafe void InjectEntry(InjectionContext.Entry entry, ComponentGroup entityGroup, ref ComponentChunkIterator iterator, int length, byte* groupStructPtr) - { - iterator.IndexInComponentGroup = entry.IndexInComponentGroup; - var data = new ComponentArray(iterator, length, entityGroup.ArchetypeManager); - UnsafeUtility.CopyStructureToPtr(ref data, groupStructPtr + entry.FieldOffset); - } - } -} diff --git a/Unity.Entities.Hybrid/Iterators/GameObjectArray.cs b/Unity.Entities.Hybrid/Iterators/GameObjectArray.cs deleted file mode 100644 index 20141d8f..00000000 --- a/Unity.Entities.Hybrid/Iterators/GameObjectArray.cs +++ /dev/null @@ -1,130 +0,0 @@ -using System; -using System.Linq; -using System.Reflection; - -using Unity.Collections.LowLevel.Unsafe; -using UnityEngine; -using UnityEngine.Scripting; - -namespace Unity.Entities -{ - public static class ComponentGroupExtensionsForGameObjectArray - { - [Obsolete("GetGameObjectArray has been deprecated. Use ComponentSystem.ForEach or ToTransformArray()[index].gameObject to access entity GameObjects.")] - public static GameObjectArray GetGameObjectArray(this ComponentGroup group) - { - int length = group.CalculateLength(); - ComponentChunkIterator iterator = group.GetComponentChunkIterator(); - iterator.IndexInComponentGroup = group.GetIndexInComponentGroup(TypeManager.GetTypeIndex()); - return new GameObjectArray(iterator, length, group.ArchetypeManager); - } - } -} - -namespace Unity.Entities -{ - [Obsolete("GameObjectArray has been deprecated. Use ComponentSystem.ForEach or ToTransformArray()[index].gameObject to access entity GameObjects.")] - public struct GameObjectArray - { - ComponentChunkIterator m_Iterator; - ComponentChunkCache m_Cache; - readonly int m_Length; - readonly ArchetypeManager m_ArchetypeManager; - - internal GameObjectArray(ComponentChunkIterator iterator, int length, ArchetypeManager typeMan) - { - m_Length = length; - m_Cache = default(ComponentChunkCache); - m_Iterator = iterator; - m_ArchetypeManager = typeMan; - } - - public GameObject this[int index] - { - get - { - //@TODO: Unnecessary.. integrate into cache instead... - if ((uint)index >= (uint)m_Length) - FailOutOfRangeError(index); - - if (index < m_Cache.CachedBeginIndex || index >= m_Cache.CachedEndIndex) - m_Iterator.MoveToEntityIndexAndUpdateCache(index, out m_Cache, true); - - var transform = (Transform) m_Iterator.GetManagedObject(m_ArchetypeManager, m_Cache.CachedBeginIndex, index); - - return transform.gameObject; - } - } - - public GameObject[] ToArray() - { - var arr = new GameObject[m_Length]; - var i = 0; - while (i < m_Length) - { - m_Iterator.MoveToEntityIndexAndUpdateCache(i, out m_Cache, true); - int start, length; - var objs = m_Iterator.GetManagedObjectRange(m_ArchetypeManager, m_Cache.CachedBeginIndex, i, out start, out length); - for (var obj = 0; obj < length; ++obj) - arr[i+obj] = ((Transform) objs[start + obj]).gameObject; - i += length; - } - return arr; - } - - void FailOutOfRangeError(int index) - { - throw new IndexOutOfRangeException($"Index {index} is out of range of '{Length}' Length."); - } - - public int Length => m_Length; - } - - [Preserve] - [CustomInjectionHook] - [Obsolete("GameObjectArray and injection have been deprecated. Use ComponentSystem.ForEach or ToTransformArray()[index].gameObject to access entity GameObjects.")] - sealed class GameObjectArrayInjectionHook : InjectionHook - { - public override Type FieldTypeOfInterest => typeof(GameObjectArray); - - public override bool IsInterestedInField(FieldInfo fieldInfo) - { - return fieldInfo.FieldType == typeof(GameObjectArray); - } - - public override string ValidateField(FieldInfo field, bool isReadOnly, InjectionContext injectionInfo) - { - if (field.FieldType != typeof(GameObjectArray)) - return null; - - if (isReadOnly) - return "[ReadOnly] may not be used on GameObjectArray, it can only be used on ComponentDataArray<>"; - - // Error on multiple GameObjectArray - if (injectionInfo.Entries.Any(i => i.FieldInfo.FieldType == typeof(GameObjectArray))) - return "A [Inject] struct, may only contain a single GameObjectArray"; - - return null; - } - - public override InjectionContext.Entry CreateInjectionInfoFor(FieldInfo field, bool isReadOnly) - { - return new InjectionContext.Entry - { - Hook = this, - FieldInfo = field, - IsReadOnly = isReadOnly, - AccessMode = isReadOnly ? ComponentType.AccessMode.ReadOnly : ComponentType.AccessMode.ReadWrite, - IndexInComponentGroup = -1, - FieldOffset = UnsafeUtility.GetFieldOffset(field), - ComponentRequirements = new[] { typeof(Transform) } - }; - } - - internal override unsafe void InjectEntry(InjectionContext.Entry entry, ComponentGroup entityGroup, ref ComponentChunkIterator iterator, int length, byte* groupStructPtr) - { - var gameObjectArray = entityGroup.GetGameObjectArray(); - UnsafeUtility.CopyStructureToPtr(ref gameObjectArray, groupStructPtr + entry.FieldOffset); - } - } -} diff --git a/Unity.Entities.Hybrid/Iterators/GameObjectArray.cs.meta b/Unity.Entities.Hybrid/Iterators/GameObjectArray.cs.meta deleted file mode 100644 index 3fca4764..00000000 --- a/Unity.Entities.Hybrid/Iterators/GameObjectArray.cs.meta +++ /dev/null @@ -1,11 +0,0 @@ -fileFormatVersion: 2 -guid: 5558e38bbc5c843679c87e532e532994 -MonoImporter: - externalObjects: {} - serializedVersion: 2 - defaultReferences: [] - executionOrder: 0 - icon: {instanceID: 0} - userData: - assetBundleName: - assetBundleVariant: diff --git a/Unity.Entities.Hybrid/Iterators/TransformAccessArrayIterator.cs b/Unity.Entities.Hybrid/Iterators/TransformAccessArrayIterator.cs index 13d19933..d1062b5f 100644 --- a/Unity.Entities.Hybrid/Iterators/TransformAccessArrayIterator.cs +++ b/Unity.Entities.Hybrid/Iterators/TransformAccessArrayIterator.cs @@ -21,9 +21,9 @@ public void Dispose() } } - public static class ComponentGroupExtensionsForTransformAccessArray + public static class EntityQueryExtensionsForTransformAccessArray { - public static unsafe TransformAccessArray GetTransformAccessArray(this ComponentGroup group) + public static unsafe TransformAccessArray GetTransformAccessArray(this EntityQuery group) { var state = (TransformAccessArrayState?)group.m_CachedState ?? new TransformAccessArrayState(); var orderVersion = group.EntityDataManager->GetComponentTypeOrderVersion(TypeManager.GetTypeIndex()); @@ -47,51 +47,3 @@ public static unsafe TransformAccessArray GetTransformAccessArray(this Component } } } - -namespace Unity.Entities -{ - [Preserve] - [CustomInjectionHook] - [Obsolete("Injection has been deprecated. Use ComponentGroup.GetTransformAccessArray instead.")] - sealed class TransformAccessArrayInjectionHook : InjectionHook - { - public override Type FieldTypeOfInterest => typeof(TransformAccessArray); - - public override bool IsInterestedInField(FieldInfo fieldInfo) - { - return fieldInfo.FieldType == typeof(TransformAccessArray); - } - - public override string ValidateField(FieldInfo field, bool isReadOnly, InjectionContext injectionInfo) - { - if (isReadOnly) - return "[ReadOnly] may not be used on a TransformAccessArray only on ComponentDataArray<>"; - - // Error on multiple TransformAccessArray - if (injectionInfo.Entries.Any(i => i.FieldInfo.FieldType == typeof(TransformAccessArray))) - return "A [Inject] struct, may only contain a single TransformAccessArray"; - - return null; - } - - public override InjectionContext.Entry CreateInjectionInfoFor(FieldInfo field, bool isReadOnly) - { - return new InjectionContext.Entry - { - Hook = this, - FieldInfo = field, - IsReadOnly = isReadOnly, - AccessMode = isReadOnly ? ComponentType.AccessMode.ReadOnly : ComponentType.AccessMode.ReadWrite, - IndexInComponentGroup = -1, - FieldOffset = UnsafeUtility.GetFieldOffset(field), - ComponentRequirements = new[] { typeof(Transform) } - }; - } - - internal override unsafe void InjectEntry(InjectionContext.Entry entry, ComponentGroup entityGroup, ref ComponentChunkIterator iterator, int length, byte* groupStructPtr) - { - var transformsArray = entityGroup.GetTransformAccessArray(); - UnsafeUtility.CopyStructureToPtr(ref transformsArray, groupStructPtr + entry.FieldOffset); - } - } -} diff --git a/Unity.Entities.PerformanceTests/EntityCommandBufferPerformanceTests.cs b/Unity.Entities.PerformanceTests/EntityCommandBufferPerformanceTests.cs index a87b83b4..260efdaf 100644 --- a/Unity.Entities.PerformanceTests/EntityCommandBufferPerformanceTests.cs +++ b/Unity.Entities.PerformanceTests/EntityCommandBufferPerformanceTests.cs @@ -23,7 +23,7 @@ public void Setup() { m_PreviousWorld = World.Active; m_World = World.Active = new World("Test World"); - m_Manager = m_World.GetOrCreateManager(); + m_Manager = m_World.EntityManager; } public struct EcsTestData : IComponentData diff --git a/Unity.Entities.PerformanceTests/EntityManagerPerformanceTests.cs b/Unity.Entities.PerformanceTests/EntityManagerPerformanceTests.cs index 88fef9fe..f48010f5 100644 --- a/Unity.Entities.PerformanceTests/EntityManagerPerformanceTests.cs +++ b/Unity.Entities.PerformanceTests/EntityManagerPerformanceTests.cs @@ -16,7 +16,7 @@ public sealed unsafe class EntityManagerPerformanceTests NativeArray entities1; NativeArray entities2; NativeArray entities3; - ComponentGroup group; + EntityQuery group; const int count = 1024 * 128; @@ -25,14 +25,14 @@ public void Setup() { m_PreviousWorld = World.Active; m_World = World.Active = new World("Test World"); - m_Manager = m_World.GetOrCreateManager(); + m_Manager = m_World.EntityManager; archetype1 = m_Manager.CreateArchetype(typeof(EcsTestData), typeof(EcsTestData2), typeof(EcsTestData3)); archetype2 = m_Manager.CreateArchetype(typeof(EcsTestData), typeof(EcsTestData2)); archetype3 = m_Manager.CreateArchetype(typeof(EcsTestData), typeof(EcsTestData3)); entities1 = new NativeArray(count, Allocator.Persistent); entities2 = new NativeArray(count, Allocator.Persistent); entities3 = new NativeArray(count, Allocator.Persistent); - group = m_Manager.CreateComponentGroup(typeof(EcsTestData)); + group = m_Manager.CreateEntityQuery(typeof(EcsTestData)); } [TearDown] diff --git a/Unity.Entities.PerformanceTests/EntitySerializationPerformanceTests.cs b/Unity.Entities.PerformanceTests/EntitySerializationPerformanceTests.cs index de478d21..683bf65c 100644 --- a/Unity.Entities.PerformanceTests/EntitySerializationPerformanceTests.cs +++ b/Unity.Entities.PerformanceTests/EntitySerializationPerformanceTests.cs @@ -22,7 +22,7 @@ public void Setup() { m_PreviousWorld = World.Active; m_World = World.Active = new World("Test World"); - m_Manager = m_World.GetOrCreateManager(); + m_Manager = m_World.EntityManager; } [TearDown] @@ -104,7 +104,7 @@ private struct SerializationJob : IJobParallelForBatch public void Execute(int startIndex, int count) { // @HACK need a reliable way having the entity manage for the given entities - var manager = World.Active.GetExistingManager(); + var manager = World.Active.EntityManager; var buffer = new StringBuffer(4096); var visitor = new JsonVisitor { StringBuffer = buffer }; diff --git a/Unity.Entities.PerformanceTests/JobProcessComponentDataPrefilteringPerformanceTests.cs b/Unity.Entities.PerformanceTests/JobForEachPrefilteringPerformanceTests.cs similarity index 78% rename from Unity.Entities.PerformanceTests/JobProcessComponentDataPrefilteringPerformanceTests.cs rename to Unity.Entities.PerformanceTests/JobForEachPrefilteringPerformanceTests.cs index 91c0abbf..543d5736 100644 --- a/Unity.Entities.PerformanceTests/JobProcessComponentDataPrefilteringPerformanceTests.cs +++ b/Unity.Entities.PerformanceTests/JobForEachPrefilteringPerformanceTests.cs @@ -13,13 +13,13 @@ namespace Unity.Entities.PerformanceTests { [TestFixture] [NUnit.Framework.Category("Performance")] - public sealed class JobProcessComponentDataPrefilteringPerformanceTests + public sealed class JobForEachPrefilteringPerformanceTests { private World m_PreviousWorld; private World m_World; private EntityManager m_Manager; - private struct ProcessJob : IJobProcessComponentData + private struct ProcessJob : IJobForEach { public void Execute(ref EcsTestData c0) { @@ -32,7 +32,7 @@ public void Setup() { m_PreviousWorld = World.Active; m_World = World.Active = new World("Test World"); - m_Manager = m_World.GetOrCreateManager(); + m_Manager = m_World.EntityManager; } #if UNITY_2019_2_OR_NEWER @@ -43,36 +43,36 @@ public void Setup() public void Prefiltering_SingleArchetype_SingleChunk_Unfiltered() { const int kEntityCount = 10; - + var archetype = m_Manager.CreateArchetype(ComponentType.ReadWrite(), ComponentType.ReadWrite()); - var group = m_Manager.CreateComponentGroup(ComponentType.ReadWrite(),ComponentType.ReadWrite()); + var group = m_Manager.CreateEntityQuery(ComponentType.ReadWrite(),ComponentType.ReadWrite()); var entities = new NativeArray(kEntityCount, Allocator.TempJob); m_Manager.CreateEntity(archetype, entities); - + var dependsOn = new JobHandle(); - + Measure.Method( () => { - dependsOn = new ProcessJob().ScheduleGroup(group, dependsOn); + dependsOn = new ProcessJob().Schedule(group, dependsOn); }) .Definition("Scheduling") .Run(); - + dependsOn.Complete(); - + Measure.Method( () => { - var job = new ProcessJob().ScheduleGroup(group); + var job = new ProcessJob().Schedule(group); job.Complete(); }) .Definition("ScheduleAndRun") .Run(); - + entities.Dispose(); } - + #if UNITY_2019_2_OR_NEWER [Test, Performance] #else @@ -81,49 +81,49 @@ public void Prefiltering_SingleArchetype_SingleChunk_Unfiltered() public void Prefiltering_SingleArchetype_TwoChunks_Filtered() { const int kEntityCount = 10; - + var archetype = m_Manager.CreateArchetype( - ComponentType.ReadWrite(), - ComponentType.ReadWrite(), + ComponentType.ReadWrite(), + ComponentType.ReadWrite(), ComponentType.ReadWrite()); - - var group = m_Manager.CreateComponentGroup( - ComponentType.ReadWrite(), + + var group = m_Manager.CreateEntityQuery( + ComponentType.ReadWrite(), ComponentType.ReadWrite(), ComponentType.ReadWrite()); - + var entities = new NativeArray(kEntityCount, Allocator.TempJob); m_Manager.CreateEntity(archetype, entities); for (int i = kEntityCount / 2; i < kEntityCount; ++i) { m_Manager.SetSharedComponentData(entities[i], new EcsTestSharedComp{value = 10}); } - + var dependsOn = new JobHandle(); group.SetFilter(new EcsTestSharedComp {value = 10}); Measure.Method( () => { - dependsOn = new ProcessJob().ScheduleGroup(group, dependsOn); + dependsOn = new ProcessJob().Schedule(group, dependsOn); }) .Definition("Scheduling") .Run(); - + dependsOn.Complete(); - + Measure.Method( () => { - var job = new ProcessJob().ScheduleGroup(group); + var job = new ProcessJob().Schedule(group); job.Complete(); }) .Definition("ScheduleAndRun") .Run(); - + entities.Dispose(); } - + #if UNITY_2019_2_OR_NEWER [Test, Performance] #else @@ -134,15 +134,15 @@ public void Prefiltering_SingleArchetype_MultipleChunks_Filtered() const int kEntityCount = 10000; var archetype = m_Manager.CreateArchetype( - ComponentType.ReadWrite(), - ComponentType.ReadWrite(), + ComponentType.ReadWrite(), + ComponentType.ReadWrite(), ComponentType.ReadWrite()); - - var group = m_Manager.CreateComponentGroup( - ComponentType.ReadWrite(), + + var group = m_Manager.CreateEntityQuery( + ComponentType.ReadWrite(), ComponentType.ReadWrite(), ComponentType.ReadWrite()); - + var entities = new NativeArray(kEntityCount, Allocator.TempJob); m_Manager.CreateEntity(archetype, entities); @@ -150,32 +150,32 @@ public void Prefiltering_SingleArchetype_MultipleChunks_Filtered() { m_Manager.SetSharedComponentData(entities[i], new EcsTestSharedComp {value = i % 10 } ); } - + var dependsOn = new JobHandle(); group.SetFilter(new EcsTestSharedComp{value = 0}); Measure.Method( () => { - dependsOn = new ProcessJob().ScheduleGroup(group, dependsOn); + dependsOn = new ProcessJob().Schedule(group, dependsOn); }) .Definition("Scheduling") .Run(); - + dependsOn.Complete(); - + Measure.Method( () => { - var job = new ProcessJob().ScheduleGroup(group); + var job = new ProcessJob().Schedule(group); job.Complete(); }) .Definition("ScheduleAndRun") .Run(); - + entities.Dispose(); } - + #if UNITY_2019_2_OR_NEWER [Test, Performance] #else @@ -198,7 +198,7 @@ public void Prefiltering_MultipleArchetype_MultipleChunks_Filtered() allArchetypes[4] = m_Manager.CreateArchetype(allTypes[0], allTypes[1], allTypes[2], allTypes[3]); allArchetypes[5] = m_Manager.CreateArchetype(allTypes[0], allTypes[1], allTypes[2], allTypes[4]); allArchetypes[6] = m_Manager.CreateArchetype(allTypes[0], allTypes[1], allTypes[3], allTypes[4]); - allArchetypes[7] = m_Manager.CreateArchetype(allTypes); + allArchetypes[7] = m_Manager.CreateArchetype(allTypes); const int kEntityCountPerArchetype = 1000; for (int i = 0; i < 8; ++i) @@ -209,36 +209,36 @@ public void Prefiltering_MultipleArchetype_MultipleChunks_Filtered() for (int j = 0; j < kEntityCountPerArchetype; ++j) { m_Manager.SetSharedComponentData(entities[i], new EcsTestSharedComp {value = i % 10 } ); - } - + } + entities.Dispose(); } - + var dependsOn = new JobHandle(); - var group = m_Manager.CreateComponentGroup( - ComponentType.ReadWrite(), + var group = m_Manager.CreateEntityQuery( + ComponentType.ReadWrite(), ComponentType.ReadWrite()); group.SetFilter(new EcsTestSharedComp{value = 0}); Measure.Method( () => { - dependsOn = new ProcessJob().ScheduleGroup(group, dependsOn); + dependsOn = new ProcessJob().Schedule(group, dependsOn); }) .Definition("Scheduling") .Run(); - + dependsOn.Complete(); - + Measure.Method( () => { - var job = new ProcessJob().ScheduleGroup(group); + var job = new ProcessJob().Schedule(group); job.Complete(); }) .Definition("ScheduleAndRun") .Run(); - + } } } diff --git a/Unity.Entities.PerformanceTests/JobProcessComponentDataPrefilteringPerformanceTests.cs.meta b/Unity.Entities.PerformanceTests/JobForEachPrefilteringPerformanceTests.cs.meta similarity index 100% rename from Unity.Entities.PerformanceTests/JobProcessComponentDataPrefilteringPerformanceTests.cs.meta rename to Unity.Entities.PerformanceTests/JobForEachPrefilteringPerformanceTests.cs.meta diff --git a/Unity.Entities.PerformanceTests/SharedComponentPerformanceTests.cs b/Unity.Entities.PerformanceTests/SharedComponentPerformanceTests.cs index d3edb6a3..238362e0 100644 --- a/Unity.Entities.PerformanceTests/SharedComponentPerformanceTests.cs +++ b/Unity.Entities.PerformanceTests/SharedComponentPerformanceTests.cs @@ -15,7 +15,7 @@ public void Setup() { m_PreviousWorld = World.Active; m_World = World.Active = new World("Test World"); - m_Manager = m_World.GetOrCreateManager(); + m_Manager = m_World.EntityManager; } [TearDown] diff --git a/Unity.Entities.Properties.Tests/EntitySerializationTests.cs b/Unity.Entities.Properties.Tests/EntitySerializationTests.cs index 3ca27dec..3bd371bc 100644 --- a/Unity.Entities.Properties.Tests/EntitySerializationTests.cs +++ b/Unity.Entities.Properties.Tests/EntitySerializationTests.cs @@ -19,7 +19,7 @@ public void Setup() { m_PreviousWorld = World.Active; m_World = World.Active = new World ("Test World"); - m_Manager = m_World.GetOrCreateManager (); + m_Manager = m_World.EntityManager; } [TearDown] diff --git a/Unity.Entities.StaticTypeRegistry/StaticTypeRegistry.cs b/Unity.Entities.StaticTypeRegistry/StaticTypeRegistry.cs index 1ffff332..d3bfc530 100644 --- a/Unity.Entities.StaticTypeRegistry/StaticTypeRegistry.cs +++ b/Unity.Entities.StaticTypeRegistry/StaticTypeRegistry.cs @@ -10,6 +10,8 @@ static internal unsafe class StaticTypeRegistry static public readonly bool[] SystemIsGroup; static public readonly string[] SystemName; // Debugging. And a reason to have a TinyReflectionSystem static public readonly int[] EntityOffsets; + static public readonly int[] BlobAssetReferenceOffsets; + static public readonly int[] WriteGroups; // This field will be generated in the replacement assembly //static public readonly TypeManager.TypeInfo[] TypeInfos; #pragma warning restore 0649 @@ -35,6 +37,20 @@ public static bool Equals(void* lhs, void* rhs, int typeIndex) throw new NotImplementedException("This function should have been replaced by the TypeRegGen build step. Ensure TypeRegGen.exe is generating a new Unity.Entities.StaticTypeRegistry assembly."); } + public static bool Equals(object lhs, object rhs, int typeIndex) + { + // empty -- dynamic reg is used. TypeRegGen will generate + // a replacement assembly + throw new NotImplementedException("This function should have been replaced by the TypeRegGen build step. Ensure TypeRegGen.exe is generating a new Unity.Entities.StaticTypeRegistry assembly."); + } + + public static bool Equals(object lhs, void* rhs, int typeIndex) + { + // empty -- dynamic reg is used. TypeRegGen will generate + // a replacement assembly + throw new NotImplementedException("This function should have been replaced by the TypeRegGen build step. Ensure TypeRegGen.exe is generating a new Unity.Entities.StaticTypeRegistry assembly."); + } + public static int GetHashCode(void* val, int typeIndex) { // empty -- dynamic reg is used. TypeRegGen will generate @@ -42,5 +58,11 @@ public static int GetHashCode(void* val, int typeIndex) throw new NotImplementedException("This function should have been replaced by the TypeRegGen build step. Ensure TypeRegGen.exe is generating a new Unity.Entities.StaticTypeRegistry assembly."); } + public static int BoxedGetHashCode(object val, int typeIndex) + { + // empty -- dynamic reg is used. TypeRegGen will generate + // a replacement assembly + throw new NotImplementedException("This function should have been replaced by the TypeRegGen build step. Ensure TypeRegGen.exe is generating a new Unity.Entities.StaticTypeRegistry assembly."); + } } } diff --git a/Unity.Entities.Tests/ArchetypeChunkArrayTests.cs b/Unity.Entities.Tests/ArchetypeChunkArrayTests.cs index 476c8c69..51314888 100644 --- a/Unity.Entities.Tests/ArchetypeChunkArrayTests.cs +++ b/Unity.Entities.Tests/ArchetypeChunkArrayTests.cs @@ -85,13 +85,13 @@ public void ACS_WriteMixed() { CreateMixedEntities(64); - var query = new EntityArchetypeQuery + var query = new EntityQueryDesc { Any = new ComponentType[] {typeof(EcsTestData2), typeof(EcsTestData)}, // any None = Array.Empty(), // none All = Array.Empty(), // all }; - var group = m_Manager.CreateComponentGroup(query); + var group = m_Manager.CreateEntityQuery(query); var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob); group.Dispose(); @@ -199,7 +199,7 @@ public void ACS_WriteMixedFilterShared() } } - var group = m_Manager.CreateComponentGroup(new EntityArchetypeQuery + var group = m_Manager.CreateEntityQuery(new EntityQueryDesc { Any = new ComponentType[] {typeof(EcsTestData2), typeof(EcsTestData)}, // any None = Array.Empty(), // none @@ -287,7 +287,7 @@ public void ACS_Buffers() { CreateEntities(128); - var group = m_Manager.CreateComponentGroup(ComponentType.ReadWrite()); + var group = m_Manager.CreateEntityQuery(ComponentType.ReadWrite()); var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob); group.Dispose(); @@ -336,11 +336,11 @@ public void Execute(int chunkIndex) } } - ComponentGroup m_Group; + EntityQuery m_Group; - protected override void OnCreateManager() + protected override void OnCreate() { - m_Group = GetComponentGroup(typeof(EcsIntElement)); + m_Group = GetEntityQuery(typeof(EcsIntElement)); } protected override void OnUpdate() @@ -365,7 +365,7 @@ public void ACS_BufferHas() { CreateEntities(128); - var group = m_Manager.CreateComponentGroup(ComponentType.ReadWrite()); + var group = m_Manager.CreateEntityQuery(ComponentType.ReadWrite()); var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob); group.Dispose(); @@ -392,7 +392,7 @@ public void ACS_BufferVersions() { CreateEntities(128); - var group = m_Manager.CreateComponentGroup(ComponentType.ReadWrite()); + var group = m_Manager.CreateEntityQuery(ComponentType.ReadWrite()); var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob); group.Dispose(); @@ -412,7 +412,7 @@ public void ACS_BufferVersions() } // Run system to bump chunk versions - var bumpChunkBufferTypeVersionSystem = World.CreateManager(); + var bumpChunkBufferTypeVersionSystem = World.CreateSystem(); bumpChunkBufferTypeVersionSystem.Update(); // Check versions after modifications @@ -430,12 +430,12 @@ public void ACS_BufferVersions() } [Test] - [StandaloneFixme] // ISharedComponentData + [StandaloneFixme] // don't know why this fails public void ACS_BuffersRO() { CreateEntities(128); - var group = m_Manager.CreateComponentGroup(ComponentType.ReadWrite()); + var group = m_Manager.CreateEntityQuery(ComponentType.ReadWrite()); var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob); group.Dispose(); var intElements = m_Manager.GetArchetypeChunkBufferType(true); @@ -450,14 +450,14 @@ public void ACS_BuffersRO() } [Test] - [StandaloneFixme] // ISharedComponentData + [StandaloneFixme] // don't know why this fails public void ACS_ChunkArchetypeTypesMatch() { var entityTypes = new ComponentType[] {typeof(EcsTestData), typeof(EcsTestSharedComp), typeof(EcsIntElement)}; CreateEntities(128); - var group = m_Manager.CreateComponentGroup(entityTypes); + var group = m_Manager.CreateEntityQuery(entityTypes); using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) @@ -481,13 +481,12 @@ public void ACS_ChunkArchetypeTypesMatch() struct Max3Capacity : IComponentData { } [Test] - [StandaloneFixme] // MaximumChunkCapacityAttribute not supported in Tiny? public void MaximumChunkCapacityIsRespected() { for (int i = 0; i != 4; i++) m_Manager.CreateEntity(typeof(Max3Capacity)); - var group = m_Manager.CreateComponentGroup(typeof(Max3Capacity)); + var group = m_Manager.CreateEntityQuery(typeof(Max3Capacity)); using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) { Assert.AreEqual(2, chunks.Length); diff --git a/Unity.Entities.Tests/BlobificationTests.cs b/Unity.Entities.Tests/BlobificationTests.cs index b681d77b..582f3766 100644 --- a/Unity.Entities.Tests/BlobificationTests.cs +++ b/Unity.Entities.Tests/BlobificationTests.cs @@ -7,7 +7,6 @@ using Unity.Entities; using Unity.Entities.Tests; -[StandaloneFixme] // Should this work for Tiny? public class BlobTests : ECSTestsFixture { //@TODO: Test Prevent NativeArray and other containers inside of Blob data @@ -122,7 +121,7 @@ public void ReadBlobDataFromJob() } - struct ValidateBlobInComponentJob : IJobProcessComponentData + struct ValidateBlobInComponentJob : IJobForEach { public bool ExpectException; @@ -156,7 +155,7 @@ public unsafe void ParallelBlobAccessFromEntityJob() var jobData = new ValidateBlobInComponentJob(); - var system = World.Active.GetOrCreateManager(); + var system = World.Active.GetOrCreateSystem(); var jobHandle = jobData.Schedule(system); ValidateBlobData(ref blob.Value); @@ -175,7 +174,7 @@ public void DestroyedBlobAccessFromEntityJobThrows() var jobData = new ValidateBlobInComponentJob(); jobData.ExpectException = true; - var system = World.Active.GetOrCreateManager(); + var system = World.Active.GetOrCreateSystem(); var jobHandle = jobData.Schedule(system); jobHandle.Complete (); diff --git a/Unity.Entities.Tests/BufferElementDataInstantiateTests.cs b/Unity.Entities.Tests/BufferElementDataInstantiateTests.cs index b00d5fa0..4d6e5c6a 100644 --- a/Unity.Entities.Tests/BufferElementDataInstantiateTests.cs +++ b/Unity.Entities.Tests/BufferElementDataInstantiateTests.cs @@ -5,7 +5,6 @@ using Unity.Collections; #pragma warning disable 0649 -#pragma warning disable 0618 #pragma warning disable 0219 // assigned but its value is never used namespace Unity.Entities.Tests @@ -144,7 +143,6 @@ public void DuplicatingEntity_WhenPrototypeHasDynamicBuffer_DoesNotWriteOutOfBou } #pragma warning restore 0649 -#pragma warning restore 0618 #pragma warning restore 0219 // assigned but its value is never used diff --git a/Unity.Entities.Tests/BufferElementDataSystemStateInstantiateTests.cs b/Unity.Entities.Tests/BufferElementDataSystemStateInstantiateTests.cs index 9dc84222..e8d23d61 100644 --- a/Unity.Entities.Tests/BufferElementDataSystemStateInstantiateTests.cs +++ b/Unity.Entities.Tests/BufferElementDataSystemStateInstantiateTests.cs @@ -5,7 +5,6 @@ namespace Unity.Entities.Tests public class BufferElementDataSystemStateInstantiateTests : ECSTestsFixture { [Test] - [StandaloneFixme] // Real issue public unsafe void InstantiateDoesNotCreatesCopy() { var original = m_Manager.CreateEntity(typeof(EcsIntStateElement)); diff --git a/Unity.Entities.Tests/BufferElementDataSystemStateTests.cs b/Unity.Entities.Tests/BufferElementDataSystemStateTests.cs index 4a683be9..7b1ce534 100644 --- a/Unity.Entities.Tests/BufferElementDataSystemStateTests.cs +++ b/Unity.Entities.Tests/BufferElementDataSystemStateTests.cs @@ -1,5 +1,6 @@ using NUnit.Framework; using Unity.Collections; +using Unity.Collections.LowLevel.Unsafe; using System; using System.Collections.Generic; using System.Linq; @@ -16,7 +17,6 @@ // ******* COPY AND PASTE WARNING ************* #pragma warning disable 0649 -#pragma warning disable 0618 #pragma warning disable 0219 // assigned but its value is never used namespace Unity.Entities.Tests @@ -282,34 +282,6 @@ public void MutateBufferData() } } - [Test] - public void BufferArrayComponentGroupIteration() - { - /*var entity64 =*/ - m_Manager.CreateEntity(typeof(EcsIntStateElement)); - /*var entity10 =*/ - m_Manager.CreateEntity(typeof(EcsIntStateElement)); - - var group = m_Manager.CreateComponentGroup(typeof(EcsIntStateElement)); - - var buffers = group.GetBufferArray(); - - Assert.AreEqual(2, buffers.Length); - Assert.AreEqual(0, buffers[0].Length); - Assert.AreEqual(8, buffers[0].Capacity); - Assert.AreEqual(0, buffers[1].Length); - Assert.AreEqual(8, buffers[1].Capacity); - - buffers[0].Add(12); - buffers[0].Add(13); - - Assert.AreEqual(2, buffers[0].Length); - Assert.AreEqual(12, buffers[0][0].Value); - Assert.AreEqual(13, buffers[0][1].Value); - - Assert.AreEqual(0, buffers[1].Length); - } - [Test] public void BufferComponentGroupChunkIteration() { @@ -318,7 +290,7 @@ public void BufferComponentGroupChunkIteration() /*var entity10 =*/ m_Manager.CreateEntity(typeof(EcsIntStateElement)); - var group = m_Manager.CreateComponentGroup(typeof(EcsIntStateElement)); + var group = m_Manager.CreateEntityQuery(typeof(EcsIntStateElement)); var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob); var buffers = chunks[0].GetBufferAccessor(m_Manager.GetArchetypeChunkBufferType(false)); @@ -355,7 +327,6 @@ public void BufferFromEntityWorks() } [Test] - [StandaloneFixme] // Real issue - Safety & Sentinel should be invalid after Destroy public void OutOfBoundsAccessThrows() { var entityInt = m_Manager.CreateEntity(typeof(EcsIntStateElement)); @@ -370,7 +341,6 @@ public void OutOfBoundsAccessThrows() } [Test] - [StandaloneFixme] // Real issue - Safety & Sentinel should be invalid after Destroy public void UseAfterStructuralChangeThrows() { var entityInt = m_Manager.CreateEntity(typeof(EcsIntStateElement)); @@ -384,7 +354,6 @@ public void UseAfterStructuralChangeThrows() } [Test] - [StandaloneFixme] // Real issue - Safety & Sentinel should be invalid after Destroy public void UseAfterStructuralChangeThrows2() { var entityInt = m_Manager.CreateEntity(typeof(EcsIntStateElement)); @@ -399,19 +368,19 @@ public void UseAfterStructuralChangeThrows2() } [Test] - [StandaloneFixme] // Real issue - Safety & Sentinel should be invalid after Add on structural change public void UseAfterStructuralChangeThrows3() { var entityInt = m_Manager.CreateEntity(typeof(EcsIntStateElement)); var buffer = m_Manager.GetBuffer(entityInt); buffer.CopyFrom(new EcsIntStateElement[] { 1, 2, 3 }); m_Manager.AddComponentData(entityInt, new EcsTestData() { value = 20 }); - Assert.Throws(() => { buffer.Add(4); }); + Assert.Throws(() => { + buffer.Add(4); + }); } [Test] - [StandaloneFixme] // Real issue - Safety & Sentinel should be invalid after Add on structural change public void WritingReadOnlyThrows() { var entityInt = m_Manager.CreateEntity(typeof(EcsIntStateElement)); @@ -453,57 +422,6 @@ public void ReinterpretWrongSizeThrows() }); } -// Injection is obsolete - [DisableAutoCreation] - public class InjectionTestSystem : JobComponentSystem - { - public struct Data - { - public readonly int Length; - public BufferArray Buffers; - } - - [Inject] Data m_Data; - - public struct MyJob : IJobParallelFor - { - public BufferArray Buffers; - - public void Execute(int i) - { - Buffers[i].Add(i * 3); - } - } - - protected override JobHandle OnUpdate(JobHandle inputDeps) - { - new MyJob { Buffers = m_Data.Buffers }.Schedule(m_Data.Length, 32, inputDeps).Complete(); - return default(JobHandle); - } - } - - [Test] - [StandaloneFixme] // IJob - InjectionTestSystem : JobComponentSystem - public void Injection() - { - var system = World.Active.GetOrCreateManager(); - - using (var entities = new NativeArray(320, Allocator.Temp)) - { - var arch = m_Manager.CreateArchetype(typeof(EcsIntStateElement)); - m_Manager.CreateEntity(arch, entities); - - system.Update(); - system.Update(); - - for (var i = 0; i < entities.Length; ++i) - { - var buf = m_Manager.GetBuffer(entities[i]); - Assert.AreEqual(2, buf.Length); - } - } - } - [Test] public void TrimExcessWorks() { @@ -566,7 +484,6 @@ public void NoCapacitySpecifiedWorks() } [Test] - [StandaloneFixme] // Real issue : buffer.AsNativeArray should invalidate the Safety public void ArrayInvalidationWorks() { var original = m_Manager.CreateEntity(typeof(EcsIntStateElement)); @@ -587,7 +504,6 @@ public void ArrayInvalidationWorks() } [Test] - [StandaloneFixme] // Real issue : buffer.AsNativeArray should invalidate the Safety public void ArrayInvalidationHappensForAllInstances() { var e0 = m_Manager.CreateEntity(typeof(EcsIntStateElement)); @@ -680,7 +596,7 @@ public void ReadWriteDynamicBuffer() var buffer = m_Manager.GetBuffer(original); buffer.Add(5); - var group = EmptySystem.GetComponentGroup(new EntityArchetypeQuery {All = new ComponentType[] {typeof(EcsIntStateElement)}}); + var group = EmptySystem.GetEntityQuery(new EntityQueryDesc {All = new ComponentType[] {typeof(EcsIntStateElement)}}); var job = new WriteJob { //@TODO: Throw exception when read only flag is not accurately passed to job for buffers... @@ -720,7 +636,7 @@ public void ReadOnlyDynamicBufferImpl(bool readOnlyType) var buffer = m_Manager.GetBuffer(original); buffer.Add(5); - var group = EmptySystem.GetComponentGroup(new EntityArchetypeQuery {All = new ComponentType[] {typeof(EcsIntStateElement)}}); + var group = EmptySystem.GetEntityQuery(new EntityQueryDesc {All = new ComponentType[] {typeof(EcsIntStateElement)}}); var job = new ReadOnlyJob { Int = EmptySystem.GetArchetypeChunkBufferType(readOnlyType) @@ -823,10 +739,99 @@ public void DynamicBuffer_FromEntity_IsCreated_IsTrue() var buffer = m_Manager.GetBuffer(entity); Assert.IsTrue(buffer.IsCreated); } + + [Test] + public void DynamicBuffer_AllocateBufferWithLongSize_DoesNotThrow() + { + var entity = m_Manager.CreateEntity(typeof(EcsIntStateElement)); + var buffer = m_Manager.GetBuffer(entity); + int capacity = (int)(((long)int.MaxValue + 1) / UnsafeUtility.SizeOf() + 1); //536870913 + Assert.DoesNotThrow(() => buffer.ResizeUninitialized(capacity)); + Assert.AreEqual(capacity, buffer.Length); + } + + [Test] + public void DynamicBuffer_Insert_BufferHasLongSize_DoesNotThrow() + { + var entity = m_Manager.CreateEntity(typeof(EcsIntStateElement)); + var buffer = m_Manager.GetBuffer(entity); + int capacity = (int)(((long)int.MaxValue + 1) / UnsafeUtility.SizeOf() + 1); //536870913 + buffer.ResizeUninitialized(capacity); + + Assert.DoesNotThrow(() => buffer.Insert(0, new EcsIntStateElement { Value = 99 })); + Assert.AreEqual(capacity + 1, buffer.Length); + } + + [Test] + public void DynamicBuffer_AddRange_NewBufferHasLongSize_DoesNotThrow() + { + var entity = m_Manager.CreateEntity(typeof(EcsIntElement)); + var buffer = m_Manager.GetBuffer(entity); + int capacity = (int)(((long)int.MaxValue + 1) / UnsafeUtility.SizeOf() + 1); //536870913 + buffer.ResizeUninitialized(capacity); + + NativeArray array = new NativeArray(10, Allocator.Temp); + Assert.DoesNotThrow(() => buffer.AddRange(array)); + Assert.AreEqual(capacity + 10, buffer.Length); + } + + [Test] + public void DynamicBuffer_RemoveRange_MovedBufferHasLongSize_DoesNotThrow() + { + var entity = m_Manager.CreateEntity(typeof(EcsIntElement)); + var buffer = m_Manager.GetBuffer(entity); + int capacity = (int)(((long)int.MaxValue + 1) / UnsafeUtility.SizeOf() + 2); + buffer.ResizeUninitialized(capacity); + + Assert.AreEqual(536870914, buffer.Length); + Assert.DoesNotThrow(() => buffer.RemoveRange(0, 1)); + Assert.AreEqual(536870913, buffer.Length); + } + + [Test] + public void DynamicBuffer_Add_NewBufferHasLongSize_DoesNotThrow() + { + var arrayType = ComponentType.ReadWrite(); + var entity = m_Manager.CreateEntity(typeof(EcsIntElement)); + var buffer = m_Manager.GetBuffer(entity); + int capacity = (int)(((long)int.MaxValue + 1) / UnsafeUtility.SizeOf() + 1); //536870913 + + buffer.ResizeUninitialized(capacity); + + Assert.DoesNotThrow(() => buffer.Add(1)); + } + + [Test] + public void DynamicBuffer_TrimExcess_NewBufferHasLongSize_DoesNotThrow() + { + var arrayType = ComponentType.ReadWrite(); + var entity = m_Manager.CreateEntity(typeof(EcsIntElement)); + var buffer = m_Manager.GetBuffer(entity); + int capacity = (int)(((long)int.MaxValue + 1) / UnsafeUtility.SizeOf() + 1); //536870913 + + buffer.ResizeUninitialized(capacity); + // cause the capacity to double + buffer.Add(1); + + Assert.DoesNotThrow(() => buffer.TrimExcess()); + Assert.AreEqual(capacity + 1, buffer.Length); + } + + [Test] + public void DynamicBuffer_Reserve_IncreasesCapacity() + { + var arrayType = ComponentType.ReadWrite(); + var entity = m_Manager.CreateEntity(typeof(EcsIntElement)); + var buffer = m_Manager.GetBuffer(entity); + + buffer.Reserve(100); + + Assert.AreEqual(100, buffer.Capacity); + Assert.AreEqual(0, buffer.Length); + } } } #pragma warning restore 0649 -#pragma warning restore 0618 #pragma warning restore 0219 // assigned but its value is never used diff --git a/Unity.Entities.Tests/BufferElementDataTests.cs b/Unity.Entities.Tests/BufferElementDataTests.cs index 7f5ecd9b..e78665d8 100644 --- a/Unity.Entities.Tests/BufferElementDataTests.cs +++ b/Unity.Entities.Tests/BufferElementDataTests.cs @@ -1,5 +1,6 @@ using NUnit.Framework; using Unity.Collections; +using Unity.Collections.LowLevel.Unsafe; using System; using System.Collections.Generic; using System.Linq; @@ -16,7 +17,6 @@ // ******* COPY AND PASTE WARNING ************* #pragma warning disable 0649 -#pragma warning disable 0618 #pragma warning disable 0219 // assigned but its value is never used namespace Unity.Entities.Tests @@ -282,34 +282,6 @@ public void MutateBufferData() } } - [Test] - public void BufferArrayComponentGroupIteration() - { - /*var entity64 =*/ - m_Manager.CreateEntity(typeof(EcsIntElement)); - /*var entity10 =*/ - m_Manager.CreateEntity(typeof(EcsIntElement)); - - var group = m_Manager.CreateComponentGroup(typeof(EcsIntElement)); - - var buffers = group.GetBufferArray(); - - Assert.AreEqual(2, buffers.Length); - Assert.AreEqual(0, buffers[0].Length); - Assert.AreEqual(8, buffers[0].Capacity); - Assert.AreEqual(0, buffers[1].Length); - Assert.AreEqual(8, buffers[1].Capacity); - - buffers[0].Add(12); - buffers[0].Add(13); - - Assert.AreEqual(2, buffers[0].Length); - Assert.AreEqual(12, buffers[0][0].Value); - Assert.AreEqual(13, buffers[0][1].Value); - - Assert.AreEqual(0, buffers[1].Length); - } - [Test] public void BufferComponentGroupChunkIteration() { @@ -318,7 +290,7 @@ public void BufferComponentGroupChunkIteration() /*var entity10 =*/ m_Manager.CreateEntity(typeof(EcsIntElement)); - var group = m_Manager.CreateComponentGroup(typeof(EcsIntElement)); + var group = m_Manager.CreateEntityQuery(typeof(EcsIntElement)); var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob); var buffers = chunks[0].GetBufferAccessor(m_Manager.GetArchetypeChunkBufferType(false)); @@ -355,7 +327,6 @@ public void BufferFromEntityWorks() } [Test] - [StandaloneFixme] // Real problem DestroyEntity should invalidate the buffers. Not sure about array bounds checking in this test public void OutOfBoundsAccessThrows() { var entityInt = m_Manager.CreateEntity(typeof(EcsIntElement)); @@ -370,7 +341,6 @@ public void OutOfBoundsAccessThrows() } [Test] - [StandaloneFixme] // Real problem DestroyEntity should invalidate the buffers public void UseAfterStructuralChangeThrows() { var entityInt = m_Manager.CreateEntity(typeof(EcsIntElement)); @@ -384,7 +354,6 @@ public void UseAfterStructuralChangeThrows() } [Test] - [StandaloneFixme] // Real problem DestroyEntity should invalidate the buffers public void UseAfterStructuralChangeThrows2() { var entityInt = m_Manager.CreateEntity(typeof(EcsIntElement)); @@ -399,7 +368,6 @@ public void UseAfterStructuralChangeThrows2() } [Test] - [StandaloneFixme] // Real problem structural change should invalidate the buffers public void UseAfterStructuralChangeThrows3() { var entityInt = m_Manager.CreateEntity(typeof(EcsIntElement)); @@ -411,7 +379,6 @@ public void UseAfterStructuralChangeThrows3() [Test] - [StandaloneFixme] // Real problem structural change should invalidate the buffers public void WritingReadOnlyThrows() { var entityInt = m_Manager.CreateEntity(typeof(EcsIntElement)); @@ -453,57 +420,6 @@ public void ReinterpretWrongSizeThrows() }); } -// Injection is obsolete - [DisableAutoCreation] - public class InjectionTestSystem : JobComponentSystem - { - public struct Data - { - public readonly int Length; - public BufferArray Buffers; - } - - [Inject] Data m_Data; - - public struct MyJob : IJobParallelFor - { - public BufferArray Buffers; - - public void Execute(int i) - { - Buffers[i].Add(i * 3); - } - } - - protected override JobHandle OnUpdate(JobHandle inputDeps) - { - new MyJob { Buffers = m_Data.Buffers }.Schedule(m_Data.Length, 32, inputDeps).Complete(); - return default(JobHandle); - } - } - - [Test] - [StandaloneFixme] // IJob - public void Injection() - { - var system = World.Active.GetOrCreateManager(); - - using (var entities = new NativeArray(320, Allocator.Temp)) - { - var arch = m_Manager.CreateArchetype(typeof(EcsIntElement)); - m_Manager.CreateEntity(arch, entities); - - system.Update(); - system.Update(); - - for (var i = 0; i < entities.Length; ++i) - { - var buf = m_Manager.GetBuffer(entities[i]); - Assert.AreEqual(2, buf.Length); - } - } - } - [Test] public void TrimExcessWorks() { @@ -566,7 +482,6 @@ public void NoCapacitySpecifiedWorks() } [Test] - [StandaloneFixme] // Real problem structural change should invalidate the buffers public void ArrayInvalidationWorks() { var original = m_Manager.CreateEntity(typeof(EcsIntElement)); @@ -588,7 +503,6 @@ public void ArrayInvalidationWorks() } [Test] - [StandaloneFixme] // Real problem structural change should invalidate the buffers public void ArrayInvalidationHappensForAllInstances() { var e0 = m_Manager.CreateEntity(typeof(EcsIntElement)); @@ -639,7 +553,7 @@ public void Execute() } [Test] - [StandaloneFixme] // Real problem structural change should invalidate the buffers && IJob + [StandaloneFixme] // IJob public void BufferInvalidationNotPossibleWhenArraysAreGivenToJobs() { var original = m_Manager.CreateEntity(typeof(EcsIntElement)); @@ -674,14 +588,13 @@ public void Execute(ArchetypeChunk chunk, int chunkIndex, int entityOffset) } [Test] - [StandaloneFixme] // IJob public void ReadWriteDynamicBuffer() { var original = m_Manager.CreateEntity(typeof(EcsIntElement)); var buffer = m_Manager.GetBuffer(original); buffer.Add(5); - var group = EmptySystem.GetComponentGroup(new EntityArchetypeQuery {All = new ComponentType[] {typeof(EcsIntElement)}}); + var group = EmptySystem.GetEntityQuery(new EntityQueryDesc {All = new ComponentType[] {typeof(EcsIntElement)}}); var job = new WriteJob { //@TODO: Throw exception when read only flag is not accurately passed to job for buffers... @@ -721,7 +634,7 @@ public void ReadOnlyDynamicBufferImpl(bool readOnlyType) var buffer = m_Manager.GetBuffer(original); buffer.Add(5); - var group = EmptySystem.GetComponentGroup(new EntityArchetypeQuery {All = new ComponentType[] {typeof(EcsIntElement)}}); + var group = EmptySystem.GetEntityQuery(new EntityQueryDesc {All = new ComponentType[] {typeof(EcsIntElement)}}); var job = new ReadOnlyJob { Int = EmptySystem.GetArchetypeChunkBufferType(readOnlyType) @@ -786,7 +699,7 @@ public void Execute() } [Test] - [StandaloneFixme] // IJob + Safety Handles + [StandaloneFixme] // IJob public void NativeArrayInJobReadOnly() { var original = m_Manager.CreateEntity(typeof(EcsIntElement)); @@ -824,10 +737,99 @@ public void DynamicBuffer_FromEntity_IsCreated_IsTrue() var buffer = m_Manager.GetBuffer(entity); Assert.IsTrue(buffer.IsCreated); } - } + + [Test] + public void DynamicBuffer_AllocateBufferWithLongSize_DoesNotThrow() + { + var entity = m_Manager.CreateEntity(typeof(EcsIntElement)); + var buffer = m_Manager.GetBuffer(entity); + int capacity = (int)(((long)int.MaxValue + 1) / UnsafeUtility.SizeOf() + 1); //536870913 + Assert.DoesNotThrow(() => buffer.ResizeUninitialized(capacity)); + Assert.AreEqual(capacity, buffer.Length); + } + + [Test] + public void DynamicBuffer_Insert_BufferHasLongSize_DoesNotThrow() + { + var entity = m_Manager.CreateEntity(typeof(EcsIntElement)); + var buffer = m_Manager.GetBuffer(entity); + int capacity = (int)(((long)int.MaxValue + 1) / UnsafeUtility.SizeOf() + 1); //536870913 + buffer.ResizeUninitialized(capacity); + + Assert.DoesNotThrow(() => buffer.Insert(0, new EcsIntElement { Value = 99 })); + Assert.AreEqual(capacity + 1, buffer.Length); + } + + [Test] + public void DynamicBuffer_AddRange_NewBufferHasLongSize_DoesNotThrow() + { + var entity = m_Manager.CreateEntity(typeof(EcsIntElement)); + var buffer = m_Manager.GetBuffer(entity); + int capacity = (int)(((long)int.MaxValue + 1) / UnsafeUtility.SizeOf() + 1); //536870913 + buffer.ResizeUninitialized(capacity); + + NativeArray array = new NativeArray(10, Allocator.Temp); + Assert.DoesNotThrow(() => buffer.AddRange(array)); + Assert.AreEqual(capacity + 10, buffer.Length); + } + + [Test] + public void DynamicBuffer_RemoveRange_MovedBufferHasLongSize_DoesNotThrow() + { + var entity = m_Manager.CreateEntity(typeof(EcsIntElement)); + var buffer = m_Manager.GetBuffer(entity); + int capacity = (int)(((long)int.MaxValue + 1) / UnsafeUtility.SizeOf() + 2); + buffer.ResizeUninitialized(capacity); + + Assert.AreEqual(536870914, buffer.Length); + Assert.DoesNotThrow(() => buffer.RemoveRange(0, 1)); + Assert.AreEqual(536870913, buffer.Length); + } + + [Test] + public void DynamicBuffer_Add_NewBufferHasLongSize_DoesNotThrow() + { + var arrayType = ComponentType.ReadWrite(); + var entity = m_Manager.CreateEntity(typeof(EcsIntElement)); + var buffer = m_Manager.GetBuffer(entity); + int capacity = (int)(((long)int.MaxValue + 1) / UnsafeUtility.SizeOf() + 1); //536870913 + + buffer.ResizeUninitialized(capacity); + + Assert.DoesNotThrow(() => buffer.Add(1)); + } + + [Test] + public void DynamicBuffer_TrimExcess_NewBufferHasLongSize_DoesNotThrow() + { + var arrayType = ComponentType.ReadWrite(); + var entity = m_Manager.CreateEntity(typeof(EcsIntElement)); + var buffer = m_Manager.GetBuffer(entity); + int capacity = (int)(((long)int.MaxValue + 1) / UnsafeUtility.SizeOf() + 1); //536870913 + + buffer.ResizeUninitialized(capacity); + // cause the capacity to double + buffer.Add(1); + + Assert.DoesNotThrow(() => buffer.TrimExcess()); + Assert.AreEqual(capacity + 1, buffer.Length); + } + + [Test] + public void DynamicBuffer_Reserve_IncreasesCapacity() + { + var arrayType = ComponentType.ReadWrite(); + var entity = m_Manager.CreateEntity(typeof(EcsIntElement)); + var buffer = m_Manager.GetBuffer(entity); + + buffer.Reserve(100); + + Assert.AreEqual(100, buffer.Capacity); + Assert.AreEqual(0, buffer.Length); + } + } } #pragma warning restore 0649 -#pragma warning restore 0618 #pragma warning restore 0219 // assigned but its value is never used diff --git a/Unity.Entities.Tests/ChangeVersionTests.cs b/Unity.Entities.Tests/ChangeVersionTests.cs index 108f8319..78b7dbf1 100644 --- a/Unity.Entities.Tests/ChangeVersionTests.cs +++ b/Unity.Entities.Tests/ChangeVersionTests.cs @@ -2,79 +2,67 @@ using Unity.Collections; using Unity.Jobs; -#pragma warning disable 618 - namespace Unity.Entities.Tests { class ChangeVersionTests : ECSTestsFixture { +#if !UNITY_ZEROPLAYER [DisableAutoCreation] class BumpVersionSystemInJob : ComponentSystem { -#pragma warning disable 649 - struct MyStruct - { - public readonly int Length; - public ComponentDataArray Data; - public ComponentDataArray Data2; - } - - [Inject] - MyStruct DataStruct; -#pragma warning restore 649 + public EntityQuery m_Group; - struct UpdateData : IJob + struct UpdateData : IJobForEach { - public int Length; - public ComponentDataArray Data; - public ComponentDataArray Data2; - - public void Execute() + public void Execute(ref EcsTestData data, ref EcsTestData2 data2) { - for (int i = 0; i < Length; ++i) - { - var d2 = Data2[i]; - d2.value0 = 10; - Data2[i] = d2; - } + data2 = new EcsTestData2 {value0 = 10}; } } protected override void OnUpdate() { - var updateDataJob = new UpdateData - { - Length = DataStruct.Length, - Data = DataStruct.Data, - Data2 = DataStruct.Data2 - }; - var updateDataJobHandle = updateDataJob.Schedule(); + var updateDataJob = new UpdateData{}; + var updateDataJobHandle = updateDataJob.Schedule(m_Group); updateDataJobHandle.Complete(); } + + protected override void OnCreate() + { + m_Group = GetEntityQuery(ComponentType.ReadWrite(), + ComponentType.ReadWrite()); + } } +#endif [DisableAutoCreation] class BumpVersionSystem : ComponentSystem { - struct MyStruct - { -#pragma warning disable 649 - public readonly int Length; - public ComponentDataArray Data; - public ComponentDataArray Data2; - } - - [Inject] - MyStruct DataStruct; -#pragma warning restore 649 + public EntityQuery m_Group; protected override void OnUpdate() { - for (int i = 0; i < DataStruct.Length; ++i) { - var d2 = DataStruct.Data2[i]; + var data = m_Group.ToComponentDataArray(Allocator.TempJob); + var data2 = m_Group.ToComponentDataArray(Allocator.TempJob); + + for (int i = 0; i < data.Length; ++i) + { + var d2 = data2[i]; d2.value0 = 10; - DataStruct.Data2[i] = d2; + data2[i] = d2; } + + m_Group.CopyFromComponentDataArray(data); + m_Group.CopyFromComponentDataArray(data2); + + data.Dispose(); + data2.Dispose(); + } + + protected override void OnCreate() + { + m_Group = GetEntityQuery(ComponentType.ReadWrite(), + ComponentType.ReadWrite()); } } @@ -97,12 +85,12 @@ public void Execute(int chunkIndex) } } - ComponentGroup m_Group; + EntityQuery m_Group; private bool m_LastAllChanged; - protected override void OnCreateManager() + protected override void OnCreate() { - m_Group = GetComponentGroup(typeof(EcsTestData)); + m_Group = GetEntityQuery(typeof(EcsTestData)); m_LastAllChanged = false; } @@ -135,60 +123,12 @@ public bool AllEcsTestDataChunksChanged() } } - [Test] - [StandaloneFixme] // IJob - public void CHG_IncrementedOnInjectionInJob() - { - var entity0 = m_Manager.CreateEntity(typeof(EcsTestData), typeof(EcsTestData2)); - var bumpSystem = World.CreateManager(); - - var oldGlobalVersion = m_Manager.GlobalSystemVersion; - - bumpSystem.Update(); - - var value0 = m_Manager.GetComponentData(entity0).value0; - Assert.AreEqual(10, value0); - - Assert.That(m_Manager.GlobalSystemVersion > oldGlobalVersion); - - unsafe { - // a system ran, the version should match the global - var chunk0 = m_Manager.Entities->GetComponentChunk(entity0); - var td2index0 = ChunkDataUtility.GetIndexInTypeArray(chunk0->Archetype, TypeManager.GetTypeIndex()); - Assert.AreEqual(m_Manager.GlobalSystemVersion, chunk0->GetChangeVersion(td2index0)); - } - } - - [Test] - [StandaloneFixme] // IJob - public void CHG_IncrementedOnInjection() - { - var entity0 = m_Manager.CreateEntity(typeof(EcsTestData), typeof(EcsTestData2)); - var bumpSystem = World.CreateManager(); - - var oldGlobalVersion = m_Manager.GlobalSystemVersion; - - bumpSystem.Update(); - - var value0 = m_Manager.GetComponentData(entity0).value0; - Assert.AreEqual(10, value0); - - Assert.That(m_Manager.GlobalSystemVersion > oldGlobalVersion); - - unsafe { - // a system ran, the version should match the global - var chunk0 = m_Manager.Entities->GetComponentChunk(entity0); - var td2index0 = ChunkDataUtility.GetIndexInTypeArray(chunk0->Archetype, TypeManager.GetTypeIndex()); - Assert.AreEqual(m_Manager.GlobalSystemVersion, chunk0->GetChangeVersion(td2index0)); - } - } - [Test] public void CHG_BumpValueChangesChunkTypeVersion() { m_Manager.CreateEntity(typeof(EcsTestData), typeof(EcsTestData2)); - var bumpChunkTypeVersionSystem = World.CreateManager(); + var bumpChunkTypeVersionSystem = World.CreateSystem(); bumpChunkTypeVersionSystem.Update(); Assert.AreEqual(true, bumpChunkTypeVersionSystem.AllEcsTestDataChunksChanged()); @@ -201,10 +141,22 @@ public void CHG_BumpValueChangesChunkTypeVersion() public void CHG_SystemVersionZeroWhenNotRun() { m_Manager.CreateEntity(typeof(EcsTestData), typeof(EcsTestData2)); - var system = World.CreateManager(); + var system = World.CreateSystem(); + Assert.AreEqual(0, system.LastSystemVersion); + system.Update(); + Assert.AreNotEqual(0, system.LastSystemVersion); + } + +#if !UNITY_ZEROPLAYER + [Test] + public void CHG_SystemVersionJob() + { + m_Manager.CreateEntity(typeof(EcsTestData), typeof(EcsTestData2)); + var system = World.CreateSystem(); Assert.AreEqual(0, system.LastSystemVersion); system.Update(); Assert.AreNotEqual(0, system.LastSystemVersion); } +#endif } } diff --git a/Unity.Entities.Tests/ChunkChangeVersionTests.cs b/Unity.Entities.Tests/ChunkChangeVersionTests.cs index f91e88ef..23e565b2 100644 --- a/Unity.Entities.Tests/ChunkChangeVersionTests.cs +++ b/Unity.Entities.Tests/ChunkChangeVersionTests.cs @@ -143,7 +143,6 @@ public void ModifyingBufferComponentMarksOnlySetTypeAsChanged() } [Test] - [StandaloneFixme] // ISharedComponentData public void AddSharedComponentMarksSrcAndDestChunkAsChanged() { var e0 = m_Manager.CreateEntity(typeof(EcsTestData), typeof(EcsTestData2)); @@ -158,7 +157,6 @@ public void AddSharedComponentMarksSrcAndDestChunkAsChanged() } [Test] - [StandaloneFixme] // ISharedComponentData public void SetSharedComponentMarksSrcAndDestChunkAsChanged() { var e0 = m_Manager.CreateEntity(typeof(EcsTestData), typeof(EcsTestSharedComp)); @@ -172,7 +170,6 @@ public void SetSharedComponentMarksSrcAndDestChunkAsChanged() } [Test] - [StandaloneFixme] // ISharedComponentData public void SwapComponentsMarksSrcAndDestChunkAsChanged() { var e0 = m_Manager.CreateEntity(typeof(EcsTestData), typeof(EcsTestSharedComp)); diff --git a/Unity.Entities.Tests/ChunkComponentTests.cs b/Unity.Entities.Tests/ChunkComponentTests.cs index f35f67a5..c83b2dd6 100644 --- a/Unity.Entities.Tests/ChunkComponentTests.cs +++ b/Unity.Entities.Tests/ChunkComponentTests.cs @@ -91,7 +91,7 @@ public void RemoveChunkComponent() public void UpdateChunkComponent() { var arch0 = m_Manager.CreateArchetype(ComponentType.ChunkComponent(), typeof(EcsTestData2)); - ComponentGroup group0 = m_Manager.CreateComponentGroup(typeof(ChunkHeader), typeof(EcsTestData)); + EntityQuery group0 = m_Manager.CreateEntityQuery(typeof(ChunkHeader), typeof(EcsTestData)); var entity0 = m_Manager.CreateEntity(arch0); var chunk0 = m_Manager.GetChunk(entity0); @@ -136,7 +136,7 @@ public void ProcessMetaChunkComponent() m_Manager.SetComponentData(entity0, new BoundsComponent{boundsMin = new float3(-10,-10,-10), boundsMax = new float3(0,0,0)}); var entity1 = m_Manager.CreateEntity(typeof(BoundsComponent), ComponentType.ChunkComponent()); m_Manager.SetComponentData(entity1, new BoundsComponent{boundsMin = new float3(0,0,0), boundsMax = new float3(10,10,10)}); - var metaGroup = m_Manager.CreateComponentGroup(typeof(ChunkBoundsComponent), typeof(ChunkHeader)); + var metaGroup = m_Manager.CreateEntityQuery(typeof(ChunkBoundsComponent), typeof(ChunkHeader)); var metaBoundsCount = metaGroup.CalculateLength(); var metaChunkHeaders = metaGroup.ToComponentDataArray(Allocator.TempJob); Assert.AreEqual(1, metaBoundsCount); @@ -166,10 +166,9 @@ public void ProcessMetaChunkComponent() #if !UNITY_ZEROPLAYER [DisableAutoCreation] [UpdateInGroup(typeof(PresentationSystemGroup))] - [StandaloneFixme] private class ChunkBoundsUpdateSystem : JobComponentSystem { - struct UpdateChunkBoundsJob : IJobProcessComponentData + struct UpdateChunkBoundsJob : IJobForEach { [ReadOnly] public ArchetypeChunkComponentType chunkComponentType; public void Execute(ref ChunkBoundsComponent chunkBounds, [ReadOnly] ref ChunkHeader chunkHeader) @@ -194,10 +193,9 @@ protected override JobHandle OnUpdate(JobHandle inputDeps) } [Test] - [StandaloneFixme] public void SystemProcessMetaChunkComponent() { - var chunkBoundsUpdateSystem = World.GetOrCreateManager (); + var chunkBoundsUpdateSystem = World.GetOrCreateSystem (); var entity0 = m_Manager.CreateEntity(typeof(BoundsComponent), ComponentType.ChunkComponent()); m_Manager.SetComponentData(entity0, new BoundsComponent{boundsMin = new float3(-10,-10,-10), boundsMax = new float3(0,0,0)}); @@ -219,8 +217,8 @@ public void ChunkHeaderMustBeQueriedExplicitly() var arch0 = m_Manager.CreateArchetype(ComponentType.ChunkComponent(), typeof(EcsTestData2)); var entity0 = m_Manager.CreateEntity(arch0); - ComponentGroup group0 = m_Manager.CreateComponentGroup(typeof(ChunkHeader), typeof(EcsTestData)); - ComponentGroup group1 = m_Manager.CreateComponentGroup(typeof(EcsTestData)); + EntityQuery group0 = m_Manager.CreateEntityQuery(typeof(ChunkHeader), typeof(EcsTestData)); + EntityQuery group1 = m_Manager.CreateEntityQuery(typeof(EcsTestData)); Assert.AreEqual(1, group0.CalculateLength()); Assert.AreEqual(0, group1.CalculateLength()); diff --git a/Unity.Entities.Tests/ComponentGroupArrayTests.cs b/Unity.Entities.Tests/ComponentGroupArrayTests.cs deleted file mode 100644 index 2caa057a..00000000 --- a/Unity.Entities.Tests/ComponentGroupArrayTests.cs +++ /dev/null @@ -1,130 +0,0 @@ -#if !UNITY_ZEROPLAYER -using NUnit.Framework; -using Unity.Jobs; -using Unity.Collections; - -#pragma warning disable 618 - -namespace Unity.Entities.Tests -{ - class ComponentGroupArrayTests : ECSTestsFixture - { - public ComponentGroupArrayTests() - { - Assert.IsTrue(Unity.Jobs.LowLevel.Unsafe.JobsUtility.JobDebuggerEnabled, "JobDebugger must be enabled for these tests"); - } - - struct TestCopy1To2Job : IJob - { - public ComponentGroupArray entities; - unsafe public void Execute() - { - foreach (var e in entities) - e.testData2->value0 = e.testData->value; - } - } - - struct TestReadOnlyJob : IJob - { - public ComponentGroupArray entities; - public void Execute() - { - foreach (var e in entities) - ; - } - } - - - //@TODO: Test for Entity setup with same component twice... - //@TODO: Test for subtractive components - //@TODO: Test for process ComponentGroupArray in job - - unsafe struct TestEntity - { -#pragma warning disable 649 - [ReadOnly] - - public EcsTestData* testData; - - public EcsTestData2* testData2; - } - - unsafe struct TestEntityReadOnly - { - [ReadOnly] - public EcsTestData* testData; - [ReadOnly] - public EcsTestData2* testData2; - } -#pragma warning restore 649 - - [Test] - [StandaloneFixme] - public void ComponentAccessAfterScheduledJobThrowsEntityArray() - { - m_Manager.CreateEntity(typeof(EcsTestData), typeof(EcsTestData2)); - - var job = new TestCopy1To2Job(); - job.entities = EmptySystem.GetEntities(); - - var fence = job.Schedule(); - - var entityArray = EmptySystem.GetEntities(); - Assert.Throws(() => { var temp = entityArray[0]; }); - - fence.Complete(); - } - - [Test] - [StandaloneFixme] - public void ComponentGroupArrayJobScheduleDetectsWriteDependency() - { - var entity = m_Manager.CreateEntity(typeof(EcsTestData), typeof(EcsTestData2)); - m_Manager.SetComponentData(entity, new EcsTestData(42)); - - var job = new TestCopy1To2Job(); - job.entities = EmptySystem.GetEntities(); - - var fence = job.Schedule(); - Assert.Throws(() => { job.Schedule(); }); - - fence.Complete(); - } - - [Test] - [StandaloneFixme] - public void ComponentGroupArrayJobScheduleReadOnlyParallelIsAllowed() - { - var entity = m_Manager.CreateEntity(typeof(EcsTestData), typeof(EcsTestData2)); - m_Manager.SetComponentData(entity, new EcsTestData(42)); - - var job = new TestReadOnlyJob(); - job.entities = EmptySystem.GetEntities(); - - var fence = job.Schedule(); - var fence2 = job.Schedule(); - - JobHandle.CompleteAll(ref fence, ref fence2); - } - - unsafe struct TestEntitySub2 - { -#pragma warning disable 649 - public EcsTestData* testData; - public ExcludeComponent testData2; -#pragma warning restore 649 - } - - [Test] - [StandaloneFixme] - public void ComponentGroupArrayExclude() - { - m_Manager.CreateEntity(typeof(EcsTestData), typeof(EcsTestData2)); - m_Manager.CreateEntity(typeof(EcsTestData)); - - var entities = EmptySystem.GetEntities(); - Assert.AreEqual(1, entities.Length); - } - } -} -#endif diff --git a/Unity.Entities.Tests/ComponentGroupArrayTests.cs.meta b/Unity.Entities.Tests/ComponentGroupArrayTests.cs.meta deleted file mode 100644 index 8e6b5fce..00000000 --- a/Unity.Entities.Tests/ComponentGroupArrayTests.cs.meta +++ /dev/null @@ -1,3 +0,0 @@ -fileFormatVersion: 2 -guid: 41659abe70864cfdbd0d142215cf0cd9 -timeCreated: 1512497553 \ No newline at end of file diff --git a/Unity.Entities.Tests/ComponentGroupDelta.cs b/Unity.Entities.Tests/ComponentGroupDelta.cs index 2d2249f4..ba163e2b 100644 --- a/Unity.Entities.Tests/ComponentGroupDelta.cs +++ b/Unity.Entities.Tests/ComponentGroupDelta.cs @@ -4,8 +4,6 @@ using Unity.Collections; using Unity.Jobs; -#pragma warning disable 618 - namespace Unity.Entities.Tests { [TestFixture] @@ -28,14 +26,15 @@ public class DeltaCheckSystem : ComponentSystem protected override void OnUpdate() { - var group = GetComponentGroup(typeof(EcsTestData)); + var group = GetEntityQuery(typeof(EcsTestData)); group.SetFilterChanged(typeof(EcsTestData)); - var actualEntityArray = group.GetEntityArray().ToArray(); + var actualEntityArray = group.ToEntityArray(Allocator.TempJob); var systemVersion = GlobalSystemVersion; var lastSystemVersion = LastSystemVersion; CollectionAssert.AreEqual(Expected, actualEntityArray); + actualEntityArray.Dispose(); } public void UpdateExpectedResults(Entity[] expected) @@ -47,11 +46,10 @@ public void UpdateExpectedResults(Entity[] expected) [Test] - [StandaloneFixme] public void CreateEntityTriggersChange() { Entity[] entity = new Entity[] { m_Manager.CreateEntity(typeof(EcsTestData)) }; - var deltaCheckSystem = World.CreateManager(); + var deltaCheckSystem = World.CreateSystem(); deltaCheckSystem.UpdateExpectedResults(entity); } @@ -59,8 +57,6 @@ public enum ChangeMode { SetComponentData, SetComponentDataFromEntity, - ComponentDataArray, - ComponentGroupArray } #pragma warning disable 649 @@ -80,15 +76,10 @@ unsafe struct GroupRO unsafe void SetValue(int index, int value, ChangeMode mode) { EmptySystem.Update(); - var entityArray = EmptySystem.GetComponentGroup(typeof(EcsTestData)).GetEntityArray(); + var entityArray = EmptySystem.GetEntityQuery(typeof(EcsTestData)).ToEntityArray(Allocator.TempJob); var entity = entityArray[index]; - if (mode == ChangeMode.ComponentDataArray) - { - var array = EmptySystem.GetComponentGroup(typeof(EcsTestData)).GetComponentDataArray(); - array[index] = new EcsTestData(value); - } - else if (mode == ChangeMode.SetComponentData) + if (mode == ChangeMode.SetComponentData) { m_Manager.SetComponentData(entity, new EcsTestData(value)); } @@ -98,27 +89,17 @@ unsafe void SetValue(int index, int value, ChangeMode mode) var array = EmptySystem.GetComponentDataFromEntity(false); array[entity] = new EcsTestData(value); } - else if (mode == ChangeMode.ComponentGroupArray) - { - *(EmptySystem.GetEntities()[index].Data) = new EcsTestData(value); - } + + entityArray.Dispose(); } // Running GetValue should not trigger any changes to chunk version. void GetValue(ChangeMode mode) { EmptySystem.Update(); - var entityArray = EmptySystem.GetComponentGroup(typeof(EcsTestData)).GetEntityArray(); + var entityArray = EmptySystem.GetEntityQuery(typeof(EcsTestData)).ToEntityArray(Allocator.TempJob); - if (mode == ChangeMode.ComponentDataArray) - { - var array = EmptySystem.GetComponentGroup(typeof(EcsTestData)).GetComponentDataArray(); - for (int i = 0; i != array.Length; i++) - { - var val = array[i]; - } - } - else if (mode == ChangeMode.SetComponentData) + if (mode == ChangeMode.SetComponentData) { for(int i = 0;i != entityArray.Length;i++) m_Manager.GetComponentData(entityArray[i]); @@ -128,22 +109,17 @@ void GetValue(ChangeMode mode) for(int i = 0;i != entityArray.Length;i++) m_Manager.GetComponentData(entityArray[i]); } - else if (mode == ChangeMode.ComponentGroupArray) - { - foreach (var e in EmptySystem.GetEntities()) - ; - } + entityArray.Dispose(); } [Test] - [StandaloneFixme] public void ChangeEntity([Values]ChangeMode mode) { var entity0 = m_Manager.CreateEntity(typeof(EcsTestData)); var entity1 = m_Manager.CreateEntity(typeof(EcsTestData), typeof(EcsTestData2)); - var deltaCheckSystem0 = World.CreateManager(); - var deltaCheckSystem1 = World.CreateManager(); + var deltaCheckSystem0 = World.CreateSystem(); + var deltaCheckSystem1 = World.CreateSystem(); // Chunk versions are considered changed upon creation and until after they're first updated. deltaCheckSystem0.UpdateExpectedResults(new Entity[] { entity0, entity1 }); @@ -172,12 +148,11 @@ public void ChangeEntity([Values]ChangeMode mode) } [Test] - [StandaloneFixme] public void GetEntityDataDoesNotChange([Values]ChangeMode mode) { var entity0 = m_Manager.CreateEntity(typeof(EcsTestData)); var entity1 = m_Manager.CreateEntity(typeof(EcsTestData), typeof(EcsTestData2)); - var deltaCheckSystem = World.CreateManager(); + var deltaCheckSystem = World.CreateSystem(); // First update of chunks after creation. SetValue(0, 2, mode); @@ -185,20 +160,19 @@ public void GetEntityDataDoesNotChange([Values]ChangeMode mode) deltaCheckSystem.UpdateExpectedResults(new Entity[] { entity0, entity1 }); deltaCheckSystem.UpdateExpectedResults(nothing); - // Now ensure that GetValue does not trigger a change on the ComponentGroup. + // Now ensure that GetValue does not trigger a change on the EntityQuery. GetValue(mode); deltaCheckSystem.UpdateExpectedResults(nothing); } [Test] - [StandaloneFixme] public void ChangeEntityWrap() { m_Manager.Debug.SetGlobalSystemVersion(uint.MaxValue-3); var entity = m_Manager.CreateEntity(typeof(EcsTestData)); - var deltaCheckSystem = World.CreateManager(); + var deltaCheckSystem = World.CreateSystem(); for (int i = 0; i != 7; i++) { @@ -210,7 +184,6 @@ public void ChangeEntityWrap() } [Test] - [StandaloneFixme] public void NoChangeEntityWrap() { m_Manager.Debug.SetGlobalSystemVersion(uint.MaxValue - 3); @@ -218,7 +191,7 @@ public void NoChangeEntityWrap() var entity = m_Manager.CreateEntity(typeof(EcsTestData)); SetValue(0, 2, ChangeMode.SetComponentData); - var deltaCheckSystem = World.CreateManager(); + var deltaCheckSystem = World.CreateSystem(); deltaCheckSystem.UpdateExpectedResults(new Entity[] { entity }); for (int i = 0; i != 7; i++) @@ -228,7 +201,7 @@ public void NoChangeEntityWrap() [DisableAutoCreation] public class DeltaProcessComponentSystem : JobComponentSystem { - struct DeltaJob : IJobProcessComponentData + struct DeltaJob : IJobForEach { public void Execute([ChangedFilter][ReadOnly]ref EcsTestData input, ref EcsTestData2 output) { @@ -244,13 +217,12 @@ protected override JobHandle OnUpdate(JobHandle deps) [Test] - [StandaloneFixme] public void IJobProcessComponentDeltaWorks() { var entity0 = m_Manager.CreateEntity(typeof(EcsTestData), typeof(EcsTestData2), typeof(EcsTestData3)); var entity1 = m_Manager.CreateEntity(typeof(EcsTestData), typeof(EcsTestData2)); - var deltaSystem = World.CreateManager(); + var deltaSystem = World.CreateSystem(); // First update of chunks after creation. SetValue(0, -100, ChangeMode.SetComponentData); @@ -276,7 +248,7 @@ public void IJobProcessComponentDeltaWorks() [DisableAutoCreation] public class DeltaProcessComponentSystemUsingRun : ComponentSystem { - struct DeltaJob : IJobProcessComponentData + struct DeltaJob : IJobForEach { public void Execute([ChangedFilter][ReadOnly]ref EcsTestData input, ref EcsTestData2 output) { @@ -291,13 +263,12 @@ protected override void OnUpdate() } [Test] - [StandaloneFixme] public void IJobProcessComponentDeltaWorksWhenUsingRun() { var entity0 = m_Manager.CreateEntity(typeof(EcsTestData), typeof(EcsTestData2), typeof(EcsTestData3)); var entity1 = m_Manager.CreateEntity(typeof(EcsTestData), typeof(EcsTestData2)); - var deltaSystem = World.CreateManager(); + var deltaSystem = World.CreateSystem(); // First update of chunks after creation. SetValue(0, -100, ChangeMode.SetComponentData); @@ -320,7 +291,6 @@ public void IJobProcessComponentDeltaWorksWhenUsingRun() #if false [Test] - [StandaloneFixme] public void IJobProcessComponentDeltaWorksWhenSetSharedComponent() { var entity0 = m_Manager.CreateEntity(typeof(EcsTestData), typeof(EcsTestData2), typeof(EcsTestData3), typeof(EcsTestSharedComp)); @@ -341,10 +311,10 @@ public void IJobProcessComponentDeltaWorksWhenSetSharedComponent() [DisableAutoCreation] public class ModifyComponentSystem1Comp : JobComponentSystem { - public ComponentGroup m_Group; + public EntityQuery m_Group; public EcsTestSharedComp m_sharedComp; - struct DeltaJob : IJobProcessComponentData + struct DeltaJob : IJobForEach { public void Execute(ref EcsTestData data) { @@ -354,28 +324,28 @@ public void Execute(ref EcsTestData data) protected override JobHandle OnUpdate(JobHandle deps) { - m_Group = GetComponentGroup( + m_Group = GetEntityQuery( typeof(EcsTestData), ComponentType.ReadOnly(typeof(EcsTestSharedComp))); m_Group.SetFilter(m_sharedComp); DeltaJob job = new DeltaJob(); - return job.ScheduleGroup(m_Group, deps); + return job.Schedule(m_Group, deps); } } [DisableAutoCreation] public class DeltaModifyComponentSystem1Comp : JobComponentSystem { - struct DeltaJobFirstRunAfterCreation : IJobProcessComponentData + struct DeltaJobFirstRunAfterCreation : IJobForEach { public void Execute([ChangedFilter]ref EcsTestData output) { output.value = 0; } } - struct DeltaJob : IJobProcessComponentData + struct DeltaJob : IJobForEach { public void Execute([ChangedFilter]ref EcsTestData output) { @@ -394,15 +364,14 @@ protected override JobHandle OnUpdate(JobHandle deps) } [Test] - [StandaloneFixme] public void ChangedFilterJobAfterAnotherJob1Comp() { var archetype = m_Manager.CreateArchetype(typeof(EcsTestData), typeof(EcsTestSharedComp)); var entities = new NativeArray(10000, Allocator.Persistent); m_Manager.CreateEntity(archetype, entities); - var modifSystem = World.CreateManager(); - var deltaSystem = World.CreateManager(); + var modifSystem = World.CreateSystem(); + var deltaSystem = World.CreateSystem(); // First update of chunks after creation. modifSystem.Update(); @@ -435,10 +404,10 @@ public void ChangedFilterJobAfterAnotherJob1Comp() [DisableAutoCreation] public class ModifyComponentSystem2Comp : JobComponentSystem { - public ComponentGroup m_Group; + public EntityQuery m_Group; public EcsTestSharedComp m_sharedComp; - struct DeltaJob : IJobProcessComponentData + struct DeltaJob : IJobForEach { public void Execute(ref EcsTestData data, ref EcsTestData2 data2) { @@ -448,7 +417,7 @@ public void Execute(ref EcsTestData data, ref EcsTestData2 data2) protected override JobHandle OnUpdate(JobHandle deps) { - m_Group = GetComponentGroup( + m_Group = GetEntityQuery( typeof(EcsTestData), typeof(EcsTestData2), ComponentType.ReadOnly(typeof(EcsTestSharedComp))); @@ -456,14 +425,14 @@ protected override JobHandle OnUpdate(JobHandle deps) m_Group.SetFilter(m_sharedComp); DeltaJob job = new DeltaJob(); - return job.ScheduleGroup(m_Group, deps); + return job.Schedule(m_Group, deps); } } [DisableAutoCreation] public class DeltaModifyComponentSystem2Comp : JobComponentSystem { - struct DeltaJobFirstRunAfterCreation : IJobProcessComponentData + struct DeltaJobFirstRunAfterCreation : IJobForEach { public void Execute(ref EcsTestData output, ref EcsTestData2 output2) { @@ -472,7 +441,7 @@ public void Execute(ref EcsTestData output, ref EcsTestData2 output2) } } - struct DeltaJobChanged0 : IJobProcessComponentData + struct DeltaJobChanged0 : IJobForEach { public void Execute([ChangedFilter]ref EcsTestData output, ref EcsTestData2 output2) { @@ -481,7 +450,7 @@ public void Execute([ChangedFilter]ref EcsTestData output, ref EcsTestData2 outp } } - struct DeltaJobChanged1 : IJobProcessComponentData + struct DeltaJobChanged1 : IJobForEach { public void Execute(ref EcsTestData output, [ChangedFilter]ref EcsTestData2 output2) { @@ -517,7 +486,6 @@ protected override JobHandle OnUpdate(JobHandle deps) } [Test] - [StandaloneFixme] public void ChangedFilterJobAfterAnotherJob2Comp([Values]DeltaModifyComponentSystem2Comp.Variant variant) { var archetype = m_Manager.CreateArchetype(typeof(EcsTestData), typeof(EcsTestData2), typeof(EcsTestSharedComp)); @@ -525,8 +493,8 @@ public void ChangedFilterJobAfterAnotherJob2Comp([Values]DeltaModifyComponentSys m_Manager.CreateEntity(archetype, entities); // All entities have just been created, so they're all technically "changed". - var modifSystem = World.CreateManager(); - var deltaSystem = World.CreateManager(); + var modifSystem = World.CreateSystem(); + var deltaSystem = World.CreateSystem(); // First update of chunks after creation. modifSystem.Update(); @@ -562,10 +530,10 @@ public void ChangedFilterJobAfterAnotherJob2Comp([Values]DeltaModifyComponentSys [DisableAutoCreation] public class ModifyComponentSystem3Comp : JobComponentSystem { - public ComponentGroup m_Group; + public EntityQuery m_Group; public EcsTestSharedComp m_sharedComp; - struct DeltaJob : IJobProcessComponentData + struct DeltaJob : IJobForEach { public void Execute(ref EcsTestData data, ref EcsTestData2 data2, ref EcsTestData3 data3) { @@ -576,7 +544,7 @@ public void Execute(ref EcsTestData data, ref EcsTestData2 data2, ref EcsTestDat protected override JobHandle OnUpdate(JobHandle deps) { - m_Group = GetComponentGroup( + m_Group = GetEntityQuery( typeof(EcsTestData), typeof(EcsTestData2), typeof(EcsTestData3), @@ -585,14 +553,14 @@ protected override JobHandle OnUpdate(JobHandle deps) m_Group.SetFilter(m_sharedComp); DeltaJob job = new DeltaJob(); - return job.ScheduleGroup(m_Group, deps); + return job.Schedule(m_Group, deps); } } [DisableAutoCreation] public class DeltaModifyComponentSystem3Comp : JobComponentSystem { - struct DeltaJobChanged0 : IJobProcessComponentData + struct DeltaJobChanged0 : IJobForEach { public void Execute([ChangedFilter]ref EcsTestData output, ref EcsTestData2 output2, ref EcsTestData3 output3) { @@ -602,7 +570,7 @@ public void Execute([ChangedFilter]ref EcsTestData output, ref EcsTestData2 outp } } - struct DeltaJobChanged1 : IJobProcessComponentData + struct DeltaJobChanged1 : IJobForEach { public void Execute(ref EcsTestData output, [ChangedFilter]ref EcsTestData2 output2, ref EcsTestData3 output3) { @@ -612,7 +580,7 @@ public void Execute(ref EcsTestData output, [ChangedFilter]ref EcsTestData2 outp } } - struct DeltaJobChanged2 : IJobProcessComponentData + struct DeltaJobChanged2 : IJobForEach { public void Execute(ref EcsTestData output, ref EcsTestData2 output2, [ChangedFilter]ref EcsTestData3 output3) { @@ -653,8 +621,8 @@ public void ChangedFilterJobAfterAnotherJob3Comp([Values]DeltaModifyComponentSys var entities = new NativeArray(10000, Allocator.Persistent); m_Manager.CreateEntity(archetype, entities); - var modifSystem = World.CreateManager(); - var deltaSystem = World.CreateManager(); + var modifSystem = World.CreateSystem(); + var deltaSystem = World.CreateSystem(); deltaSystem.variant = variant; @@ -687,7 +655,7 @@ public void ChangedFilterJobAfterAnotherJob3Comp([Values]DeltaModifyComponentSys [DisableAutoCreation] class ChangeFilter1TestSystem : JobComponentSystem { - struct ChangedFilterJob : IJobProcessComponentData + struct ChangedFilterJob : IJobForEach { public void Execute(ref EcsTestData output, [ChangedFilter]ref EcsTestData2 output2) { @@ -706,7 +674,7 @@ protected override JobHandle OnUpdate(JobHandle inputDeps) public void ChangeFilterWorksWithOneTypes() { var e = m_Manager.CreateEntity(); - var system = World.GetOrCreateManager(); + var system = World.GetOrCreateSystem(); m_Manager.AddComponentData(e, new EcsTestData(0)); m_Manager.AddComponentData(e, new EcsTestData2(1)); @@ -733,7 +701,7 @@ public void ChangeFilterWorksWithOneTypes() [DisableAutoCreation] class ChangeFilter2TestSystem : JobComponentSystem { - struct ChangedFilterJob : IJobProcessComponentData + struct ChangedFilterJob : IJobForEach { public void Execute(ref EcsTestData output, [ChangedFilter]ref EcsTestData2 output2, [ChangedFilter]ref EcsTestData3 output3) { @@ -752,7 +720,7 @@ protected override JobHandle OnUpdate(JobHandle inputDeps) public void ChangeFilterWorksWithTwoTypes() { var e = m_Manager.CreateEntity(); - var system = World.GetOrCreateManager(); + var system = World.GetOrCreateSystem(); m_Manager.AddComponentData(e, new EcsTestData(0)); m_Manager.AddComponentData(e, new EcsTestData2(1)); m_Manager.AddComponentData(e, new EcsTestData3(2)); diff --git a/Unity.Entities.Tests/ComponentGroupTests.cs b/Unity.Entities.Tests/ComponentGroupTests.cs index fde05527..1a3242a4 100644 --- a/Unity.Entities.Tests/ComponentGroupTests.cs +++ b/Unity.Entities.Tests/ComponentGroupTests.cs @@ -14,7 +14,7 @@ ArchetypeChunk[] CreateEntitiesAndReturnChunks(EntityArchetype archetype, int en { var entities = new NativeArray(entityCount, Allocator.Temp); m_Manager.CreateEntity(archetype, entities); -#if UNITY_CSHARP_TINY +#if NET_DOTS var managedEntities = new Entity[entities.Length]; for (int i = 0; i < entities.Length; i++) { @@ -45,8 +45,8 @@ public void CreateArchetypeChunkArray() var allCreatedChunks = createdChunks1.Concat(createdChunks2).Concat(createdChunks12); - var group1 = m_Manager.CreateComponentGroup(typeof(EcsTestData)); - var group12 = m_Manager.CreateComponentGroup(typeof(EcsTestData), typeof(EcsTestData2)); + var group1 = m_Manager.CreateEntityQuery(typeof(EcsTestData)); + var group12 = m_Manager.CreateEntityQuery(typeof(EcsTestData), typeof(EcsTestData2)); var queriedChunks1 = group1.CreateArchetypeChunkArray(Allocator.TempJob); var queriedChunks12 = group12.CreateArchetypeChunkArray(Allocator.TempJob); @@ -67,7 +67,6 @@ void SetShared(Entity e, int i) } [Test] - [StandaloneFixme] // ISharedComponentData public void CreateArchetypeChunkArray_FiltersSharedComponents() { var archetype1 = m_Manager.CreateArchetype(typeof(EcsTestData), typeof(EcsTestSharedComp)); @@ -78,7 +77,7 @@ public void CreateArchetypeChunkArray_FiltersSharedComponents() var createdChunks3 = CreateEntitiesAndReturnChunks(archetype1, 5000, e => SetShared(e, 2)); var createdChunks4 = CreateEntitiesAndReturnChunks(archetype2, 5000, e => SetShared(e, 2)); - var group = m_Manager.CreateComponentGroup(typeof(EcsTestSharedComp)); + var group = m_Manager.CreateEntityQuery(typeof(EcsTestSharedComp)); group.SetFilter(new EcsTestSharedComp(1)); @@ -103,7 +102,6 @@ void SetShared(Entity e, int i, int j) } [Test] - [StandaloneFixme] // ISharedComponentData public void CreateArchetypeChunkArray_FiltersTwoSharedComponents() { var archetype1 = m_Manager.CreateArchetype(typeof(EcsTestData), typeof(EcsTestSharedComp), typeof(EcsTestSharedComp2)); @@ -118,7 +116,7 @@ public void CreateArchetypeChunkArray_FiltersTwoSharedComponents() var createdChunks7 = CreateEntitiesAndReturnChunks(archetype1, 5000, e => SetShared(e, 2,8)); var createdChunks8 = CreateEntitiesAndReturnChunks(archetype2, 5000, e => SetShared(e, 2,8)); - var group = m_Manager.CreateComponentGroup(typeof(EcsTestSharedComp), typeof(EcsTestSharedComp2)); + var group = m_Manager.CreateEntityQuery(typeof(EcsTestSharedComp), typeof(EcsTestSharedComp2)); group.SetFilter(new EcsTestSharedComp(1), new EcsTestSharedComp2(7)); var queriedChunks1 = group.CreateArchetypeChunkArray(Allocator.TempJob); @@ -164,7 +162,7 @@ public void CreateArchetypeChunkArray_FiltersChangeVersions() m_ManagerDebug.SetGlobalSystemVersion(40); var createdChunks3 = CreateEntitiesAndReturnChunks(archetype3, 5000, e => SetData(e, 3)); - var group = m_Manager.CreateComponentGroup(typeof(EcsTestData)); + var group = m_Manager.CreateEntityQuery(typeof(EcsTestData)); group.SetFilterChanged(typeof(EcsTestData)); @@ -213,7 +211,7 @@ public void CreateArchetypeChunkArray_FiltersTwoChangeVersions() m_ManagerDebug.SetGlobalSystemVersion(40); var createdChunks3 = CreateEntitiesAndReturnChunks(archetype3, 5000, e => SetData(e, 3, 6)); - var group = m_Manager.CreateComponentGroup(typeof(EcsTestData), typeof(EcsTestData2)); + var group = m_Manager.CreateEntityQuery(typeof(EcsTestData), typeof(EcsTestData2)); group.SetFilterChanged(new ComponentType[]{typeof(EcsTestData), typeof(EcsTestData2)}); @@ -258,9 +256,9 @@ public void TestIssue1098() using ( - var group = m_Manager.CreateComponentGroup + var group = m_Manager.CreateEntityQuery ( - new EntityArchetypeQuery + new EntityQueryDesc { All = new ComponentType[] {typeof(EcsTestData)} } @@ -276,7 +274,7 @@ public void TestIssue1098() [AlwaysUpdateSystem] public class WriteEcsTestDataSystem : JobComponentSystem { - private struct WriteJob : IJobProcessComponentData + private struct WriteJob : IJobForEach { public void Execute(ref EcsTestData c0) {} } @@ -289,13 +287,13 @@ protected override JobHandle OnUpdate(JobHandle input) } [Test] - public void CreateArchetypeChunkArray_SyncsChangeFilterTypes() + public unsafe void CreateArchetypeChunkArray_SyncsChangeFilterTypes() { - var group = m_Manager.CreateComponentGroup(typeof(EcsTestData)); + var group = m_Manager.CreateEntityQuery(typeof(EcsTestData)); group.SetFilterChanged(typeof(EcsTestData)); - var ws1 = World.GetOrCreateManager(); + var ws1 = World.GetOrCreateSystem(); ws1.Update(); - var safetyHandle = m_Manager.ComponentJobSafetyManager.GetSafetyHandle(TypeManager.GetTypeIndex(), false); + var safetyHandle = m_Manager.ComponentJobSafetyManager->GetSafetyHandle(TypeManager.GetTypeIndex(), false); Assert.Throws(() => AtomicSafetyHandle.CheckWriteAndThrow(safetyHandle)); var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob); @@ -306,13 +304,13 @@ public void CreateArchetypeChunkArray_SyncsChangeFilterTypes() } [Test] - public void CalculateLength_SyncsChangeFilterTypes() + public unsafe void CalculateLength_SyncsChangeFilterTypes() { - var group = m_Manager.CreateComponentGroup(typeof(EcsTestData)); + var group = m_Manager.CreateEntityQuery(typeof(EcsTestData)); group.SetFilterChanged(typeof(EcsTestData)); - var ws1 = World.GetOrCreateManager(); + var ws1 = World.GetOrCreateSystem(); ws1.Update(); - var safetyHandle = m_Manager.ComponentJobSafetyManager.GetSafetyHandle(TypeManager.GetTypeIndex(), false); + var safetyHandle = m_Manager.ComponentJobSafetyManager->GetSafetyHandle(TypeManager.GetTypeIndex(), false); Assert.Throws(() => AtomicSafetyHandle.CheckWriteAndThrow(safetyHandle)); group.CalculateLength(); @@ -323,7 +321,6 @@ public void CalculateLength_SyncsChangeFilterTypes() #endif [Test] - [StandaloneFixme] // ISharedComponentData public void ToEntityArrayOnFilteredGroup() { // Note - test is setup so that each entity is in its own chunk, this checks that entity indices are correct @@ -335,7 +332,7 @@ public void ToEntityArrayOnFilteredGroup() m_Manager.SetSharedComponentData(b, new EcsTestSharedComp {value = 456}); m_Manager.SetSharedComponentData(c, new EcsTestSharedComp {value = 123}); - using (var group = m_Manager.CreateComponentGroup(typeof(EcsTestSharedComp))) + using (var group = m_Manager.CreateEntityQuery(typeof(EcsTestSharedComp))) { group.SetFilter(new EcsTestSharedComp {value = 123}); using (var entities = group.ToEntityArray(Allocator.TempJob)) @@ -344,7 +341,7 @@ public void ToEntityArrayOnFilteredGroup() } } - using (var group = m_Manager.CreateComponentGroup(typeof(EcsTestSharedComp))) + using (var group = m_Manager.CreateEntityQuery(typeof(EcsTestSharedComp))) { group.SetFilter(new EcsTestSharedComp {value = 456}); using (var entities = group.ToEntityArray(Allocator.TempJob)) diff --git a/Unity.Entities.Tests/ComponentOrderVersionTests.cs b/Unity.Entities.Tests/ComponentOrderVersionTests.cs index 1918ccda..4fce6690 100644 --- a/Unity.Entities.Tests/ComponentOrderVersionTests.cs +++ b/Unity.Entities.Tests/ComponentOrderVersionTests.cs @@ -32,10 +32,10 @@ void AddEvenOddTestData() } } - void ActionEvenOdd(Action even, Action odd) + void ActionEvenOdd(Action even, Action odd) { var uniqueTypes = new List(10); - var group = m_Manager.CreateComponentGroup(typeof(EcsTestData), typeof(SharedData1)); + var group = m_Manager.CreateEntityQuery(typeof(EcsTestData), typeof(SharedData1)); group.CompleteDependency(); m_Manager.GetAllUniqueSharedComponentData(uniqueTypes); @@ -61,7 +61,6 @@ void ActionEvenOdd(Action even, Action } [Test] - [StandaloneFixme] // ISharedComponentData public void SharedComponentNoChangeVersionUnchanged() { AddEvenOddTestData(); @@ -69,7 +68,7 @@ public void SharedComponentNoChangeVersionUnchanged() (version, group) => { Assert.AreEqual(version, 1); }); } - void TestSourceEvenValues(int version, ComponentGroup group) + void TestSourceEvenValues(int version, EntityQuery group) { var testData = group.ToComponentDataArray(Allocator.TempJob); @@ -83,7 +82,7 @@ void TestSourceEvenValues(int version, ComponentGroup group) testData.Dispose(); } - void TestSourceOddValues(int version, ComponentGroup group) + void TestSourceOddValues(int version, EntityQuery group) { var testData = group.ToComponentDataArray(Allocator.TempJob); @@ -98,14 +97,13 @@ void TestSourceOddValues(int version, ComponentGroup group) } [Test] - [StandaloneFixme] // ISharedComponentData public void SharedComponentNoChangeValuesUnchanged() { AddEvenOddTestData(); ActionEvenOdd(TestSourceEvenValues, TestSourceOddValues); } - void ChangeGroupOrder(int version, ComponentGroup group) + void ChangeGroupOrder(int version, EntityQuery group) { var entities = group.ToEntityArray(Allocator.TempJob); @@ -123,7 +121,6 @@ void ChangeGroupOrder(int version, ComponentGroup group) } [Test] - [StandaloneFixme] // ISharedComponentData public void SharedComponentChangeOddGroupOrderOnlyOddVersionChanged() { AddEvenOddTestData(); @@ -134,7 +131,6 @@ public void SharedComponentChangeOddGroupOrderOnlyOddVersionChanged() } [Test] - [StandaloneFixme] // ISharedComponentData public void SharedComponentChangeOddGroupOrderEvenValuesUnchanged() { AddEvenOddTestData(); @@ -143,7 +139,7 @@ public void SharedComponentChangeOddGroupOrderEvenValuesUnchanged() ActionEvenOdd(TestSourceEvenValues, (version, group) => { }); } - void DestroyAllButOneEntityInGroup(int version, ComponentGroup group) + void DestroyAllButOneEntityInGroup(int version, EntityQuery group) { var entities = group.ToEntityArray(Allocator.TempJob); @@ -157,7 +153,6 @@ void DestroyAllButOneEntityInGroup(int version, ComponentGroup group) } [Test] - [StandaloneFixme] // ISharedComponentData public void SharedComponentDestroyAllButOneEntityInOddGroupOnlyOddVersionChanged() { AddEvenOddTestData(); @@ -168,7 +163,6 @@ public void SharedComponentDestroyAllButOneEntityInOddGroupOnlyOddVersionChanged } [Test] - [StandaloneFixme] // ISharedComponentData public void SharedComponentDestroyAllButOneEntityInOddGroupEvenValuesUnchanged() { AddEvenOddTestData(); @@ -228,7 +222,6 @@ public void ChangedOnlyAffectedArchetype() } [Test] - [StandaloneFixme] // ISharedComponentData public void SetSharedComponent() { var entity = m_Manager.CreateEntity(typeof(SharedData1), typeof(SharedData2)); @@ -241,7 +234,6 @@ public void SetSharedComponent() } [Test] - [StandaloneFixme] // ISharedComponentData public void DestroySharedComponentEntity() { var sharedData = new SharedData1(1); @@ -258,7 +250,6 @@ public void DestroySharedComponentEntity() } [Test] - [StandaloneFixme] // ISharedComponentData public void DestroySharedComponentDataSetsOrderVersionToZero() { var sharedData = new SharedData1(1); diff --git a/Unity.Entities.Tests/ComponentSystemGroupTests.cs b/Unity.Entities.Tests/ComponentSystemGroupTests.cs index b34b2a1f..cf931688 100644 --- a/Unity.Entities.Tests/ComponentSystemGroupTests.cs +++ b/Unity.Entities.Tests/ComponentSystemGroupTests.cs @@ -18,7 +18,7 @@ class TestGroup : ComponentSystemGroup } [DisableAutoCreation] -#if UNITY_CSHARP_TINY +#if NET_DOTS private class TestSystemBase :ComponentSystem { protected override void OnUpdate() => throw new System.NotImplementedException(); @@ -46,8 +46,8 @@ class TestSystem : TestSystemBase [Test] public void SortOneChildSystem() { - var parent = World.CreateManager(); - var child = World.CreateManager(); + var parent = World.CreateSystem(); + var child = World.CreateSystem(); parent.AddSystemToUpdateList(child); parent.SortSystemUpdateList(); CollectionAssert.AreEqual(new[] {child}, parent.Systems); @@ -66,9 +66,9 @@ class Sibling2System : TestSystemBase [Test] public void SortTwoChildSystems_CorrectOrder() { - var parent = World.CreateManager(); - var child1 = World.CreateManager(); - var child2 = World.CreateManager(); + var parent = World.CreateSystem(); + var child1 = World.CreateSystem(); + var child2 = World.CreateSystem(); parent.AddSystemToUpdateList(child1); parent.AddSystemToUpdateList(child2); parent.SortSystemUpdateList(); @@ -114,18 +114,18 @@ class Circle6System : TestSystemBase } [Test] -#if UNITY_CSHARP_TINY +#if NET_DOTS [Ignore("Tiny pre-compiles systems. Many tests will fail if they exist, not just this one.")] #endif public void DetectCircularDependency_Throws() { - var parent = World.CreateManager(); - var child1 = World.CreateManager(); - var child2 = World.CreateManager(); - var child3 = World.CreateManager(); - var child4 = World.CreateManager(); - var child5 = World.CreateManager(); - var child6 = World.CreateManager(); + var parent = World.CreateSystem(); + var child1 = World.CreateSystem(); + var child2 = World.CreateSystem(); + var child3 = World.CreateSystem(); + var child4 = World.CreateSystem(); + var child5 = World.CreateSystem(); + var child6 = World.CreateSystem(); parent.AddSystemToUpdateList(child3); parent.AddSystemToUpdateList(child6); parent.AddSystemToUpdateList(child2); @@ -175,11 +175,11 @@ class Unconstrained4System : TestSystemBase [Test] public void SortUnconstrainedSystems_IsDeterministic() { - var parent = World.CreateManager(); - var child1 = World.CreateManager(); - var child2 = World.CreateManager(); - var child3 = World.CreateManager(); - var child4 = World.CreateManager(); + var parent = World.CreateSystem(); + var child1 = World.CreateSystem(); + var child2 = World.CreateSystem(); + var child3 = World.CreateSystem(); + var child4 = World.CreateSystem(); parent.AddSystemToUpdateList(child2); parent.AddSystemToUpdateList(child4); parent.AddSystemToUpdateList(child3); @@ -219,14 +219,14 @@ protected override void OnUpdate() } } -#if !UNITY_CSHARP_TINY // Tiny precompiles systems, and lacks a Regex overload for LogAssert.Expect() +#if !NET_DOTS // Tiny precompiles systems, and lacks a Regex overload for LogAssert.Expect() [Test] public void SystemInGroupThrows_LaterSystemsRun() { - var parent = World.CreateManager(); - var child1 = World.CreateManager(); - var child2 = World.CreateManager(); - var child3 = World.CreateManager(); + var parent = World.CreateSystem(); + var child1 = World.CreateSystem(); + var child2 = World.CreateSystem(); + var child3 = World.CreateSystem(); parent.AddSystemToUpdateList(child1); parent.AddSystemToUpdateList(child2); parent.AddSystemToUpdateList(child3); diff --git a/Unity.Entities.Tests/ComponentSystemInjectionTests.cs b/Unity.Entities.Tests/ComponentSystemInjectionTests.cs deleted file mode 100644 index fe4df969..00000000 --- a/Unity.Entities.Tests/ComponentSystemInjectionTests.cs +++ /dev/null @@ -1,71 +0,0 @@ -#if !UNITY_ZEROPLAYER -using NUnit.Framework; - -// Injection is deprecated -#pragma warning disable 618 - -namespace Unity.Entities.Tests -{ - class ComponentSystemInjectionTests : ECSTestsFixture - { - [DisableAutoCreation] - class TestSystem : ComponentSystem - { - protected override void OnUpdate() - { - } - } - - [DisableAutoCreation] - class AttributeInjectionSystem : ComponentSystem - { - [Inject] -#pragma warning disable 649 - public TestSystem test; -#pragma warning restore 649 - - protected override void OnUpdate() - { - } - } - - [DisableAutoCreation] - class ConstructorInjectionSystem : ComponentSystem - { - public string test; - - public ConstructorInjectionSystem(string value) - { - this.test = value; - } - - protected override void OnUpdate() - { - } - } - - [Test] - public void ConstructorInjection() - { - var hello = "HelloWorld"; - var system = World.CreateManager(hello); - Assert.AreEqual(hello, system.test); - } - - [Test] - public void AttributeInjectionCreates() - { - var system = World.CreateManager(); - Assert.AreEqual(World.GetOrCreateManager(), system.test); - } - - [Test] - public void AttributeInjectionAlreadyCreated() - { - var test = World.CreateManager(); - var system = World.CreateManager(); - Assert.AreEqual(test, system.test); - } - } -} -#endif \ No newline at end of file diff --git a/Unity.Entities.Tests/ComponentSystemInjectionTests.cs.meta b/Unity.Entities.Tests/ComponentSystemInjectionTests.cs.meta deleted file mode 100644 index ab977e91..00000000 --- a/Unity.Entities.Tests/ComponentSystemInjectionTests.cs.meta +++ /dev/null @@ -1,3 +0,0 @@ -fileFormatVersion: 2 -guid: 83110af8b51c435eb9e2d0a50d4ae5de -timeCreated: 1512086766 \ No newline at end of file diff --git a/Unity.Entities.Tests/ComponentSystemStartStopRunningTests.cs b/Unity.Entities.Tests/ComponentSystemStartStopRunningTests.cs index 53239666..5f329284 100644 --- a/Unity.Entities.Tests/ComponentSystemStartStopRunningTests.cs +++ b/Unity.Entities.Tests/ComponentSystemStartStopRunningTests.cs @@ -10,7 +10,7 @@ class ComponentSystemStartStopRunningTests : ECSTestsFixture [DisableAutoCreation] class TestSystem : ComponentSystem { - public ComponentGroup m_TestGroup; + public EntityQuery m_TestGroup; public const string OnStartRunningString = nameof(TestSystem) + ".OnStartRunning()"; @@ -45,9 +45,9 @@ protected override void OnStopRunning() base.OnStopRunning(); } - protected override void OnCreateManager() + protected override void OnCreate() { - m_TestGroup = GetComponentGroup(ComponentType.ReadWrite()); + m_TestGroup = GetEntityQuery(ComponentType.ReadWrite()); } } @@ -71,7 +71,7 @@ public void ShouldRunSystem(bool shouldRun) public override void Setup() { base.Setup(); - system = World.Active.GetOrCreateManager(); + system = World.Active.GetOrCreateSystem(); ShouldRunSystem(true); } @@ -84,7 +84,7 @@ public override void TearDown() } if (system != null) { - World.Active.DestroyManager(system); + World.Active.DestroySystem(system); system = null; } @@ -282,7 +282,7 @@ public void OnStopRunning_WhenDestroyingActiveManager_CalledOnce() system.Update(); LogAssert.Expect(LogType.Log, TestSystem.OnStopRunningString); - World.Active.DestroyManager(system); + World.Active.DestroySystem(system); system = null; LogAssert.NoUnexpectedReceived(); @@ -294,7 +294,7 @@ public void OnStopRunning_WhenDestroyingInactiveManager_NotCalled() system.Enabled = false; system.Update(); - World.Active.DestroyManager(system); + World.Active.DestroySystem(system); system = null; LogAssert.NoUnexpectedReceived(); diff --git a/Unity.Entities.Tests/ComponentSystemTests.cs b/Unity.Entities.Tests/ComponentSystemTests.cs index cdd80ec4..6670c9af 100644 --- a/Unity.Entities.Tests/ComponentSystemTests.cs +++ b/Unity.Entities.Tests/ComponentSystemTests.cs @@ -17,12 +17,12 @@ protected override void OnUpdate() { } - protected override void OnCreateManager() + protected override void OnCreate() { Created = true; } - protected override void OnDestroyManager() + protected override void OnDestroy() { Created = false; } @@ -39,7 +39,7 @@ protected override void OnUpdate() [DisableAutoCreation] class ThrowExceptionSystem : TestSystem { - protected override void OnCreateManager() + protected override void OnCreate() { throw new System.Exception(); } @@ -65,7 +65,7 @@ protected override JobHandle OnUpdate(JobHandle inputDeps) return new Job(){ test = test }.Schedule(inputDeps); } - protected override void OnDestroyManager() + protected override void OnDestroy() { // We expect this to not throw an exception since the jobs scheduled // by this system should be synced before the system is destroyed @@ -76,38 +76,38 @@ protected override void OnDestroyManager() [Test] public void Create() { - var system = World.CreateManager(); - Assert.AreEqual(system, World.GetExistingManager()); + var system = World.CreateSystem(); + Assert.AreEqual(system, World.GetExistingSystem()); Assert.IsTrue(system.Created); } [Test] public void CreateAndDestroy() { - var system = World.CreateManager(); - World.DestroyManager(system); - Assert.AreEqual(null, World.GetExistingManager()); + var system = World.CreateSystem(); + World.DestroySystem(system); + Assert.AreEqual(null, World.GetExistingSystem()); Assert.IsFalse(system.Created); } [Test] - public void GetOrCreateManagerReturnsSameSystem() + public void GetOrCreateSystemReturnsSameSystem() { - var system = World.GetOrCreateManager(); - Assert.AreEqual(system, World.GetOrCreateManager()); + var system = World.GetOrCreateSystem(); + Assert.AreEqual(system, World.GetOrCreateSystem()); } [Test] public void InheritedSystem() { - var system = World.CreateManager(); - Assert.AreEqual(system, World.GetExistingManager()); - Assert.AreEqual(system, World.GetExistingManager()); + var system = World.CreateSystem(); + Assert.AreEqual(system, World.GetExistingSystem()); + Assert.AreEqual(system, World.GetExistingSystem()); - World.DestroyManager(system); + World.DestroySystem(system); - Assert.AreEqual(null, World.GetExistingManager()); - Assert.AreEqual(null, World.GetExistingManager()); + Assert.AreEqual(null, World.GetExistingSystem()); + Assert.AreEqual(null, World.GetExistingSystem()); Assert.IsFalse(system.Created); } @@ -116,142 +116,142 @@ public void InheritedSystem() [Test] public void CreateNonSystemThrows() { - Assert.Throws(() => { World.CreateManager(typeof(Entity)); }); + Assert.Throws(() => { World.CreateSystem(typeof(Entity)); }); } [Test] public void GetOrCreateNonSystemThrows() { - Assert.Throws(() => { World.GetOrCreateManager(typeof(Entity)); }); + Assert.Throws(() => { World.GetOrCreateSystem(typeof(Entity)); }); } #endif [Test] public void OnCreateThrowRemovesSystem() { - Assert.Throws(() => { World.CreateManager(); }); - Assert.AreEqual(null, World.GetExistingManager()); + Assert.Throws(() => { World.CreateSystem(); }); + Assert.AreEqual(null, World.GetExistingSystem()); } [Test] [StandaloneFixme] // IJob public void DestroySystemWhileJobUsingArrayIsRunningWorks() { - var system = World.CreateManager(); + var system = World.CreateSystem(); system.Update(); - World.DestroyManager(system); + World.DestroySystem(system); } [Test] public void DisposeSystemComponentGroupThrows() { - var system = World.CreateManager(); - var group = system.GetComponentGroup(typeof(EcsTestData)); + var system = World.CreateSystem(); + var group = system.GetEntityQuery(typeof(EcsTestData)); Assert.Throws(() => group.Dispose()); } [Test] - public void DestroyManagerTwiceThrows() + public void DestroySystemTwiceThrows() { - var system = World.CreateManager(); - World.DestroyManager(system); - Assert.Throws(() => World.DestroyManager(system) ); + var system = World.CreateSystem(); + World.DestroySystem(system); + Assert.Throws(() => World.DestroySystem(system) ); } [Test] public void CreateTwoSystemsOfSameType() { - var systemA = World.CreateManager(); - var systemB = World.CreateManager(); - // CreateManager makes a new manager + var systemA = World.CreateSystem(); + var systemB = World.CreateSystem(); + // CreateSystem makes a new system Assert.AreNotEqual(systemA, systemB); // Return first system - Assert.AreEqual(systemA, World.GetOrCreateManager()); + Assert.AreEqual(systemA, World.GetOrCreateSystem()); } [Test] public void CreateTwoSystemsAfterDestroyReturnSecond() { - var systemA = World.CreateManager(); - var systemB = World.CreateManager(); - World.DestroyManager(systemA); + var systemA = World.CreateSystem(); + var systemB = World.CreateSystem(); + World.DestroySystem(systemA); - Assert.AreEqual(systemB, World.GetExistingManager());; + Assert.AreEqual(systemB, World.GetExistingSystem());; } [Test] public void CreateTwoSystemsAfterDestroyReturnFirst() { - var systemA = World.CreateManager(); - var systemB = World.CreateManager(); - World.DestroyManager(systemB); + var systemA = World.CreateSystem(); + var systemB = World.CreateSystem(); + World.DestroySystem(systemB); - Assert.AreEqual(systemA, World.GetExistingManager());; + Assert.AreEqual(systemA, World.GetExistingSystem());; } [Test] - public void GetComponentGroup() + public void GetEntityQuery() { ComponentType[] ro_rw = { ComponentType.ReadOnly(), typeof(EcsTestData2) }; ComponentType[] rw_rw = { typeof(EcsTestData), typeof(EcsTestData2) }; ComponentType[] rw = { typeof(EcsTestData) }; - var ro_rw0_system = EmptySystem.GetComponentGroup(ro_rw); - var rw_rw_system = EmptySystem.GetComponentGroup(rw_rw); - var rw_system = EmptySystem.GetComponentGroup(rw); + var ro_rw0_system = EmptySystem.GetEntityQuery(ro_rw); + var rw_rw_system = EmptySystem.GetEntityQuery(rw_rw); + var rw_system = EmptySystem.GetEntityQuery(rw); - Assert.AreEqual(ro_rw0_system, EmptySystem.GetComponentGroup(ro_rw)); - Assert.AreEqual(rw_rw_system, EmptySystem.GetComponentGroup(rw_rw)); - Assert.AreEqual(rw_system, EmptySystem.GetComponentGroup(rw)); + Assert.AreEqual(ro_rw0_system, EmptySystem.GetEntityQuery(ro_rw)); + Assert.AreEqual(rw_rw_system, EmptySystem.GetEntityQuery(rw_rw)); + Assert.AreEqual(rw_system, EmptySystem.GetEntityQuery(rw)); - Assert.AreEqual(3, EmptySystem.ComponentGroups.Length); + Assert.AreEqual(3, EmptySystem.EntityQueries.Length); } [Test] public void GetComponentGroupArchetypeQuery() { var query1 = new ComponentType[] { typeof(EcsTestData) }; - var query2 = new EntityArchetypeQuery { All = new ComponentType[] {typeof(EcsTestData)} }; - var query3 = new EntityArchetypeQuery { All = new ComponentType[] {typeof(EcsTestData), typeof(EcsTestData2)} }; + var query2 = new EntityQueryDesc { All = new ComponentType[] {typeof(EcsTestData)} }; + var query3 = new EntityQueryDesc { All = new ComponentType[] {typeof(EcsTestData), typeof(EcsTestData2)} }; - var group1 = EmptySystem.GetComponentGroup(query1); - var group2 = EmptySystem.GetComponentGroup(query2); - var group3 = EmptySystem.GetComponentGroup(query3); + var group1 = EmptySystem.GetEntityQuery(query1); + var group2 = EmptySystem.GetEntityQuery(query2); + var group3 = EmptySystem.GetEntityQuery(query3); - Assert.AreEqual(group1, EmptySystem.GetComponentGroup(query1)); - Assert.AreEqual(group2, EmptySystem.GetComponentGroup(query2)); - Assert.AreEqual(group3, EmptySystem.GetComponentGroup(query3)); + Assert.AreEqual(group1, EmptySystem.GetEntityQuery(query1)); + Assert.AreEqual(group2, EmptySystem.GetEntityQuery(query2)); + Assert.AreEqual(group3, EmptySystem.GetEntityQuery(query3)); - Assert.AreEqual(2, EmptySystem.ComponentGroups.Length); + Assert.AreEqual(2, EmptySystem.EntityQueries.Length); } [Test] public void GetComponentGroupComponentTypeArchetypeQueryEquality() { var query1 = new ComponentType[] { typeof(EcsTestData) }; - var query2 = new EntityArchetypeQuery { All = new ComponentType[] {typeof(EcsTestData)} }; - var query3 = new EntityArchetypeQuery { All = new [] {ComponentType.ReadWrite()} }; + var query2 = new EntityQueryDesc { All = new ComponentType[] {typeof(EcsTestData)} }; + var query3 = new EntityQueryDesc { All = new [] {ComponentType.ReadWrite()} }; - var group1 = EmptySystem.GetComponentGroup(query1); - var group2 = EmptySystem.GetComponentGroup(query2); - var group3 = EmptySystem.GetComponentGroup(query3); + var group1 = EmptySystem.GetEntityQuery(query1); + var group2 = EmptySystem.GetEntityQuery(query2); + var group3 = EmptySystem.GetEntityQuery(query3); Assert.AreEqual(group1, group2); Assert.AreEqual(group2, group3); - Assert.AreEqual(1, EmptySystem.ComponentGroups.Length); + Assert.AreEqual(1, EmptySystem.EntityQueries.Length); } [Test] public void GetComponentGroupRespectsRWAccessInequality() { - var query1 = new EntityArchetypeQuery { All = new [] {ComponentType.ReadOnly(), ComponentType.ReadWrite()} }; - var query2 = new EntityArchetypeQuery { All = new [] {ComponentType.ReadOnly(), ComponentType.ReadOnly()} }; + var query1 = new EntityQueryDesc { All = new [] {ComponentType.ReadOnly(), ComponentType.ReadWrite()} }; + var query2 = new EntityQueryDesc { All = new [] {ComponentType.ReadOnly(), ComponentType.ReadOnly()} }; - var group1 = EmptySystem.GetComponentGroup(query1); - var group2 = EmptySystem.GetComponentGroup(query2); + var group1 = EmptySystem.GetEntityQuery(query1); + var group2 = EmptySystem.GetEntityQuery(query2); Assert.AreNotEqual(group1, group2); - Assert.AreEqual(2, EmptySystem.ComponentGroups.Length); + Assert.AreEqual(2, EmptySystem.EntityQueries.Length); } [Test] @@ -260,20 +260,20 @@ public void GetComponentGroupOrderIndependent() var query1 = new ComponentType[] { typeof(EcsTestData), typeof(EcsTestData2) }; var query2 = new ComponentType[] { typeof(EcsTestData2), typeof(EcsTestData) }; - var group1 = EmptySystem.GetComponentGroup(query1); - var group2 = EmptySystem.GetComponentGroup(query2); + var group1 = EmptySystem.GetEntityQuery(query1); + var group2 = EmptySystem.GetEntityQuery(query2); Assert.AreEqual(group1, group2); - Assert.AreEqual(1, EmptySystem.ComponentGroups.Length); + Assert.AreEqual(1, EmptySystem.EntityQueries.Length); - var query3 = new EntityArchetypeQuery { All = new ComponentType[] {typeof(EcsTestData2), typeof(EcsTestData3)} }; - var query4 = new EntityArchetypeQuery { All = new ComponentType[] {typeof(EcsTestData3), typeof(EcsTestData2)} }; + var query3 = new EntityQueryDesc { All = new ComponentType[] {typeof(EcsTestData2), typeof(EcsTestData3)} }; + var query4 = new EntityQueryDesc { All = new ComponentType[] {typeof(EcsTestData3), typeof(EcsTestData2)} }; - var group3 = EmptySystem.GetComponentGroup(query3); - var group4 = EmptySystem.GetComponentGroup(query4); + var group3 = EmptySystem.GetEntityQuery(query3); + var group4 = EmptySystem.GetEntityQuery(query4); Assert.AreEqual(group3, group4); - Assert.AreEqual(2, EmptySystem.ComponentGroups.Length); + Assert.AreEqual(2, EmptySystem.EntityQueries.Length); } //@TODO: Behaviour is a slightly dodgy... Should probably just ignore and return same as single typeof(EcsTestData) @@ -281,8 +281,8 @@ public void GetComponentGroupOrderIndependent() public void GetComponentGroupWithEntityThrows() { ComponentType[] e = { typeof(Entity), typeof(EcsTestData) }; - EmptySystem.GetComponentGroup(e); - Assert.Throws(() => EmptySystem.GetComponentGroup(e)); + EmptySystem.GetEntityQuery(e); + Assert.Throws(() => EmptySystem.GetEntityQuery(e)); } [Test] @@ -290,22 +290,22 @@ public void GetComponentGroupWithDuplicates() { // Currently duplicates will create two seperate groups doing the same thing... ComponentType[] dup_1 = { typeof(EcsTestData2) }; - ComponentType[] dup_2 = { typeof(EcsTestData2), typeof(EcsTestData2) }; + ComponentType[] dup_2 = { typeof(EcsTestData2), typeof(EcsTestData3) }; - var dup1_system = EmptySystem.GetComponentGroup(dup_1); - var dup2_system = EmptySystem.GetComponentGroup(dup_2); + var dup1_system = EmptySystem.GetEntityQuery(dup_1); + var dup2_system = EmptySystem.GetEntityQuery(dup_2); - Assert.AreEqual(dup1_system, EmptySystem.GetComponentGroup(dup_1)); - Assert.AreEqual(dup2_system, EmptySystem.GetComponentGroup(dup_2)); + Assert.AreEqual(dup1_system, EmptySystem.GetEntityQuery(dup_1)); + Assert.AreEqual(dup2_system, EmptySystem.GetEntityQuery(dup_2)); - Assert.AreEqual(2, EmptySystem.ComponentGroups.Length); + Assert.AreEqual(2, EmptySystem.EntityQueries.Length); } [Test] public void UpdateDestroyedSystemThrows() { var system = EmptySystem; - World.DestroyManager(system); + World.DestroySystem(system); Assert.Throws(system.Update); } } diff --git a/Unity.Entities.Tests/CreateAndDestroyTests.cs b/Unity.Entities.Tests/CreateAndDestroyTests.cs index 426daf0d..1d0cb764 100644 --- a/Unity.Entities.Tests/CreateAndDestroyTests.cs +++ b/Unity.Entities.Tests/CreateAndDestroyTests.cs @@ -1,3 +1,4 @@ +using System; using NUnit.Framework; using Unity.Collections; @@ -215,7 +216,7 @@ public void ReadOnlyAndNonReadOnlyArchetypeAreEqual() [Test] public void ExcludeArchetypeReactToAddRemoveComponent() { - var subtractiveArch = m_Manager.CreateComponentGroup(ComponentType.Exclude(typeof(EcsTestData)), typeof(EcsTestData2)); + var subtractiveArch = m_Manager.CreateEntityQuery(ComponentType.Exclude(typeof(EcsTestData)), typeof(EcsTestData2)); var archetype = m_Manager.CreateArchetype(typeof(EcsTestData), typeof(EcsTestData2)); @@ -345,7 +346,6 @@ public void AddComponentsWithTypeIndicesWorks() } [Test] - [StandaloneFixme] // ISharedComponentData public void AddComponentsWithSharedComponentsWorks() { var archetype = m_Manager.CreateArchetype(typeof(EcsTestData), typeof(EcsTestSharedComp)); @@ -415,7 +415,6 @@ public EcsSharedStateForcedOrder(int value) } [Test] - [StandaloneFixme] // ISharedComponentData public void InstantiateWithSharedSystemStateComponent() { var srcEntity = m_Manager.CreateEntity(); @@ -443,5 +442,80 @@ public void InstantiateWithSharedSystemStateComponent() Assert.AreNotEqual(versionSharedBefore, versionSharedAfter); Assert.AreEqual(versionSystemBefore, versionSystemAfter); } + + [Test] + public void AddTagComponentTwiceByValue() + { + var entity = m_Manager.CreateEntity(); + + m_Manager.AddComponentData(entity, new EcsTestTag()); + m_Manager.AddComponentData(entity, new EcsTestTag()); + } + + [Test] + public void AddTagComponentTwiceByType() + { + var entity = m_Manager.CreateEntity(); + + m_Manager.AddComponent(entity, ComponentType.ReadWrite()); + m_Manager.AddComponent(entity, ComponentType.ReadWrite()); + } + + [Test] + public void AddTagComponentTwiceToGroup() + { + m_Manager.CreateEntity(); + + m_Manager.AddComponent(m_Manager.UniversalQuery, ComponentType.ReadWrite()); + Assert.Throws(() => m_Manager.AddComponent(m_Manager.UniversalQuery, ComponentType.ReadWrite())); + + // Failure because the component type is expected to be explicitly excluded from the group. + } + + [Test] + public void AddTagComponentTwiceByTypeArray() + { + var entity = m_Manager.CreateEntity(); + + m_Manager.AddComponents(entity, new ComponentTypes(ComponentType.ReadWrite())); + m_Manager.AddComponents(entity, new ComponentTypes(ComponentType.ReadWrite())); + + m_Manager.AddComponents(entity, new ComponentTypes(ComponentType.ReadWrite())); + Assert.Throws(() => m_Manager.AddComponents(entity, new ComponentTypes(ComponentType.ReadWrite()))); + } + + [Test] + public void AddChunkComponentTwice() + { + var entity = m_Manager.CreateEntity(); + + m_Manager.AddChunkComponentData(entity); + m_Manager.AddChunkComponentData(entity); + + m_Manager.AddChunkComponentData(entity); + m_Manager.AddChunkComponentData(entity); + } + + [Test] + public void AddChunkComponentToGroupTwice() + { + m_Manager.CreateEntity(); + + m_Manager.AddChunkComponentData(m_Manager.UniversalQuery, new EcsTestTag()); + Assert.Throws(() => m_Manager.AddChunkComponentData(m_Manager.UniversalQuery, new EcsTestTag())); + + m_Manager.AddChunkComponentData(m_Manager.UniversalQuery, new EcsTestData{value = 123}); + Assert.Throws(() => m_Manager.AddChunkComponentData(m_Manager.UniversalQuery, new EcsTestData{value = 123})); + Assert.Throws(() => m_Manager.AddChunkComponentData(m_Manager.UniversalQuery, new EcsTestData{value = 456})); + } + + [Test] + public void AddSharedComponentTwice() + { + var entity = m_Manager.CreateEntity(); + + m_Manager.AddSharedComponentData(entity, new EcsTestSharedComp()); + Assert.Throws(() => m_Manager.AddSharedComponentData(entity, new EcsTestSharedComp())); + } } } diff --git a/Unity.Entities.Tests/DisableComponentTests.cs b/Unity.Entities.Tests/DisableComponentTests.cs index 6abb82d2..66122542 100644 --- a/Unity.Entities.Tests/DisableComponentTests.cs +++ b/Unity.Entities.Tests/DisableComponentTests.cs @@ -7,12 +7,12 @@ namespace Unity.Entities.Tests class DisableComponentTests : ECSTestsFixture { [Test] - public void DIS_DontFindDisabledInComponentGroup() + public void DIS_DontFindDisabledInEntityQuery() { var archetype0 = m_Manager.CreateArchetype(typeof(EcsTestData)); var archetype1 = m_Manager.CreateArchetype(typeof(EcsTestData), typeof(Disabled)); - var group = m_Manager.CreateComponentGroup(typeof(EcsTestData)); + var group = m_Manager.CreateEntityQuery(typeof(EcsTestData)); var entity0 = m_Manager.CreateEntity(archetype0); var entity1 = m_Manager.CreateEntity(archetype1); @@ -33,7 +33,7 @@ public void DIS_DontFindDisabledInChunkIterator() var entity0 = m_Manager.CreateEntity(archetype0); var entity1 = m_Manager.CreateEntity(archetype1); - var group = m_Manager.CreateComponentGroup(ComponentType.ReadWrite()); + var group = m_Manager.CreateEntityQuery(ComponentType.ReadWrite()); var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob); group.Dispose(); var count = ArchetypeChunkArray.CalculateEntityCount(chunks); @@ -46,12 +46,12 @@ public void DIS_DontFindDisabledInChunkIterator() } [Test] - public void DIS_FindDisabledIfRequestedInComponentGroup() + public void DIS_FindDisabledIfRequestedInEntityQuery() { var archetype0 = m_Manager.CreateArchetype(typeof(EcsTestData)); var archetype1 = m_Manager.CreateArchetype(typeof(EcsTestData), typeof(Disabled)); - var group = m_Manager.CreateComponentGroup(ComponentType.ReadWrite(), ComponentType.ReadWrite()); + var group = m_Manager.CreateEntityQuery(ComponentType.ReadWrite(), ComponentType.ReadWrite()); var entity0 = m_Manager.CreateEntity(archetype0); var entity1 = m_Manager.CreateEntity(archetype1); @@ -75,7 +75,7 @@ public void DIS_FindDisabledIfRequestedInChunkIterator() var entity1 = m_Manager.CreateEntity(archetype1); var entity2 = m_Manager.CreateEntity(archetype1); - var group = m_Manager.CreateComponentGroup(ComponentType.ReadWrite(), ComponentType.ReadWrite()); + var group = m_Manager.CreateEntityQuery(ComponentType.ReadWrite(), ComponentType.ReadWrite()); var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob); group.Dispose(); var count = ArchetypeChunkArray.CalculateEntityCount(chunks); @@ -115,15 +115,15 @@ public void PrefabAndDisabledQueryOptions() m_Manager.CreateEntity(typeof(EcsTestData), typeof(Disabled)); m_Manager.CreateEntity(typeof(EcsTestData), typeof(Disabled), typeof(Prefab)); - CheckPrefabAndDisabledQueryOptions(EntityArchetypeQueryOptions.Default, 0); - CheckPrefabAndDisabledQueryOptions(EntityArchetypeQueryOptions.IncludePrefab, 1); - CheckPrefabAndDisabledQueryOptions(EntityArchetypeQueryOptions.IncludeDisabled, 1); - CheckPrefabAndDisabledQueryOptions(EntityArchetypeQueryOptions.IncludeDisabled | EntityArchetypeQueryOptions.IncludePrefab, 3); + CheckPrefabAndDisabledQueryOptions(EntityQueryOptions.Default, 0); + CheckPrefabAndDisabledQueryOptions(EntityQueryOptions.IncludePrefab, 1); + CheckPrefabAndDisabledQueryOptions(EntityQueryOptions.IncludeDisabled, 1); + CheckPrefabAndDisabledQueryOptions(EntityQueryOptions.IncludeDisabled | EntityQueryOptions.IncludePrefab, 3); } - void CheckPrefabAndDisabledQueryOptions(EntityArchetypeQueryOptions options, int expected) + void CheckPrefabAndDisabledQueryOptions(EntityQueryOptions options, int expected) { - var group = m_Manager.CreateComponentGroup(new EntityArchetypeQuery { All = new[] {ComponentType.ReadWrite()}, Options = options }); + var group = m_Manager.CreateEntityQuery(new EntityQueryDesc { All = new[] {ComponentType.ReadWrite()}, Options = options }); Assert.AreEqual(expected, group.CalculateLength()); group.Dispose(); } diff --git a/Unity.Entities.Tests/ECSTestsFixture.cs b/Unity.Entities.Tests/ECSTestsFixture.cs index ea4ef8df..8f88c040 100644 --- a/Unity.Entities.Tests/ECSTestsFixture.cs +++ b/Unity.Entities.Tests/ECSTestsFixture.cs @@ -1,3 +1,4 @@ +using System.Linq; using NUnit.Framework; using Unity.Collections; using Unity.Jobs; @@ -6,25 +7,25 @@ namespace Unity.Entities.Tests { [DisableAutoCreation] -#if UNITY_CSHARP_TINY +#if NET_DOTS public class EmptySystem : ComponentSystem { protected override void OnUpdate() { } - public new ComponentGroup GetComponentGroup(params EntityArchetypeQuery[] queries) + public new EntityQuery GetEntityQuery(params EntityQueryDesc[] queriesDesc) { - return base.GetComponentGroup(queries); + return base.GetEntityQuery(queriesDesc); } - public new ComponentGroup GetComponentGroup(params ComponentType[] componentTypes) + public new EntityQuery GetEntityQuery(params ComponentType[] componentTypes) { - return base.GetComponentGroup(componentTypes); + return base.GetEntityQuery(componentTypes); } - public new ComponentGroup GetComponentGroup(NativeArray componentTypes) + public new EntityQuery GetEntityQuery(NativeArray componentTypes) { - return base.GetComponentGroup(componentTypes); + return base.GetEntityQuery(componentTypes); } public BufferFromEntity GetBufferFromEntity(bool isReadOnly = false) where T : struct, IBufferElementData { @@ -38,27 +39,19 @@ public class EmptySystem : JobComponentSystem protected override JobHandle OnUpdate(JobHandle dep) { return dep; } - new public ComponentGroup GetComponentGroup(params EntityArchetypeQuery[] queries) + new public EntityQuery GetEntityQuery(params EntityQueryDesc[] queriesDesc) { - return base.GetComponentGroup(queries); + return base.GetEntityQuery(queriesDesc); } - new public ComponentGroup GetComponentGroup(params ComponentType[] componentTypes) + new public EntityQuery GetEntityQuery(params ComponentType[] componentTypes) { - return base.GetComponentGroup(componentTypes); + return base.GetEntityQuery(componentTypes); } - new public ComponentGroup GetComponentGroup(NativeArray componentTypes) + new public EntityQuery GetEntityQuery(NativeArray componentTypes) { - return base.GetComponentGroup(componentTypes); + return base.GetEntityQuery(componentTypes); } -#if !UNITY_ZEROPLAYER - #pragma warning disable 618 - new public ComponentGroupArray GetEntities() where T : struct - { - return base.GetEntities(); - } - #pragma warning restore 618 -#endif } #endif public class ECSTestsFixture @@ -80,7 +73,7 @@ public virtual void Setup() World = DefaultTinyWorldInitialization.Initialize("Test World"); #endif - m_Manager = World.GetOrCreateManager(); + m_Manager = World.EntityManager; m_ManagerDebug = new EntityManager.EntityManagerDebug(m_Manager); #if !UNITY_ZEROPLAYER @@ -100,11 +93,9 @@ public virtual void TearDown() { // Clean up systems before calling CheckInternalConsistency because we might have filters etc // holding on SharedComponentData making checks fail - var system = World.GetExistingManager(); - while (system != null) + while (World.Systems.Any()) { - World.DestroyManager(system); - system = World.GetExistingManager(); + World.DestroySystem(World.Systems.First()); } m_ManagerDebug.CheckInternalConsistency(); @@ -166,7 +157,7 @@ public EmptySystem EmptySystem { get { - return World.Active.GetOrCreateManager(); + return World.Active.GetOrCreateSystem(); } } } diff --git a/Unity.Entities.Tests/EntityArchetypeQueryTests.cs b/Unity.Entities.Tests/EntityArchetypeQueryTests.cs new file mode 100644 index 00000000..cb2bedcf --- /dev/null +++ b/Unity.Entities.Tests/EntityArchetypeQueryTests.cs @@ -0,0 +1,112 @@ +using System; +using NUnit.Framework; + +namespace Unity.Entities.Tests +{ + class EntityArchetypeQueryTests : ECSTestsFixture + { + [Test] + public void EntityQueryFilter_IdenticalIds_InDifferentFilters_Throws() + { + var query = new EntityQueryDesc + { + All = new ComponentType[] {typeof(EcsTestData)}, + None = new ComponentType[] {typeof(EcsTestData)} + }; + + Assert.Throws(() => + { + query.Validate(); + }); + } + + [Test] + public void EntityQueryFilter_IdenticalIds_InSameFilter_Throws() + { + var query = new EntityQueryDesc + { + All = new ComponentType[] {typeof(EcsTestData), typeof(EcsTestData)} + }; + + Assert.Throws(() => + { + query.Validate(); + }); + } + + [Test] + public void EntityQueryFilter_MultipleIdenticalIds_Throws() + { + var query = new EntityQueryDesc + { + All = new ComponentType[] {typeof(EcsTestData), typeof(EcsTestData2)}, + None = new ComponentType[] {typeof(EcsTestData3), typeof(EcsTestData)}, + Any = new ComponentType[] {typeof(EcsTestData), typeof(EcsTestData4)}, + }; + + Assert.Throws(() => + { + query.Validate(); + }); + } + + [Test] + public void EntityQueryFilter_SeparatedIds() + { + var query = new EntityQueryDesc + { + All = new ComponentType[] {typeof(EcsTestData), typeof(EcsTestData2)}, + None = new ComponentType[] {typeof(EcsTestData3), typeof(EcsTestData4)}, + Any = new ComponentType[] {typeof(EcsTestData5)}, + }; + + Assert.DoesNotThrow(() => + { + query.Validate(); + }); + } + + [Test] + public void EntityQueryFilter_CannotContainExcludeComponentType_All_Throws() + { + var query = new EntityQueryDesc + { + All = new ComponentType[] {typeof(EcsTestData), ComponentType.Exclude() }, + }; + + Assert.Throws(() => + { + query.Validate(); + }); + } + + [Test] + public void EntityQueryFilterCannotContainExcludeComponentType_Any_Throws() + { + var query = new EntityQueryDesc + { + Any = new ComponentType[] {typeof(EcsTestData), ComponentType.Exclude() }, + }; + + Assert.Throws(() => + { + query.Validate(); + }); + } + + [Test] + public void EntityQueryFilterCannotContainExcludeComponentType_None_Throws() + { + var query = new EntityQueryDesc + { + All = new ComponentType[] {typeof(EcsTestData) }, + None = new ComponentType[] {typeof(EcsTestData3), ComponentType.Exclude() }, + }; + + Assert.Throws(() => + { + query.Validate(); + }); + } + } +} diff --git a/Unity.Entities.Tests/EntityArchetypeQueryTests.cs.meta b/Unity.Entities.Tests/EntityArchetypeQueryTests.cs.meta new file mode 100644 index 00000000..9594c267 --- /dev/null +++ b/Unity.Entities.Tests/EntityArchetypeQueryTests.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 64c6e77bc4a4d6346b0b1a89c7fa09ff +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/Unity.Entities.Tests/EntityCommandBufferTests.cs b/Unity.Entities.Tests/EntityCommandBufferTests.cs index ea5cbcec..5b3d562e 100644 --- a/Unity.Entities.Tests/EntityCommandBufferTests.cs +++ b/Unity.Entities.Tests/EntityCommandBufferTests.cs @@ -47,7 +47,7 @@ public void SingleWriterEnforced() cmds.Playback(m_Manager); cmds.Dispose(); - var group = m_Manager.CreateComponentGroup(typeof(EcsTestData)); + var group = m_Manager.CreateEntityQuery(typeof(EcsTestData)); var arr = group.ToComponentDataArray(Allocator.TempJob); Assert.AreEqual(2, arr.Length); Assert.AreEqual(42, arr[0].value); @@ -153,7 +153,7 @@ public void CreateEntity() cmds.Playback(m_Manager); cmds.Dispose(); - var group = m_Manager.CreateComponentGroup(typeof(EcsTestData)); + var group = m_Manager.CreateEntityQuery(typeof(EcsTestData)); var arr = group.ToComponentDataArray(Allocator.TempJob); Assert.AreEqual(1, arr.Length); Assert.AreEqual(12, arr[0].value); @@ -172,7 +172,7 @@ public void CreateEntityWithArchetype() cmds.Playback(m_Manager); cmds.Dispose(); - var group = m_Manager.CreateComponentGroup(typeof(EcsTestData)); + var group = m_Manager.CreateEntityQuery(typeof(EcsTestData)); var arr = group.ToComponentDataArray(Allocator.TempJob); Assert.AreEqual(1, arr.Length); Assert.AreEqual(12, arr[0].value); @@ -191,7 +191,7 @@ public void CreateTwoComponents() cmds.Dispose(); { - var group = m_Manager.CreateComponentGroup(typeof(EcsTestData)); + var group = m_Manager.CreateEntityQuery(typeof(EcsTestData)); var arr = group.ToComponentDataArray(Allocator.TempJob); Assert.AreEqual(1, arr.Length); Assert.AreEqual(12, arr[0].value); @@ -200,7 +200,7 @@ public void CreateTwoComponents() } { - var group = m_Manager.CreateComponentGroup(typeof(EcsTestData2)); + var group = m_Manager.CreateEntityQuery(typeof(EcsTestData2)); var arr = group.ToComponentDataArray(Allocator.TempJob); Assert.AreEqual(1, arr.Length); Assert.AreEqual(1, arr[0].value0); @@ -229,7 +229,7 @@ public void TestMultiChunks() cmds.Dispose(); { - var group = m_Manager.CreateComponentGroup(typeof(EcsTestData), typeof(EcsTestData2)); + var group = m_Manager.CreateEntityQuery(typeof(EcsTestData), typeof(EcsTestData2)); var arr = group.ToComponentDataArray(Allocator.TempJob); var arr2 = group.ToComponentDataArray(Allocator.TempJob); Assert.AreEqual(count, arr.Length); @@ -246,7 +246,6 @@ public void TestMultiChunks() } [Test] - [StandaloneFixme] // ISharedComponentData public void AddSharedComponent() { var cmds = new EntityCommandBuffer(Allocator.TempJob); @@ -264,7 +263,6 @@ public void AddSharedComponent() } [Test] - [StandaloneFixme] // ISharedComponentData public void AddSharedComponentDefault() { var cmds = new EntityCommandBuffer(Allocator.TempJob); @@ -292,7 +290,6 @@ public void AddSharedComponentDefault() } [Test] - [StandaloneFixme] // ISharedComponentData public void SetSharedComponent() { var cmds = new EntityCommandBuffer(Allocator.TempJob); @@ -313,7 +310,6 @@ public void SetSharedComponent() } [Test] - [StandaloneFixme] // ISharedComponentData public void SetSharedComponentDefault() { var cmds = new EntityCommandBuffer(Allocator.TempJob); @@ -334,7 +330,6 @@ public void SetSharedComponentDefault() } [Test] - [StandaloneFixme] // ISharedComponentData public void RemoveSharedComponent() { var cmds = new EntityCommandBuffer(Allocator.TempJob); @@ -700,7 +695,6 @@ public void ConcurrentRecordInstantiate() } [Test] - [StandaloneFixme] // // Real problem: Atomic Safety public void PlaybackInvalidatesBuffers() { var cmds = new EntityCommandBuffer(Allocator.TempJob); @@ -718,7 +712,6 @@ public void PlaybackInvalidatesBuffers() } [Test] - [StandaloneFixme] // Real problem: Atomic Safety public void ArrayAliasesOfPendingBuffersAreInvalidateOnResize() { var cmds = new EntityCommandBuffer(Allocator.TempJob); @@ -925,10 +918,9 @@ public void BufferCopyFromDoesNotThrowInJob() #if ENABLE_UNITY_COLLECTIONS_CHECKS [Test] - [StandaloneFixme] public void EntityCommandBufferSystemPlaybackExceptionIsolation() { - var entityCommandBufferSystem = World.GetOrCreateManager(); + var entityCommandBufferSystem = World.GetOrCreateSystem(); var buf1 = entityCommandBufferSystem.CreateCommandBuffer(); var buf2 = entityCommandBufferSystem.CreateCommandBuffer(); @@ -944,13 +936,13 @@ public void EntityCommandBufferSystemPlaybackExceptionIsolation() // We exp both command buffers to execute, and an exception thrown afterwards // Essentially we want isolation of two systems that might fail independently. Assert.Throws(() => { entityCommandBufferSystem.Update(); }); - Assert.AreEqual(2, EmptySystem.GetComponentGroup(typeof(EcsTestData)).CalculateLength()); + Assert.AreEqual(2, EmptySystem.GetEntityQuery(typeof(EcsTestData)).CalculateLength()); // On second run, we expect all buffers to be removed... // So no more exceptions thrown. entityCommandBufferSystem.Update(); - Assert.AreEqual(2, EmptySystem.GetComponentGroup(typeof(EcsTestData)).CalculateLength()); + Assert.AreEqual(2, EmptySystem.GetEntityQuery(typeof(EcsTestData)).CalculateLength()); } #endif @@ -959,7 +951,7 @@ public void EntityCommandBufferSystemPlaybackExceptionIsolation() [StandaloneFixme] // IJob public void EntityCommandBufferSystem_OmitAddJobHandleForProducer_ThrowArgumentException() { - var barrier = World.GetOrCreateManager(); + var barrier = World.GetOrCreateSystem(); var cmds = barrier.CreateCommandBuffer(); const int kCreateCount = 10000; var job = new TestParallelJob @@ -974,7 +966,6 @@ public void EntityCommandBufferSystem_OmitAddJobHandleForProducer_ThrowArgumentE #endif [Test] - [StandaloneFixme] // ISharedComponentData public void AddSharedComponent_WhenComponentHasEntityField_ThrowsArgumentException() { var cmds = new EntityCommandBuffer(Allocator.TempJob); @@ -1000,7 +991,7 @@ public void AddComponent_WhenDataContainsDeferredEntity_DeferredEntityIsResolved cmds.Playback(m_Manager); cmds.Dispose(); - var group = m_Manager.CreateComponentGroup(typeof(EcsTestDataEntity)); + var group = m_Manager.CreateEntityQuery(typeof(EcsTestDataEntity)); var arr = group.ToComponentDataArray(Allocator.TempJob); Assert.AreEqual(1, arr.Length); @@ -1052,6 +1043,63 @@ public void InstantiateEntity_BatchMode_DisabledIfEntityDirty() Assert.AreEqual(12, m_Manager.GetComponentData(realDst1).value1); } + [Test] + public void UninitializedEntityCommandBufferThrows() + { + EntityCommandBuffer cmds = new EntityCommandBuffer(); + var exception = Assert.Throws(() => cmds.CreateEntity()); + Assert.AreEqual(exception.Message, "The EntityCommandBuffer has not been initialized!"); + } + + [Test] + public void UninitializedConcurrentEntityCommandBufferThrows() + { + EntityCommandBuffer.Concurrent cmds = new EntityCommandBuffer.Concurrent(); + var exception = Assert.Throws(() => cmds.CreateEntity(0)); + Assert.AreEqual(exception.Message, "The EntityCommandBuffer has not been initialized!"); + } + + [Test] + public void AddOrSetBufferWithEntity_NeedsFixup_Works([Values(true,false)] bool setBuffer) + { + EntityCommandBuffer cmds = new EntityCommandBuffer(Allocator.TempJob); + + Entity e0 = m_Manager.CreateEntity(); + Entity e1 = m_Manager.CreateEntity(); + Entity e2 = m_Manager.CreateEntity(); + + if (setBuffer) + m_Manager.AddComponent(e1, typeof(EcsComplexEntityRefElement)); + + { + var deferred0 = cmds.CreateEntity(); + var deferred1 = cmds.CreateEntity(); + var deferred2 = cmds.CreateEntity(); + + cmds.AddComponent(e0, new EcsTestDataEntity() { value1 = deferred0 }); + cmds.AddComponent(e1, new EcsTestDataEntity() { value1 = deferred1 }); + cmds.AddComponent(e2, new EcsTestDataEntity() { value1 = deferred2 }); + + var buf = setBuffer ? cmds.SetBuffer(e1) : cmds.AddBuffer(e1); + buf.Add(new EcsComplexEntityRefElement() {Entity = e0}); + buf.Add(new EcsComplexEntityRefElement() {Entity = deferred1}); + buf.Add(new EcsComplexEntityRefElement() {Entity = deferred2}); + buf.Add(new EcsComplexEntityRefElement() {Entity = deferred0}); + cmds.Playback(m_Manager); + cmds.Dispose(); + } + { + var outbuf = m_Manager.GetBuffer(e1); + Assert.AreEqual(4, outbuf.Length); + var expect0 = m_Manager.GetComponentData(e0).value1; + var expect1 = m_Manager.GetComponentData(e1).value1; + var expect2 = m_Manager.GetComponentData(e2).value1; + Assert.AreEqual(e0, outbuf[0].Entity); + Assert.AreEqual(expect1, outbuf[1].Entity); + Assert.AreEqual(expect2, outbuf[2].Entity); + Assert.AreEqual(expect0, outbuf[3].Entity); + } + } } } diff --git a/Unity.Entities.Tests/EntityManagerBugTests.cs b/Unity.Entities.Tests/EntityManagerBugTests.cs index 70c255d0..9720128e 100644 --- a/Unity.Entities.Tests/EntityManagerBugTests.cs +++ b/Unity.Entities.Tests/EntityManagerBugTests.cs @@ -34,6 +34,8 @@ class EntityBag [Test] public void TestIssue149() { + m_OurTypes = new ComponentType[] {typeof(Issue149Data)}; + m_Archetype = m_Manager.CreateArchetype(typeof(Issue149Data)); for (int i = 0; i < Bags.Length; ++i) @@ -91,19 +93,17 @@ void RecycleEntities(EntityBag bag) } } - private static readonly ComponentType[] s_OurTypes = new ComponentType[] { - typeof(Issue149Data) - }; + private ComponentType[] m_OurTypes; // Walk all accessible entity data and check that the versions match what we // believe the generation numbers should be. private void SanityCheckVersions() { - var group = m_Manager.CreateComponentGroup(new EntityArchetypeQuery + var group = m_Manager.CreateEntityQuery(new EntityQueryDesc { Any = Array.Empty(), None = Array.Empty(), - All = s_OurTypes, + All = m_OurTypes, }); var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob); group.Dispose(); @@ -138,7 +138,7 @@ class Bug476 : ECSTestsFixture public void EntityArchetypeQueryMembersHaveSensibleDefaults() { ComponentType[] types = {typeof(Issue476Data)}; - var group = m_Manager.CreateComponentGroup(types); + var group = m_Manager.CreateEntityQuery(types); var temp = group.CreateArchetypeChunkArray(Allocator.TempJob); group.Dispose(); temp.Dispose(); @@ -156,7 +156,7 @@ public void Test1() World w = DefaultTinyWorldInitialization.Initialize("TestWorld"); #endif World.Active = w; - EntityManager em = World.Active.GetOrCreateManager(); + EntityManager em = World.Active.EntityManager; List remember = new List(); for (int i = 0; i < 5; i++) { @@ -185,7 +185,7 @@ public void Test2() World w = DefaultTinyWorldInitialization.Initialize("TestWorld"); #endif World.Active = w; - EntityManager em = World.Active.GetOrCreateManager(); + EntityManager em = World.Active.EntityManager; List remember = new List(); for (int i = 0; i < 5; i++) @@ -202,7 +202,7 @@ public void Test2() w = DefaultTinyWorldInitialization.Initialize("TestWorld"); #endif World.Active = w; - em = World.Active.GetOrCreateManager(); + em = World.Active.EntityManager; var allEnt = em.GetAllEntities(Allocator.Temp); Assert.AreEqual(0, allEnt.Length); allEnt.Dispose(); diff --git a/Unity.Entities.Tests/EntityManagerComponentGroupOperationsTests.cs b/Unity.Entities.Tests/EntityManagerComponentGroupOperationsTests.cs index 004a2bed..91af710e 100644 --- a/Unity.Entities.Tests/EntityManagerComponentGroupOperationsTests.cs +++ b/Unity.Entities.Tests/EntityManagerComponentGroupOperationsTests.cs @@ -8,13 +8,13 @@ class EntityManagerComponentGroupOperationsTests : ECSTestsFixture [Test] public void AddRemoveChunkComponentWithGroupWorks() { - var metaChunkGroup = m_Manager.CreateComponentGroup(typeof(ChunkHeader)); + var metaChunkGroup = m_Manager.CreateEntityQuery(typeof(ChunkHeader)); var entity1 = m_Manager.CreateEntity(typeof(EcsTestData)); var entity2 = m_Manager.CreateEntity(typeof(EcsTestData), typeof(EcsTestData2)); var entity3 = m_Manager.CreateEntity(typeof(EcsTestData2)); - var group1 = m_Manager.CreateComponentGroup(ComponentType.ReadWrite()); + var group1 = m_Manager.CreateEntityQuery(ComponentType.ReadWrite()); m_Manager.AddChunkComponentData(group1, new EcsTestData3(7)); @@ -32,7 +32,7 @@ public void AddRemoveChunkComponentWithGroupWorks() m_ManagerDebug.CheckInternalConsistency(); - var group2 = m_Manager.CreateComponentGroup(ComponentType.ReadWrite(), ComponentType.ChunkComponent()); + var group2 = m_Manager.CreateEntityQuery(ComponentType.ReadWrite(), ComponentType.ChunkComponent()); m_Manager.RemoveChunkComponentData(group2); @@ -49,7 +49,7 @@ public void AddRemoveSharedComponentWithGroupWorks() var entity2 = m_Manager.CreateEntity(typeof(EcsTestData), typeof(EcsTestData2)); var entity3 = m_Manager.CreateEntity(typeof(EcsTestData2)); - var group1 = m_Manager.CreateComponentGroup(ComponentType.ReadWrite()); + var group1 = m_Manager.CreateEntityQuery(ComponentType.ReadWrite()); m_Manager.AddSharedComponentData(group1, new EcsTestSharedComp(7)); @@ -65,90 +65,106 @@ public void AddRemoveSharedComponentWithGroupWorks() m_ManagerDebug.CheckInternalConsistency(); - var group2 = m_Manager.CreateComponentGroup(ComponentType.ReadWrite(), ComponentType.ReadWrite()); + var group2 = m_Manager.CreateEntityQuery(ComponentType.ReadWrite(), ComponentType.ReadWrite()); m_Manager.RemoveComponent(group2, typeof(EcsTestSharedComp)); Assert.IsFalse(m_Manager.HasComponent(entity2, typeof(EcsTestSharedComp))); } - public static ComponentType[] GetTestTypes() - { - TypeManager.Initialize(); - return new ComponentType[] { typeof(EcsTestTag), typeof(EcsTestData4), ComponentType.ChunkComponent(), typeof(EcsTestSharedComp) }; - } - [Test] [StandaloneFixme] // ISharedComponentData - public void AddRemoveAnyComponentWithGroupWorksWith([ValueSource(nameof(GetTestTypes))] ComponentType type) + public void AddRemoveAnyComponentWithGroupWorksWithVariousTypes() { - var metaChunkGroup = m_Manager.CreateComponentGroup(typeof(ChunkHeader)); + TypeManager.Initialize(); + var componentTypes = new ComponentType[] { typeof(EcsTestTag), typeof(EcsTestData4), ComponentType.ChunkComponent(), typeof(EcsTestSharedComp) }; - var entity1 = m_Manager.CreateEntity(typeof(EcsTestData)); - var entity2 = m_Manager.CreateEntity(typeof(EcsTestData), typeof(EcsTestData2)); - var entity3 = m_Manager.CreateEntity(typeof(EcsTestData2)); + foreach (var type in componentTypes) + { + // We want a clean slate for the m_manager so teardown and setup before the test + TearDown(); + Setup(); - var group1 = m_Manager.CreateComponentGroup(ComponentType.ReadWrite()); + var metaChunkGroup = m_Manager.CreateEntityQuery(typeof(ChunkHeader)); - m_Manager.AddComponent(group1, type); + var entity1 = m_Manager.CreateEntity(typeof(EcsTestData)); + var entity2 = m_Manager.CreateEntity(typeof(EcsTestData), typeof(EcsTestData2)); + var entity3 = m_Manager.CreateEntity(typeof(EcsTestData2)); - Assert.IsTrue(m_Manager.HasComponent(entity1, type)); - Assert.IsTrue(m_Manager.HasComponent(entity2, type)); - Assert.IsFalse(m_Manager.HasComponent(entity3, type)); + var group1 = m_Manager.CreateEntityQuery(ComponentType.ReadWrite()); - if(type.IsChunkComponent) - Assert.AreEqual(2, metaChunkGroup.CalculateLength()); + m_Manager.AddComponent(group1, type); - if (type == ComponentType.ReadWrite()) - { - m_Manager.SetSharedComponentData(entity1, new EcsTestSharedComp(1)); - m_Manager.SetSharedComponentData(entity2, new EcsTestSharedComp(2)); - } + Assert.IsTrue(m_Manager.HasComponent(entity1, type)); + Assert.IsTrue(m_Manager.HasComponent(entity2, type)); + Assert.IsFalse(m_Manager.HasComponent(entity3, type)); - m_ManagerDebug.CheckInternalConsistency(); + if (type.IsChunkComponent) + Assert.AreEqual(2, metaChunkGroup.CalculateLength()); - var group2 = m_Manager.CreateComponentGroup(ComponentType.ReadWrite(), type); + if (type == ComponentType.ReadWrite()) + { + m_Manager.SetSharedComponentData(entity1, new EcsTestSharedComp(1)); + m_Manager.SetSharedComponentData(entity2, new EcsTestSharedComp(2)); + } - m_Manager.RemoveComponent(group2, type); + m_ManagerDebug.CheckInternalConsistency(); - Assert.IsFalse(m_Manager.HasComponent(entity2, ComponentType.ChunkComponent())); + var group2 = m_Manager.CreateEntityQuery(ComponentType.ReadWrite(), type); + + m_Manager.RemoveComponent(group2, type); - if(type.IsChunkComponent) - Assert.AreEqual(1, metaChunkGroup.CalculateLength()); + Assert.IsFalse(m_Manager.HasComponent(entity2, ComponentType.ChunkComponent())); + + if (type.IsChunkComponent) + Assert.AreEqual(1, metaChunkGroup.CalculateLength()); + } + TypeManager.Shutdown(); } [Test] [StandaloneFixme] // ISharedComponentData - public void RemoveAnyComponentWithGroupIgnoresChunksThatDontHaveTheComponent([ValueSource(nameof(GetTestTypes))] ComponentType type) + public void RemoveAnyComponentWithGroupIgnoresChunksThatDontHaveTheComponent() { - var metaChunkGroup = m_Manager.CreateComponentGroup(typeof(ChunkHeader)); + TypeManager.Initialize(); + var componentTypes = new ComponentType[] { typeof(EcsTestTag), typeof(EcsTestData4), ComponentType.ChunkComponent(), typeof(EcsTestSharedComp) }; - var entity1 = m_Manager.CreateEntity(typeof(EcsTestData)); - var entity2 = m_Manager.CreateEntity(typeof(EcsTestData), typeof(EcsTestData2)); - var entity3 = m_Manager.CreateEntity(typeof(EcsTestData2)); + foreach (var type in componentTypes) + { + // We want a clean slate for the m_manager so teardown and setup before the test + TearDown(); + Setup(); - var group1 = m_Manager.CreateComponentGroup(ComponentType.ReadWrite()); + var metaChunkGroup = m_Manager.CreateEntityQuery(typeof(ChunkHeader)); - m_Manager.AddComponent(group1, type); + var entity1 = m_Manager.CreateEntity(typeof(EcsTestData)); + var entity2 = m_Manager.CreateEntity(typeof(EcsTestData), typeof(EcsTestData2)); + var entity3 = m_Manager.CreateEntity(typeof(EcsTestData2)); - Assert.IsTrue(m_Manager.HasComponent(entity1, type)); - Assert.IsTrue(m_Manager.HasComponent(entity2, type)); - Assert.IsFalse(m_Manager.HasComponent(entity3, type)); + var group1 = m_Manager.CreateEntityQuery(ComponentType.ReadWrite()); - if(type.IsChunkComponent) - Assert.AreEqual(2, metaChunkGroup.CalculateLength()); + m_Manager.AddComponent(group1, type); - if (type == ComponentType.ReadWrite()) - { - m_Manager.SetSharedComponentData(entity1, new EcsTestSharedComp(1)); - m_Manager.SetSharedComponentData(entity2, new EcsTestSharedComp(2)); - } + Assert.IsTrue(m_Manager.HasComponent(entity1, type)); + Assert.IsTrue(m_Manager.HasComponent(entity2, type)); + Assert.IsFalse(m_Manager.HasComponent(entity3, type)); - m_ManagerDebug.CheckInternalConsistency(); + if (type.IsChunkComponent) + Assert.AreEqual(2, metaChunkGroup.CalculateLength()); + + if (type == ComponentType.ReadWrite()) + { + m_Manager.SetSharedComponentData(entity1, new EcsTestSharedComp(1)); + m_Manager.SetSharedComponentData(entity2, new EcsTestSharedComp(2)); + } - m_Manager.RemoveComponent(m_Manager.UniversalGroup, type); + m_ManagerDebug.CheckInternalConsistency(); - Assert.AreEqual(0, m_Manager.CreateComponentGroup(type).CalculateLength()); + m_Manager.RemoveComponent(m_Manager.UniversalQuery, type); + + Assert.AreEqual(0, m_Manager.CreateEntityQuery(type).CalculateLength()); + } + TypeManager.Shutdown(); } uint GetComponentDataVersion(Entity e) where T : struct, IComponentData @@ -162,7 +178,7 @@ uint GetSharedComponentDataVersion(Entity e) where T : struct, ISharedCompone } [Test] - [StandaloneFixme] // ISharedComponentData + [StandaloneFixme] // Don't know why fails? public void AddRemoveComponentWithGroupPreservesChangeVersions() { m_ManagerDebug.SetGlobalSystemVersion(10); @@ -182,11 +198,11 @@ public void AddRemoveComponentWithGroupPreservesChangeVersions() m_ManagerDebug.SetGlobalSystemVersion(30); - m_Manager.AddSharedComponentData(m_Manager.UniversalGroup, new EcsTestSharedComp(1)); + m_Manager.AddSharedComponentData(m_Manager.UniversalQuery, new EcsTestSharedComp(1)); m_ManagerDebug.SetGlobalSystemVersion(40); - m_Manager.AddComponent(m_Manager.UniversalGroup, typeof(EcsTestTag)); + m_Manager.AddComponent(m_Manager.UniversalQuery, typeof(EcsTestTag)); Assert.AreEqual(30, GetSharedComponentDataVersion(entity1)); Assert.AreEqual(30, GetSharedComponentDataVersion(entity2)); @@ -198,7 +214,7 @@ public void AddRemoveComponentWithGroupPreservesChangeVersions() m_ManagerDebug.SetGlobalSystemVersion(50); - m_Manager.RemoveComponent(m_Manager.UniversalGroup, typeof(EcsTestSharedComp2)); + m_Manager.RemoveComponent(m_Manager.UniversalQuery, typeof(EcsTestSharedComp2)); Assert.AreEqual(30, GetSharedComponentDataVersion(entity1)); Assert.AreEqual(30, GetSharedComponentDataVersion(entity2)); @@ -206,7 +222,7 @@ public void AddRemoveComponentWithGroupPreservesChangeVersions() m_ManagerDebug.SetGlobalSystemVersion(60); - m_Manager.RemoveComponent(m_Manager.UniversalGroup, typeof(EcsTestSharedComp)); + m_Manager.RemoveComponent(m_Manager.UniversalQuery, typeof(EcsTestSharedComp)); Assert.AreEqual(10, GetComponentDataVersion(entity1)); Assert.AreEqual(10, GetComponentDataVersion(entity2)); diff --git a/Unity.Entities.Tests/EntityManagerPrefabTests.cs b/Unity.Entities.Tests/EntityManagerPrefabTests.cs index 535d3002..a0150152 100644 --- a/Unity.Entities.Tests/EntityManagerPrefabTests.cs +++ b/Unity.Entities.Tests/EntityManagerPrefabTests.cs @@ -70,7 +70,7 @@ public void InstantiateLinkedGroup() CheckLinkedGroup(clone, srcLinked, external); // Make sure that instantiated objects are found by component group (They are no longer prefabs) - Assert.AreEqual(4, m_Manager.CreateComponentGroup(typeof(EcsTestDataEntity)).CalculateLength()); + Assert.AreEqual(4, m_Manager.CreateEntityQuery(typeof(EcsTestDataEntity)).CalculateLength()); srcLinked.Dispose(); } @@ -91,7 +91,7 @@ public void InstantiateLinkedGroupStressTest([Values(1, 1023)]int count) } // Make sure that instantiated objects are found by component group (They are no longer prefabs) - Assert.AreEqual(3 * 4 * count, m_Manager.CreateComponentGroup(typeof(EcsTestDataEntity)).CalculateLength()); + Assert.AreEqual(3 * 4 * count, m_Manager.CreateEntityQuery(typeof(EcsTestDataEntity)).CalculateLength()); clones.Dispose(); srcLinked.Dispose(); diff --git a/Unity.Entities.Tests/EntityManagerTests.cs b/Unity.Entities.Tests/EntityManagerTests.cs index 27876954..43177a79 100644 --- a/Unity.Entities.Tests/EntityManagerTests.cs +++ b/Unity.Entities.Tests/EntityManagerTests.cs @@ -138,7 +138,7 @@ public unsafe void ComponentsWithBool() var hash = new NativeHashMap(count, Allocator.Temp); - var cg = m_Manager.CreateComponentGroup(ComponentType.ReadWrite()); + var cg = m_Manager.CreateEntityQuery(ComponentType.ReadWrite()); using (var chunks = cg.CreateArchetypeChunkArray(Allocator.TempJob)) { var boolsType = m_Manager.GetArchetypeChunkComponentType(false); @@ -166,5 +166,56 @@ public unsafe void ComponentsWithBool() array.Dispose(); hash.Dispose(); } + + + struct BigComponentWithAlign1A : IComponentData + { + NativeString4096 bs; + unsafe fixed byte val[3]; + } + + struct ComponentWithAlign8 : IComponentData { + double val; + } + + struct BigComponentWithAlign1B : IComponentData { + NativeString4096 bs; + unsafe fixed byte val[3]; + } + + [Test] + public unsafe void ChunkComponentRunIsAligned() + { + // We need to make sure that the stricter-alignment component (WithAlign8) comes after the simpler one + Type oneByteAlignmentType = typeof(BigComponentWithAlign1A); + int bigTypeIndex = TypeManager.GetTypeIndex(); + if ((TypeManager.GetTypeIndex() & TypeManager.ClearFlagsMask) > + (bigTypeIndex & TypeManager.ClearFlagsMask)) + { + // must be the other one + oneByteAlignmentType = typeof(BigComponentWithAlign1B); + } + + var oneByteInfo = TypeManager.GetTypeInfo(TypeManager.GetTypeIndex(oneByteAlignmentType)); + int bigAlignment = TypeManager.GetTypeInfo().AlignmentInBytes; + + //Assert.AreEqual(4, TypeManager.GetTypeInfo(TypeManager.GetTypeIndex(oneByteAlignmentType)).AlignmentInBytes); + Assert.AreEqual(8, bigAlignment); + + // Create an entity + var archetype = m_Manager.CreateArchetype(oneByteAlignmentType, typeof(ComponentWithAlign8)); + var entity = m_Manager.CreateEntity(archetype); + // Get a pointer to the first bigger-aligned component + var p2 = m_Manager.GetComponentDataRawRW(entity, bigTypeIndex); + + // p2 needs to be aligned properly + Assert.AreEqual(0, (long)p2 & (bigAlignment - 1)); + + // But let's verify that we didn't get lucky. If you see this assertion fire due to a change, + // it's because chunk layout chanked such that the 8-byte-aligned chunk would naturally fall + // on its proper alignment. Play with the BigComponent sizes (by adding/removing members) above + // until you get this to pass. + Assert.AreNotEqual(0, (archetype.Archetype->ChunkCapacity * oneByteInfo.SizeInChunk) % 8); + } } } diff --git a/Unity.Entities.Tests/EntityQueryBuilderTests.cs b/Unity.Entities.Tests/EntityQueryBuilderTests.cs index 97f9f68d..65e508b1 100644 --- a/Unity.Entities.Tests/EntityQueryBuilderTests.cs +++ b/Unity.Entities.Tests/EntityQueryBuilderTests.cs @@ -11,7 +11,7 @@ class EntityQueryBuilderTestFixture : ECSTestsFixture protected class TestComponentSystem : ComponentSystem { protected override void OnUpdate() { } } - protected static TestComponentSystem TestSystem => World.Active.GetOrCreateManager(); + protected static TestComponentSystem TestSystem => World.Active.GetOrCreateSystem(); } class EntityQueryBuilderTests : EntityQueryBuilderTestFixture @@ -20,7 +20,7 @@ class EntityQueryBuilderTests : EntityQueryBuilderTestFixture class TestComponentSystem2 : ComponentSystem { protected override void OnUpdate() { } } - static TestComponentSystem2 TestSystem2 => World.Active.GetOrCreateManager(); + static TestComponentSystem2 TestSystem2 => World.Active.GetOrCreateSystem(); [Test] public void WithGroup_WithNullGroup_Throws() => @@ -29,8 +29,8 @@ public void WithGroup_WithNullGroup_Throws() => [Test] public void WithGroup_WithExistingGroup_Throws() { - var group0 = TestSystem.GetComponentGroup(ComponentType.ReadWrite()); - var group1 = TestSystem.GetComponentGroup(ComponentType.ReadOnly()); + var group0 = TestSystem.GetEntityQuery(ComponentType.ReadWrite()); + var group1 = TestSystem.GetEntityQuery(ComponentType.ReadOnly()); var query = TestSystem.Entities.With(group0); @@ -40,7 +40,7 @@ public void WithGroup_WithExistingGroup_Throws() [Test] public void WithGroup_WithExistingSpec_Throws() { - var group = TestSystem.GetComponentGroup(ComponentType.ReadWrite()); + var group = TestSystem.GetEntityQuery(ComponentType.ReadWrite()); Assert.Throws(() => TestSystem.Entities.WithAny().With(group)); Assert.Throws(() => TestSystem.Entities.WithNone().With(group)); @@ -50,7 +50,7 @@ public void WithGroup_WithExistingSpec_Throws() [Test] public void WithSpec_WithExistingGroup_Throws() { - var group = TestSystem.GetComponentGroup(ComponentType.ReadWrite()); + var group = TestSystem.GetEntityQuery(ComponentType.ReadWrite()); Assert.Throws(() => TestSystem.Entities.With(group).WithAny()); Assert.Throws(() => TestSystem.Entities.With(group).WithNone()); @@ -91,8 +91,8 @@ public void Equals_WithSlightlyDifferentlyConstructedBuilders_ReturnsFalse() [Test] public void Equals_WithDifferentGroups_ReturnsFalse() { - var group0 = TestSystem.GetComponentGroup(ComponentType.ReadWrite()); - var group1 = TestSystem.GetComponentGroup(ComponentType.ReadOnly()); + var group0 = TestSystem.GetEntityQuery(ComponentType.ReadWrite()); + var group1 = TestSystem.GetEntityQuery(ComponentType.ReadOnly()); var builder0 = TestSystem.Entities.With(group0); var builder1 = TestSystem.Entities.With(group1); @@ -145,7 +145,7 @@ public void Equals_WithMismatchedBuilders_ReturnsFalse() } { - var group = TestSystem.GetComponentGroup(ComponentType.ReadWrite()); + var group = TestSystem.GetEntityQuery(ComponentType.ReadWrite()); var builder0 = TestSystem.Entities.With(group); var builder1 = TestSystem.Entities; Assert.IsFalse(builder0.ShallowEquals(ref builder1)); @@ -159,7 +159,7 @@ public void ToEntityArchetypeQuery_WithFluentSpec_ReturnsQueryAsSpecified() .WithAll() .WithAny() .WithNone() - .ToEntityArchetypeQuery(); + .ToEntityQueryDesc(); CollectionAssert.AreEqual( new[] { ComponentType.ReadWrite() }, @@ -177,7 +177,7 @@ public void ToComponentGroup_OnceCached_StaysCached() { // this will cause the group to get cached in the query var query = TestSystem.Entities.WithAll(); - query.ToComponentGroup(); + query.ToEntityQuery(); // this will throw because we're trying to modify the spec, yet we already have a group cached Assert.Throws(() => query.WithNone()); @@ -203,7 +203,7 @@ public void ForEach_WithReusedQueryButDifferentDelegateParams_Throws() Assert.IsTrue(oldQuery.ShallowEquals(ref query)); - var eaq = query.ToEntityArchetypeQuery(); + var eaq = query.ToEntityQueryDesc(); CollectionAssert.AreEqual( new[] { ComponentType.ReadWrite(), ComponentType.ReadWrite() }, eaq.All); diff --git a/Unity.Entities.Tests/EntityQueryCacheTests.cs b/Unity.Entities.Tests/EntityQueryCacheTests.cs index aad117cb..f0a13f87 100644 --- a/Unity.Entities.Tests/EntityQueryCacheTests.cs +++ b/Unity.Entities.Tests/EntityQueryCacheTests.cs @@ -8,6 +8,18 @@ namespace Unity.Entities.Tests { unsafe class EntityQueryCacheTests { + [SetUp] + public void Setup() + { + TypeManager.Initialize(); + } + + [TearDown] + public void TearDown() + { + TypeManager.Shutdown(); + } + [Test] public void Ctor_WithCacheSize0_Throws() { @@ -17,7 +29,7 @@ public void Ctor_WithCacheSize0_Throws() // ReSharper restore ObjectCreationAsStatement } - static void SimpleWrapCreateCachedQuery(EntityQueryCache cache, uint hash, ComponentGroup group) + static void SimpleWrapCreateCachedQuery(EntityQueryCache cache, uint hash, EntityQuery group) { #if ENABLE_UNITY_COLLECTIONS_CHECKS var builder = new EntityQueryBuilder(); @@ -27,6 +39,32 @@ static void SimpleWrapCreateCachedQuery(EntityQueryCache cache, uint hash, Compo #endif } + [Test] + public void CalcUsedCacheCount_WithEmptyCache_ReturnsZero() + { + var cache = new EntityQueryCache(1); + + Assert.AreEqual(0, cache.CalcUsedCacheCount()); + } + + [Test] + public void CalcUsedCacheCount_WithSomeInCache_ReturnsCorrectNumber() + { + var cache = new EntityQueryCache(2); + SimpleWrapCreateCachedQuery(cache, 0, k_DummyGroup); + + Assert.AreEqual(1, cache.CalcUsedCacheCount()); + } + + [Test] + public void CalcUsedCacheCount_WithFullCache_ReturnsCorrectNumber() + { + var cache = new EntityQueryCache(1); + SimpleWrapCreateCachedQuery(cache, 0, k_DummyGroup); + + Assert.AreEqual(1, cache.CalcUsedCacheCount()); + } + [Test] public void FindQueryInCache_WithEmptyCache_ReturnsErrorIndex() { @@ -61,7 +99,7 @@ public void FindQueryInCache_WithHashFound_ReturnsFoundIndex() } readonly Regex k_ResizeError = new Regex(".*is too small to hold the current number of queries.*"); - readonly ComponentGroup k_DummyGroup = new ComponentGroup(null, null, null, null); + readonly EntityQuery k_DummyGroup = new EntityQuery(null, null, null, null); [Test] public void CreateCachedQuery_WithNullGroup_Throws() @@ -127,6 +165,7 @@ public void GetCachedQuery_WithInvalidIndex_Throws() Assert.Throws(() => cache.GetCachedQuery(1)); } +#if ENABLE_UNITY_COLLECTIONS_CHECKS [Test] public void ValidateMatchesCache_WithValidMatch_DoesNotThrow() @@ -191,5 +230,6 @@ public void ValidateMatchesCache_WithMismatchedDelegateTypeIndices_Throws() catch (InvalidOperationException x) { testException1 = x; } Assert.NotNull(testException1); } +#endif } } diff --git a/Unity.Entities.Tests/EntityRemapUtilityTests.cs b/Unity.Entities.Tests/EntityRemapUtilityTests.cs index 0376bb27..fef0936e 100644 --- a/Unity.Entities.Tests/EntityRemapUtilityTests.cs +++ b/Unity.Entities.Tests/EntityRemapUtilityTests.cs @@ -10,6 +10,7 @@ public class EntityRemapUtilityTests [SetUp] public void Setup() { + TypeManager.Initialize(); m_Remapping = new NativeArray(100, Allocator.Persistent); } @@ -17,6 +18,7 @@ public void Setup() public void TearDown() { m_Remapping.Dispose(); + TypeManager.Shutdown(); } [Test] diff --git a/Unity.Entities.Tests/EntityTransactionTests.cs b/Unity.Entities.Tests/EntityTransactionTests.cs index a661d6ed..397c7cdf 100644 --- a/Unity.Entities.Tests/EntityTransactionTests.cs +++ b/Unity.Entities.Tests/EntityTransactionTests.cs @@ -13,7 +13,7 @@ namespace Unity.Entities.Tests [StandaloneFixme] // Asserts on JobDebugger in constructor class EntityTransactionTests : ECSTestsFixture { - ComponentGroup m_Group; + EntityQuery m_Group; public EntityTransactionTests() { @@ -25,7 +25,7 @@ public override void Setup() { base.Setup(); - m_Group = m_Manager.CreateComponentGroup(typeof(EcsTestData)); + m_Group = m_Manager.CreateEntityQuery(typeof(EcsTestData)); // Archetypes can't be created on a job m_Manager.CreateArchetype(typeof(EcsTestData)); @@ -84,7 +84,6 @@ public void CreateEntitiesChainedJob() [Test] - [StandaloneFixme] public void CommitAfterNotRegisteredTransactionJobLogsError() { var job = new CreateEntityJob(); diff --git a/Unity.Entities/Injection.meta b/Unity.Entities.Tests/ForEach.meta similarity index 77% rename from Unity.Entities/Injection.meta rename to Unity.Entities.Tests/ForEach.meta index 0b6e859b..ed678d6a 100644 --- a/Unity.Entities/Injection.meta +++ b/Unity.Entities.Tests/ForEach.meta @@ -1,5 +1,5 @@ fileFormatVersion: 2 -guid: 8bf694bfa86dd41fa8c549c11468bc71 +guid: e118a56a80e4c6749bc6caeb2ff6a56a folderAsset: yes DefaultImporter: externalObjects: {} diff --git a/Unity.Entities.Tests/ForEach/ForEachBasicMemoryTests.cs b/Unity.Entities.Tests/ForEach/ForEachBasicMemoryTests.cs new file mode 100644 index 00000000..e95c91d2 --- /dev/null +++ b/Unity.Entities.Tests/ForEach/ForEachBasicMemoryTests.cs @@ -0,0 +1,41 @@ +#if !UNITY_ZEROPLAYER +using System; +using NUnit.Framework; + +// ReSharper disable MemberCanBeMadeStatic.Local + +namespace Unity.Entities.Tests.ForEach +{ + // these tests are intended to confirm that compiled code allocs exactly how we expect + + class ForEachBasicMemoryTests : ForEachMemoryTestFixtureBase + { + // SANITY + + [Test] + public void StackAllocValueType_ShouldRecordNoGCAllocs() + { + AllocRecorder.enabled = true; + + // ReSharper disable once NotAccessedVariable + var i = 0; + ++i; + + AllocRecorder.enabled = false; + Assert.Zero(AllocRecorder.sampleBlockCount); + } + + [Test] + public void NewArray_ShouldRecordOneGCAlloc() + { + AllocRecorder.enabled = true; + + var t = new int[1]; + t[0] = 1; + + AllocRecorder.enabled = false; + Assert.AreEqual(1, AllocRecorder.sampleBlockCount); + } + } +} +#endif // !UNITY_ZEROPLAYER diff --git a/Unity.Entities.Tests/ForEach/ForEachBasicMemoryTests.cs.meta b/Unity.Entities.Tests/ForEach/ForEachBasicMemoryTests.cs.meta new file mode 100644 index 00000000..13e52c7f --- /dev/null +++ b/Unity.Entities.Tests/ForEach/ForEachBasicMemoryTests.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 047b97d3d30df7241bf8322367a2edc4 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/Unity.Entities.Tests/ForEach/ForEachDelegateMemoryTests.gen.cs b/Unity.Entities.Tests/ForEach/ForEachDelegateMemoryTests.gen.cs new file mode 100644 index 00000000..0e80c806 --- /dev/null +++ b/Unity.Entities.Tests/ForEach/ForEachDelegateMemoryTests.gen.cs @@ -0,0 +1,729 @@ +#if !UNITY_ZEROPLAYER +//------------------------------------------------------------------------------ +// +// This code was generated by a tool. +// +// Changes to this file may cause incorrect behavior and will be lost if +// the code is regenerated. +// +//------------------------------------------------------------------------------ +// Generated by ForEachDelegateMemoryTests.tt + +using NUnit.Framework; + +/* +The purpose of these tests is to validate and make invariant assumptions about +compiler codegen. ForEach needs to receive a delegate, and there are many ways +to implicitly construct one, which are tested below. What we want is to ensure +that we understand the scenarios that generate garbage every frame so we can +detect and warn about them in user code, or replace with our own codegen. + +There are two paths to garbage: + +1. ForEach needs to receive a delegate to do its work, which is represented in + the runtime as a MulticastDelegate. This needs to be allocated from the + gc, and it is not done statically, but on-demand. + + The result of this `newobj` may always be cached manually, and sometimes is + cached automatically. That's one of the things the below tests validate. + +2. In a closure, something needs to associate all captured non-global + variables (including `this`) with the delegate, typically a codegen'd class + with captures as members. This will involve a gc alloc per-call. + + Note that local functions use a struct for their captures when called in + the containing method, but this optimization does not apply if the local is + converted to a delegate. + +Categories: (each has a fixture) + + Lambda: Using the () => {} syntax. Current C# compilers (at least Mono and + Roslyn) will + + This is the key scenario we care about, + because it's the most natural to use with ForEach. (The other scenarios we + include because users may adopt them as workarounds.) + + LocalFunction: Using a local function defined in the same function where the + ForEach is called. + + (Static)Method: An ordinary (static) class method. Aside from lack of access + to locals (and `this` for statics), these should behave identically to local + functions. + + "Cached": Because the compilers don't currently cache delegates auto-created + for anything except lambdas, we'll generate garbage every frame for all + operations. So we manually cache in order to test just the closure alloc + behavior. + +Tests: (some combos that are impossible are left out) + + Empty: just an empty code block + + WithLocal: a closure that captures a local + + WithThis: a closure that captures `this` + + WithStatic: a method that uses a static +*/ + +namespace Unity.Entities.Tests.ForEach +{ + class ForEachLambdaMemoryTests : ForEachMemoryTestFixtureBase + { + int InvokeEmpty() + { + TestInvoke(v => { }); + return 0; + } + + [Test] + public void Empty_ShouldAllocFirstCallOnly() + { + AllocRecorder.enabled = true; + var firstCallTotal = InvokeEmpty(); + AllocRecorder.enabled = false; + var firstCallAllocCount = AllocRecorder.sampleBlockCount; + + AllocRecorder.enabled = true; + var loopedCallTotal = 0; + for (var i = 0; i < 5; ++i) + loopedCallTotal = InvokeEmpty(); + AllocRecorder.enabled = false; + var loopedCallAllocCount = AllocRecorder.sampleBlockCount; + + Assert.Greater(firstCallAllocCount, 0); + Assert.Zero(loopedCallAllocCount); + + Assert.AreEqual(0, firstCallTotal); + Assert.AreEqual(0, loopedCallTotal); + } + + int InvokeWithLocal() + { + int local = 2; + TestInvoke(v => { local += v; }); + return local; + } + + [Test] + public void WithLocal_ShouldAllocEveryCall() + { + AllocRecorder.enabled = true; + var firstCallTotal = InvokeWithLocal(); + AllocRecorder.enabled = false; + var firstCallAllocCount = AllocRecorder.sampleBlockCount; + + AllocRecorder.enabled = true; + var loopedCallTotal = 0; + for (var i = 0; i < 5; ++i) + loopedCallTotal += InvokeWithLocal(); + AllocRecorder.enabled = false; + var loopedCallAllocCount = AllocRecorder.sampleBlockCount; + + Assert.Greater(firstCallAllocCount, 0); + Assert.AreEqual(firstCallAllocCount * 5, loopedCallAllocCount); + + Assert.AreEqual(4, firstCallTotal); + Assert.AreEqual(20, loopedCallTotal); + } + + int InvokeWithThis() + { + TestInvoke(v => { m_Field += v; }); + return m_Field; + } + + [Test] + public void WithThis_ShouldAllocEveryCall() + { + AllocRecorder.enabled = true; + var firstCallTotal = InvokeWithThis(); + AllocRecorder.enabled = false; + var firstCallAllocCount = AllocRecorder.sampleBlockCount; + + AllocRecorder.enabled = true; + var loopedCallTotal = 0; + for (var i = 0; i < 5; ++i) + loopedCallTotal = InvokeWithThis(); + AllocRecorder.enabled = false; + var loopedCallAllocCount = AllocRecorder.sampleBlockCount; + + Assert.Greater(firstCallAllocCount, 0); + Assert.AreEqual(firstCallAllocCount * 5, loopedCallAllocCount); + + Assert.AreEqual(5, firstCallTotal); + Assert.AreEqual(15, loopedCallTotal); + } + + int InvokeWithStatic() + { + TestInvoke(v => { s_Static += v; }); + return s_Static; + } + + [Test] + public void WithStatic_ShouldAllocFirstCallOnly() + { + AllocRecorder.enabled = true; + var firstCallTotal = InvokeWithStatic(); + AllocRecorder.enabled = false; + var firstCallAllocCount = AllocRecorder.sampleBlockCount; + + AllocRecorder.enabled = true; + var loopedCallTotal = 0; + for (var i = 0; i < 5; ++i) + loopedCallTotal = InvokeWithStatic(); + AllocRecorder.enabled = false; + var loopedCallAllocCount = AllocRecorder.sampleBlockCount; + + Assert.Greater(firstCallAllocCount, 0); + Assert.Zero(loopedCallAllocCount); + + Assert.AreEqual(6, firstCallTotal); + Assert.AreEqual(16, loopedCallTotal); + } + + } + + class ForEachLocalFunctionMemoryTests : ForEachMemoryTestFixtureBase + { + int InvokeEmpty() + { + void Empty(int v) { } + TestInvoke(Empty); + return 0; + } + + [Test] + [Ignore("Unstable")] + public void Empty_ShouldAllocFirstCallOnly() + { + AllocRecorder.enabled = true; + var firstCallTotal = InvokeEmpty(); + AllocRecorder.enabled = false; + var firstCallAllocCount = AllocRecorder.sampleBlockCount; + + AllocRecorder.enabled = true; + var loopedCallTotal = 0; + for (var i = 0; i < 5; ++i) + loopedCallTotal = InvokeEmpty(); + AllocRecorder.enabled = false; + var loopedCallAllocCount = AllocRecorder.sampleBlockCount; + + Assert.Greater(firstCallAllocCount, 0); + Assert.Zero(loopedCallAllocCount); + + Assert.AreEqual(0, firstCallTotal); + Assert.AreEqual(0, loopedCallTotal); + } + + int InvokeWithLocal() + { + int local = 2; + void WithLocal(int v) { local += v; } + TestInvoke(WithLocal); + return local; + } + + [Test] + [Ignore("Unstable")] + public void WithLocal_ShouldAllocEveryCall() + { + AllocRecorder.enabled = true; + var firstCallTotal = InvokeWithLocal(); + AllocRecorder.enabled = false; + var firstCallAllocCount = AllocRecorder.sampleBlockCount; + + AllocRecorder.enabled = true; + var loopedCallTotal = 0; + for (var i = 0; i < 5; ++i) + loopedCallTotal += InvokeWithLocal(); + AllocRecorder.enabled = false; + var loopedCallAllocCount = AllocRecorder.sampleBlockCount; + + Assert.Greater(firstCallAllocCount, 0); + Assert.AreEqual(firstCallAllocCount * 5, loopedCallAllocCount); + + Assert.AreEqual(0, firstCallTotal); + Assert.AreEqual(0, loopedCallTotal); + } + + int InvokeWithThis() + { + void WithThis(int v) { m_Field += v; } + TestInvoke(WithThis); + return m_Field; + } + + [Test] + public void WithThis_ShouldAllocEveryCall() + { + AllocRecorder.enabled = true; + var firstCallTotal = InvokeWithThis(); + AllocRecorder.enabled = false; + var firstCallAllocCount = AllocRecorder.sampleBlockCount; + + AllocRecorder.enabled = true; + var loopedCallTotal = 0; + for (var i = 0; i < 5; ++i) + loopedCallTotal = InvokeWithThis(); + AllocRecorder.enabled = false; + var loopedCallAllocCount = AllocRecorder.sampleBlockCount; + + Assert.Greater(firstCallAllocCount, 0); + Assert.AreEqual(firstCallAllocCount * 5, loopedCallAllocCount); + + Assert.AreEqual(5, firstCallTotal); + Assert.AreEqual(15, loopedCallTotal); + } + + int InvokeWithStatic() + { + void WithStatic(int v) { s_Static += v; } + TestInvoke(WithStatic); + return s_Static; + } + + [Test] + [Ignore("Unstable")] + public void WithStatic_ShouldAllocEveryCall() + { + AllocRecorder.enabled = true; + var firstCallTotal = InvokeWithStatic(); + AllocRecorder.enabled = false; + var firstCallAllocCount = AllocRecorder.sampleBlockCount; + + AllocRecorder.enabled = true; + var loopedCallTotal = 0; + for (var i = 0; i < 5; ++i) + loopedCallTotal = InvokeWithStatic(); + AllocRecorder.enabled = false; + var loopedCallAllocCount = AllocRecorder.sampleBlockCount; + + Assert.Greater(firstCallAllocCount, 0); + Assert.AreEqual(firstCallAllocCount * 5, loopedCallAllocCount); + + Assert.AreEqual(6, firstCallTotal); + Assert.AreEqual(16, loopedCallTotal); + } + + } + + class ForEachLocalFunctionCachedMemoryTests : ForEachMemoryTestFixtureBase + { + TestDelegate m_EmptyDelegate; + int InvokeEmpty() + { + void Empty(int v) { } + TestInvoke(m_EmptyDelegate = m_EmptyDelegate ?? Empty); + return 0; + } + + [Test] + public void Empty_ShouldAllocFirstCallOnly() + { + AllocRecorder.enabled = true; + var firstCallTotal = InvokeEmpty(); + AllocRecorder.enabled = false; + var firstCallAllocCount = AllocRecorder.sampleBlockCount; + + AllocRecorder.enabled = true; + var loopedCallTotal = 0; + for (var i = 0; i < 5; ++i) + loopedCallTotal = InvokeEmpty(); + AllocRecorder.enabled = false; + var loopedCallAllocCount = AllocRecorder.sampleBlockCount; + + Assert.Greater(firstCallAllocCount, 0); + Assert.Zero(loopedCallAllocCount); + + Assert.AreEqual(0, firstCallTotal); + Assert.AreEqual(0, loopedCallTotal); + } + + TestDelegate m_WithThisDelegate; + int InvokeWithThis() + { + void WithThis(int v) { m_Field += v; } + TestInvoke(m_WithThisDelegate = m_WithThisDelegate ?? WithThis); + return m_Field; + } + + [Test] + public void WithThis_ShouldAllocFirstCallOnly() + { + AllocRecorder.enabled = true; + var firstCallTotal = InvokeWithThis(); + AllocRecorder.enabled = false; + var firstCallAllocCount = AllocRecorder.sampleBlockCount; + + AllocRecorder.enabled = true; + var loopedCallTotal = 0; + for (var i = 0; i < 5; ++i) + loopedCallTotal = InvokeWithThis(); + AllocRecorder.enabled = false; + var loopedCallAllocCount = AllocRecorder.sampleBlockCount; + + Assert.Greater(firstCallAllocCount, 0); + Assert.Zero(loopedCallAllocCount); + + Assert.AreEqual(5, firstCallTotal); + Assert.AreEqual(15, loopedCallTotal); + } + + TestDelegate m_WithStaticDelegate; + int InvokeWithStatic() + { + void WithStatic(int v) { s_Static += v; } + TestInvoke(m_WithStaticDelegate = m_WithStaticDelegate ?? WithStatic); + return s_Static; + } + + [Test] + public void WithStatic_ShouldAllocFirstCallOnly() + { + AllocRecorder.enabled = true; + var firstCallTotal = InvokeWithStatic(); + AllocRecorder.enabled = false; + var firstCallAllocCount = AllocRecorder.sampleBlockCount; + + AllocRecorder.enabled = true; + var loopedCallTotal = 0; + for (var i = 0; i < 5; ++i) + loopedCallTotal = InvokeWithStatic(); + AllocRecorder.enabled = false; + var loopedCallAllocCount = AllocRecorder.sampleBlockCount; + + Assert.Greater(firstCallAllocCount, 0); + Assert.Zero(loopedCallAllocCount); + + Assert.AreEqual(6, firstCallTotal); + Assert.AreEqual(16, loopedCallTotal); + } + + } + + class ForEachMethodMemoryTests : ForEachMemoryTestFixtureBase + { + void MethodEmpty(int v) { } + + int InvokeEmpty() + { + TestInvoke(MethodEmpty); + return 0; + } + + [Test] + public void Empty_ShouldAllocEveryCall() + { + AllocRecorder.enabled = true; + var firstCallTotal = InvokeEmpty(); + AllocRecorder.enabled = false; + var firstCallAllocCount = AllocRecorder.sampleBlockCount; + + AllocRecorder.enabled = true; + var loopedCallTotal = 0; + for (var i = 0; i < 5; ++i) + loopedCallTotal = InvokeEmpty(); + AllocRecorder.enabled = false; + var loopedCallAllocCount = AllocRecorder.sampleBlockCount; + + Assert.Greater(firstCallAllocCount, 0); + Assert.AreEqual(firstCallAllocCount * 5, loopedCallAllocCount); + + Assert.AreEqual(0, firstCallTotal); + Assert.AreEqual(0, loopedCallTotal); + } + + void MethodWithThis(int v) { m_Field += v; } + + int InvokeWithThis() + { + TestInvoke(MethodWithThis); + return m_Field; + } + + [Test] + public void WithThis_ShouldAllocEveryCall() + { + AllocRecorder.enabled = true; + var firstCallTotal = InvokeWithThis(); + AllocRecorder.enabled = false; + var firstCallAllocCount = AllocRecorder.sampleBlockCount; + + AllocRecorder.enabled = true; + var loopedCallTotal = 0; + for (var i = 0; i < 5; ++i) + loopedCallTotal = InvokeWithThis(); + AllocRecorder.enabled = false; + var loopedCallAllocCount = AllocRecorder.sampleBlockCount; + + Assert.Greater(firstCallAllocCount, 0); + Assert.AreEqual(firstCallAllocCount * 5, loopedCallAllocCount); + + Assert.AreEqual(5, firstCallTotal); + Assert.AreEqual(15, loopedCallTotal); + } + + void MethodWithStatic(int v) { s_Static += v; } + + int InvokeWithStatic() + { + TestInvoke(MethodWithStatic); + return s_Static; + } + + [Test] + public void WithStatic_ShouldAllocEveryCall() + { + AllocRecorder.enabled = true; + var firstCallTotal = InvokeWithStatic(); + AllocRecorder.enabled = false; + var firstCallAllocCount = AllocRecorder.sampleBlockCount; + + AllocRecorder.enabled = true; + var loopedCallTotal = 0; + for (var i = 0; i < 5; ++i) + loopedCallTotal = InvokeWithStatic(); + AllocRecorder.enabled = false; + var loopedCallAllocCount = AllocRecorder.sampleBlockCount; + + Assert.Greater(firstCallAllocCount, 0); + Assert.AreEqual(firstCallAllocCount * 5, loopedCallAllocCount); + + Assert.AreEqual(6, firstCallTotal); + Assert.AreEqual(16, loopedCallTotal); + } + + } + + class ForEachMethodCachedMemoryTests : ForEachMemoryTestFixtureBase + { + TestDelegate m_EmptyDelegate; + void MethodEmpty(int v) { } + + int InvokeEmpty() + { + TestInvoke(m_EmptyDelegate = m_EmptyDelegate ?? MethodEmpty); + return 0; + } + + [Test] + public void Empty_ShouldAllocFirstCallOnly() + { + AllocRecorder.enabled = true; + var firstCallTotal = InvokeEmpty(); + AllocRecorder.enabled = false; + var firstCallAllocCount = AllocRecorder.sampleBlockCount; + + AllocRecorder.enabled = true; + var loopedCallTotal = 0; + for (var i = 0; i < 5; ++i) + loopedCallTotal = InvokeEmpty(); + AllocRecorder.enabled = false; + var loopedCallAllocCount = AllocRecorder.sampleBlockCount; + + Assert.Greater(firstCallAllocCount, 0); + Assert.Zero(loopedCallAllocCount); + + Assert.AreEqual(0, firstCallTotal); + Assert.AreEqual(0, loopedCallTotal); + } + + TestDelegate m_WithThisDelegate; + void MethodWithThis(int v) { m_Field += v; } + + int InvokeWithThis() + { + TestInvoke(m_WithThisDelegate = m_WithThisDelegate ?? MethodWithThis); + return m_Field; + } + + [Test] + public void WithThis_ShouldAllocFirstCallOnly() + { + AllocRecorder.enabled = true; + var firstCallTotal = InvokeWithThis(); + AllocRecorder.enabled = false; + var firstCallAllocCount = AllocRecorder.sampleBlockCount; + + AllocRecorder.enabled = true; + var loopedCallTotal = 0; + for (var i = 0; i < 5; ++i) + loopedCallTotal = InvokeWithThis(); + AllocRecorder.enabled = false; + var loopedCallAllocCount = AllocRecorder.sampleBlockCount; + + Assert.Greater(firstCallAllocCount, 0); + Assert.Zero(loopedCallAllocCount); + + Assert.AreEqual(5, firstCallTotal); + Assert.AreEqual(15, loopedCallTotal); + } + + TestDelegate m_WithStaticDelegate; + void MethodWithStatic(int v) { s_Static += v; } + + int InvokeWithStatic() + { + TestInvoke(m_WithStaticDelegate = m_WithStaticDelegate ?? MethodWithStatic); + return s_Static; + } + + [Test] + public void WithStatic_ShouldAllocFirstCallOnly() + { + AllocRecorder.enabled = true; + var firstCallTotal = InvokeWithStatic(); + AllocRecorder.enabled = false; + var firstCallAllocCount = AllocRecorder.sampleBlockCount; + + AllocRecorder.enabled = true; + var loopedCallTotal = 0; + for (var i = 0; i < 5; ++i) + loopedCallTotal = InvokeWithStatic(); + AllocRecorder.enabled = false; + var loopedCallAllocCount = AllocRecorder.sampleBlockCount; + + Assert.Greater(firstCallAllocCount, 0); + Assert.Zero(loopedCallAllocCount); + + Assert.AreEqual(6, firstCallTotal); + Assert.AreEqual(16, loopedCallTotal); + } + + } + + class ForEachStaticMethodMemoryTests : ForEachMemoryTestFixtureBase + { + static void StaticMethodEmpty(int v) { } + + int InvokeEmpty() + { + TestInvoke(StaticMethodEmpty); + return 0; + } + + [Test] + public void Empty_ShouldAllocEveryCall() + { + AllocRecorder.enabled = true; + var firstCallTotal = InvokeEmpty(); + AllocRecorder.enabled = false; + var firstCallAllocCount = AllocRecorder.sampleBlockCount; + + AllocRecorder.enabled = true; + var loopedCallTotal = 0; + for (var i = 0; i < 5; ++i) + loopedCallTotal = InvokeEmpty(); + AllocRecorder.enabled = false; + var loopedCallAllocCount = AllocRecorder.sampleBlockCount; + + Assert.Greater(firstCallAllocCount, 0); + Assert.AreEqual(firstCallAllocCount * 5, loopedCallAllocCount); + + Assert.AreEqual(0, firstCallTotal); + Assert.AreEqual(0, loopedCallTotal); + } + + static void StaticMethodWithStatic(int v) { s_Static += v; } + + int InvokeWithStatic() + { + TestInvoke(StaticMethodWithStatic); + return s_Static; + } + + [Test] + public void WithStatic_ShouldAllocEveryCall() + { + AllocRecorder.enabled = true; + var firstCallTotal = InvokeWithStatic(); + AllocRecorder.enabled = false; + var firstCallAllocCount = AllocRecorder.sampleBlockCount; + + AllocRecorder.enabled = true; + var loopedCallTotal = 0; + for (var i = 0; i < 5; ++i) + loopedCallTotal = InvokeWithStatic(); + AllocRecorder.enabled = false; + var loopedCallAllocCount = AllocRecorder.sampleBlockCount; + + Assert.Greater(firstCallAllocCount, 0); + Assert.AreEqual(firstCallAllocCount * 5, loopedCallAllocCount); + + Assert.AreEqual(6, firstCallTotal); + Assert.AreEqual(16, loopedCallTotal); + } + + } + + class ForEachStaticMethodCachedMemoryTests : ForEachMemoryTestFixtureBase + { + TestDelegate m_EmptyDelegate; + static void StaticMethodEmpty(int v) { } + + int InvokeEmpty() + { + TestInvoke(m_EmptyDelegate = m_EmptyDelegate ?? StaticMethodEmpty); + return 0; + } + + [Test] + public void Empty_ShouldAllocFirstCallOnly() + { + AllocRecorder.enabled = true; + var firstCallTotal = InvokeEmpty(); + AllocRecorder.enabled = false; + var firstCallAllocCount = AllocRecorder.sampleBlockCount; + + AllocRecorder.enabled = true; + var loopedCallTotal = 0; + for (var i = 0; i < 5; ++i) + loopedCallTotal = InvokeEmpty(); + AllocRecorder.enabled = false; + var loopedCallAllocCount = AllocRecorder.sampleBlockCount; + + Assert.Greater(firstCallAllocCount, 0); + Assert.Zero(loopedCallAllocCount); + + Assert.AreEqual(0, firstCallTotal); + Assert.AreEqual(0, loopedCallTotal); + } + + TestDelegate m_WithStaticDelegate; + static void StaticMethodWithStatic(int v) { s_Static += v; } + + int InvokeWithStatic() + { + TestInvoke(m_WithStaticDelegate = m_WithStaticDelegate ?? StaticMethodWithStatic); + return s_Static; + } + + [Test] + public void WithStatic_ShouldAllocFirstCallOnly() + { + AllocRecorder.enabled = true; + var firstCallTotal = InvokeWithStatic(); + AllocRecorder.enabled = false; + var firstCallAllocCount = AllocRecorder.sampleBlockCount; + + AllocRecorder.enabled = true; + var loopedCallTotal = 0; + for (var i = 0; i < 5; ++i) + loopedCallTotal = InvokeWithStatic(); + AllocRecorder.enabled = false; + var loopedCallAllocCount = AllocRecorder.sampleBlockCount; + + Assert.Greater(firstCallAllocCount, 0); + Assert.Zero(loopedCallAllocCount); + + Assert.AreEqual(6, firstCallTotal); + Assert.AreEqual(16, loopedCallTotal); + } + + } + +} +#endif // !UNITY_ZEROPLAYER diff --git a/Unity.Entities.Tests/ForEach/ForEachDelegateMemoryTests.gen.cs.meta b/Unity.Entities.Tests/ForEach/ForEachDelegateMemoryTests.gen.cs.meta new file mode 100644 index 00000000..ea41a907 --- /dev/null +++ b/Unity.Entities.Tests/ForEach/ForEachDelegateMemoryTests.gen.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: b99fa878950225049b0139e443740796 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/Unity.Entities.Tests/ForEach/ForEachDelegateMemoryTests.tt b/Unity.Entities.Tests/ForEach/ForEachDelegateMemoryTests.tt new file mode 100644 index 00000000..d78a315b --- /dev/null +++ b/Unity.Entities.Tests/ForEach/ForEachDelegateMemoryTests.tt @@ -0,0 +1,214 @@ +<#/*THIS IS A T4 FILE - see HACKING.md for what it is and how to run codegen*/#> +<#@ assembly name="System.Collections" #> +<#@ assembly name="System.Core" #> +<#@ assembly name="System.Linq" #> +<#@ import namespace="System.Collections.Generic" #> +<#@ import namespace="System.Linq" #> +<#@ output extension=".gen.cs" #> +#if !UNITY_ZEROPLAYER +//------------------------------------------------------------------------------ +// +// This code was generated by a tool. +// +// Changes to this file may cause incorrect behavior and will be lost if +// the code is regenerated. +// +//------------------------------------------------------------------------------ +// Generated by ForEachDelegateMemoryTests.tt + +using NUnit.Framework; + +/* +The purpose of these tests is to validate and make invariant assumptions about +compiler codegen. ForEach needs to receive a delegate, and there are many ways +to implicitly construct one, which are tested below. What we want is to ensure +that we understand the scenarios that generate garbage every frame so we can +detect and warn about them in user code, or replace with our own codegen. + +There are two paths to garbage: + +1. ForEach needs to receive a delegate to do its work, which is represented in + the runtime as a MulticastDelegate. This needs to be allocated from the + gc, and it is not done statically, but on-demand. + + The result of this `newobj` may always be cached manually, and sometimes is + cached automatically. That's one of the things the below tests validate. + +2. In a closure, something needs to associate all captured non-global + variables (including `this`) with the delegate, typically a codegen'd class + with captures as members. This will involve a gc alloc per-call. + + Note that local functions use a struct for their captures when called in + the containing method, but this optimization does not apply if the local is + converted to a delegate. + +Categories: (each has a fixture) + + Lambda: Using the () => {} syntax. Current C# compilers (at least Mono and + Roslyn) will + + This is the key scenario we care about, + because it's the most natural to use with ForEach. (The other scenarios we + include because users may adopt them as workarounds.) + + LocalFunction: Using a local function defined in the same function where the + ForEach is called. + + (Static)Method: An ordinary (static) class method. Aside from lack of access + to locals (and `this` for statics), these should behave identically to local + functions. + + "Cached": Because the compilers don't currently cache delegates auto-created + for anything except lambdas, we'll generate garbage every frame for all + operations. So we manually cache in order to test just the closure alloc + behavior. + +Tests: (some combos that are impossible are left out) + + Empty: just an empty code block + + WithLocal: a closure that captures a local + + WithThis: a closure that captures `this` + + WithStatic: a method that uses a static +*/ + +namespace Unity.Entities.Tests.ForEach +{ +<# +const int k_Loops = 5; + +foreach (var fixture in new[] { + // spec: bool allocEveryCall, int firstCallTotal, int loopedCallTotal + // invoke: 0 = test.name, 1 = test.code + (type: "Lambda", invoke: "v => {1}", + empty: new Spec(false, 0, 0), + local: new Spec( true, 4, 20), + thiz: new Spec( true, 5, 15), + statik: new Spec(false, 6, 16)), + (type: "LocalFunction", invoke: "{0}", + empty: new Spec(false, 0, 0) { Disabled = "Unstable" }, + local: new Spec( true, 0, 0) { Disabled = "Unstable" }, + thiz: new Spec( true, 5, 15), + statik: new Spec( true, 6, 16) { Disabled = "Unstable" }), // << loopedCallAllocCount doesn't match expected + (type: "LocalFunctionCached", invoke: "m_{0}Delegate = m_{0}Delegate ?? {0}", + empty: new Spec(false, 0, 0), + local: null, // new Spec( true, 0, 0), + thiz: new Spec(false, 5, 15), + statik: new Spec(false, 6, 16)), + (type: "Method", invoke: "Method{0}", + empty: new Spec( true, 0, 0), + local: null, + thiz: new Spec( true, 5, 15), + statik: new Spec( true, 6, 16)), + (type: "MethodCached", invoke: "m_{0}Delegate = m_{0}Delegate ?? Method{0}", + empty: new Spec(false, 0, 0), + local: null, + thiz: new Spec(false, 5, 15), + statik: new Spec(false, 6, 16)), + (type: "StaticMethod", invoke: "StaticMethod{0}", + empty: new Spec( true, 0, 0), + local: null, + thiz: null, + statik: new Spec( true, 6, 16)), + (type: "StaticMethodCached", invoke: "m_{0}Delegate = m_{0}Delegate ?? StaticMethod{0}", + empty: new Spec(false, 0, 0), + local: null, + thiz: null, + statik: new Spec(false, 6, 16)) }) { + + // TODO: fix unstable commented-out Spec stuff above + + var tests = new[] { + (name: "Empty", spec: fixture.empty, code: "{ }", retval: "0"), + (name: "WithLocal", spec: fixture.local, code: "{ local += v; }", retval: "local"), + (name: "WithThis", spec: fixture.thiz, code: "{ m_Field += v; }", retval: "m_Field"), + (name: "WithStatic", spec: fixture.statik, code: "{ s_Static += v; }", retval: "s_Static") }; +#> + class ForEach<#=fixture.type#>MemoryTests : ForEachMemoryTestFixtureBase + { +<# + foreach (var test in tests) { + if (test.spec == null) + continue; + string member = null; + if (fixture.type.StartsWith("Method")) + member = $"void Method{test.name}(int v) {test.code}"; + else if (fixture.type.StartsWith("StaticMethod")) + member = $"static void StaticMethod{test.name}(int v) {test.code}"; + if (fixture.type.EndsWith("Cached")) {#> + TestDelegate m_<#=test.name#>Delegate; +<# } + if (member != null) {#> + <#=member#> + +<# }#> + int Invoke<#=test.name#>() + { +<# if (test.name == "WithLocal") {#> + int local = 2; +<# }#> +<# if (fixture.type.StartsWith("LocalFunction")) {#> + void <#=test.name#>(int v) <#=test.code#> +<# }#> + TestInvoke(<#=string.Format(fixture.invoke, test.name, test.code)#>); + return <#=test.retval#>; + } + + [Test] +<# if (test.spec.Disabled != null) {#> + [Ignore("<#=test.spec.Disabled#>")] +<# }#> + public void <#=test.name#>_ShouldAlloc<#=test.spec.AllocEveryCall ? "EveryCall" : "FirstCallOnly"#>() + { + AllocRecorder.enabled = true; + var firstCallTotal = Invoke<#=test.name#>(); + AllocRecorder.enabled = false; + var firstCallAllocCount = AllocRecorder.sampleBlockCount; + + AllocRecorder.enabled = true; + var loopedCallTotal = 0; + for (var i = 0; i < <#=k_Loops#>; ++i) + loopedCallTotal <#=test.name == "WithLocal" ? "+=" : "="#> Invoke<#=test.name#>(); + AllocRecorder.enabled = false; + var loopedCallAllocCount = AllocRecorder.sampleBlockCount; + + Assert.Greater(firstCallAllocCount, 0); +<# if (test.spec.AllocEveryCall) {#> + Assert.AreEqual(firstCallAllocCount * <#=k_Loops#>, loopedCallAllocCount); +<# } else {#> + Assert.Zero(loopedCallAllocCount); +<# }#> + + Assert.AreEqual(<#=test.spec.FirstCallTotal#>, firstCallTotal); + Assert.AreEqual(<#=test.spec.LoopedCallTotal#>, loopedCallTotal); + } + +<# }#> + } + +<#}#> +} +#endif // !UNITY_ZEROPLAYER +<#+ + +class Spec +{ + public bool AllocEveryCall; + + // these counts are just used to validate that the delegates are called the expected number of times + public int FirstCallTotal; + public int LoopedCallTotal; + + public string Disabled; + + public Spec(bool allocEveryCall, int firstCallTotal, int loopedCallTotal) + { + AllocEveryCall = allocEveryCall; + FirstCallTotal = firstCallTotal; + LoopedCallTotal = loopedCallTotal; + } +} + +#> diff --git a/Unity.Entities/EntityQueryBuilder.tt.meta b/Unity.Entities.Tests/ForEach/ForEachDelegateMemoryTests.tt.meta similarity index 74% rename from Unity.Entities/EntityQueryBuilder.tt.meta rename to Unity.Entities.Tests/ForEach/ForEachDelegateMemoryTests.tt.meta index ec9bb529..3a692fd5 100644 --- a/Unity.Entities/EntityQueryBuilder.tt.meta +++ b/Unity.Entities.Tests/ForEach/ForEachDelegateMemoryTests.tt.meta @@ -1,5 +1,5 @@ fileFormatVersion: 2 -guid: df0a5c4fe61ce2c42bcab398bb71d5ce +guid: 2593083696dc6c341b5525d68f1597bc DefaultImporter: externalObjects: {} userData: diff --git a/Unity.Entities.Tests/ForEachTests.cs b/Unity.Entities.Tests/ForEach/ForEachGeneralTests.cs similarity index 55% rename from Unity.Entities.Tests/ForEachTests.cs rename to Unity.Entities.Tests/ForEach/ForEachGeneralTests.cs index 0156601e..21d0e92f 100644 --- a/Unity.Entities.Tests/ForEachTests.cs +++ b/Unity.Entities.Tests/ForEach/ForEachGeneralTests.cs @@ -1,7 +1,7 @@ using System; using NUnit.Framework; -namespace Unity.Entities.Tests +namespace Unity.Entities.Tests.ForEach { class ForEachBasicTests : EntityQueryBuilderTestFixture { @@ -9,10 +9,47 @@ class ForEachBasicTests : EntityQueryBuilderTestFixture public void CreateTestEntities() { m_Manager.AddComponentData(m_Manager.CreateEntity(), new EcsTestData(5)); + m_Manager.CreateEntity(typeof(EcsTestTag)); m_Manager.AddSharedComponentData(m_Manager.CreateEntity(), new SharedData1(7)); m_Manager.CreateEntity(typeof(EcsIntElement)); } + [Test, Repeat(2)] + public void Ensure_CacheClearedBetweenTests() // sanity check + { + Assert.IsNull(TestSystem.EntityQueryCache); + + var counter = 0; + TestSystem.Entities.ForEach(e => ++counter); + Assert.AreEqual(4, counter); + } + + [Test] + public void Ensure_EachElementOfEntityQueryBuilder_UsedInHashing() + { + // re-run should not change hash + for (var i = 0; i < 2; ++i) + { + // variance in 'all' should change hash + TestSystem.Entities.WithAll().ForEach(e => { }); + TestSystem.Entities.WithAll().ForEach(e => { }); + + // variance in 'any' should change hash + TestSystem.Entities.WithAny().ForEach(e => { }); + TestSystem.Entities.WithAny().ForEach(e => { }); + + // variance in 'none' should change hash + TestSystem.Entities.WithNone().ForEach(e => { }); + TestSystem.Entities.WithNone().ForEach(e => { }); + + // variance in delegate params should change hash + TestSystem.Entities.WithNone().ForEach((ref EcsTestData2 d) => { }); + TestSystem.Entities.WithNone().ForEach((ref EcsTestData3 d) => { }); + + Assert.AreEqual(8, TestSystem.EntityQueryCache.CalcUsedCacheCount()); + } + } + [Test] public void All() { @@ -22,7 +59,7 @@ public void All() Assert.IsTrue(m_Manager.Exists(entity)); counter++; }); - Assert.AreEqual(3, counter); + Assert.AreEqual(4, counter); } [Test] @@ -78,6 +115,14 @@ public void DynamicBuffer() }); Assert.AreEqual(1, counter); } + + [Test] + public void EmptyComponentData() // "tag" + { + var counter = 0; + TestSystem.Entities.WithAll().ForEach(entity => ++counter); + Assert.AreEqual(1, counter); + } } class ForEachTests : EntityQueryBuilderTestFixture @@ -106,6 +151,57 @@ public void Many() Assert.AreEqual(1, counter); } + [Test] + public void MixingWithAllAndForEach_Throws() + { + Assert.Throws(() => + { + TestSystem.Entities.WithAll().ForEach((Entity e, ref EcsTestData2 t1) => + { + Assert.Fail(); + }); + }); + } + + [Test] + public void MixingNoneAndAll_Throws() + { + Assert.Throws(() => + { + TestSystem.Entities.WithNone().ForEach((Entity e, ref EcsTestData2 t1) => + { + Assert.Fail(); + }); + }); + } + + [Test] + public void UsingAnyInForEach_Matches() + { + m_Manager.CreateEntity(typeof(EcsTestData)); + m_Manager.CreateEntity(typeof(EcsTestData2)); + m_Manager.CreateEntity(typeof(EcsTestData), typeof(EcsTestData2)); + + var counter = 0; + TestSystem.Entities + .WithAny() + .ForEach(e => ++counter); + Assert.AreEqual(counter, 3); + + } + + [Test] + public void MixingAnyWithForEachDelegateParams_Throws() + { + Assert.Throws(() => + { + TestSystem.Entities.WithAny().ForEach((ref EcsTestData t1) => + { + Assert.Fail(); + }); + }); + } + [Test] public void Safety() { @@ -127,8 +223,9 @@ public void Safety() { TestSystem.Entities.ForEach((Entity e, ref EcsTestData t0) => throw new ArgumentException()); }); - +#if ENABLE_UNITY_COLLECTIONS_CHECKS Assert.IsFalse(m_Manager.IsInsideForEach); +#endif } //@TODO: Class iterator test coverage... diff --git a/Unity.Entities.Tests/ForEach/ForEachGeneralTests.cs.meta b/Unity.Entities.Tests/ForEach/ForEachGeneralTests.cs.meta new file mode 100644 index 00000000..3e8abc4c --- /dev/null +++ b/Unity.Entities.Tests/ForEach/ForEachGeneralTests.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 9723439e4511f2a4aa9aa750e04775cd +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/Unity.Entities.Tests/ForEach/ForEachMemoryTestFixtureBase.cs b/Unity.Entities.Tests/ForEach/ForEachMemoryTestFixtureBase.cs new file mode 100644 index 00000000..02508a26 --- /dev/null +++ b/Unity.Entities.Tests/ForEach/ForEachMemoryTestFixtureBase.cs @@ -0,0 +1,62 @@ +#if !UNITY_ZEROPLAYER +using System; +using System.Linq; +using System.Reflection; +using NUnit.Framework; +using UnityEngine.Profiling; + +namespace Unity.Entities.Tests.ForEach +{ + class ForEachMemoryTestFixtureBase : EntityQueryBuilderTestFixture + { + // TODO: implement equivalent in DOTS runtime and remove #if !UNITY_ZEROPLAYER from all ForEach memory tests + Recorder m_AllocRecorder; + + protected int m_Field; + protected static int s_Static; + + [SetUp] + public void SetUp() + { + m_AllocRecorder = Recorder.Get("GC.Alloc"); + m_AllocRecorder.FilterToCurrentThread(); + m_AllocRecorder.enabled = false; + + m_Field = 3; + s_Static = 4; + } + + [TearDown] + public new void TearDown() + { + // this code is necessary to make our tests re-runnable without requiring a domain reload. + // lambdas only provide code, and we're operating with delegates in ForEach, which require + // an alloc to create. the c# compiler generates code to alloc the delegate on first use, + // and then stores it in a static for later reuse. we need to clear those statics in order + // to rerun the tests. + + var cache = GetType().GetNestedType("<>c", BindingFlags.NonPublic); + if (cache != null) + { + foreach (var field in cache + .GetFields(BindingFlags.Static | BindingFlags.Public) + .Where(f => typeof(MulticastDelegate).IsAssignableFrom(f.FieldType))) + { + field.SetValue(null, null); + } + } + } + + protected Recorder AllocRecorder => m_AllocRecorder; + + protected delegate void TestDelegate(int i); + + protected void TestInvoke(TestDelegate op) + { + // simulate more than one iteration + op(1); + op(1); + } + } +} +#endif // !UNITY_ZEROPLAYER diff --git a/Unity.Entities.Tests/ForEach/ForEachMemoryTestFixtureBase.cs.meta b/Unity.Entities.Tests/ForEach/ForEachMemoryTestFixtureBase.cs.meta new file mode 100644 index 00000000..fe51948f --- /dev/null +++ b/Unity.Entities.Tests/ForEach/ForEachMemoryTestFixtureBase.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 60f00a90d39638042a2984d05e17c25e +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/Unity.Entities.Tests/ForEachTests.cs.meta b/Unity.Entities.Tests/ForEachTests.cs.meta deleted file mode 100644 index a5567c60..00000000 --- a/Unity.Entities.Tests/ForEachTests.cs.meta +++ /dev/null @@ -1,11 +0,0 @@ -fileFormatVersion: 2 -guid: fbe615a9e67b4d049bdc94d3655cf536 -MonoImporter: - externalObjects: {} - serializedVersion: 2 - defaultReferences: [] - executionOrder: 0 - icon: {instanceID: 0} - userData: - assetBundleName: - assetBundleVariant: diff --git a/Unity.Entities.Tests/IJobChunkTests.cs b/Unity.Entities.Tests/IJobChunkTests.cs index 37987ee2..a0cb2ea7 100644 --- a/Unity.Entities.Tests/IJobChunkTests.cs +++ b/Unity.Entities.Tests/IJobChunkTests.cs @@ -25,7 +25,7 @@ public void Execute(ArchetypeChunk chunk, int chunkIndex, int entityOffset) public void IJobChunkProcess() { var archetype = m_Manager.CreateArchetype(typeof(EcsTestData), typeof(EcsTestData2)); - var group = m_Manager.CreateComponentGroup(new EntityArchetypeQuery + var group = m_Manager.CreateEntityQuery(new EntityQueryDesc { Any = Array.Empty(), None = Array.Empty(), @@ -47,7 +47,7 @@ public void IJobChunkProcess() public void IJobChunkProcessFiltered() { var archetype = m_Manager.CreateArchetype(typeof(EcsTestData), typeof(EcsTestData2), typeof(SharedData1)); - var group = m_Manager.CreateComponentGroup(typeof(EcsTestData), typeof(EcsTestData2), typeof(SharedData1)); + var group = m_Manager.CreateEntityQuery(typeof(EcsTestData), typeof(EcsTestData2), typeof(SharedData1)); var entity1 = m_Manager.CreateEntity(archetype); var entity2 = m_Manager.CreateEntity(archetype); @@ -76,7 +76,7 @@ public void IJobChunkProcessFiltered() public void IJobChunkWithEntityOffsetCopy() { var archetype = m_Manager.CreateArchetype(typeof(EcsTestData), typeof(EcsTestData2)); - var group = m_Manager.CreateComponentGroup(typeof(EcsTestData), typeof(EcsTestData2)); + var group = m_Manager.CreateEntityQuery(typeof(EcsTestData), typeof(EcsTestData2)); var entities = new NativeArray(50000, Allocator.Temp); m_Manager.CreateEntity(archetype, entities); @@ -127,7 +127,7 @@ public void Execute(ArchetypeChunk chunk, int chunkIndex, int entityOffset) public void IJobChunkProcessChunkIndex() { var archetype = m_Manager.CreateArchetype(typeof(EcsTestData), typeof(EcsTestData2), typeof(SharedData1)); - var group = m_Manager.CreateComponentGroup(typeof(EcsTestData), typeof(EcsTestData2), typeof(SharedData1)); + var group = m_Manager.CreateEntityQuery(typeof(EcsTestData), typeof(EcsTestData2), typeof(SharedData1)); var entity1 = m_Manager.CreateEntity(archetype); var entity2 = m_Manager.CreateEntity(archetype); @@ -160,7 +160,7 @@ public void IJobChunkProcessChunkIndex() public void IJobChunkProcessEntityOffset() { var archetype = m_Manager.CreateArchetype(typeof(EcsTestData), typeof(EcsTestData2), typeof(SharedData1)); - var group = m_Manager.CreateComponentGroup(typeof(EcsTestData), typeof(EcsTestData2), typeof(SharedData1)); + var group = m_Manager.CreateEntityQuery(typeof(EcsTestData), typeof(EcsTestData2), typeof(SharedData1)); var entity1 = m_Manager.CreateEntity(archetype); var entity2 = m_Manager.CreateEntity(archetype); @@ -203,7 +203,7 @@ public void IJobChunkProcessChunkMultiArchetype() var entityC = m_Manager.CreateEntity(archetypeC); - var group = m_Manager.CreateComponentGroup(typeof(EcsTestData)); + var group = m_Manager.CreateEntityQuery(typeof(EcsTestData)); m_Manager.SetComponentData(entity1A, new EcsTestData { value = -1 }); m_Manager.SetComponentData(entity2A, new EcsTestData { value = -1 }); @@ -222,5 +222,75 @@ public void IJobChunkProcessChunkMultiArchetype() group.Dispose(); } + + #if !UNITY_ZEROPLAYER + struct ProcessChunkWriteIndex : IJobChunk + { + public ArchetypeChunkComponentType ecsTestType; + + public void Execute(ArchetypeChunk chunk, int chunkIndex, int entityOffset) + { + var testDataArray = chunk.GetNativeArray(ecsTestType); + for (int i = 0; i < chunk.Count; ++i) + { + testDataArray[i] = new EcsTestData + { + value = entityOffset + i + }; + } + } + } + + struct ForEachComponentData : IJobForEachWithEntity + { + public void Execute(Entity entity, int index, ref EcsTestData c0) + { + c0 = new EcsTestData { value = index }; + } + } + + [Test] + public void FilteredIJobChunkProcessesSameChunksAsFilteredJobForEach() + { + var archetypeA = m_Manager.CreateArchetype(typeof(EcsTestData), typeof(EcsTestSharedComp)); + var archetypeB = m_Manager.CreateArchetype(typeof(EcsTestData), typeof(EcsTestData2), typeof(EcsTestSharedComp)); + + var entitiesA = new NativeArray(5000, Allocator.Temp); + m_Manager.CreateEntity(archetypeA, entitiesA); + + var entitiesB = new NativeArray(5000, Allocator.Temp); + m_Manager.CreateEntity(archetypeA, entitiesB); + + for (int i = 0; i < 5000; ++i) + { + m_Manager.SetSharedComponentData(entitiesA[i], new EcsTestSharedComp { value = i % 8 }); + m_Manager.SetSharedComponentData(entitiesB[i], new EcsTestSharedComp { value = i % 8 }); + } + + var group = m_Manager.CreateEntityQuery(typeof(EcsTestData), typeof(EcsTestSharedComp)); + + group.SetFilter(new EcsTestSharedComp { value = 1 }); + + var jobChunk = new ProcessChunkWriteIndex + { + ecsTestType = m_Manager.GetArchetypeChunkComponentType(false) + }; + jobChunk.Schedule(group).Complete(); + + var componentArrayA = group.ToComponentDataArray(Allocator.TempJob); + + var jobProcess = new ForEachComponentData + {}; + jobProcess.Schedule(group).Complete(); + + var componentArrayB = group.ToComponentDataArray(Allocator.TempJob); + + CollectionAssert.AreEqual(componentArrayA.ToArray(), componentArrayB.ToArray()); + + componentArrayA.Dispose(); + componentArrayB.Dispose(); + group.Dispose(); + } +#endif } } diff --git a/Unity.Entities.Tests/IJobProcessComponentDataCombinationsTests.cs b/Unity.Entities.Tests/IJobForEachCombinationsTests.cs similarity index 83% rename from Unity.Entities.Tests/IJobProcessComponentDataCombinationsTests.cs rename to Unity.Entities.Tests/IJobForEachCombinationsTests.cs index 15658523..f4c4c298 100644 --- a/Unity.Entities.Tests/IJobProcessComponentDataCombinationsTests.cs +++ b/Unity.Entities.Tests/IJobForEachCombinationsTests.cs @@ -1,12 +1,11 @@ -#if !UNITY_ZEROPLAYER -using NUnit.Framework; +using NUnit.Framework; using Unity.Collections; namespace Unity.Entities.Tests { - class IJobProcessComponentDataCombinationsTests : ECSTestsFixture + class IJobForEachCombinationsTests : ECSTestsFixture { - struct Process1 : IJobProcessComponentData + struct Process1 : IJobForEach { public void Execute(ref EcsTestData value) { @@ -14,7 +13,7 @@ public void Execute(ref EcsTestData value) } } - struct Process2 : IJobProcessComponentData + struct Process2 : IJobForEach { public void Execute([ReadOnly]ref EcsTestData src, ref EcsTestData2 dst) { @@ -22,7 +21,7 @@ public void Execute([ReadOnly]ref EcsTestData src, ref EcsTestData2 dst) } } - struct Process3 : IJobProcessComponentData + struct Process3 : IJobForEach { public void Execute([ReadOnly]ref EcsTestData src, ref EcsTestData2 dst1, ref EcsTestData3 dst2) { @@ -30,7 +29,7 @@ public void Execute([ReadOnly]ref EcsTestData src, ref EcsTestData2 dst1, ref Ec } } - struct Process4 : IJobProcessComponentData + struct Process4 : IJobForEach { public void Execute([ReadOnly]ref EcsTestData src, ref EcsTestData2 dst1, ref EcsTestData3 dst2, ref EcsTestData4 dst3) { @@ -38,7 +37,7 @@ public void Execute([ReadOnly]ref EcsTestData src, ref EcsTestData2 dst1, ref Ec } } - struct Process1Entity : IJobProcessComponentDataWithEntity + struct Process1Entity : IJobForEachWithEntity { public void Execute(Entity entity, int index, ref EcsTestData value) { @@ -46,7 +45,7 @@ public void Execute(Entity entity, int index, ref EcsTestData value) } } - struct Process2Entity : IJobProcessComponentDataWithEntity + struct Process2Entity : IJobForEachWithEntity { public void Execute(Entity entity, int index, [ReadOnly]ref EcsTestData src, ref EcsTestData2 dst) { @@ -54,7 +53,7 @@ public void Execute(Entity entity, int index, [ReadOnly]ref EcsTestData src, ref } } - struct Process3Entity : IJobProcessComponentDataWithEntity + struct Process3Entity : IJobForEachWithEntity { public void Execute(Entity entity, int index, [ReadOnly]ref EcsTestData src, ref EcsTestData2 dst1, ref EcsTestData3 dst2) { @@ -62,7 +61,7 @@ public void Execute(Entity entity, int index, [ReadOnly]ref EcsTestData src, ref } } - struct Process4Entity : IJobProcessComponentDataWithEntity + struct Process4Entity : IJobForEachWithEntity { public void Execute(Entity entity, int index, [ReadOnly]ref EcsTestData src, ref EcsTestData2 dst1, ref EcsTestData3 dst2, ref EcsTestData4 dst3) { @@ -77,14 +76,16 @@ public enum ProcessMode Run } - void Schedule(ProcessMode mode) where T : struct, JobProcessComponentDataExtensions.IBaseJobProcessComponentData + void Schedule(ProcessMode mode) where T : struct, JobForEachExtensions.IBaseJobForEach { +#if !UNITY_ZEROPLAYER if (mode == ProcessMode.Parallel) new T().Schedule(EmptySystem).Complete(); else if (mode == ProcessMode.Run) new T().Run(EmptySystem); else new T().ScheduleSingle(EmptySystem).Complete(); +#endif } NativeArray PrepareData(int entityCount) @@ -138,14 +139,17 @@ public void JobProcessStress_1([Values]ProcessMode mode, [Values(0, 1, 1000)]int var entities = new NativeArray(entityCount, Allocator.Temp); m_Manager.CreateEntity(archetype, entities); +#if UNITY_ZEROPLAYER + (new Process1()).Schedule(EmptySystem); +#else Schedule(mode); - +#endif for (int i = 0; i < entities.Length; i++) Assert.AreEqual(1, m_Manager.GetComponentData(entities[i]).value); entities.Dispose(); } - + [Test] public void JobProcessStress_1_WithEntity([Values]ProcessMode mode, [Values(0, 1, 1000)]int entityCount) { @@ -153,7 +157,11 @@ public void JobProcessStress_1_WithEntity([Values]ProcessMode mode, [Values(0, 1 var entities = new NativeArray(entityCount, Allocator.Temp); m_Manager.CreateEntity(archetype, entities); +#if UNITY_ZEROPLAYER + (new Process1Entity()).Schedule(EmptySystem); +#else Schedule(mode); +#endif for (int i = 0; i < entities.Length; i++) Assert.AreEqual(i + entities[i].Index, m_Manager.GetComponentData(entities[i]).value); @@ -166,8 +174,11 @@ public void JobProcessStress_2([Values]ProcessMode mode, [Values(0, 1, 1000)]int { var entities = PrepareData(entityCount); +#if UNITY_ZEROPLAYER + (new Process2()).Schedule(EmptySystem); +#else Schedule(mode); - +#endif CheckResultsAndDispose(entities, 2, false); } @@ -176,7 +187,11 @@ public void JobProcessStress_2_WithEntity([Values]ProcessMode mode, [Values(0, 1 { var entities = PrepareData(entityCount); +#if UNITY_ZEROPLAYER + (new Process2Entity()).Schedule(EmptySystem); +#else Schedule(mode); +#endif CheckResultsAndDispose(entities, 2, true); @@ -187,7 +202,11 @@ public void JobProcessStress_3([Values]ProcessMode mode, [Values(0, 1, 1000)]int { var entities = PrepareData(entityCount); +#if UNITY_ZEROPLAYER + (new Process3()).Schedule(EmptySystem); +#else Schedule(mode); +#endif CheckResultsAndDispose(entities, 3, false); } @@ -197,8 +216,11 @@ public void JobProcessStress_3_WithEntity([Values]ProcessMode mode, [Values(0, 1 { var entities = PrepareData(entityCount); +#if UNITY_ZEROPLAYER + (new Process3Entity()).Schedule(EmptySystem); +#else Schedule(mode); - +#endif CheckResultsAndDispose(entities, 3, true); } @@ -207,8 +229,11 @@ public void JobProcessStress_4([Values]ProcessMode mode, [Values(0, 1, 1000)]int { var entities = PrepareData(entityCount); +#if UNITY_ZEROPLAYER + (new Process4()).Schedule(EmptySystem); +#else Schedule(mode); - +#endif CheckResultsAndDispose(entities, 4, false); } @@ -217,10 +242,12 @@ public void JobProcessStress_4_WithEntity([Values]ProcessMode mode, [Values(0, 1 { var entities = PrepareData(entityCount); +#if UNITY_ZEROPLAYER + (new Process4Entity()).Schedule(EmptySystem); +#else Schedule(mode); - +#endif CheckResultsAndDispose(entities, 4, true); } } } -#endif \ No newline at end of file diff --git a/Unity.Entities.Tests/IJobProcessComponentDataCombinationsTests.cs.meta b/Unity.Entities.Tests/IJobForEachCombinationsTests.cs.meta similarity index 100% rename from Unity.Entities.Tests/IJobProcessComponentDataCombinationsTests.cs.meta rename to Unity.Entities.Tests/IJobForEachCombinationsTests.cs.meta diff --git a/Unity.Entities.Tests/IJobProcessComponentDataTests.cs b/Unity.Entities.Tests/IJobForEachTests.cs similarity index 78% rename from Unity.Entities.Tests/IJobProcessComponentDataTests.cs rename to Unity.Entities.Tests/IJobForEachTests.cs index d2298ef7..20fd6f15 100644 --- a/Unity.Entities.Tests/IJobProcessComponentDataTests.cs +++ b/Unity.Entities.Tests/IJobForEachTests.cs @@ -6,9 +6,9 @@ namespace Unity.Entities.Tests { - public class IJobProcessComponentDataTests :ECSTestsFixture + public class IJobForEachTests :ECSTestsFixture { - struct Process1 : IJobProcessComponentData + struct Process1 : IJobForEach { public void Execute(ref EcsTestData dst) { @@ -16,7 +16,7 @@ public void Execute(ref EcsTestData dst) } } - struct Process2 : IJobProcessComponentData + struct Process2 : IJobForEach { public void Execute([ReadOnly]ref EcsTestData src, ref EcsTestData2 dst) { @@ -24,14 +24,14 @@ public void Execute([ReadOnly]ref EcsTestData src, ref EcsTestData2 dst) } } - struct Process3Entity : IJobProcessComponentDataWithEntity + struct Process3Entity : IJobForEachWithEntity { public void Execute(Entity entity, int index, [ReadOnly]ref EcsTestData src, ref EcsTestData2 dst1, ref EcsTestData3 dst2) { dst1.value1 = dst2.value2 = src.value + index + entity.Index; } } - + [Test] public void JobProcessSimple() { @@ -49,11 +49,11 @@ public void JobProcessComponentGroupCorrect() ComponentType[] expectedTypes = { ComponentType.ReadOnly(), ComponentType.ReadWrite() }; new Process2().Run(EmptySystem); - var group = EmptySystem.GetComponentGroup(expectedTypes); + var group = EmptySystem.GetEntityQuery(expectedTypes); - Assert.AreEqual(1, EmptySystem.ComponentGroups.Length); - Assert.IsTrue(EmptySystem.ComponentGroups[0].CompareComponents(expectedTypes)); - Assert.AreEqual(group, EmptySystem.ComponentGroups[0]); + Assert.AreEqual(1, EmptySystem.EntityQueries.Length); + Assert.IsTrue(EmptySystem.EntityQueries[0].CompareComponents(expectedTypes)); + Assert.AreEqual(group, EmptySystem.EntityQueries[0]); } @@ -72,11 +72,11 @@ public void JobProcessComponentGroupCorrectNativeArrayOfComponentTypes() componentTypes[0] = ComponentType.ReadOnly(componentTypes[0].TypeIndex); - var group = EmptySystem.GetComponentGroup(componentTypes); + var group = EmptySystem.GetEntityQuery(componentTypes); - Assert.AreEqual(1, EmptySystem.ComponentGroups.Length); - Assert.IsTrue(EmptySystem.ComponentGroups[0].CompareComponents(componentTypes)); - Assert.AreEqual(group, EmptySystem.ComponentGroups[0]); + Assert.AreEqual(1, EmptySystem.EntityQueries.Length); + Assert.IsTrue(EmptySystem.EntityQueries[0].CompareComponents(componentTypes)); + Assert.AreEqual(group, EmptySystem.EntityQueries[0]); componentTypes.Dispose(); } @@ -87,11 +87,11 @@ public void JobProcessComponentWithEntityGroupCorrect() ComponentType[] expectedTypes = { ComponentType.ReadOnly(), ComponentType.ReadWrite(), ComponentType.ReadWrite() }; new Process3Entity().Run(EmptySystem); - var group = EmptySystem.GetComponentGroup(expectedTypes); + var group = EmptySystem.GetEntityQuery(expectedTypes); - Assert.AreEqual(1, EmptySystem.ComponentGroups.Length); - Assert.IsTrue(EmptySystem.ComponentGroups[0].CompareComponents(expectedTypes)); - Assert.AreEqual(group, EmptySystem.ComponentGroups[0]); + Assert.AreEqual(1, EmptySystem.EntityQueries.Length); + Assert.IsTrue(EmptySystem.EntityQueries[0].CompareComponents(expectedTypes)); + Assert.AreEqual(group, EmptySystem.EntityQueries[0]); } @@ -106,10 +106,10 @@ protected override JobHandle OnUpdate(JobHandle inputDeps) } } [Test] - public void MultipleJobProcessComponentDataCanChain() + public void MultipleJobForEachCanChain() { var entity = m_Manager.CreateEntity(typeof(EcsTestData), typeof(EcsTestData2)); - var system = World.GetOrCreateManager(); + var system = World.GetOrCreateSystem(); system.Update(); Assert.AreEqual(7, m_Manager.GetComponentData(entity).value1); } @@ -133,7 +133,7 @@ public void JobWithMissingDependency() [ExcludeComponent(typeof(EcsTestData3))] [RequireComponentTag(typeof(EcsTestData4))] - struct ProcessTagged : IJobProcessComponentData + struct ProcessTagged : IJobForEach { public void Execute(ref EcsTestData src, ref EcsTestData2 dst) { @@ -165,7 +165,7 @@ public void JobProcessAdditionalRequirements() var entityProcess = m_Manager.CreateEntity(typeof(EcsTestData), typeof(EcsTestData2), typeof(EcsTestData4)); Test(true, entityProcess); } - struct ProcessFilteredData : IJobProcessComponentData + struct ProcessFilteredData : IJobForEach { public void Execute(ref EcsTestData c0) { @@ -174,7 +174,7 @@ public void Execute(ref EcsTestData c0) } [Test] - public void JobProcessWithFilteredComponentGroup() + public void JobProcessWithFilteredEntityQuery() { var archetype = m_Manager.CreateArchetype(typeof(EcsTestData), typeof(EcsTestSharedComp)); @@ -186,11 +186,11 @@ public void JobProcessWithFilteredComponentGroup() m_Manager.SetSharedComponentData(entityInGroupA, new EcsTestSharedComp { value = 1} ); m_Manager.SetSharedComponentData(entityInGroupB, new EcsTestSharedComp { value = 2} ); - var group = EmptySystem.GetComponentGroup(typeof(EcsTestData), typeof(EcsTestSharedComp)); + var group = EmptySystem.GetEntityQuery(typeof(EcsTestData), typeof(EcsTestSharedComp)); group.SetFilter(new EcsTestSharedComp { value = 1}); var processJob = new ProcessFilteredData(); - processJob.ScheduleGroup(group).Complete(); + processJob.Schedule(group).Complete(); Assert.AreEqual(10, m_Manager.GetComponentData(entityInGroupA).value); Assert.AreEqual(5, m_Manager.GetComponentData(entityInGroupB).value); @@ -214,9 +214,19 @@ public void JobCalculateEntityCount() job2.Schedule(EmptySystem).Complete(); } + [Test] + public void JobExcludedComponentExplicitQuery() + { + var group = EmptySystem.GetEntityQuery(ComponentType.Exclude()); + + var handle = new JobHandle(); + Assert.Throws(() => handle = new Process1().Schedule(group)); + handle.Complete(); + } + [Test] [Ignore("TODO")] - public void TestCoverageFor_ComponentSystemBase_InjectNestedIJobProcessComponentDataJobs() + public void TestCoverageFor_ComponentSystemBase_InjectNestedIJobForEachJobs() { } @@ -227,4 +237,4 @@ public void DuplicateComponentTypeParametersThrows() } } } -#endif \ No newline at end of file +#endif diff --git a/Unity.Entities.Tests/IJobProcessComponentDataTests.cs.meta b/Unity.Entities.Tests/IJobForEachTests.cs.meta similarity index 100% rename from Unity.Entities.Tests/IJobProcessComponentDataTests.cs.meta rename to Unity.Entities.Tests/IJobForEachTests.cs.meta diff --git a/Unity.Entities.Tests/IJobProcessComponentInjection.cs b/Unity.Entities.Tests/IJobProcessComponentInjection.cs index 14ca80ff..49f50358 100644 --- a/Unity.Entities.Tests/IJobProcessComponentInjection.cs +++ b/Unity.Entities.Tests/IJobProcessComponentInjection.cs @@ -9,7 +9,7 @@ class IJobProcessComponentInjection : ECSTestsFixture [DisableAutoCreation] class TestSystem : JobComponentSystem { - private struct Process1 : IJobProcessComponentData + private struct Process1 : IJobForEach { public void Execute(ref EcsTestData value) { @@ -17,14 +17,14 @@ public void Execute(ref EcsTestData value) } } - public struct Process2 : IJobProcessComponentData + public struct Process2 : IJobForEach { public void Execute(ref EcsTestData src, ref EcsTestData2 dst) { dst.value1 = src.value; } } - + protected override JobHandle OnUpdate(JobHandle inputDeps) { inputDeps = new Process1().Schedule(this, inputDeps); @@ -34,12 +34,12 @@ protected override JobHandle OnUpdate(JobHandle inputDeps) } [Test] - public void NestedIJobProcessComponentDataAreInjectedDuringOnCreateManager() + public void NestedIJobForEachAreInjectedDuringOnCreate() { m_Manager.CreateEntity(typeof(EcsTestData), typeof(EcsTestData2)); - var system = World.GetOrCreateManager(); - Assert.AreEqual(2, system.ComponentGroups.Length); + var system = World.GetOrCreateSystem(); + Assert.AreEqual(2, system.EntityQueries.Length); } } } -#endif \ No newline at end of file +#endif diff --git a/Unity.Entities.Tests/InjectComponentGroupTests.cs b/Unity.Entities.Tests/InjectComponentGroupTests.cs deleted file mode 100644 index 5182d26e..00000000 --- a/Unity.Entities.Tests/InjectComponentGroupTests.cs +++ /dev/null @@ -1,339 +0,0 @@ -using NUnit.Framework; -using System; -using Unity.Collections; -using Unity.Jobs; - -#pragma warning disable 649 -// Injection is deprecated -#pragma warning disable 618 - -namespace Unity.Entities.Tests -{ - class InjectComponentGroupTests : ECSTestsFixture - { - [DisableAutoCreation] - [AlwaysUpdateSystem] - public class PureEcsTestSystem : ComponentSystem - { - public struct DataAndEntites - { - public ComponentDataArray Data; - public EntityArray Entities; - public readonly int Length; - } - - [Inject] - public DataAndEntites Group; - - protected override void OnUpdate() - { - } - } - - [DisableAutoCreation] - [AlwaysUpdateSystem] - public class PureReadOnlySystem : ComponentSystem - { - public struct Datas - { - [ReadOnly] - public ComponentDataArray Data; - } - - [Inject] - public Datas Group; - - protected override void OnUpdate() - { - } - } - - - - public struct SharedData : ISharedComponentData - { - public int value; - - public SharedData(int val) { value = val; } - } - - [DisableAutoCreation] - [AlwaysUpdateSystem] - public class SharedComponentSystem : ComponentSystem - { - public struct Datas - { - public ComponentDataArray Data; - [ReadOnly] public SharedComponentDataArray SharedData; - } - - [Inject] - public Datas Group; - - protected override void OnUpdate() - { - } - } - - [DisableAutoCreation] - [AlwaysUpdateSystem] - public class ComponentGroupAsPartOfInjectedGroupSystem : ComponentSystem - { - public struct GroupStruct0 - { - public ComponentDataArray Data; - public ComponentDataArray Data2; - public readonly int GroupIndex; - } - - public struct GroupStruct1 - { - public ComponentDataArray Data; - [ReadOnly] public SharedComponentDataArray Shared; - public readonly int GroupIndex; - } - - [Inject] - public GroupStruct0 Group0; - [Inject] - public GroupStruct1 Group1; - - protected override void OnCreateManager() - { - ComponentGroups[Group1.GroupIndex].SetFilter(new EcsTestSharedComp(123)); - } - - protected override void OnUpdate() - { - } - } - - [Test] - [StandaloneFixme] // ISharedComponentData - public void ComponentGroupFromInjectedGroup() - { - var system = World.GetOrCreateManager(); - - var entity0 = m_Manager.CreateEntity(typeof(EcsTestData), typeof(EcsTestSharedComp), typeof(EcsTestData2)); - var entity1 = m_Manager.CreateEntity(typeof(EcsTestData), typeof(EcsTestSharedComp)); - var entity2 = m_Manager.CreateEntity(typeof(EcsTestData), typeof(EcsTestSharedComp)); - - m_Manager.SetSharedComponentData(entity0, new EcsTestSharedComp(123)); - m_Manager.SetSharedComponentData(entity1, new EcsTestSharedComp(456)); - m_Manager.SetSharedComponentData(entity2, new EcsTestSharedComp(123)); - - var group0 = system.ComponentGroups[system.Group0.GroupIndex]; - var group1 = system.ComponentGroups[system.Group1.GroupIndex]; - - var data0 = group0.GetComponentDataArray(); - var data1 = group1.GetComponentDataArray(); - - Assert.AreEqual(1, data0.Length); - Assert.AreEqual(2, data1.Length); - } - - [Test] - [StandaloneFixme] // AtomicSafety - public void ReadOnlyComponentDataArray() - { - var readOnlySystem = World.GetOrCreateManager (); - - var go = m_Manager.CreateEntity (new ComponentType[0]); - m_Manager.AddComponentData (go, new EcsTestData(2)); - - readOnlySystem.Update (); - Assert.AreEqual (2, readOnlySystem.Group.Data[0].value); - Assert.Throws(()=> { readOnlySystem.Group.Data[0] = new EcsTestData(); }); - } - - - - [Test] - [StandaloneFixme] // ISharedComponentData - public void SharedComponentDataArray() - { - var sharedComponentSystem = World.GetOrCreateManager (); - - var go = m_Manager.CreateEntity(new ComponentType[0]); - m_Manager.AddComponentData (go, new EcsTestData(2)); - m_Manager.AddSharedComponentData(go, new SharedData(3)); - - sharedComponentSystem.Update (); - Assert.AreEqual (1, sharedComponentSystem.Group.Data.Length); - Assert.AreEqual (2, sharedComponentSystem.Group.Data[0].value); - Assert.AreEqual (3, sharedComponentSystem.Group.SharedData[0].value); - } - - - [Test] - [StandaloneFixme] // Real problem - Test failure - public void RemoveComponentGroupTracking() - { - var pureSystem = World.GetOrCreateManager (); - - var go0 = m_Manager.CreateEntity (new ComponentType[0]); - m_Manager.AddComponentData (go0, new EcsTestData(10)); - - var go1 = m_Manager.CreateEntity (); - m_Manager.AddComponentData (go1, new EcsTestData(20)); - - pureSystem.Update (); - Assert.AreEqual (2, pureSystem.Group.Length); - Assert.AreEqual (10, pureSystem.Group.Data[0].value); - Assert.AreEqual (20, pureSystem.Group.Data[1].value); - - m_Manager.RemoveComponent (go0); - - pureSystem.Update (); - Assert.AreEqual (1, pureSystem.Group.Length); - Assert.AreEqual (20, pureSystem.Group.Data[0].value); - - m_Manager.RemoveComponent (go1); - pureSystem.Update (); - Assert.AreEqual (0, pureSystem.Group.Length); - } - - [Test] - [StandaloneFixme] // Real problem - Test failure - public void EntityGroupTracking() - { - var pureSystem = World.GetOrCreateManager (); - - var go = m_Manager.CreateEntity (new ComponentType[0]); - m_Manager.AddComponentData (go, new EcsTestData(2)); - - pureSystem.Update (); - Assert.AreEqual (1, pureSystem.Group.Length); - Assert.AreEqual (1, pureSystem.Group.Data.Length); - Assert.AreEqual (1, pureSystem.Group.Entities.Length); - Assert.AreEqual (2, pureSystem.Group.Data[0].value); - Assert.AreEqual (go, pureSystem.Group.Entities[0]); - } - - [DisableAutoCreation] - public class FromEntitySystemIncrementInJob : JobComponentSystem - { - public struct IncrementValueJob : IJob - { - public Entity entity; - - public ComponentDataFromEntity ecsTestDataFromEntity; - public BufferFromEntity intArrayFromEntity; - - public void Execute() - { - DynamicBuffer array = intArrayFromEntity[entity]; - for (int i = 0; i < array.Length; i++) - array[i] = new EcsIntElement { Value = array[i].Value + 1 }; - - var value = ecsTestDataFromEntity[entity]; - value.value++; - ecsTestDataFromEntity[entity] = value; - } - } - -#pragma warning disable 649 - [Inject] - BufferFromEntity intArrayFromEntity; - - [Inject] - ComponentDataFromEntity ecsTestDataFromEntity; -#pragma warning restore 649 - - public Entity entity; - - protected override JobHandle OnUpdate(JobHandle inputDeps) - { - var job = new IncrementValueJob(); - job.entity = entity; - job.ecsTestDataFromEntity = ecsTestDataFromEntity; - job.intArrayFromEntity = intArrayFromEntity; - - return job.Schedule(inputDeps); - } - } - - [Test] - [StandaloneFixme] // IJob - public void FromEntitySystemIncrementInJobWorks() - { - var system = World.GetOrCreateManager(); - - var entity = m_Manager.CreateEntity(typeof(EcsTestData), typeof(EcsIntElement)); - - m_Manager.GetBuffer(entity).CopyFrom(new EcsIntElement[] { 0, -1, Int32.MinValue }); - - system.entity = entity; - system.Update(); - system.Update(); - - Assert.AreEqual(2, m_Manager.GetComponentData(entity).value); - Assert.AreEqual(2, m_Manager.GetBuffer(entity)[0].Value); - } - - [DisableAutoCreation] - public class OnCreateManagerComponentGroupInjectionSystem : JobComponentSystem - { - public struct Group - { - public ComponentDataArray Data; - } - - [Inject] - public Group group; - - protected override void OnCreateManager() - { - Assert.AreEqual(1, group.Data.Length); - Assert.AreEqual(42, group.Data[0].value); - } - - protected override JobHandle OnUpdate(JobHandle inputDeps) - { - return inputDeps; - } - } - - [Test] - [StandaloneFixme] // IJob - public void OnCreateManagerComponentGroupInjectionWorks() - { - var entity = m_Manager.CreateEntity (typeof(EcsTestData)); - m_Manager.SetComponentData(entity, new EcsTestData(42)); - World.GetOrCreateManager(); - } - - [DisableAutoCreation] - public class OnDestroyManagerComponentGroupInjectionSystem : JobComponentSystem - { - public struct Group - { - public ComponentDataArray Data; - } - - [Inject] - public Group group; - - protected override void OnDestroyManager() - { - Assert.AreEqual(1, group.Data.Length); - Assert.AreEqual(42, group.Data[0].value); - } - - protected override JobHandle OnUpdate(JobHandle inputDeps) - { - return inputDeps; - } - } - - [Test] - [StandaloneFixme] // IJob - public void OnDestroyManagerComponentGroupInjectionWorks() - { - var system = World.GetOrCreateManager(); - var entity = m_Manager.CreateEntity (typeof(EcsTestData)); - m_Manager.SetComponentData(entity, new EcsTestData(42)); - World.DestroyManager(system); - } - } -} diff --git a/Unity.Entities.Tests/IterationTests.cs b/Unity.Entities.Tests/IterationTests.cs index 20dff2bf..d5950d41 100644 --- a/Unity.Entities.Tests/IterationTests.cs +++ b/Unity.Entities.Tests/IterationTests.cs @@ -8,11 +8,11 @@ namespace Unity.Entities.Tests class IterationTests : ECSTestsFixture { [Test] - public void CreateComponentGroup() + public void CreateEntityQuery() { var archetype = m_Manager.CreateArchetype(typeof(EcsTestData), typeof(EcsTestData2)); - var group = m_Manager.CreateComponentGroup(typeof(EcsTestData), typeof(EcsTestData2)); + var group = m_Manager.CreateEntityQuery(typeof(EcsTestData), typeof(EcsTestData2)); Assert.AreEqual(0, group.CalculateLength()); var entity = m_Manager.CreateEntity(archetype); @@ -33,7 +33,7 @@ struct TempComponentNeverInstantiated : IComponentData [Test] public void IterateEmptyArchetype() { - var group = m_Manager.CreateComponentGroup(typeof(TempComponentNeverInstantiated)); + var group = m_Manager.CreateEntityQuery(typeof(TempComponentNeverInstantiated)); Assert.AreEqual(0, group.CalculateLength()); var archetype = m_Manager.CreateArchetype(typeof(TempComponentNeverInstantiated)); @@ -45,12 +45,12 @@ public void IterateEmptyArchetype() Assert.AreEqual(0, group.CalculateLength()); } [Test] - public void IterateChunkedComponentGroup() + public void IterateChunkedEntityQuery() { var archetype1 = m_Manager.CreateArchetype(typeof(EcsTestData)); var archetype2 = m_Manager.CreateArchetype(typeof(EcsTestData), typeof(EcsTestData2)); - var group = m_Manager.CreateComponentGroup(typeof(EcsTestData)); + var group = m_Manager.CreateEntityQuery(typeof(EcsTestData)); Assert.AreEqual(0, group.CalculateLength()); Entity[] entities = new Entity[10000]; @@ -87,7 +87,7 @@ public void IterateChunkedComponentGroupBackwards() var archetype1 = m_Manager.CreateArchetype(typeof(EcsTestData)); var archetype2 = m_Manager.CreateArchetype(typeof(EcsTestData), typeof(EcsTestData2)); - var group = m_Manager.CreateComponentGroup(typeof(EcsTestData)); + var group = m_Manager.CreateEntityQuery(typeof(EcsTestData)); Assert.AreEqual(0, group.CalculateLength()); Entity[] entities = new Entity[10000]; @@ -127,7 +127,7 @@ public void IterateChunkedComponentGroupAfterDestroy() var archetype1 = m_Manager.CreateArchetype(typeof(EcsTestData)); var archetype2 = m_Manager.CreateArchetype(typeof(EcsTestData), typeof(EcsTestData2)); - var group = m_Manager.CreateComponentGroup(typeof(EcsTestData)); + var group = m_Manager.CreateEntityQuery(typeof(EcsTestData)); Assert.AreEqual(0, group.CalculateLength()); Entity[] entities = new Entity[10000]; @@ -188,70 +188,6 @@ public void IterateChunkedComponentGroupAfterDestroy() } arr.Dispose(); } - -#pragma warning disable 618 - [Test] - public void IterateEntityArray() - { - var archetype1 = m_Manager.CreateArchetype(typeof(EcsTestData)); - var archetype2 = m_Manager.CreateArchetype(typeof(EcsTestData), typeof(EcsTestData2)); - - var group = m_Manager.CreateComponentGroup(typeof(EcsTestData)); - var arr = group.GetEntityArray(); - Assert.AreEqual(0, arr.Length); - - Entity[] entities = new Entity[10000]; - for (int i = 0; i < entities.Length/2;i++) - { - entities[i] = m_Manager.CreateEntity(archetype1); - m_Manager.SetComponentData(entities[i], new EcsTestData(i)); - } - for (int i = entities.Length/2; i < entities.Length;i++) - { - entities[i] = m_Manager.CreateEntity(archetype2); - m_Manager.SetComponentData(entities[i], new EcsTestData(i)); - } - - arr = group.GetEntityArray(); - Assert.AreEqual(entities.Length, arr.Length); - var values = new HashSet(); - for (int i = 0; i < arr.Length;i++) - { - Entity val = arr[i]; - Assert.IsFalse(values.Contains(val)); - values.Add(val); - } - - for (int i = 0; i < entities.Length;i++) - m_Manager.DestroyEntity(entities[i]); - } - - [Test] - public void ComponentDataArrayCopy() - { - var entity = m_Manager.CreateEntity(typeof(EcsTestData)); - - var entities = new NativeArray(20000, Allocator.Persistent); - m_Manager.Instantiate(entity, entities); - - var ecsArray = m_Manager.CreateComponentGroup(typeof(EcsTestData)).GetComponentDataArray(); - - for (int i = 0; i < ecsArray.Length; i++) - ecsArray[i] = new EcsTestData(i); - - var copied = new NativeArray(entities.Length - 11 + 1, Allocator.Persistent); - ecsArray.CopyTo(copied, 11); - - for (int i = 0; i < copied.Length; i++) - { - if (copied[i].value != i) - Assert.AreEqual(i + 11, copied[i].value); - } - - copied.Dispose(); - entities.Dispose(); - } -#pragma warning restore 618 [Test] public void GroupCopyFromNativeArray() @@ -274,7 +210,7 @@ public void GroupCopyFromNativeArray() } - var group = m_Manager.CreateComponentGroup(typeof(EcsTestData)); + var group = m_Manager.CreateEntityQuery(typeof(EcsTestData)); group.CopyFromComponentDataArray(dataToCopyA); for (int i = 0; i < dataToCopyA.Length; ++i) @@ -297,7 +233,7 @@ public void ComponentGroupFilteredEntityIndexWithMultipleArchetypes() var archetypeA = m_Manager.CreateArchetype(typeof(EcsTestData), typeof(EcsTestData2), typeof(EcsTestSharedComp)); var archetypeB = m_Manager.CreateArchetype(typeof(EcsTestData), typeof(EcsTestSharedComp)); - var group = m_Manager.CreateComponentGroup(typeof(EcsTestData), typeof(EcsTestSharedComp)); + var group = m_Manager.CreateEntityQuery(typeof(EcsTestData), typeof(EcsTestSharedComp)); var entity1A = m_Manager.CreateEntity(archetypeA); var entity2A = m_Manager.CreateEntity(archetypeA); @@ -314,7 +250,7 @@ public void ComponentGroupFilteredEntityIndexWithMultipleArchetypes() iterator.MoveToChunkWithoutFiltering(2); // 2 is index of chunk iterator.GetCurrentChunkRange(out var begin, out var end ); - Assert.AreEqual(1, begin); // 1 is index of entity in filtered ComponentGroup + Assert.AreEqual(1, begin); // 1 is index of entity in filtered EntityQuery group.Dispose(); } @@ -325,7 +261,7 @@ public void ComponentGroupFilteredChunkCount() { var archetypeA = m_Manager.CreateArchetype(typeof(EcsTestData), typeof(EcsTestData2), typeof(EcsTestSharedComp)); - var group = m_Manager.CreateComponentGroup(typeof(EcsTestData), typeof(EcsTestSharedComp)); + var group = m_Manager.CreateEntityQuery(typeof(EcsTestData), typeof(EcsTestSharedComp)); for (int i = 0; i < archetypeA.ChunkCapacity * 2; ++i) { diff --git a/Unity.Entities.Tests/JobComponentSystemDependencyTests.cs b/Unity.Entities.Tests/JobComponentSystemDependencyTests.cs index 99bd0d8da..8658cb86 100644 --- a/Unity.Entities.Tests/JobComponentSystemDependencyTests.cs +++ b/Unity.Entities.Tests/JobComponentSystemDependencyTests.cs @@ -11,9 +11,9 @@ class JobComponentSystemDependencyTests : ECSTestsFixture [DisableAutoCreation] public class ReadSystem1 : JobComponentSystem { - public ComponentGroup m_ReadGroup; + public EntityQuery m_ReadGroup; - struct ReadJob : IJobProcessComponentData + struct ReadJob : IJobForEach { public void Execute([ReadOnly]ref EcsTestData c0) { @@ -23,24 +23,24 @@ public void Execute([ReadOnly]ref EcsTestData c0) protected override JobHandle OnUpdate(JobHandle input) { var job = new ReadJob() {}; - return job.ScheduleGroup(m_ReadGroup); + return job.Schedule(m_ReadGroup); } - protected override void OnCreateManager() + protected override void OnCreate() { - m_ReadGroup = GetComponentGroup(ComponentType.ReadOnly()); + m_ReadGroup = GetEntityQuery(ComponentType.ReadOnly()); } } [DisableAutoCreation] public class ReadSystem2 : JobComponentSystem { - public ComponentGroup m_ReadGroup; + public EntityQuery m_ReadGroup; public bool returnWrongJob = false; public bool ignoreInputDeps = false; - private struct ReadJob : IJobProcessComponentData + private struct ReadJob : IJobForEach { public void Execute([ReadOnly]ref EcsTestData c0) { @@ -55,46 +55,46 @@ protected override JobHandle OnUpdate(JobHandle input) if (ignoreInputDeps) { - h = job.ScheduleGroup(m_ReadGroup); + h = job.Schedule(m_ReadGroup); } else { - h = job.ScheduleGroup(m_ReadGroup, input); + h = job.Schedule(m_ReadGroup, input); } return returnWrongJob ? input : h; } - protected override void OnCreateManager() + protected override void OnCreate() { - m_ReadGroup = GetComponentGroup(ComponentType.ReadOnly()); + m_ReadGroup = GetEntityQuery(ComponentType.ReadOnly()); } } [DisableAutoCreation] public class ReadSystem3 : JobComponentSystem { - public ComponentGroup m_ReadGroup; + public EntityQuery m_ReadGroup; protected override JobHandle OnUpdate(JobHandle input) { return input; } - protected override void OnCreateManager() + protected override void OnCreate() { - m_ReadGroup = GetComponentGroup(ComponentType.ReadOnly()); + m_ReadGroup = GetEntityQuery(ComponentType.ReadOnly()); } } [DisableAutoCreation] public class WriteSystem : JobComponentSystem { - public ComponentGroup m_WriteGroup; + public EntityQuery m_WriteGroup; public bool SkipJob = false; - private struct WriteJob : IJobProcessComponentData + private struct WriteJob : IJobForEach { public void Execute(ref EcsTestData c0) { @@ -106,7 +106,7 @@ protected override JobHandle OnUpdate(JobHandle input) if (!SkipJob) { var job = new WriteJob() {}; - return job.ScheduleGroup(m_WriteGroup); + return job.Schedule(m_WriteGroup); } else { @@ -114,9 +114,9 @@ protected override JobHandle OnUpdate(JobHandle input) } } - protected override void OnCreateManager() + protected override void OnCreate() { - m_WriteGroup = GetComponentGroup(ComponentType.ReadWrite()); + m_WriteGroup = GetEntityQuery(ComponentType.ReadWrite()); } } @@ -125,8 +125,8 @@ public void ReturningWrongJobThrowsInCorrectSystemUpdate() { var entity = m_Manager.CreateEntity (typeof(EcsTestData)); m_Manager.SetComponentData(entity, new EcsTestData(42)); - ReadSystem1 rs1 = World.GetOrCreateManager(); - ReadSystem2 rs2 = World.GetOrCreateManager(); + ReadSystem1 rs1 = World.GetOrCreateSystem(); + ReadSystem2 rs2 = World.GetOrCreateSystem(); rs2.returnWrongJob = true; @@ -139,8 +139,8 @@ public void IgnoredInputDepsThrowsInCorrectSystemUpdate() { var entity = m_Manager.CreateEntity (typeof(EcsTestData)); m_Manager.SetComponentData(entity, new EcsTestData(42)); - WriteSystem ws1 = World.GetOrCreateManager(); - ReadSystem2 rs2 = World.GetOrCreateManager(); + WriteSystem ws1 = World.GetOrCreateSystem(); + ReadSystem2 rs2 = World.GetOrCreateSystem(); rs2.ignoreInputDeps = true; @@ -153,7 +153,7 @@ public void NotSchedulingWriteJobIsHarmless() { var entity = m_Manager.CreateEntity (typeof(EcsTestData)); m_Manager.SetComponentData(entity, new EcsTestData(42)); - WriteSystem ws1 = World.GetOrCreateManager(); + WriteSystem ws1 = World.GetOrCreateSystem(); ws1.Update(); ws1.SkipJob = true; @@ -165,20 +165,20 @@ public void NotUsingDataIsHarmless() { var entity = m_Manager.CreateEntity (typeof(EcsTestData)); m_Manager.SetComponentData(entity, new EcsTestData(42)); - ReadSystem1 rs1 = World.GetOrCreateManager(); - ReadSystem3 rs3 = World.GetOrCreateManager(); + ReadSystem1 rs1 = World.GetOrCreateSystem(); + ReadSystem3 rs3 = World.GetOrCreateSystem(); rs1.Update(); rs3.Update(); } [Test] - public void ReadAfterWrite_JobProcessComponentDataGroup_Works() + public void ReadAfterWrite_JobForEachGroup_Works() { var entity = m_Manager.CreateEntity (typeof(EcsTestData)); m_Manager.SetComponentData(entity, new EcsTestData(42)); - var ws = World.GetOrCreateManager(); - var rs = World.GetOrCreateManager(); + var ws = World.GetOrCreateSystem(); + var rs = World.GetOrCreateSystem(); ws.Update(); rs.Update(); @@ -205,13 +205,13 @@ protected override JobHandle OnUpdate(JobHandle dep) } // The writer dependency on EcsTestData is not predeclared during - // OnCreateManager, but we still expect the code to work correctly. + // OnCreate, but we still expect the code to work correctly. // This should result in a sync point when adding the dependency for the first time. [Test] public void AddingDependencyTypeDuringOnUpdateSyncsDependency() { - var systemA = World.CreateManager(); - var systemB = World.CreateManager(); + var systemA = World.CreateSystem(); + var systemB = World.CreateSystem(); systemA.Update(); systemB.Update(); @@ -241,7 +241,7 @@ protected override JobHandle OnUpdate(JobHandle dep) var handle = new EmptyJob { TestDataType = GetArchetypeChunkComponentType() - }.Schedule(m_EntityManager.UniversalGroup, dep); + }.Schedule(m_EntityManager.UniversalQuery, dep); return handle; } } @@ -251,8 +251,8 @@ public void EmptySystemAfterNonEmptySystemDoesntThrow() { m_Manager.CreateEntity(typeof(EcsTestData)); - var systemA = World.CreateManager(); - var systemB = World.CreateManager(); + var systemA = World.CreateSystem(); + var systemB = World.CreateSystem(); systemA.Update(); systemB.Update(); diff --git a/Unity.Entities.Tests/JobSafetyTests.cs b/Unity.Entities.Tests/JobSafetyTests.cs index 202057ec..c2b9980f 100644 --- a/Unity.Entities.Tests/JobSafetyTests.cs +++ b/Unity.Entities.Tests/JobSafetyTests.cs @@ -35,7 +35,7 @@ public void Execute() [Test] public void ComponentAccessAfterScheduledJobThrows() { - var group = m_Manager.CreateComponentGroup(typeof(EcsTestData)); + var group = m_Manager.CreateEntityQuery(typeof(EcsTestData)); var entity = m_Manager.CreateEntity(typeof(EcsTestData)); m_Manager.SetComponentData(entity, new EcsTestData(42)); @@ -64,7 +64,7 @@ public void ComponentAccessAfterScheduledJobThrows() public void GetComponentCompletesJob() { var entity = m_Manager.CreateEntity(typeof(EcsTestData)); - var group = m_Manager.CreateComponentGroup(typeof(EcsTestData)); + var group = m_Manager.CreateEntityQuery(typeof(EcsTestData)); var job = new TestIncrementJob(); job.entities = group.ToEntityArray(Allocator.TempJob); @@ -82,7 +82,7 @@ public void DestroyEntityCompletesScheduledJobs() { var entity = m_Manager.CreateEntity(typeof(EcsTestData)); /*var entity2 =*/ m_Manager.CreateEntity(typeof(EcsTestData)); - var group = m_Manager.CreateComponentGroup(typeof(EcsTestData)); + var group = m_Manager.CreateEntityQuery(typeof(EcsTestData)); var job = new TestIncrementJob(); job.entities = group.ToEntityArray(Allocator.TempJob); @@ -110,7 +110,7 @@ public void EntityManagerDestructionDetectsUnregisteredJob() #endif /*var entity =*/ m_Manager.CreateEntity(typeof(EcsTestData)); - var group = m_Manager.CreateComponentGroup(typeof(EcsTestData)); + var group = m_Manager.CreateEntityQuery(typeof(EcsTestData)); var job = new TestIncrementJob(); job.entities = group.ToEntityArray(Allocator.TempJob); @@ -126,7 +126,7 @@ public void EntityManagerDestructionDetectsUnregisteredJob() public void DestroyEntityDetectsUnregisteredJob() { var entity = m_Manager.CreateEntity(typeof(EcsTestData)); - var group = m_Manager.CreateComponentGroup(typeof(EcsTestData)); + var group = m_Manager.CreateEntityQuery(typeof(EcsTestData)); var job = new TestIncrementJob(); job.entities = group.ToEntityArray(Allocator.TempJob); @@ -143,7 +143,7 @@ public void DestroyEntityDetectsUnregisteredJob() public void GetComponentDetectsUnregisteredJob() { var entity = m_Manager.CreateEntity(typeof(EcsTestData)); - var group = m_Manager.CreateComponentGroup(typeof(EcsTestData)); + var group = m_Manager.CreateEntityQuery(typeof(EcsTestData)); var job = new TestIncrementJob(); job.entities = group.ToEntityArray(Allocator.TempJob); @@ -185,7 +185,7 @@ class EntityOnlyDependencySystem : JobComponentSystem protected override JobHandle OnUpdate(JobHandle inputDeps) { EntityManager.CreateEntity(typeof(EcsTestData)); - var group = GetComponentGroup(new ComponentType[]{}); + var group = GetEntityQuery(new ComponentType[]{}); var job = new EntityOnlyDependencyJob { entityType = m_EntityManager.GetArchetypeChunkEntityType() @@ -202,7 +202,7 @@ class NoComponentDependenciesSystem : JobComponentSystem protected override JobHandle OnUpdate(JobHandle inputDeps) { EntityManager.CreateEntity(typeof(EcsTestData)); - var group = GetComponentGroup(new ComponentType[]{}); + var group = GetEntityQuery(new ComponentType[]{}); var job = new NoDependenciesJob{}; JobHandle = job.Schedule(group, inputDeps); @@ -225,18 +225,18 @@ protected override JobHandle OnUpdate(JobHandle inputDeps) [Test] public void StructuralChangeCompletesEntityOnlyDependencyJob() { - var system = World.GetOrCreateManager(); + var system = World.GetOrCreateSystem(); system.Update(); - World.GetOrCreateManager().Update(); + World.GetOrCreateSystem().Update(); Assert.IsTrue(JobHandle.CheckFenceIsDependencyOrDidSyncFence(system.JobHandle, new JobHandle())); } [Test] public void StructuralChangeCompletesNoComponentDependenciesJob() { - var system = World.GetOrCreateManager(); + var system = World.GetOrCreateSystem(); system.Update(); - World.GetOrCreateManager().Update(); + World.GetOrCreateSystem().Update(); Assert.IsTrue(JobHandle.CheckFenceIsDependencyOrDidSyncFence(system.JobHandle, new JobHandle())); } @@ -245,7 +245,7 @@ public void StructuralChangeAfterSchedulingNoDependenciesJobThrows() { var archetype = m_Manager.CreateArchetype(typeof(EcsTestData)); var entity = m_Manager.CreateEntity(archetype); - var group = EmptySystem.GetComponentGroup(typeof(EcsTestData)); + var group = EmptySystem.GetEntityQuery(typeof(EcsTestData)); var handle = new NoDependenciesJob().Schedule(group); Assert.Throws(() => m_Manager.DestroyEntity(entity)); handle.Complete(); @@ -256,7 +256,7 @@ public void StructuralChangeAfterSchedulingEntityOnlyDependencyJobThrows() { var archetype = m_Manager.CreateArchetype(typeof(EcsTestData)); var entity = m_Manager.CreateEntity(archetype); - var group = EmptySystem.GetComponentGroup(typeof(EcsTestData)); + var group = EmptySystem.GetEntityQuery(typeof(EcsTestData)); var handle = new EntityOnlyDependencyJob{entityType = m_Manager.GetArchetypeChunkEntityType()}.Schedule(group); Assert.Throws(() => m_Manager.DestroyEntity(entity)); handle.Complete(); @@ -265,9 +265,9 @@ public void StructuralChangeAfterSchedulingEntityOnlyDependencyJobThrows() [DisableAutoCreation] class SharedComponentSystem : JobComponentSystem { - ComponentGroup group; - protected override void OnCreateManager() { - group = GetComponentGroup(ComponentType.ReadOnly()); + EntityQuery group; + protected override void OnCreate() { + group = GetEntityQuery(ComponentType.ReadOnly()); } struct SharedComponentJobChunk : IJobChunk { @@ -292,7 +292,7 @@ public void JobsUsingArchetypeChunkSharedComponentTypeSyncOnStructuralChange() var archetype = m_Manager.CreateArchetype(typeof(EcsTestData), typeof(EcsTestSharedComp)); var entity = m_Manager.CreateEntity(archetype); - var sharedComponentSystem = World.CreateManager(); + var sharedComponentSystem = World.CreateSystem(); sharedComponentSystem.Update(); // DestroyEntity should sync the job and not cause any safety error diff --git a/Unity.Entities.Tests/LockedEntityTests.cs b/Unity.Entities.Tests/LockedEntityTests.cs index 8cdf4524..cb5f6f9a 100644 --- a/Unity.Entities.Tests/LockedEntityTests.cs +++ b/Unity.Entities.Tests/LockedEntityTests.cs @@ -73,13 +73,13 @@ public void LockedEntityRestrictions() m_Manager.LockChunk(chunk); Assert.Throws(() => m_Manager.AddComponentData(entity, new EcsTestTag()) ); - Assert.Throws(() => m_Manager.AddComponent(m_Manager.UniversalGroup, typeof(EcsTestTag)) ); + Assert.Throws(() => m_Manager.AddComponent(m_Manager.UniversalQuery, typeof(EcsTestTag)) ); Assert.Throws(() => m_Manager.RemoveComponent(entity) ); - Assert.Throws(() => m_Manager.RemoveComponent(m_Manager.UniversalGroup, typeof(EcsTestData))); + Assert.Throws(() => m_Manager.RemoveComponent(m_Manager.UniversalQuery, typeof(EcsTestData))); Assert.Throws(() => m_Manager.DestroyEntity(entity)); - Assert.Throws(() => m_Manager.DestroyEntity(m_Manager.UniversalGroup)); + Assert.Throws(() => m_Manager.DestroyEntity(m_Manager.UniversalQuery)); } [Test] @@ -90,10 +90,10 @@ public void LockedEntityOrderRestrictions() m_Manager.LockChunkOrder(chunk); Assert.Throws(() => m_Manager.AddComponentData(entity, new EcsFooTest())); - Assert.Throws(() => m_Manager.AddComponent(m_Manager.UniversalGroup, typeof(EcsFooTest))); + Assert.Throws(() => m_Manager.AddComponent(m_Manager.UniversalQuery, typeof(EcsFooTest))); Assert.Throws(() => m_Manager.RemoveComponent(entity) ); - Assert.Throws(() => m_Manager.RemoveComponent(m_Manager.UniversalGroup, typeof(EcsTestData))); + Assert.Throws(() => m_Manager.RemoveComponent(m_Manager.UniversalQuery, typeof(EcsTestData))); Assert.Throws(() => m_Manager.DestroyEntity(entity)); } @@ -105,10 +105,10 @@ public void LockedEntityOrderAllowsNonMovingOperations() var chunk = m_Manager.GetChunk(entity); m_Manager.LockChunkOrder(chunk); - m_Manager.AddComponent(m_Manager.UniversalGroup, typeof(EcsTestTag)); - m_Manager.RemoveComponent(m_Manager.UniversalGroup, typeof(EcsTestTag)); - m_Manager.AddChunkComponentData(m_Manager.UniversalGroup, new EcsFooTest()); - m_Manager.DestroyEntity(m_Manager.UniversalGroup); + m_Manager.AddComponent(m_Manager.UniversalQuery, typeof(EcsTestTag)); + m_Manager.RemoveComponent(m_Manager.UniversalQuery, typeof(EcsTestTag)); + m_Manager.AddChunkComponentData(m_Manager.UniversalQuery, new EcsFooTest()); + m_Manager.DestroyEntity(m_Manager.UniversalQuery); } [Test] diff --git a/Unity.Entities.Tests/MoveEntitiesFromTests.cs b/Unity.Entities.Tests/MoveEntitiesFromTests.cs index 2ea4f833..9aedf9a0 100644 --- a/Unity.Entities.Tests/MoveEntitiesFromTests.cs +++ b/Unity.Entities.Tests/MoveEntitiesFromTests.cs @@ -19,7 +19,7 @@ public void MoveEntitiesToSameEntityManagerThrows() public void MoveEntities() { var creationWorld = new World("CreationWorld"); - var creationManager = creationWorld.GetOrCreateManager(); + var creationManager = creationWorld.EntityManager; var archetype = creationManager.CreateArchetype(typeof(EcsTestData), typeof(EcsTestData2)); @@ -39,9 +39,9 @@ public void MoveEntities() m_Manager.Debug.CheckInternalConsistency(); creationManager.Debug.CheckInternalConsistency(); - var group = m_Manager.CreateComponentGroup(typeof(EcsTestData)); + var group = m_Manager.CreateEntityQuery(typeof(EcsTestData)); Assert.AreEqual(entities.Length, group.CalculateLength()); - Assert.AreEqual(0, creationManager.CreateComponentGroup(typeof(EcsTestData)).CalculateLength()); + Assert.AreEqual(0, creationManager.CreateEntityQuery(typeof(EcsTestData)).CalculateLength()); // We expect that the order of the crated entities is the same as in the creation scene var testDataArray = group.ToComponentDataArray(Allocator.TempJob); @@ -54,11 +54,10 @@ public void MoveEntities() } [Test] - [StandaloneFixme] // ISharedComponentData public void MoveEntitiesWithSharedComponentData() { var creationWorld = new World("CreationWorld"); - var creationManager = creationWorld.GetOrCreateManager(); + var creationManager = creationWorld.EntityManager; var archetype = creationManager.CreateArchetype(typeof(EcsTestData), typeof(EcsTestData2), typeof(SharedData1)); @@ -78,9 +77,9 @@ public void MoveEntitiesWithSharedComponentData() m_Manager.Debug.CheckInternalConsistency(); creationManager.Debug.CheckInternalConsistency(); - var group = m_Manager.CreateComponentGroup(typeof(EcsTestData), typeof(SharedData1)); + var group = m_Manager.CreateEntityQuery(typeof(EcsTestData), typeof(SharedData1)); Assert.AreEqual(entities.Length, group.CalculateLength()); - Assert.AreEqual(0, creationManager.CreateComponentGroup(typeof(EcsTestData)).CalculateLength()); + Assert.AreEqual(0, creationManager.CreateEntityQuery(typeof(EcsTestData)).CalculateLength()); // We expect that the shared component data matches the correct entities var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob); @@ -104,7 +103,7 @@ public void MoveEntitiesWithSharedComponentData() public void MoveEntitiesWithChunkComponents() { var creationWorld = new World("CreationWorld"); - var creationManager = creationWorld.GetOrCreateManager(); + var creationManager = creationWorld.EntityManager; var archetype = creationManager.CreateArchetype(typeof(EcsTestData), ComponentType.ChunkComponent()); @@ -130,9 +129,9 @@ public void MoveEntitiesWithChunkComponents() m_Manager.Debug.CheckInternalConsistency(); creationManager.Debug.CheckInternalConsistency(); - var group = m_Manager.CreateComponentGroup(typeof(EcsTestData), ComponentType.ChunkComponent()); + var group = m_Manager.CreateEntityQuery(typeof(EcsTestData), ComponentType.ChunkComponent()); Assert.AreEqual(entities.Length, group.CalculateLength()); - Assert.AreEqual(0, creationManager.CreateComponentGroup(typeof(EcsTestData)).CalculateLength()); + Assert.AreEqual(0, creationManager.CreateEntityQuery(typeof(EcsTestData)).CalculateLength()); var movedEntities = group.ToEntityArray(Allocator.TempJob); for (int i = 0; i < movedEntities.Length; ++i) @@ -149,10 +148,10 @@ public void MoveEntitiesWithChunkComponents() } [Test] - public void MoveEntitiesWithComponentGroup() + public void MoveEntitiesWithEntityQuery() { var creationWorld = new World("CreationWorld"); - var creationManager = creationWorld.GetOrCreateManager(); + var creationManager = creationWorld.EntityManager; var archetype = creationManager.CreateArchetype(typeof(EcsTestData), typeof(EcsTestData2), typeof(SharedData1)); @@ -167,7 +166,7 @@ public void MoveEntitiesWithComponentGroup() m_Manager.Debug.CheckInternalConsistency(); creationManager.Debug.CheckInternalConsistency(); - var filteredComponentGroup = creationManager.CreateComponentGroup(typeof(EcsTestData), typeof(SharedData1)); + var filteredComponentGroup = creationManager.CreateEntityQuery(typeof(EcsTestData), typeof(SharedData1)); filteredComponentGroup.SetFilter(new SharedData1(2)); var entityRemapping = creationManager.CreateEntityRemapArray(Allocator.TempJob); @@ -179,9 +178,9 @@ public void MoveEntitiesWithComponentGroup() m_Manager.Debug.CheckInternalConsistency(); creationManager.Debug.CheckInternalConsistency(); - var group = m_Manager.CreateComponentGroup(typeof(EcsTestData), typeof(SharedData1)); + var group = m_Manager.CreateEntityQuery(typeof(EcsTestData), typeof(SharedData1)); Assert.AreEqual(2000, group.CalculateLength()); - Assert.AreEqual(8000, creationManager.CreateComponentGroup(typeof(EcsTestData)).CalculateLength()); + Assert.AreEqual(8000, creationManager.CreateEntityQuery(typeof(EcsTestData)).CalculateLength()); // We expect that the shared component data matches the correct entities var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob); @@ -206,11 +205,10 @@ public void MoveEntitiesWithComponentGroup() } [Test] - [StandaloneFixme] // ISharedComponentData public void MoveEntitiesWithComponentGroupMovesChunkComponents() { var creationWorld = new World("CreationWorld"); - var creationManager = creationWorld.GetOrCreateManager(); + var creationManager = creationWorld.EntityManager; var archetype = creationManager.CreateArchetype(typeof(EcsTestData), ComponentType.ChunkComponent(), typeof(SharedData1)); var entities = new NativeArray(10000, Allocator.Temp); @@ -221,7 +219,7 @@ public void MoveEntitiesWithComponentGroupMovesChunkComponents() creationManager.SetSharedComponentData(entities[i], new SharedData1(i % 5)); } - var srcGroup = creationManager.CreateComponentGroup(typeof(EcsTestData), ComponentType.ChunkComponent(), typeof(SharedData1)); + var srcGroup = creationManager.CreateEntityQuery(typeof(EcsTestData), ComponentType.ChunkComponent(), typeof(SharedData1)); var chunksPerValue = new int[5]; var chunks = srcGroup.CreateArchetypeChunkArray(Allocator.TempJob); @@ -241,7 +239,7 @@ public void MoveEntitiesWithComponentGroupMovesChunkComponents() m_Manager.Debug.CheckInternalConsistency(); creationManager.Debug.CheckInternalConsistency(); - var filteredComponentGroup = creationManager.CreateComponentGroup(typeof(EcsTestData), typeof(SharedData1)); + var filteredComponentGroup = creationManager.CreateEntityQuery(typeof(EcsTestData), typeof(SharedData1)); filteredComponentGroup.SetFilter(new SharedData1(2)); var entityRemapping = creationManager.CreateEntityRemapArray(Allocator.TempJob); @@ -253,15 +251,15 @@ public void MoveEntitiesWithComponentGroupMovesChunkComponents() m_Manager.Debug.CheckInternalConsistency(); creationManager.Debug.CheckInternalConsistency(); - var dstGroup = m_Manager.CreateComponentGroup(typeof(EcsTestData), ComponentType.ChunkComponent(), typeof(SharedData1)); + var dstGroup = m_Manager.CreateEntityQuery(typeof(EcsTestData), ComponentType.ChunkComponent(), typeof(SharedData1)); Assert.AreEqual(2000, dstGroup.CalculateLength()); - Assert.AreEqual(8000, creationManager.CreateComponentGroup(typeof(EcsTestData)).CalculateLength()); + Assert.AreEqual(8000, creationManager.CreateEntityQuery(typeof(EcsTestData)).CalculateLength()); int expectedMovedChunkCount = chunksPerValue[2]; int expectedRemainingChunkCount = chunksPerValue.Sum() - expectedMovedChunkCount; - var movedChunkHeaderGroup = m_Manager.CreateComponentGroup(typeof(EcsTestData2), typeof(ChunkHeader)); - var remainingChunkHeaderGroup = creationManager.CreateComponentGroup(typeof(EcsTestData2), typeof(ChunkHeader)); + var movedChunkHeaderGroup = m_Manager.CreateEntityQuery(typeof(EcsTestData2), typeof(ChunkHeader)); + var remainingChunkHeaderGroup = creationManager.CreateEntityQuery(typeof(EcsTestData2), typeof(ChunkHeader)); Assert.AreEqual(expectedMovedChunkCount, movedChunkHeaderGroup.CalculateLength()); Assert.AreEqual(expectedRemainingChunkCount, remainingChunkHeaderGroup.CalculateLength()); @@ -303,7 +301,7 @@ public void MoveEntitiesWithComponentGroupMovesChunkComponents() public void MoveEntitiesWithChunkHeaderChunksThrows() { var creationWorld = new World("CreationWorld"); - var creationManager = creationWorld.GetOrCreateManager(); + var creationManager = creationWorld.EntityManager; var archetype = creationManager.CreateArchetype(typeof(EcsTestData), ComponentType.ChunkComponent()); @@ -314,7 +312,7 @@ public void MoveEntitiesWithChunkHeaderChunksThrows() m_Manager.Debug.CheckInternalConsistency(); creationManager.Debug.CheckInternalConsistency(); - var componentGroupToMove = creationManager.CreateComponentGroup(typeof(EcsTestData2), typeof(ChunkHeader)); + var componentGroupToMove = creationManager.CreateEntityQuery(typeof(EcsTestData2), typeof(ChunkHeader)); var entityRemapping = creationManager.CreateEntityRemapArray(Allocator.TempJob); Assert.Throws(() => m_Manager.MoveEntitiesFrom(creationManager, componentGroupToMove, entityRemapping)); @@ -337,7 +335,7 @@ public void MoveEntitiesAppendsToExistingEntities() m_Manager.SetComponentData(targetEntities[i], new EcsTestData(i)); var sourceWorld = new World("SourceWorld"); - var sourceManager = sourceWorld.GetOrCreateManager(); + var sourceManager = sourceWorld.EntityManager; var sourceArchetype = sourceManager.CreateArchetype(typeof(EcsTestData), typeof(EcsTestData2)); var sourceEntities = new NativeArray(numberOfEntitiesPerManager, Allocator.Temp); sourceManager.CreateEntity(sourceArchetype, sourceEntities); @@ -352,9 +350,9 @@ public void MoveEntitiesAppendsToExistingEntities() m_Manager.Debug.CheckInternalConsistency(); sourceManager.Debug.CheckInternalConsistency(); - var group = m_Manager.CreateComponentGroup(typeof(EcsTestData)); + var group = m_Manager.CreateEntityQuery(typeof(EcsTestData)); Assert.AreEqual(numberOfEntitiesPerManager * 2, group.CalculateLength()); - Assert.AreEqual(0, sourceManager.CreateComponentGroup(typeof(EcsTestData)).CalculateLength()); + Assert.AreEqual(0, sourceManager.CreateEntityQuery(typeof(EcsTestData)).CalculateLength()); // We expect that the order of the crated entities is the same as in the creation scene var testDataArray = group.ToComponentDataArray(Allocator.TempJob); @@ -379,7 +377,7 @@ public void MoveEntitiesPatchesEntityReferences() m_Manager.SetComponentData(targetEntities[i], new EcsTestDataEntity(i, targetEntities[i])); var sourceWorld = new World("SourceWorld"); - var sourceManager = sourceWorld.GetOrCreateManager(); + var sourceManager = sourceWorld.EntityManager; var sourceArchetype = sourceManager.CreateArchetype(typeof(EcsTestDataEntity)); var sourceEntities = new NativeArray(numberOfEntitiesPerManager, Allocator.Temp); sourceManager.CreateEntity(sourceArchetype, sourceEntities); @@ -394,9 +392,9 @@ public void MoveEntitiesPatchesEntityReferences() m_Manager.Debug.CheckInternalConsistency(); sourceManager.Debug.CheckInternalConsistency(); - var group = m_Manager.CreateComponentGroup(typeof(EcsTestDataEntity)); + var group = m_Manager.CreateEntityQuery(typeof(EcsTestDataEntity)); Assert.AreEqual(numberOfEntitiesPerManager * 2, group.CalculateLength()); - Assert.AreEqual(0, sourceManager.CreateComponentGroup(typeof(EcsTestDataEntity)).CalculateLength()); + Assert.AreEqual(0, sourceManager.CreateEntityQuery(typeof(EcsTestDataEntity)).CalculateLength()); var testDataArray = group.ToComponentDataArray(Allocator.TempJob); for (int i = 0; i != testDataArray.Length; i++) @@ -409,7 +407,6 @@ public void MoveEntitiesPatchesEntityReferences() } [Test] - [StandaloneFixme] // ISharedComponentData [Ignore("This behaviour is currently not intended. It prevents streaming efficiently.")] public void MoveEntitiesPatchesEntityReferencesInSharedComponentData() { @@ -426,7 +423,7 @@ public void MoveEntitiesPatchesEntityReferencesInSharedComponentData() } var sourceWorld = new World("SourceWorld"); - var sourceManager = sourceWorld.GetOrCreateManager(); + var sourceManager = sourceWorld.EntityManager; var sourceArchetype = sourceManager.CreateArchetype(typeof(EcsTestData), typeof(EcsTestSharedCompEntity)); var sourceEntities = new NativeArray(numberOfEntitiesPerManager, Allocator.Temp); sourceManager.CreateEntity(sourceArchetype, sourceEntities); @@ -444,9 +441,9 @@ public void MoveEntitiesPatchesEntityReferencesInSharedComponentData() m_Manager.Debug.CheckInternalConsistency(); sourceManager.Debug.CheckInternalConsistency(); - var group = m_Manager.CreateComponentGroup(typeof(EcsTestData), typeof(EcsTestSharedCompEntity)); + var group = m_Manager.CreateEntityQuery(typeof(EcsTestData), typeof(EcsTestSharedCompEntity)); Assert.AreEqual(numberOfEntitiesPerManager * 2, group.CalculateLength()); - Assert.AreEqual(0, sourceManager.CreateComponentGroup(typeof(EcsTestData)).CalculateLength()); + Assert.AreEqual(0, sourceManager.CreateEntityQuery(typeof(EcsTestData)).CalculateLength()); var testDataArray = group.ToComponentDataArray(Allocator.TempJob); var entities = group.ToEntityArray(Allocator.TempJob); @@ -469,7 +466,7 @@ public void MoveEntitiesPatchesEntityReferencesInSharedComponentData() public void MoveEntitiesFromCanReturnEntities() { var creationWorld = new World("CreationWorld"); - var creationManager = creationWorld.GetOrCreateManager(); + var creationManager = creationWorld.EntityManager; var archetype = creationManager.CreateArchetype(typeof(EcsTestData), typeof(EcsTestData2)); @@ -513,7 +510,7 @@ public void MoveEntitiesFromCanReturnEntities() public unsafe void MoveEntitiesArchetypeChunkCountMatches() { var creationWorld = new World("CreationWorld"); - var creationManager = creationWorld.GetOrCreateManager(); + var creationManager = creationWorld.EntityManager; var archetype = creationManager.CreateArchetype(typeof(EcsTestData), typeof(EcsTestData2)); @@ -549,11 +546,11 @@ public unsafe void MoveEntitiesArchetypeChunkCountMatches() public void MoveEntitiesFromChunksAreConsideredChangedOnlyOnce() { var creationWorld = new World("CreationWorld"); - var creationManager = creationWorld.GetOrCreateManager(); + var creationManager = creationWorld.EntityManager; var entity = creationManager.CreateEntity(); creationManager.AddComponentData(entity, new EcsTestData(42)); - var system = World.GetOrCreateManager(); + var system = World.GetOrCreateSystem(); system.Update(); Assert.AreEqual(0, system.NumChanged); diff --git a/Unity.Entities.Tests/PrefabComponentTests.cs b/Unity.Entities.Tests/PrefabComponentTests.cs index 554e0969..04693c7b 100644 --- a/Unity.Entities.Tests/PrefabComponentTests.cs +++ b/Unity.Entities.Tests/PrefabComponentTests.cs @@ -7,12 +7,12 @@ namespace Unity.Entities.Tests class PrefabComponentTests : ECSTestsFixture { [Test] - public void PFB_DontFindPrefabInComponentGroup() + public void PFB_DontFindPrefabInEntityQuery() { var archetype0 = m_Manager.CreateArchetype(typeof(EcsTestData)); var archetype1 = m_Manager.CreateArchetype(typeof(EcsTestData), typeof(Prefab)); - var group = m_Manager.CreateComponentGroup(typeof(EcsTestData)); + var group = m_Manager.CreateEntityQuery(typeof(EcsTestData)); var entity0 = m_Manager.CreateEntity(archetype0); var entity1 = m_Manager.CreateEntity(archetype1); @@ -32,7 +32,7 @@ public void PFB_DontFindPrefabInChunkIterator() var entity0 = m_Manager.CreateEntity(archetype0); var entity1 = m_Manager.CreateEntity(archetype1); - var group = m_Manager.CreateComponentGroup(ComponentType.ReadWrite()); + var group = m_Manager.CreateEntityQuery(ComponentType.ReadWrite()); var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob); group.Dispose(); var count = ArchetypeChunkArray.CalculateEntityCount(chunks); @@ -45,12 +45,12 @@ public void PFB_DontFindPrefabInChunkIterator() } [Test] - public void PFB_FindPrefabIfRequestedInComponentGroup() + public void PFB_FindPrefabIfRequestedInEntityQuery() { var archetype0 = m_Manager.CreateArchetype(typeof(EcsTestData)); var archetype1 = m_Manager.CreateArchetype(typeof(EcsTestData), typeof(Prefab)); - var group = m_Manager.CreateComponentGroup(typeof(EcsTestData), typeof(Prefab)); + var group = m_Manager.CreateEntityQuery(typeof(EcsTestData), typeof(Prefab)); var entity0 = m_Manager.CreateEntity(archetype0); var entity1 = m_Manager.CreateEntity(archetype1); @@ -73,7 +73,7 @@ public void PFB_FindPrefabIfRequestedInChunkIterator() var entity1 = m_Manager.CreateEntity(archetype1); var entity2 = m_Manager.CreateEntity(archetype1); - var group = m_Manager.CreateComponentGroup(ComponentType.ReadWrite(), + var group = m_Manager.CreateEntityQuery(ComponentType.ReadWrite(), ComponentType.ReadWrite()); var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob); group.Dispose(); @@ -117,7 +117,7 @@ public void PFB_InstantiatedWithoutPrefab() Assert.AreEqual(true, m_Manager.HasComponent(entity0)); Assert.AreEqual(false, m_Manager.HasComponent(entity1)); - var group = m_Manager.CreateComponentGroup(ComponentType.ReadWrite()); + var group = m_Manager.CreateEntityQuery(ComponentType.ReadWrite()); var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob); group.Dispose(); var count = ArchetypeChunkArray.CalculateEntityCount(chunks); diff --git a/Unity.Entities.Tests/SafetyTests.cs b/Unity.Entities.Tests/SafetyTests.cs index c49ba7ad..719fb465 100644 --- a/Unity.Entities.Tests/SafetyTests.cs +++ b/Unity.Entities.Tests/SafetyTests.cs @@ -20,77 +20,6 @@ public void RemoveEntityComponentThrows() Assert.IsTrue(m_Manager.HasComponent(entity)); } -#pragma warning disable 618 - [Test] - public void ComponentArrayChunkSliceOutOfBoundsThrowsException() - { - for (int i = 0;i<10;i++) - m_Manager.CreateEntity(typeof(EcsTestData), typeof(EcsTestData2)); - - var group = m_Manager.CreateComponentGroup(typeof(EcsTestData)); - var testData = group.GetComponentDataArray(); - - Assert.AreEqual(0, testData.GetChunkArray(5, 0).Length); - Assert.AreEqual(10, testData.GetChunkArray(0, 10).Length); - - Assert.Throws(() => { testData.GetChunkArray(-1, 1); }); - Assert.Throws(() => { testData.GetChunkArray(5, 6); }); - Assert.Throws(() => { testData.GetChunkArray(10, 1); }); - } - - - [Test] - [StandaloneFixme] // Real problem : Atomic Safety - public void ReadOnlyComponentDataArray() - { - var group = m_Manager.CreateComponentGroup(typeof(EcsTestData2), ComponentType.ReadOnly(typeof(EcsTestData))); - - var entity = m_Manager.CreateEntity(typeof(EcsTestData), typeof(EcsTestData2)); - m_Manager.SetComponentData(entity, new EcsTestData(42)); - - // EcsTestData is read only - var arr = group.GetComponentDataArray(); - Assert.AreEqual(1, arr.Length); - Assert.AreEqual(42, arr[0].value); - Assert.Throws(() => { arr[0] = new EcsTestData(0); }); - - // EcsTestData2 can be written to - var arr2 = group.GetComponentDataArray(); - Assert.AreEqual(1, arr2.Length); - arr2[0] = new EcsTestData2(55); - Assert.AreEqual(55, arr2[0].value0); - } - - [Test] - [StandaloneFixme] // Real problem : Atomic Safety - public void AccessComponentArrayAfterCreationThrowsException() - { - CreateEntityWithDefaultData(0); - - var group = m_Manager.CreateComponentGroup(typeof(EcsTestData)); - var arr = group.GetComponentDataArray(); - - CreateEntityWithDefaultData(1); - - Assert.Throws(() => { var value = arr[0]; }); - } - - [Test] - [StandaloneFixme] // Real problem : Atomic Safety - public void CreateEntityInvalidatesArray() - { - CreateEntityWithDefaultData(0); - - var group = m_Manager.CreateComponentGroup(typeof(EcsTestData)); - var arr = group.GetComponentDataArray(); - - CreateEntityWithDefaultData(1); - - Assert.Throws(() => { var value = arr[0]; }); - } - - #pragma warning restore 618 - [Test] public void GetSetComponentThrowsIfNotExist() { @@ -106,7 +35,6 @@ public void GetSetComponentThrowsIfNotExist() } [Test] - [StandaloneFixme] // Real problem : Atomic Safety public void ComponentDataArrayFromEntityThrowsIfNotExist() { var entity = m_Manager.CreateEntity(typeof(EcsTestData)); @@ -131,15 +59,6 @@ public void AddComponentTwiceThrows() Assert.Throws(() => { m_Manager.AddComponentData(entity, new EcsTestData(1)); }); } - [Test] - public void AddChunkComponentTwiceOnEntityThrows() - { - var entity = m_Manager.CreateEntity(); - - m_Manager.AddChunkComponentData(entity); - Assert.Throws(() => { m_Manager.AddChunkComponentData(entity); }); - } - [Test] public void AddComponentOnDestroyedEntityThrows() { @@ -210,7 +129,6 @@ unsafe struct BigComponentData2 : IComponentData } [Test] - [StandaloneFixme] // Real problem - sizeof(BigComponentData1) = 4 vs 40000 expected public void CreateTooBigArchetypeThrows() { Assert.Throws(() => diff --git a/Unity.Entities.Tests/SerializeTests.cs b/Unity.Entities.Tests/SerializeTests.cs index 5ccd1497..3afc0c3b 100644 --- a/Unity.Entities.Tests/SerializeTests.cs +++ b/Unity.Entities.Tests/SerializeTests.cs @@ -121,7 +121,7 @@ public unsafe void SerializeEntities() var reader = new TestBinaryReader(writer); var deserializedWorld = new World("SerializeEntities Test World 3"); - var entityManager = deserializedWorld.GetOrCreateManager(); + var entityManager = deserializedWorld.EntityManager; SerializeUtility.DeserializeWorld(entityManager.BeginExclusiveEntityTransaction(), reader, 0); entityManager.EndExclusiveEntityTransaction(); @@ -134,20 +134,20 @@ public unsafe void SerializeEntities() Assert.AreEqual(4, count); - var group1 = entityManager.CreateComponentGroup(typeof(EcsTestData), typeof(EcsTestData2), + var group1 = entityManager.CreateEntityQuery(typeof(EcsTestData), typeof(EcsTestData2), typeof(TestComponentData1)); - var group2 = entityManager.CreateComponentGroup(typeof(EcsTestData), typeof(EcsTestData2), + var group2 = entityManager.CreateEntityQuery(typeof(EcsTestData), typeof(EcsTestData2), typeof(TestComponentData2)); - var group3 = entityManager.CreateComponentGroup(typeof(EcsTestData), + var group3 = entityManager.CreateEntityQuery(typeof(EcsTestData), typeof(TestComponentData1), typeof(TestComponentData2)); - var group4 = entityManager.CreateComponentGroup(typeof(EcsComplexEntityRefElement)); + var group4 = entityManager.CreateEntityQuery(typeof(EcsComplexEntityRefElement)); Assert.AreEqual(1, group1.CalculateLength()); Assert.AreEqual(1, group2.CalculateLength()); Assert.AreEqual(1, group3.CalculateLength()); Assert.AreEqual(1, group4.CalculateLength()); - var everythingGroup = entityManager.CreateComponentGroup(Array.Empty()); + var everythingGroup = entityManager.CreateEntityQuery(Array.Empty()); var chunks = everythingGroup.CreateArchetypeChunkArray(Allocator.TempJob); Assert.AreEqual(4, chunks.Length); everythingGroup.Dispose(); @@ -249,7 +249,7 @@ public void SerializeEntitiesSupportsNonASCIIComponentTypeNames() var reader = new TestBinaryReader(writer); var deserializedWorld = new World("SerializeEntitiesSupportsNonASCIIComponentTypeNames Test World"); - var entityManager = deserializedWorld.GetOrCreateManager(); + var entityManager = deserializedWorld.EntityManager; SerializeUtility.DeserializeWorld(entityManager.BeginExclusiveEntityTransaction(), reader, 0); entityManager.EndExclusiveEntityTransaction(); @@ -262,7 +262,7 @@ public void SerializeEntitiesSupportsNonASCIIComponentTypeNames() Assert.AreEqual(1, count); - var group1 = entityManager.CreateComponentGroup(typeof(测试)); + var group1 = entityManager.CreateEntityQuery(typeof(测试)); Assert.AreEqual(1, group1.CalculateLength()); @@ -308,7 +308,7 @@ public unsafe void SerializeEntitiesRemapsEntitiesInBuffers() var reader = new TestBinaryReader(writer); var deserializedWorld = new World("SerializeEntities Test World 3"); - var entityManager = deserializedWorld.GetOrCreateManager(); + var entityManager = deserializedWorld.EntityManager; SerializeUtility.DeserializeWorld(entityManager.BeginExclusiveEntityTransaction(), reader, 0); entityManager.EndExclusiveEntityTransaction(); @@ -316,8 +316,8 @@ public unsafe void SerializeEntitiesRemapsEntitiesInBuffers() try { - var group1 = entityManager.CreateComponentGroup(typeof(EcsTestData), typeof(TestBufferElement)); - var group2 = entityManager.CreateComponentGroup(typeof(EcsTestData2), typeof(TestBufferElement)); + var group1 = entityManager.CreateEntityQuery(typeof(EcsTestData), typeof(TestBufferElement)); + var group2 = entityManager.CreateEntityQuery(typeof(EcsTestData2), typeof(TestBufferElement)); Assert.AreEqual(1, group1.CalculateLength()); Assert.AreEqual(1, group2.CalculateLength()); @@ -376,15 +376,15 @@ public unsafe void SerializeEntitiesWorksWithChunkComponents() var reader = new TestBinaryReader(writer); var deserializedWorld = new World("SerializeEntities Test World 3"); - var entityManager = deserializedWorld.GetOrCreateManager(); + var entityManager = deserializedWorld.EntityManager; SerializeUtility.DeserializeWorld(entityManager.BeginExclusiveEntityTransaction(), reader, 0); entityManager.EndExclusiveEntityTransaction(); try { - var group1 = entityManager.CreateComponentGroup(typeof(EcsTestData)); - var group2 = entityManager.CreateComponentGroup(typeof(EcsTestData2)); + var group1 = entityManager.CreateEntityQuery(typeof(EcsTestData)); + var group2 = entityManager.CreateEntityQuery(typeof(EcsTestData2)); Assert.AreEqual(1, group1.CalculateLength()); Assert.AreEqual(1, group2.CalculateLength()); @@ -468,7 +468,7 @@ ExternalSharedComponentValue[] ExtractSharedComponentValues(int[] indices, Entit { object value = manager.m_SharedComponentManager.GetSharedComponentDataNonDefaultBoxed(indices[i]); int typeIndex = TypeManager.GetTypeIndex(value.GetType()); - int hash = SharedComponentDataManager.GetHashCodeFast(value, typeIndex); + int hash = TypeManager.GetHashCode(value, typeIndex); values[i] = new ExternalSharedComponentValue {obj = value, hashcode = hash, typeIndex = typeIndex}; } return values; @@ -536,7 +536,7 @@ public unsafe void SerializeEntitiesWorksWithBlobAssetReferences() var sharedComponents = ExtractSharedComponentValues(sharedData, m_Manager); - m_Manager.DestroyEntity(m_Manager.UniversalGroup); + m_Manager.DestroyEntity(m_Manager.UniversalQuery); arrayComponent.array.Release(); for(int i=0; i(); + var entityManager = deserializedWorld.EntityManager; InsertSharedComponentValues(sharedComponents, entityManager); @@ -558,8 +558,8 @@ public unsafe void SerializeEntitiesWorksWithBlobAssetReferences() try { - var group1 = entityManager.CreateComponentGroup(typeof(EcsTestDataBlobAssetArray)); - var group2 = entityManager.CreateComponentGroup(typeof(EcsTestDataBlobAssetRef)); + var group1 = entityManager.CreateEntityQuery(typeof(EcsTestDataBlobAssetArray)); + var group2 = entityManager.CreateEntityQuery(typeof(EcsTestDataBlobAssetRef)); var entities1 = group1.ToEntityArray(Allocator.TempJob); Assert.AreEqual(entityCount, entities1.Length); @@ -598,7 +598,7 @@ public void DeserializedChunksAreConsideredChangedOnlyOnce() TestBinaryReader CreateSerializedData() { var world = new World("DeserializedChunksAreConsideredChangedOnlyOnce World"); - var manager = world.GetOrCreateManager(); + var manager = world.EntityManager; var entity = manager.CreateEntity(); manager.AddComponentData(entity, new EcsTestData(42)); var writer = new TestBinaryWriter(); @@ -610,8 +610,8 @@ TestBinaryReader CreateSerializedData() var reader = CreateSerializedData(); var deserializedWorld = new World("DeserializedChunksAreConsideredChangedOnlyOnce World 2"); - var deserializedManager = deserializedWorld.GetOrCreateManager(); - var system = deserializedWorld.GetOrCreateManager(); + var deserializedManager = deserializedWorld.EntityManager; + var system = deserializedWorld.GetOrCreateSystem(); system.Update(); Assert.AreEqual(0, system.NumChanged); diff --git a/Unity.Entities.Tests/SharedComponentDataTests.cs b/Unity.Entities.Tests/SharedComponentDataTests.cs index 47f5be08..757eb246 100644 --- a/Unity.Entities.Tests/SharedComponentDataTests.cs +++ b/Unity.Entities.Tests/SharedComponentDataTests.cs @@ -3,8 +3,6 @@ using System.Collections.Generic; using Unity.Collections; -#pragma warning disable 618 - namespace Unity.Entities.Tests { struct SharedData1 : ISharedComponentData @@ -20,7 +18,7 @@ struct SharedData2 : ISharedComponentData public SharedData2(int val) { value = val; } } - [StandaloneFixme] // ISharedComponentData + class SharedComponentDataTests : ECSTestsFixture { //@TODO: No tests for invalid shared components / destroyed shared component data @@ -32,13 +30,13 @@ public void SetSharedComponent() { var archetype = m_Manager.CreateArchetype(typeof(SharedData1), typeof(EcsTestData), typeof(SharedData2)); - var group1 = m_Manager.CreateComponentGroup(typeof(EcsTestData), typeof(SharedData1)); - var group2 = m_Manager.CreateComponentGroup(typeof(EcsTestData), typeof(SharedData2)); - var group12 = m_Manager.CreateComponentGroup(typeof(EcsTestData), typeof(SharedData2), typeof(SharedData1)); + var group1 = m_Manager.CreateEntityQuery(typeof(EcsTestData), typeof(SharedData1)); + var group2 = m_Manager.CreateEntityQuery(typeof(EcsTestData), typeof(SharedData2)); + var group12 = m_Manager.CreateEntityQuery(typeof(EcsTestData), typeof(SharedData2), typeof(SharedData1)); - var group1_filter_0 = m_Manager.CreateComponentGroup(typeof(EcsTestData), typeof(SharedData1)); + var group1_filter_0 = m_Manager.CreateEntityQuery(typeof(EcsTestData), typeof(SharedData1)); group1_filter_0.SetFilter(new SharedData1(0)); - var group1_filter_20 = m_Manager.CreateComponentGroup(typeof(EcsTestData), typeof(SharedData1)); + var group1_filter_20 = m_Manager.CreateEntityQuery(typeof(EcsTestData), typeof(SharedData1)); group1_filter_20.SetFilter(new SharedData1(20)); Assert.AreEqual(0, group1.CalculateLength()); @@ -53,65 +51,42 @@ public void SetSharedComponent() Entity e2 = m_Manager.CreateEntity(archetype); m_Manager.SetComponentData(e2, new EcsTestData(243)); + var group1_filter0_data = group1_filter_0.ToComponentDataArray(Allocator.TempJob); + Assert.AreEqual(2, group1_filter_0.CalculateLength()); Assert.AreEqual(0, group1_filter_20.CalculateLength()); - Assert.AreEqual(117, group1_filter_0.GetComponentDataArray()[0].value); - Assert.AreEqual(243, group1_filter_0.GetComponentDataArray()[1].value); + Assert.AreEqual(117, group1_filter0_data[0].value); + Assert.AreEqual(243, group1_filter0_data[1].value); m_Manager.SetSharedComponentData(e1, new SharedData1(20)); + + group1_filter0_data.Dispose(); + group1_filter0_data = group1_filter_0.ToComponentDataArray(Allocator.TempJob); + var group1_filter20_data = group1_filter_20.ToComponentDataArray(Allocator.TempJob); Assert.AreEqual(1, group1_filter_0.CalculateLength()); Assert.AreEqual(1, group1_filter_20.CalculateLength()); - Assert.AreEqual(117, group1_filter_20.GetComponentDataArray()[0].value); - Assert.AreEqual(243, group1_filter_0.GetComponentDataArray()[0].value); + Assert.AreEqual(117, group1_filter20_data[0].value); + Assert.AreEqual(243, group1_filter0_data[0].value); m_Manager.SetSharedComponentData(e2, new SharedData1(20)); + + group1_filter20_data.Dispose(); + group1_filter20_data = group1_filter_20.ToComponentDataArray(Allocator.TempJob); Assert.AreEqual(0, group1_filter_0.CalculateLength()); Assert.AreEqual(2, group1_filter_20.CalculateLength()); - Assert.AreEqual(117, group1_filter_20.GetComponentDataArray()[0].value); - Assert.AreEqual(243, group1_filter_20.GetComponentDataArray()[1].value); + Assert.AreEqual(117, group1_filter20_data[0].value); + Assert.AreEqual(243, group1_filter20_data[1].value); group1.Dispose(); group2.Dispose(); group12.Dispose(); group1_filter_0.Dispose(); group1_filter_20.Dispose(); - } - - - [Test] - public void GetComponentArray() - { - var archetype1 = m_Manager.CreateArchetype(typeof(SharedData1), typeof(EcsTestData)); - var archetype2 = m_Manager.CreateArchetype(typeof(SharedData1), typeof(EcsTestData), typeof(SharedData2)); - - const int entitiesPerValue = 5000; - for (int i = 0; i < entitiesPerValue*8; ++i) - { - Entity e = m_Manager.CreateEntity((i % 2 == 0) ? archetype1 : archetype2); - m_Manager.SetComponentData(e, new EcsTestData(i)); - m_Manager.SetSharedComponentData(e, new SharedData1(i%8)); - } - - var group = m_Manager.CreateComponentGroup(typeof(EcsTestData), typeof(SharedData1)); - - for (int sharedValue = 0; sharedValue < 8; ++sharedValue) - { - bool[] foundEntities = new bool[entitiesPerValue]; - group.SetFilter(new SharedData1(sharedValue)); - var componentArray = group.GetComponentDataArray(); - Assert.AreEqual(entitiesPerValue, componentArray.Length); - for (int i = 0; i < entitiesPerValue; ++i) - { - int index = componentArray[i].value; - Assert.AreEqual(sharedValue, index % 8); - Assert.IsFalse(foundEntities[index/8]); - foundEntities[index/8] = true; - } - } - - group.Dispose(); + + group1_filter0_data.Dispose(); + group1_filter20_data.Dispose(); } [Test] @@ -235,50 +210,13 @@ public void RemoveSharedComponent() } [Test] - public void GetSharedComponentArray() - { - var archetype1 = m_Manager.CreateArchetype(typeof(SharedData1), typeof(EcsTestData)); - var archetype2 = m_Manager.CreateArchetype(typeof(SharedData1), typeof(EcsTestData), typeof(SharedData2)); - - const int entitiesPerValue = 5000; - for (int i = 0; i < entitiesPerValue*8; ++i) - { - Entity e = m_Manager.CreateEntity((i % 2 == 0) ? archetype1 : archetype2); - m_Manager.SetComponentData(e, new EcsTestData(i)); - m_Manager.SetSharedComponentData(e, new SharedData1(i%8)); - } - - var group = m_Manager.CreateComponentGroup(typeof(EcsTestData), typeof(SharedData1)); - - var foundEntities = new bool[8, entitiesPerValue]; - - - var sharedComponentDataArray = group.GetSharedComponentDataArray(); - var componentArray = group.GetComponentDataArray(); - - Assert.AreEqual(entitiesPerValue*8, sharedComponentDataArray.Length); - Assert.AreEqual(entitiesPerValue*8, componentArray.Length); - - for (int i = 0; i < entitiesPerValue*8; ++i) - { - var sharedValue = sharedComponentDataArray[i].value; - int index = componentArray[i].value; - Assert.AreEqual(sharedValue, index % 8); - Assert.IsFalse(foundEntities[sharedValue, index/8]); - foundEntities[sharedValue, index/8] = true; - } - - group.Dispose(); - } - - [Test] - public void SCG_DoesNotMatchRemovedSharedComponentInComponentGroup() + public void SCG_DoesNotMatchRemovedSharedComponentInEntityQuery() { var archetype0 = m_Manager.CreateArchetype(typeof(SharedData1), typeof(EcsTestData)); var archetype1 = m_Manager.CreateArchetype(typeof(SharedData1), typeof(EcsTestData), typeof(SharedData2)); - var group0 = m_Manager.CreateComponentGroup(typeof(SharedData1)); - var group1 = m_Manager.CreateComponentGroup(typeof(SharedData2)); + var group0 = m_Manager.CreateEntityQuery(typeof(SharedData1)); + var group1 = m_Manager.CreateEntityQuery(typeof(SharedData2)); m_Manager.CreateEntity(archetype0); var entity1 = m_Manager.CreateEntity(archetype1); @@ -301,8 +239,8 @@ public void SCG_DoesNotMatchRemovedSharedComponentInChunkQuery() var archetype0 = m_Manager.CreateArchetype(typeof(SharedData1), typeof(EcsTestData)); var archetype1 = m_Manager.CreateArchetype(typeof(SharedData1), typeof(EcsTestData), typeof(SharedData2)); - var group0 = m_Manager.CreateComponentGroup(ComponentType.ReadWrite()); - var group1 = m_Manager.CreateComponentGroup(ComponentType.ReadWrite()); + var group0 = m_Manager.CreateEntityQuery(ComponentType.ReadWrite()); + var group1 = m_Manager.CreateEntityQuery(ComponentType.ReadWrite()); m_Manager.CreateEntity(archetype0); var entity1 = m_Manager.CreateEntity(archetype1); @@ -329,6 +267,7 @@ public void SCG_DoesNotMatchRemovedSharedComponentInChunkQuery() postChunks1.Dispose(); } +#if !NET_DOTS [Test] public void GetSharedComponentDataWithTypeIndex() { @@ -347,5 +286,6 @@ public void GetSharedComponentDataWithTypeIndex() Assert.AreEqual(typeof(SharedData1), sharedComponentValue.GetType()); Assert.AreEqual(17, ((SharedData1)sharedComponentValue).value); } +#endif } } diff --git a/Unity.Entities.Tests/SingletonTests.cs b/Unity.Entities.Tests/SingletonTests.cs index a919687f..b098c7c8 100644 --- a/Unity.Entities.Tests/SingletonTests.cs +++ b/Unity.Entities.Tests/SingletonTests.cs @@ -6,7 +6,6 @@ namespace Unity.Entities.Tests class SingletonTests : ECSTestsFixture { [Test] - [StandaloneFixme] public void GetSetSingleton() { var entity = m_Manager.CreateEntity(typeof(EcsTestData)); @@ -16,7 +15,6 @@ public void GetSetSingleton() } [Test] - [StandaloneFixme] public void GetSetSingletonZeroThrows() { Assert.Throws(() => EmptySystem.SetSingleton(new EcsTestData())); @@ -24,7 +22,6 @@ public void GetSetSingletonZeroThrows() } [Test] - [StandaloneFixme] public void GetSetSingletonMultipleThrows() { m_Manager.CreateEntity(typeof(EcsTestData)); @@ -35,11 +32,11 @@ public void GetSetSingletonMultipleThrows() } [Test] - [StandaloneFixme] + [StandaloneFixme] // EmptySystem.ShouldRunSystem is always true in ZeroPlayer public void RequireSingletonWorks() { EmptySystem.RequireSingletonForUpdate(); - EmptySystem.GetComponentGroup(typeof(EcsTestData2)); + EmptySystem.GetEntityQuery(typeof(EcsTestData2)); m_Manager.CreateEntity(typeof(EcsTestData2)); Assert.IsFalse(EmptySystem.ShouldRunSystem()); @@ -48,7 +45,6 @@ public void RequireSingletonWorks() } [Test] - [StandaloneFixme] public void HasSingletonWorks() { Assert.IsFalse(EmptySystem.HasSingleton()); diff --git a/Unity.Entities.Tests/SizeTests.cs b/Unity.Entities.Tests/SizeTests.cs index 4f433c9a..631a5408 100644 --- a/Unity.Entities.Tests/SizeTests.cs +++ b/Unity.Entities.Tests/SizeTests.cs @@ -80,17 +80,6 @@ public void SIZ_TagCanAddComponentData() Assert.IsTrue(m_Manager.HasComponent(entity)); } - #pragma warning disable 618 - [Test] - public void SIZ_TagCannotGetComponentDataArray() - { - var group = m_Manager.CreateComponentGroup(typeof(EcsTestTag)); - var entity0 = m_Manager.CreateEntity(typeof(EcsTestTag)); - - Assert.Throws(() => { group.GetComponentDataArray(); }); - } - #pragma warning restore 618 - [Test] public void SIZ_TagThrowsOnComponentDataFromEntity() { @@ -104,7 +93,7 @@ public void SIZ_TagThrowsOnComponentDataFromEntity() public void SIZ_TagCannotGetNativeArrayFromArchetypeChunk() { m_Manager.CreateEntity(typeof(EcsTestTag)); - var group = m_Manager.CreateComponentGroup(ComponentType.ReadWrite()); + var group = m_Manager.CreateEntityQuery(ComponentType.ReadWrite()); var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob); group.Dispose(); diff --git a/Unity.Entities.Tests/StandaloneFixme.cs b/Unity.Entities.Tests/StandaloneFixme.cs index 6be96359..e26c3db2 100644 --- a/Unity.Entities.Tests/StandaloneFixme.cs +++ b/Unity.Entities.Tests/StandaloneFixme.cs @@ -3,7 +3,7 @@ namespace Unity.Entities.Tests { -#if UNITY_CSHARP_TINY +#if NET_DOTS public class StandaloneFixmeAttribute : IgnoreAttribute { public StandaloneFixmeAttribute() : base("Need to fix for Tiny.") diff --git a/Unity.Entities.Tests/SystemStateBufferElementTests.cs b/Unity.Entities.Tests/SystemStateBufferElementTests.cs index 34eed534..07a2d5ed 100644 --- a/Unity.Entities.Tests/SystemStateBufferElementTests.cs +++ b/Unity.Entities.Tests/SystemStateBufferElementTests.cs @@ -26,7 +26,7 @@ class SystemStateBufferElementTests : ECSTestsFixture void VerifyComponentCount(int expectedCount) where T : IComponentData { - var group = m_Manager.CreateComponentGroup(ComponentType.ReadWrite()); + var group = m_Manager.CreateEntityQuery(ComponentType.ReadWrite()); var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob); group.Dispose(); Assert.AreEqual(expectedCount, ArchetypeChunkArray.CalculateEntityCount(chunks)); @@ -36,14 +36,14 @@ void VerifyComponentCount(int expectedCount) void VerifyBufferCount(int expectedCount) where T : ISystemStateBufferElementData { - var group = m_Manager.CreateComponentGroup(ComponentType.ReadWrite()); + var group = m_Manager.CreateEntityQuery(ComponentType.ReadWrite()); var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob); group.Dispose(); Assert.AreEqual(expectedCount, ArchetypeChunkArray.CalculateEntityCount(chunks)); chunks.Dispose(); } - void VerifyQueryCount(ComponentGroup group, int expectedCount) + void VerifyQueryCount(EntityQuery group, int expectedCount) { var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob); Assert.AreEqual(expectedCount, ArchetypeChunkArray.CalculateEntityCount(chunks)); @@ -51,7 +51,7 @@ void VerifyQueryCount(ComponentGroup group, int expectedCount) } [Test] - [StandaloneFixme] + [StandaloneFixme] // ISharedComponentData public void DeleteWhenEmpty() { var entity = m_Manager.CreateEntity( @@ -80,7 +80,7 @@ public void DeleteWhenEmpty() } [Test] - [StandaloneFixme] + [StandaloneFixme] // ISharedComponentData public void DeleteWhenEmptyArray() { var entities = new Entity[512]; @@ -110,7 +110,7 @@ public void DeleteWhenEmptyArray() VerifyComponentCount(256); VerifyBufferCount(512); - VerifyQueryCount(m_Manager.CreateComponentGroup( + VerifyQueryCount(m_Manager.CreateEntityQuery( ComponentType.Exclude(), ComponentType.ReadWrite()), 256); @@ -136,7 +136,7 @@ public void DeleteWhenEmptyArray() } [Test] - [StandaloneFixme] + [StandaloneFixme] // ISharedComponentData public void DeleteWhenEmptyArray2() { var entities = new Entity[512]; @@ -166,7 +166,7 @@ public void DeleteWhenEmptyArray2() VerifyComponentCount(256); VerifyBufferCount(512); - VerifyQueryCount(m_Manager.CreateComponentGroup( + VerifyQueryCount(m_Manager.CreateEntityQuery( ComponentType.Exclude(), ComponentType.ReadWrite()), 256); @@ -192,7 +192,7 @@ public void DeleteWhenEmptyArray2() } [Test] - [StandaloneFixme] + [StandaloneFixme] // ISharedComponentData public void DoNotInstantiateSystemState() { var entity0 = m_Manager.CreateEntity( @@ -219,7 +219,7 @@ public void InstantiateResidueEntityThrows() } [Test] - [StandaloneFixme] + [StandaloneFixme] // Test Error public void DeleteFromEntity() { var entities = new Entity[512]; @@ -247,7 +247,7 @@ public void DeleteFromEntity() VerifyComponentCount(0); VerifyBufferCount(512); - var group = m_Manager.CreateComponentGroup( + var group = m_Manager.CreateEntityQuery( ComponentType.Exclude(), ComponentType.ReadWrite()); @@ -267,8 +267,7 @@ public void DeleteFromEntity() } [Test] - [StandaloneFixme] - public void DeleteFromComponentGroup() + public void DeleteFromEntityQuery() { var entities = new Entity[512]; @@ -296,7 +295,7 @@ public void DeleteFromComponentGroup() VerifyComponentCount(0); VerifyBufferCount(512); - var group = m_Manager.CreateComponentGroup( + var group = m_Manager.CreateEntityQuery( ComponentType.Exclude(), ComponentType.ReadWrite()); @@ -312,8 +311,7 @@ public void DeleteFromComponentGroup() } [Test] - [StandaloneFixme] - public void DeleteTagFromComponentGroup() + public void DeleteTagFromEntityQuery() { var entities = new Entity[512]; @@ -339,7 +337,7 @@ public void DeleteTagFromComponentGroup() VerifyComponentCount(0); VerifyBufferCount(512); - var group = m_Manager.CreateComponentGroup( + var group = m_Manager.CreateEntityQuery( ComponentType.Exclude(), ComponentType.ReadWrite()); diff --git a/Unity.Entities.Tests/SystemStateComponentTests.cs b/Unity.Entities.Tests/SystemStateComponentTests.cs index cdf4740b..197044c8 100644 --- a/Unity.Entities.Tests/SystemStateComponentTests.cs +++ b/Unity.Entities.Tests/SystemStateComponentTests.cs @@ -26,14 +26,14 @@ class SystemStateComponentTests : ECSTestsFixture void VerifyComponentCount(int expectedCount) where T : IComponentData { - var group = m_Manager.CreateComponentGroup(ComponentType.ReadWrite()); + var group = m_Manager.CreateEntityQuery(ComponentType.ReadWrite()); var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob); group.Dispose(); Assert.AreEqual(expectedCount, ArchetypeChunkArray.CalculateEntityCount(chunks)); chunks.Dispose(); } - void VerifyQueryCount(ComponentGroup group, int expectedCount) + void VerifyQueryCount(EntityQuery group, int expectedCount) { var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob); Assert.AreEqual(expectedCount, ArchetypeChunkArray.CalculateEntityCount(chunks)); @@ -41,7 +41,7 @@ void VerifyQueryCount(ComponentGroup group, int expectedCount) } [Test] - [StandaloneFixme] + [StandaloneFixme] // ISharedComponentData public void DeleteWhenEmpty() { var entity = m_Manager.CreateEntity( @@ -69,7 +69,7 @@ public void DeleteWhenEmpty() } [Test] - [StandaloneFixme] + [StandaloneFixme] // ISharedComponentData public void DeleteWhenEmptyArray() { var entities = new Entity[512]; @@ -98,7 +98,7 @@ public void DeleteWhenEmptyArray() VerifyComponentCount(256); VerifyComponentCount(512); - VerifyQueryCount(m_Manager.CreateComponentGroup( + VerifyQueryCount(m_Manager.CreateEntityQuery( ComponentType.Exclude(), ComponentType.ReadWrite()), 256); @@ -124,7 +124,7 @@ public void DeleteWhenEmptyArray() } [Test] - [StandaloneFixme] + [StandaloneFixme] // ISharedComponentData public void DeleteWhenEmptyArray2() { var entities = new Entity[512]; @@ -153,7 +153,7 @@ public void DeleteWhenEmptyArray2() VerifyComponentCount(256); VerifyComponentCount(512); - VerifyQueryCount(m_Manager.CreateComponentGroup( + VerifyQueryCount(m_Manager.CreateEntityQuery( ComponentType.Exclude(), ComponentType.ReadWrite()), 256); @@ -232,7 +232,7 @@ public void DeleteFromEntity() VerifyComponentCount(0); VerifyComponentCount(512); - var group = m_Manager.CreateComponentGroup( + var group = m_Manager.CreateEntityQuery( ComponentType.Exclude(), ComponentType.ReadWrite()); @@ -252,7 +252,7 @@ public void DeleteFromEntity() } [Test] - public void DeleteFromComponentGroup() + public void DeleteFromEntityQuery() { var entities = new Entity[512]; @@ -279,7 +279,7 @@ public void DeleteFromComponentGroup() VerifyComponentCount(0); VerifyComponentCount(512); - var group = m_Manager.CreateComponentGroup( + var group = m_Manager.CreateEntityQuery( ComponentType.Exclude(), ComponentType.ReadWrite()); @@ -295,7 +295,7 @@ public void DeleteFromComponentGroup() } [Test] - public void DeleteTagFromComponentGroup() + public void DeleteTagFromEntityQuery() { var entities = new Entity[512]; @@ -321,7 +321,7 @@ public void DeleteTagFromComponentGroup() VerifyComponentCount(0); VerifyComponentCount(512); - var group = m_Manager.CreateComponentGroup( + var group = m_Manager.CreateEntityQuery( ComponentType.Exclude(), ComponentType.ReadWrite()); diff --git a/Unity.Entities.Tests/TestComponentSystems.cs b/Unity.Entities.Tests/TestComponentSystems.cs index a3b473c0..78e50654 100644 --- a/Unity.Entities.Tests/TestComponentSystems.cs +++ b/Unity.Entities.Tests/TestComponentSystems.cs @@ -6,10 +6,10 @@ namespace Unity.Entities.Tests public class TestEcsChangeSystem : JobComponentSystem { public int NumChanged; - ComponentGroup ChangeGroup; - protected override void OnCreateManager() + EntityQuery ChangeGroup; + protected override void OnCreate() { - ChangeGroup = GetComponentGroup(typeof(EcsTestData)); + ChangeGroup = GetEntityQuery(typeof(EcsTestData)); ChangeGroup.SetFilterChanged(typeof(EcsTestData)); } diff --git a/Unity.Entities.Tests/TypeIndexOrderTests.cs b/Unity.Entities.Tests/TypeIndexOrderTests.cs index f3a735f8..c81a1bc5 100644 --- a/Unity.Entities.Tests/TypeIndexOrderTests.cs +++ b/Unity.Entities.Tests/TypeIndexOrderTests.cs @@ -113,7 +113,6 @@ void MatchesChunkTypes(params ComponentTypeInArchetype[] types) } [Test] - [StandaloneFixme] public unsafe void TypesInArchetypeAreOrderedAsExpected() { var archetype = m_Manager.CreateArchetype( diff --git a/Unity.Entities.Tests/WorldDebuggingToolsTests.cs b/Unity.Entities.Tests/WorldDebuggingToolsTests.cs index 721030d4..097e8e09 100644 --- a/Unity.Entities.Tests/WorldDebuggingToolsTests.cs +++ b/Unity.Entities.Tests/WorldDebuggingToolsTests.cs @@ -11,33 +11,33 @@ class WorldDebuggingToolsTests : ECSTestsFixture [DisableAutoCreation] class RegularSystem : ComponentSystem { - public ComponentGroup entities; - + public EntityQuery entities; + protected override void OnUpdate() { throw new NotImplementedException(); } - protected override void OnCreateManager() + protected override void OnCreate() { - entities = GetComponentGroup(ComponentType.ReadWrite()); + entities = GetEntityQuery(ComponentType.ReadWrite()); } } [DisableAutoCreation] class ExcludeSystem : ComponentSystem { - public ComponentGroup entities; - + public EntityQuery entities; + protected override void OnUpdate() { throw new NotImplementedException(); } - - protected override void OnCreateManager() + + protected override void OnCreate() { - entities = GetComponentGroup( - ComponentType.ReadWrite(), + entities = GetEntityQuery( + ComponentType.ReadWrite(), ComponentType.Exclude()); } } @@ -45,33 +45,33 @@ protected override void OnCreateManager() [Test] public void SystemInclusionList_MatchesComponents() { - var system = World.Active.GetOrCreateManager(); - + var system = World.Active.GetOrCreateSystem(); + var entity = m_Manager.CreateEntity(typeof(EcsTestData), typeof(EcsTestData2)); - var matchList = new List>>(); - - WorldDebuggingTools.MatchEntityInComponentGroups(World.Active, entity, matchList); - + var matchList = new List>>(); + + WorldDebuggingTools.MatchEntityInEntityQueries(World.Active, entity, matchList); + Assert.AreEqual(1, matchList.Count); Assert.AreEqual(system, matchList[0].Item1); - Assert.AreEqual(system.ComponentGroups[0], matchList[0].Item2[0]); + Assert.AreEqual(system.EntityQueries[0], matchList[0].Item2[0]); } [Test] public void SystemInclusionList_IgnoresSubtractedComponents() { - World.Active.GetOrCreateManager(); - + World.Active.GetOrCreateSystem(); + var entity = m_Manager.CreateEntity(typeof(EcsTestData), typeof(EcsTestData2)); - var matchList = new List>>(); - - WorldDebuggingTools.MatchEntityInComponentGroups(World.Active, entity, matchList); - + var matchList = new List>>(); + + WorldDebuggingTools.MatchEntityInEntityQueries(World.Active, entity, matchList); + Assert.AreEqual(0, matchList.Count); } - + } } #endif \ No newline at end of file diff --git a/Unity.Entities.Tests/WorldDiffTests.cs b/Unity.Entities.Tests/WorldDiffTests.cs index 2f434d14..d6535bbf 100644 --- a/Unity.Entities.Tests/WorldDiffTests.cs +++ b/Unity.Entities.Tests/WorldDiffTests.cs @@ -29,8 +29,8 @@ public void SetUp() m_After = new World("After"); m_DstWorld = new World("DstWorld"); - m_Manager = m_After.GetOrCreateManager(); - m_DstManager = m_DstWorld.GetOrCreateManager(); + m_Manager = m_After.EntityManager; + m_DstManager = m_DstWorld.EntityManager; } @@ -39,9 +39,9 @@ public void TearDown() { World.Active = m_PreviousWorld; - m_Shadow.GetOrCreateManager().Debug.CheckInternalConsistency(); - m_After.GetOrCreateManager().Debug.CheckInternalConsistency(); - m_DstWorld.GetOrCreateManager().Debug.CheckInternalConsistency(); + m_Shadow.EntityManager.Debug.CheckInternalConsistency(); + m_After.EntityManager.Debug.CheckInternalConsistency(); + m_DstWorld.EntityManager.Debug.CheckInternalConsistency(); m_Shadow.Dispose(); m_After.Dispose(); diff --git a/Unity.Entities.Tests/WorldTests.cs b/Unity.Entities.Tests/WorldTests.cs index 39d5cff7..eff0b9e7 100644 --- a/Unity.Entities.Tests/WorldTests.cs +++ b/Unity.Entities.Tests/WorldTests.cs @@ -21,7 +21,7 @@ public virtual void TearDown() World.Active = m_PreviousWorld; } - + [Test] [StandaloneFixme] public void ActiveWorldResets() @@ -30,21 +30,21 @@ public void ActiveWorldResets() var worldA = new World("WorldA"); var worldB = new World("WorldB"); - World.Active = worldB; - + World.Active = worldB; + Assert.AreEqual(worldB, World.Active); Assert.AreEqual(count + 2, World.AllWorlds.Count()); Assert.AreEqual(worldA, World.AllWorlds[World.AllWorlds.Count()-2]); Assert.AreEqual(worldB, World.AllWorlds[World.AllWorlds.Count()-1]); - + worldB.Dispose(); - + Assert.IsFalse(worldB.IsCreated); Assert.IsTrue(worldA.IsCreated); Assert.AreEqual(null, World.Active); - + worldA.Dispose(); - + Assert.AreEqual(count, World.AllWorlds.Count()); } @@ -63,20 +63,20 @@ public void WorldVersionIsConsistent() Assert.AreEqual(0, world.Version); var version = world.Version; - world.GetOrCreateManager(); + world.GetOrCreateSystem(); Assert.AreNotEqual(version, world.Version); version = world.Version; - var manager = world.GetOrCreateManager(); + var manager = world.GetOrCreateSystem(); Assert.AreEqual(version, world.Version); version = world.Version; - world.DestroyManager(manager); + world.DestroySystem(manager); Assert.AreNotEqual(version, world.Version); - + world.Dispose(); } - + [Test] [StandaloneFixme] public void UsingDisposedWorldThrows() @@ -84,16 +84,16 @@ public void UsingDisposedWorldThrows() var world = new World("WorldX"); world.Dispose(); - Assert.Throws(() => world.GetExistingManager()); + Assert.Throws(() => world.GetExistingSystem()); } - + [DisableAutoCreation] class AddWorldDuringConstructorThrowsSystem : ComponentSystem { public AddWorldDuringConstructorThrowsSystem() { Assert.AreEqual(null, World); - World.Active.AddManager(this); + World.Active.AddSystem(this); } protected override void OnUpdate() { } @@ -105,18 +105,18 @@ public void AddWorldDuringConstructorThrows () var world = new World("WorldX"); World.Active = world; // Adding a manager during construction is not allowed - Assert.Throws(() => world.CreateManager()); + Assert.Throws(() => world.CreateSystem()); // The manager will not be added to the list of managers if throws - Assert.AreEqual(0, world.BehaviourManagers.Count()); - + Assert.AreEqual(0, world.Systems.Count()); + world.Dispose(); } - - + + [DisableAutoCreation] - class SystemThrowingInOnCreateManagerIsRemovedSystem : ComponentSystem + class SystemThrowingInOnCreateIsRemovedSystem : ComponentSystem { - protected override void OnCreateManager() + protected override void OnCreate() { throw new AssertionException(""); } @@ -125,28 +125,27 @@ protected override void OnUpdate() { } } [Test] [StandaloneFixme] - public void SystemThrowingInOnCreateManagerIsRemoved() + public void SystemThrowingInOnCreateIsRemoved() { var world = new World("WorldX"); - world.GetOrCreateManager(); - Assert.AreEqual(1, world.BehaviourManagers.Count()); + Assert.AreEqual(0, world.Systems.Count()); - Assert.Throws(() => world.GetOrCreateManager()); + Assert.Throws(() => world.GetOrCreateSystem()); // throwing during OnCreateManager does not add the manager to the behaviour manager list - Assert.AreEqual(1, world.BehaviourManagers.Count()); - + Assert.AreEqual(0, world.Systems.Count()); + world.Dispose(); } [DisableAutoCreation] class SystemIsAccessibleDuringOnCreateManagerSystem : ComponentSystem { - protected override void OnCreateManager() + protected override void OnCreate() { - Assert.AreEqual(this, World.GetOrCreateManager()); + Assert.AreEqual(this, World.GetOrCreateSystem()); } - + protected override void OnUpdate() { } } [Test] @@ -154,14 +153,13 @@ protected override void OnUpdate() { } public void SystemIsAccessibleDuringOnCreateManager () { var world = new World("WorldX"); - world.GetOrCreateManager(); - Assert.AreEqual(1, world.BehaviourManagers.Count()); - world.CreateManager(); - Assert.AreEqual(2, world.BehaviourManagers.Count()); - + Assert.AreEqual(0, world.Systems.Count()); + world.CreateSystem(); + Assert.AreEqual(1, world.Systems.Count()); + world.Dispose(); } - - //@TODO: Test for adding a manager from one world to another. + + //@TODO: Test for adding a manager from one world to another. } } diff --git a/Unity.Entities.Tests/WriteGroupTests.cs b/Unity.Entities.Tests/WriteGroupTests.cs index e4dffe55..525eff75 100644 --- a/Unity.Entities.Tests/WriteGroupTests.cs +++ b/Unity.Entities.Tests/WriteGroupTests.cs @@ -47,13 +47,12 @@ struct TestInputD : IComponentData [Test] - [StandaloneFixme] public void WG_AllOnlyMatchesExplicit() { var archetype0 = m_Manager.CreateArchetype(typeof(TestOutputA), typeof(TestInputB), typeof(TestInputC)); var archetype1 = m_Manager.CreateArchetype(typeof(TestOutputA), typeof(TestInputB), typeof(TestInputC), typeof(TestInputD)); - var group0 = m_Manager.CreateComponentGroup(new EntityArchetypeQuery() + var group0 = m_Manager.CreateEntityQuery(new EntityQueryDesc() { All = new ComponentType[] { @@ -61,7 +60,7 @@ public void WG_AllOnlyMatchesExplicit() ComponentType.ReadOnly(), ComponentType.ReadOnly(), }, - Options = EntityArchetypeQueryOptions.FilterWriteGroup + Options = EntityQueryOptions.FilterWriteGroup }); m_Manager.CreateEntity(archetype0); @@ -75,10 +74,9 @@ public void WG_AllOnlyMatchesExplicit() } [Test] - [StandaloneFixme] public void WG_AllOnlyMatchesExplicitLateDefinition() { - var group0 = m_Manager.CreateComponentGroup(new EntityArchetypeQuery() + var group0 = m_Manager.CreateEntityQuery(new EntityQueryDesc() { All = new ComponentType[] { @@ -86,7 +84,7 @@ public void WG_AllOnlyMatchesExplicitLateDefinition() ComponentType.ReadOnly(), ComponentType.ReadOnly(), }, - Options = EntityArchetypeQueryOptions.FilterWriteGroup + Options = EntityQueryOptions.FilterWriteGroup }); var archetype0 = m_Manager.CreateArchetype(typeof(TestOutputA), typeof(TestInputB), typeof(TestInputC)); @@ -109,7 +107,7 @@ public void WG_AllOnlyMatchesExtended() var archetype0 = m_Manager.CreateArchetype(typeof(TestOutputA), typeof(TestInputB), typeof(TestInputC)); var archetype1 = m_Manager.CreateArchetype(typeof(TestOutputA), typeof(TestInputB), typeof(TestInputC), typeof(TestInputD)); - var group0 = m_Manager.CreateComponentGroup(new EntityArchetypeQuery() + var group0 = m_Manager.CreateEntityQuery(new EntityQueryDesc() { All = new ComponentType[] { @@ -118,7 +116,7 @@ public void WG_AllOnlyMatchesExtended() ComponentType.ReadOnly(), ComponentType.ReadOnly(), }, - Options = EntityArchetypeQueryOptions.FilterWriteGroup + Options = EntityQueryOptions.FilterWriteGroup }); m_Manager.CreateEntity(archetype0); @@ -132,13 +130,12 @@ public void WG_AllOnlyMatchesExtended() } [Test] - [StandaloneFixme] public void WG_AnyOnlyMatchesExplicit() { var archetype0 = m_Manager.CreateArchetype(typeof(TestOutputA), typeof(TestInputB), typeof(TestInputC)); var archetype1 = m_Manager.CreateArchetype(typeof(TestOutputA), typeof(TestInputB), typeof(TestInputC), typeof(TestInputD)); - var group0 = m_Manager.CreateComponentGroup(new EntityArchetypeQuery() + var group0 = m_Manager.CreateEntityQuery(new EntityQueryDesc() { Any = new ComponentType[] { @@ -146,7 +143,7 @@ public void WG_AnyOnlyMatchesExplicit() ComponentType.ReadOnly(), ComponentType.ReadOnly(), }, - Options = EntityArchetypeQueryOptions.FilterWriteGroup + Options = EntityQueryOptions.FilterWriteGroup }); m_Manager.CreateEntity(archetype0); @@ -165,7 +162,7 @@ public void WG_AnyMatchesAll() var archetype0 = m_Manager.CreateArchetype(typeof(TestOutputA), typeof(TestInputB), typeof(TestInputC)); var archetype1 = m_Manager.CreateArchetype(typeof(TestOutputA), typeof(TestInputB), typeof(TestInputC), typeof(TestInputD)); - var group0 = m_Manager.CreateComponentGroup(new EntityArchetypeQuery() + var group0 = m_Manager.CreateEntityQuery(new EntityQueryDesc() { Any = new ComponentType[] { @@ -174,7 +171,7 @@ public void WG_AnyMatchesAll() ComponentType.ReadOnly(), ComponentType.ReadOnly(), }, - Options = EntityArchetypeQueryOptions.FilterWriteGroup + Options = EntityQueryOptions.FilterWriteGroup }); m_Manager.CreateEntity(archetype0); @@ -193,7 +190,7 @@ public void WG_AnyExplicitlyExcludesExtension() var archetype0 = m_Manager.CreateArchetype(typeof(TestOutputA), typeof(TestInputB), typeof(TestInputC)); var archetype1 = m_Manager.CreateArchetype(typeof(TestOutputA), typeof(TestInputB), typeof(TestInputC), typeof(TestInputD)); - var group0 = m_Manager.CreateComponentGroup(new EntityArchetypeQuery() + var group0 = m_Manager.CreateEntityQuery(new EntityQueryDesc() { Any = new ComponentType[] { @@ -205,7 +202,7 @@ public void WG_AnyExplicitlyExcludesExtension() { ComponentType.ReadOnly(), }, - Options = EntityArchetypeQueryOptions.FilterWriteGroup + Options = EntityQueryOptions.FilterWriteGroup }); m_Manager.CreateEntity(archetype0); @@ -225,14 +222,14 @@ public void WG_AllAllowsDependentWriteGroups() typeof(TestInputC)); var archetype1 = m_Manager.CreateArchetype(typeof(TestOutputA), typeof(TestOutputB), typeof(TestInputB), typeof(TestInputC), typeof(TestInputD)); - var group0 = m_Manager.CreateComponentGroup(new EntityArchetypeQuery() + var group0 = m_Manager.CreateEntityQuery(new EntityQueryDesc() { All = new ComponentType[] { typeof(TestOutputA), ComponentType.ReadOnly() }, - Options = EntityArchetypeQueryOptions.FilterWriteGroup + Options = EntityQueryOptions.FilterWriteGroup }); m_Manager.CreateEntity(archetype0); @@ -246,21 +243,20 @@ public void WG_AllAllowsDependentWriteGroups() } [Test] - [StandaloneFixme] public void WG_AllExcludesFromDependentWriteGroup() { var archetype0 = m_Manager.CreateArchetype(typeof(TestOutputA), typeof(TestOutputB), typeof(TestInputB), typeof(TestInputC)); var archetype1 = m_Manager.CreateArchetype(typeof(TestOutputA), typeof(TestOutputB), typeof(TestInputB), typeof(TestInputC), typeof(TestInputD)); - var group0 = m_Manager.CreateComponentGroup(new EntityArchetypeQuery() + var group0 = m_Manager.CreateEntityQuery(new EntityQueryDesc() { All = new ComponentType[] { typeof(TestOutputA), ComponentType.ReadOnly() }, - Options = EntityArchetypeQueryOptions.FilterWriteGroup + Options = EntityQueryOptions.FilterWriteGroup }); m_Manager.CreateEntity(archetype0); @@ -280,8 +276,8 @@ public void WG_NotExcludesWhenOverrideWriteGroup() typeof(TestInputC)); var archetype1 = m_Manager.CreateArchetype(typeof(TestOutputA), typeof(TestOutputB), typeof(TestInputB), typeof(TestInputC), typeof(TestInputD)); - // Not specified Options = EntityArchetypeQueryOptions.FilterWriteGroup means that WriteGroup is being overridden (ignored) - var group0 = m_Manager.CreateComponentGroup(new EntityArchetypeQuery() + // Not specified Options = EntityQueryOptions.FilterWriteGroup means that WriteGroup is being overridden (ignored) + var group0 = m_Manager.CreateEntityQuery(new EntityQueryDesc() { All = new ComponentType[] { diff --git a/Unity.Entities.Tests/ZeroPlayerTests.cs b/Unity.Entities.Tests/ZeroPlayerTests.cs new file mode 100644 index 00000000..fcd68fea --- /dev/null +++ b/Unity.Entities.Tests/ZeroPlayerTests.cs @@ -0,0 +1,56 @@ +using System; +using System.Diagnostics; +using Unity.Jobs; +using NUnit.Framework; +using NUnit.Framework.Constraints; +using Unity.Collections.LowLevel.Unsafe; + +namespace Unity.Entities.Tests +{ + + class ZeroPlayerTests + { +#if ENABLE_UNITY_COLLECTIONS_CHECKS + [Test] + public void AllowSecondaryWriting() + { + AtomicSafetyHandle handle = AtomicSafetyHandle.Create(); + AtomicSafetyHandle.SetAllowSecondaryVersionWriting(handle, false); + AtomicSafetyHandle.UseSecondaryVersion(ref handle); + AtomicSafetyHandle.CheckReadAndThrow(handle); + Assert.Throws(() => AtomicSafetyHandle.CheckWriteAndThrow(handle)); + } + + [Test] + public void ReleaseShouldThrow() + { + AtomicSafetyHandle handle = AtomicSafetyHandle.Create(); + AtomicSafetyHandle.Release(handle); + Assert.Throws(() => AtomicSafetyHandle.CheckReadAndThrow(handle)); + Assert.Throws(() => AtomicSafetyHandle.CheckWriteAndThrow(handle)); + Assert.Throws(() => AtomicSafetyHandle.CheckExistsAndThrow(handle)); + + } + + [Test] + public void CloneAndReleaseOriginalShouldThrow() + { + AtomicSafetyHandle handle = AtomicSafetyHandle.Create(); + AtomicSafetyHandle clone = handle; + AtomicSafetyHandle.Release(handle); + Assert.Throws(() => AtomicSafetyHandle.CheckReadAndThrow(clone)); + Assert.Throws(() => AtomicSafetyHandle.CheckWriteAndThrow(clone)); + } + + [Test] + public void CheckDeallocateShouldThrow() + { + AtomicSafetyHandle handle = AtomicSafetyHandle.Create(); + AtomicSafetyHandle clone = handle; + AtomicSafetyHandle.CheckDeallocateAndThrow(clone); + AtomicSafetyHandle.Release(handle); + Assert.Throws(() => AtomicSafetyHandle.CheckDeallocateAndThrow(clone)); + } +#endif + } +} diff --git a/Unity.Entities.Tests/ZeroPlayerTests.cs.meta b/Unity.Entities.Tests/ZeroPlayerTests.cs.meta new file mode 100644 index 00000000..c75d6b03 --- /dev/null +++ b/Unity.Entities.Tests/ZeroPlayerTests.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 52cee78bb878f224daba76919fd06a82 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/Unity.Entities/ArchetypeManager.cs b/Unity.Entities/ArchetypeManager.cs index dcb62765..39441ad0 100644 --- a/Unity.Entities/ArchetypeManager.cs +++ b/Unity.Entities/ArchetypeManager.cs @@ -45,7 +45,7 @@ internal struct ComponentTypeInArchetype public bool IsSystemStateComponent => (TypeIndex & TypeManager.SystemStateTypeFlag) != 0; public bool IsSystemStateSharedComponent => (TypeIndex & TypeManager.SystemStateSharedComponentTypeFlag) == TypeManager.SystemStateSharedComponentTypeFlag; public bool IsSharedComponent => (TypeIndex & TypeManager.SharedComponentTypeFlag) != 0; - public bool IsZeroSized => (TypeIndex & TypeManager.ZeroSizeTypeFlag) != 0; + public bool IsZeroSized => (TypeIndex & TypeManager.ZeroSizeInChunkTypeFlag) != 0; public bool IsChunkComponent => (TypeIndex & TypeManager.ChunkComponentTypeFlag) != 0; public bool HasEntityReferences => (TypeIndex & TypeManager.HasNoEntityReferencesFlag) == 0; @@ -146,45 +146,53 @@ internal enum ChunkFlags LockedEntityOrder = 1 << 1 } - [StructLayout(LayoutKind.Sequential)] + [StructLayout(LayoutKind.Explicit)] internal unsafe struct Chunk { - // offset of field in architecture with // 64 / 32 bits // Chunk header START - public Archetype* Archetype; // 0 / 0 - public Entity metaChunkEntity; // 8 / 4 + [FieldOffset(0)] + public Archetype* Archetype; + // 4-byte padding on 32-bit architectures here + + [FieldOffset(8)] + public Entity metaChunkEntity; // This is meant as read-only. // ArchetypeManager.SetChunkCount should be used to change the count. - public int Count; // 16 / 12 - public int Capacity; // 20 / 16 + [FieldOffset(16)] + public int Count; + [FieldOffset(20)] + public int Capacity; // In hybrid mode, archetypes can contain non-ECS-type components which are managed objects. // In order to access them without a lot of overhead we conceptually store an Object[] in each chunk which contains the managed components. // The chunk does not really own the array though since we cannot store managed references in unmanaged memory, // so instead the ArchetypeManager has a list of Object[]s and the chunk just has an int to reference an Object[] by index in that list. - public int ManagedArrayIndex; // 24 / 20 + [FieldOffset(24)] + public int ManagedArrayIndex; - public int ListIndex; // 28 / 24 - public int ListWithEmptySlotsIndex; // 32 / 28 + [FieldOffset(28)] + public int ListIndex; + [FieldOffset(32)] + public int ListWithEmptySlotsIndex; // Incrementing automatically for each chunk - public uint SequenceNumber; // 36 / 32 + [FieldOffset(36)] + public uint SequenceNumber; // Special chunk behaviors - public uint Flags; // 40 / 36 + [FieldOffset(40)] + public uint Flags; - public int Padding1; // 44 / 40 + // 4-byte padding here, available for an int -#if UNITY_32 - public int Padding2; // * / 44 -#endif // Chunk header END // Component data buffer // This is where the actual chunk data starts. // It's declared like this so we can skip the header part of the chunk and just get to the data. - public fixed byte Buffer[4]; // 48 / 48 (must be multiple of 16) + [FieldOffset(48)] // (must be multiple of 16) + public fixed byte Buffer[4]; public const int kChunkSize = 16 * 1024 - 256; // allocate a bit less to allow for header overhead public const int kMaximumEntitiesPerChunk = kChunkSize / 8; @@ -216,21 +224,21 @@ public static int GetChunkBufferSize() return kChunkSize - (sizeof(Chunk) - 4); } - public bool MatchesFilter(MatchingArchetype* match, ref ComponentGroupFilter filter) + public bool MatchesFilter(MatchingArchetype* match, ref EntityQueryFilter filter) { if ((filter.Type & FilterType.SharedComponent) != 0) { var sharedComponentsInChunk = SharedComponentValues; var filteredCount = filter.Shared.Count; - fixed (int* indexInComponentGroupPtr = filter.Shared.IndexInComponentGroup, sharedComponentIndexPtr = + fixed (int* indexInEntityQueryPtr = filter.Shared.IndexInEntityQuery, sharedComponentIndexPtr = filter.Shared.SharedComponentIndex) { for (var i = 0; i < filteredCount; ++i) { - var indexInComponentGroup = indexInComponentGroupPtr[i]; + var indexInEntityQuery = indexInEntityQueryPtr[i]; var sharedComponentIndex = sharedComponentIndexPtr[i]; - var componentIndexInArcheType = match->IndexInArchetype[indexInComponentGroup]; + var componentIndexInArcheType = match->IndexInArchetype[indexInEntityQuery]; var componentIndexInChunk = componentIndexInArcheType - match->Archetype->FirstSharedComponent; if (sharedComponentsInChunk[componentIndexInChunk] != sharedComponentIndex) return false; @@ -245,11 +253,11 @@ public bool MatchesFilter(MatchingArchetype* match, ref ComponentGroupFilter fil var changedCount = filter.Changed.Count; var requiredVersion = filter.RequiredChangeVersion; - fixed (int* indexInComponentGroupPtr = filter.Changed.IndexInComponentGroup) + fixed (int* indexInEntityQueryPtr = filter.Changed.IndexInEntityQuery) { for (var i = 0; i < changedCount; ++i) { - var indexInArchetype = match->IndexInArchetype[indexInComponentGroupPtr[i]]; + var indexInArchetype = match->IndexInArchetype[indexInEntityQueryPtr[i]]; var changeVersion = GetChangeVersion(indexInArchetype); if (ChangeVersionUtility.DidChange(changeVersion, requiredVersion)) @@ -263,9 +271,9 @@ public bool MatchesFilter(MatchingArchetype* match, ref ComponentGroupFilter fil return true; } - public int GetSharedComponentIndex(MatchingArchetype* match, int indexInComponentGroup) + public int GetSharedComponentIndex(MatchingArchetype* match, int indexInEntityQuery) { - var componentIndexInArcheType = match->IndexInArchetype[indexInComponentGroup]; + var componentIndexInArcheType = match->IndexInArchetype[indexInEntityQuery]; var componentIndexInChunk = componentIndexInArcheType - match->Archetype->FirstSharedComponent; return GetSharedComponentValue(componentIndexInChunk); } @@ -601,10 +609,18 @@ public ArchetypeManager(SharedComponentDataManager sharedComponentManager, Entit m_ChunksBySequenceNumber = new NativeHashMap(4096, Allocator.Persistent); m_EmptyChunks = new ChunkList(); m_Archetypes = new ArchetypeList(); + + // Sanity check a few alignments #if UNITY_ASSERTIONS // Buffer should be 16 byte aligned to ensure component data layout itself can gurantee being aligned var offset = UnsafeUtility.GetFieldOffset(typeof(Chunk).GetField("Buffer")); - Assert.IsTrue(offset % 16 == 0, "Chunk buffer must be 16 byte aligned"); + Assert.IsTrue(offset % TypeManager.MaximumSupportedAlignment == 0, $"Chunk buffer must be {TypeManager.MaximumSupportedAlignment} byte aligned (buffer offset at {offset})"); + Assert.IsTrue(sizeof(Entity) == 8, $"Unity.Entities.Entity is expected to be 8 bytes in size (is {sizeof(Entity)}); if this changes, update Chunk explicit layout"); +#endif +#if ENABLE_UNITY_COLLECTIONS_CHECKS + var bufHeaderSize = UnsafeUtility.SizeOf(); + Assert.IsTrue(bufHeaderSize % TypeManager.MaximumSupportedAlignment == 0, + $"BufferHeader total struct size must be a multiple of the max supported alignment ({TypeManager.MaximumSupportedAlignment})"); #endif } @@ -714,6 +730,14 @@ public static void AssertArchetypeComponents(ComponentTypeInArchetype* types, in if(indexInTypeArray != null) *indexInTypeArray = t; + + if(archetype->Types[t] == componentType) + { + Assert.IsTrue(addedComponentType.IgnoreDuplicateAdd, $"{addedComponentType} is already part of the archetype."); + // Tag component type is already there, no new archetype required. + return null; + } + newTypes[t] = componentType; while (t < archetype->TypesCount) { @@ -956,7 +980,11 @@ void ChunkAllocate(void* pointer, int count = 1) where T : struct type->BytesPerInstance = 0; - int maxCapacity = int.MaxValue; + // number of bytes we'll reserve for potential alignment + int alignExtraSpace = 0; + var alignments = stackalloc int[count]; + + int maxCapacity = TypeManager.MaximumChunkCapacity; for (var i = 0; i < count; ++i) { var cType = TypeManager.GetTypeInfo(types[i].TypeIndex); @@ -969,15 +997,17 @@ void ChunkAllocate(void* pointer, int count = 1) where T : struct type->BufferCapacities[i] = cType.BufferCapacity; type->BytesPerInstance += sizeOf; - #if !UNITY_CSHARP_TINY - //@TODO: not supported in tiny type manager maxCapacity = math.min(cType.MaximumChunkCapacity, maxCapacity); - #endif + + // explicitly 0 here for sizeof == 0, so that the usedBytes + // calculation below properly ignores 0-sized components + alignments[i] = sizeOf == 0 ? 0 : cType.AlignmentInChunkInBytes; + alignExtraSpace += alignments[i]; } Assert.IsTrue(maxCapacity >= 1, "MaximumChunkCapacity must be larger than 1"); - type->ChunkCapacity = math.min(chunkDataSize / type->BytesPerInstance, maxCapacity); + type->ChunkCapacity = math.min((chunkDataSize - alignExtraSpace) / type->BytesPerInstance, maxCapacity); #if ENABLE_UNITY_COLLECTIONS_CHECKS if (type->BytesPerInstance > chunkDataSize) @@ -1012,6 +1042,11 @@ void ChunkAllocate(void* pointer, int count = 1) where T : struct var index = type->TypeMemoryOrder[i]; var sizeOf = type->SizeOfs[index]; + // align usedBytes upwards (eating into alignExtraSpace) so that + // this component actually starts at its required alignment. + // Assumption is that the start of the entire data segment is at the + // maximum possible alignment. + usedBytes = TypeManager.AlignUp(usedBytes, alignments[index]); type->Offsets[index] = usedBytes; usedBytes += sizeOf * type->ChunkCapacity; @@ -1048,7 +1083,7 @@ void ChunkAllocate(void* pointer, int count = 1) where T : struct for (var i = 0; i != count; i++) { var ct = TypeManager.GetTypeInfo(types[i].TypeIndex); - #if !UNITY_CSHARP_TINY + #if !NET_DOTS ulong handle = ~0UL; var offsets = ct.EntityOffsets == null ? null : (TypeManager.EntityOffsetInfo*) UnsafeUtility.PinGCArrayAndGetDataAddress(ct.EntityOffsets, out handle); var offsetCount = ct.EntityOffsetCount; @@ -1066,7 +1101,7 @@ void ChunkAllocate(void* pointer, int count = 1) where T : struct scalarPatchInfo = EntityRemapUtility.AppendEntityPatches(scalarPatchInfo, offsets, offsetCount, type->Offsets[i], type->SizeOfs[i]); } - #if !UNITY_CSHARP_TINY + #if !NET_DOTS if(offsets != null) UnsafeUtility.ReleaseGCObject(handle); #endif @@ -1507,7 +1542,7 @@ public void Execute(int index) dstArray[i] = globalSystemVersion; } } - + // Copy chunk count array var dstCountArray = dstArchetype->Chunks.GetChunkEntityCountArray() + dstChunkCount; UnsafeUtility.MemCpy(dstCountArray, srcArchetype->Chunks.GetChunkEntityCountArray(), sizeof(int) * srcChunkCount); diff --git a/Unity.Entities/AssemblyInfo.cs b/Unity.Entities/AssemblyInfo.cs index acde701d..d1b9addc 100644 --- a/Unity.Entities/AssemblyInfo.cs +++ b/Unity.Entities/AssemblyInfo.cs @@ -3,5 +3,6 @@ [assembly: InternalsVisibleTo("Unity.Entities.Editor")] [assembly: InternalsVisibleTo("Unity.Entities.Editor.Tests")] [assembly: InternalsVisibleTo("Unity.Entities.Hybrid")] +[assembly: InternalsVisibleTo("Unity.Entities.Hybrid.Tests")] [assembly: InternalsVisibleTo("Unity.Entities.StaticTypeRegistry")] [assembly: InternalsVisibleTo("Unity.Entities.CPlusPlus")] diff --git a/Unity.Entities/BinarySerialization.cs b/Unity.Entities/BinarySerialization.cs index 4b9b6a64..e3a303c0 100644 --- a/Unity.Entities/BinarySerialization.cs +++ b/Unity.Entities/BinarySerialization.cs @@ -79,7 +79,7 @@ public static void ReadArray(this BinaryReader reader, NativeArray element } } -#if !UNITY_CSHARP_TINY +#if !NET_DOTS public unsafe class StreamBinaryReader : BinaryReader { private Stream stream; diff --git a/Unity.Entities/ComponentJobManager.cs b/Unity.Entities/ComponentJobManager.cs index ea346b5c..563c1ba8 100644 --- a/Unity.Entities/ComponentJobManager.cs +++ b/Unity.Entities/ComponentJobManager.cs @@ -22,13 +22,13 @@ namespace Unity.Entities /// Job component systems that have no other type dependencies have their JobHandles registered on the Entity type /// to ensure that they are completed by CompleteAllJobsAndInvalidateArrays /// - internal unsafe class ComponentJobSafetyManager + internal unsafe struct ComponentJobSafetyManager { private const int kMaxReadJobHandles = 17; private const int kMaxTypes = TypeManager.MaximumTypesCount; - private readonly JobHandle* m_JobDependencyCombineBuffer; - private readonly int m_JobDependencyCombineBufferCount; + private JobHandle* m_JobDependencyCombineBuffer; + private int m_JobDependencyCombineBufferCount; private ComponentSafetyHandle* m_ComponentSafetyHandles; private JobHandle m_ExclusiveTransactionDependency; @@ -40,7 +40,7 @@ internal unsafe class ComponentJobSafetyManager private const int EntityTypeIndex = 1; - public ComponentJobSafetyManager() + public void OnCreate() { #if ENABLE_UNITY_COLLECTIONS_CHECKS m_TempSafety = AtomicSafetyHandle.Create(); @@ -63,6 +63,8 @@ public ComponentJobSafetyManager() #endif m_HasCleanHandles = true; + IsInTransaction = false; + m_ExclusiveTransactionDependency = default(JobHandle); } public bool IsInTransaction { get; private set; } @@ -489,7 +491,7 @@ private struct ComponentSafetyHandle } #if ENABLE_UNITY_COLLECTIONS_CHECKS - private readonly AtomicSafetyHandle m_TempSafety; + private AtomicSafetyHandle m_TempSafety; #endif } } diff --git a/Unity.Entities/ComponentSystem.cs b/Unity.Entities/ComponentSystem.cs index da8b45d5..cfc4a94a 100644 --- a/Unity.Entities/ComponentSystem.cs +++ b/Unity.Entities/ComponentSystem.cs @@ -8,6 +8,7 @@ using System.ComponentModel; using System.Diagnostics; using System.Linq; +using UnityEngine; namespace Unity.Entities { @@ -20,36 +21,159 @@ internal unsafe struct IntList public int Capacity; } - - public unsafe abstract class ComponentSystemBase : ScriptBehaviourManager + public unsafe abstract partial class ComponentSystemBase { -#if !UNITY_ZEROPLAYER - InjectComponentGroupData[] m_InjectedComponentGroups; - InjectFromEntityData m_InjectFromEntityData; -#endif -#if !UNITY_CSHARP_TINY - ComponentGroupArrayStaticCache[] m_CachedComponentGroupArrays; -#endif - ComponentGroup[] m_ComponentGroups; - ComponentGroup[] m_RequiredComponentGroups; + EntityQuery[] m_EntityQueries; + EntityQuery[] m_RequiredEntityQueries; - internal IntList m_JobDependencyForReadingManagers; - internal IntList m_JobDependencyForWritingManagers; + internal IntList m_JobDependencyForReadingSystems; + internal IntList m_JobDependencyForWritingSystems; - uint m_LastSystemVersion; + uint m_LastSystemVersion; - internal ComponentJobSafetyManager m_SafetyManager; - internal EntityManager m_EntityManager; - World m_World; + internal ComponentJobSafetyManager* m_SafetyManager; + internal EntityManager m_EntityManager; + World m_World; - bool m_AlwaysUpdateSystem; - internal bool m_PreviouslyEnabled; + bool m_AlwaysUpdateSystem; + internal bool m_PreviouslyEnabled; public bool Enabled { get; set; } = true; - public ComponentGroup[] ComponentGroups => m_ComponentGroups; + public EntityQuery[] EntityQueries => m_EntityQueries; public uint GlobalSystemVersion => m_EntityManager.GlobalSystemVersion; - public uint LastSystemVersion => m_LastSystemVersion; + public uint LastSystemVersion => m_LastSystemVersion; + + // ============ + +#if UNITY_EDITOR + private UnityEngine.Profiling.CustomSampler m_Sampler; +#endif + +#if !NET_DOTS +#if ENABLE_UNITY_COLLECTIONS_CHECKS + private static HashSet s_ObsoleteAPICheckedTypes = new HashSet(); + + void CheckForObsoleteAPI() + { + var type = this.GetType(); + while (type != typeof(ComponentSystemBase)) + { + if (s_ObsoleteAPICheckedTypes.Contains(type)) + break; + + if (type.GetMethod("OnCreateManager", BindingFlags.DeclaredOnly | BindingFlags.Instance) != null) + { + Debug.LogWarning($"The OnCreateManager overload in {type} is obsolete; please rename it to OnCreate. OnCreateManager will stop being called in a future release."); + } + + if (type.GetMethod("OnDestroyManager", BindingFlags.DeclaredOnly | BindingFlags.Instance) != null) + { + Debug.LogWarning($"The OnDestroyManager overload in {type} is obsolete; please rename it to OnDestroy. OnDestroyManager will stop being called in a future release."); + } + + s_ObsoleteAPICheckedTypes.Add(type); + + type = type.BaseType; + } + } + + protected ComponentSystemBase() + { + CheckForObsoleteAPI(); + } +#endif +#endif + + internal void CreateInstance(World world) + { + OnBeforeCreateInternal(world); + try + { + OnCreateManager(); // DELETE after obsolete period! + OnCreate(); +#if UNITY_EDITOR + var type = GetType(); + m_Sampler = UnityEngine.Profiling.CustomSampler.Create($"{world.Name} {type.FullName}"); +#endif + } + catch + { + OnBeforeDestroyInternal(); + OnAfterDestroyInternal(); + throw; + } + } + + internal void DestroyInstance() + { + OnBeforeDestroyInternal(); + OnDestroy(); + OnDestroyManager(); // DELETE after obsolete period! + OnAfterDestroyInternal(); + } + + protected virtual void OnCreateManager() + { + } + + protected virtual void OnDestroyManager() + { + } + + /// + /// Called when the ComponentSystem/JobComponentSystem is created. + /// When scripts are reloaded, OnCreate will be invoked before the + /// ComponentSystems receives its first OnUpdate call. + /// + protected virtual void OnCreate() + { + } + + /// + /// OnStartRunning is called when the ComponentSystem starts running + /// for the first time, and every time after it was disabled due to no matching entities. + /// + protected virtual void OnStartRunning() + { + } + + /// + /// OnStopRunning is called when no entities would match the system's + /// EntityQueries (and OnStartRunning was executed before). + /// + protected virtual void OnStopRunning() + { + } + + /// + /// Called when the ComponentSystem/JobComponentSystem is destroyed. + /// Will be called when scripts are reloaded or before Play Mode exits before + /// the system is destroyed. + /// + protected virtual void OnDestroy() + { + } + + /// + /// Execute the system immediately. + /// + public void Update() + { +#if UNITY_EDITOR + m_Sampler?.Begin(); +#endif + InternalUpdate(); + +#if UNITY_EDITOR + m_Sampler?.End(); +#endif + } + + protected internal EntityManager EntityManager => m_EntityManager; + protected internal World World => m_World; + + // =================== #if ENABLE_UNITY_COLLECTIONS_CHECKS internal int m_SystemID; @@ -57,7 +181,7 @@ public unsafe abstract class ComponentSystemBase : ScriptBehaviourManager internal ComponentSystemBase GetSystemFromSystemID(World world, int systemID) { - foreach(var m in world.BehaviourManagers) + foreach(var m in world.Systems) { var system = m as ComponentSystemBase; if (system == null) @@ -70,8 +194,11 @@ internal ComponentSystemBase GetSystemFromSystemID(World world, int systemID) } #endif - ref UnsafeList JobDependencyForReadingManagersUnsafeList => ref *(UnsafeList*)UnsafeUtility.AddressOf(ref m_JobDependencyForReadingManagers); - ref UnsafeList JobDependencyForWritingManagersUnsafeList => ref *(UnsafeList*)UnsafeUtility.AddressOf(ref m_JobDependencyForWritingManagers); + ref UnsafeList JobDependencyForReadingSystemsUnsafeList => + ref *(UnsafeList*) UnsafeUtility.AddressOf(ref m_JobDependencyForReadingSystems); + + ref UnsafeList JobDependencyForWritingSystemsUnsafeList => + ref *(UnsafeList*) UnsafeUtility.AddressOf(ref m_JobDependencyForWritingSystems); [Conditional("ENABLE_UNITY_COLLECTIONS_CHECKS")] internal void CheckExists() @@ -89,11 +216,11 @@ public bool ShouldRunSystem() if (m_AlwaysUpdateSystem) return true; - if (m_RequiredComponentGroups != null) + if (m_RequiredEntityQueries != null) { - for (int i = 0;i != m_RequiredComponentGroups.Length;i++) + for (int i = 0; i != m_RequiredEntityQueries.Length; i++) { - if (m_RequiredComponentGroups[i].IsEmptyIgnoreFilter) + if (m_RequiredEntityQueries[i].IsEmptyIgnoreFilter) return false; } @@ -101,17 +228,17 @@ public bool ShouldRunSystem() } else { - // Systems without component groups should always run. Specifically, - // IJobProcessComponentData adds its component group the first time it's run. - var length = m_ComponentGroups != null ? m_ComponentGroups.Length : 0; + // Systems without queriesDesc should always run. Specifically, + // IJobForEach adds its queriesDesc the first time it's run. + var length = m_EntityQueries != null ? m_EntityQueries.Length : 0; if (length == 0) return true; - // If all the groups are empty, skip it. + // If all the queriesDesc are empty, skip it. // (There’s no way to know what they key value is without other markup) - for (int i = 0;i != length;i++) + for (int i = 0; i != length; i++) { - if (!m_ComponentGroups[i].IsEmptyIgnoreFilter) + if (!m_EntityQueries[i].IsEmptyIgnoreFilter) return true; } @@ -119,106 +246,77 @@ public bool ShouldRunSystem() } } - protected override void OnBeforeCreateManagerInternal(World world) + internal virtual void OnBeforeCreateInternal(World world) { #if ENABLE_UNITY_COLLECTIONS_CHECKS m_SystemID = World.AllocateSystemID(); #endif m_World = world; - m_EntityManager = world.GetOrCreateManager(); + m_EntityManager = world.EntityManager; m_SafetyManager = m_EntityManager.ComponentJobSafetyManager; - m_ComponentGroups = new ComponentGroup[0]; -#if !UNITY_CSHARP_TINY - m_CachedComponentGroupArrays = new ComponentGroupArrayStaticCache[0]; + m_EntityQueries = new EntityQuery[0]; +#if !NET_DOTS m_AlwaysUpdateSystem = GetType().GetCustomAttributes(typeof(AlwaysUpdateSystemAttribute), true).Length != 0; #else m_AlwaysUpdateSystem = true; #endif -#if !UNITY_ZEROPLAYER - ComponentSystemInjection.Inject(this, world, m_EntityManager, out m_InjectedComponentGroups, out m_InjectFromEntityData); - m_InjectFromEntityData.ExtractJobDependencyTypes(this); -#endif - InjectNestedIJobProcessComponentDataJobs(); - - UpdateInjectedComponentGroups(); + InjectNestedIJobForEachJobs(); } - void InjectNestedIJobProcessComponentDataJobs() + void InjectNestedIJobForEachJobs() { #if !UNITY_ZEROPLAYER - // Create ComponentGroup for all nested IJobProcessComponentData jobs - foreach (var nestedType in GetType().GetNestedTypes(BindingFlags.FlattenHierarchy | BindingFlags.NonPublic | BindingFlags.Public)) - JobProcessComponentDataExtensions.GetComponentGroupForIJobProcessComponentData(this, nestedType); + // Create EntityQuery for all nested IJobForEach jobs + foreach (var nestedType in GetType() + .GetNestedTypes(BindingFlags.FlattenHierarchy | BindingFlags.NonPublic | BindingFlags.Public)) + JobForEachExtensions.GetEntityQueryForIJobForEach(this, nestedType); #endif } - protected sealed override void OnAfterDestroyManagerInternal() + internal void OnAfterDestroyInternal() { - foreach (var group in m_ComponentGroups) + foreach (var query in m_EntityQueries) { - #if ENABLE_UNITY_COLLECTIONS_CHECKS - group.DisallowDisposing = null; - #endif - group.Dispose(); +#if ENABLE_UNITY_COLLECTIONS_CHECKS + query.DisallowDisposing = null; +#endif + query.Dispose(); } - m_ComponentGroups = null; + + m_EntityQueries = null; m_EntityManager = null; m_World = null; m_SafetyManager = null; -#if !UNITY_ZEROPLAYER - m_InjectedComponentGroups = null; -#endif -#if !UNITY_CSHARP_TINY - m_CachedComponentGroupArrays = null; -#endif - JobDependencyForReadingManagersUnsafeList.Dispose(); - JobDependencyForWritingManagersUnsafeList.Dispose(); + JobDependencyForReadingSystemsUnsafeList.Dispose(); + JobDependencyForWritingSystemsUnsafeList.Dispose(); } - internal override void InternalUpdate() - { - throw new NotImplementedException(); - } + internal abstract void InternalUpdate(); - protected override void OnBeforeDestroyManagerInternal() + internal virtual void OnBeforeDestroyInternal() { if (m_PreviouslyEnabled) { m_PreviouslyEnabled = false; OnStopRunning(); } - CompleteDependencyInternal(); - UpdateInjectedComponentGroups(); } - protected virtual void OnStartRunning() - { - - } - - protected virtual void OnStopRunning() - { - - } - - protected internal void BeforeUpdateVersioning() + internal void BeforeUpdateVersioning() { m_EntityManager.Entities->IncrementGlobalSystemVersion(); - foreach (var group in m_ComponentGroups) - group.SetFilterChangedRequiredVersion(m_LastSystemVersion); + foreach (var query in m_EntityQueries) + query.SetFilterChangedRequiredVersion(m_LastSystemVersion); } - protected internal void AfterUpdateVersioning() + internal void AfterUpdateVersioning() { m_LastSystemVersion = EntityManager.Entities->GlobalSystemVersion; } - protected internal EntityManager EntityManager => m_EntityManager; - protected internal World World => m_World; - // TODO: this should be made part of UnityEngine? static void ArrayUtilityAdd(ref T[] array, T item) { @@ -257,180 +355,150 @@ public ComponentDataFromEntity GetComponentDataFromEntity(bool isReadOnly return EntityManager.GetComponentDataFromEntity(isReadOnly); } - public void RequireForUpdate(ComponentGroup group) + public void RequireForUpdate(EntityQuery query) { - if (m_RequiredComponentGroups == null) - m_RequiredComponentGroups =new ComponentGroup[1] { group }; + if (m_RequiredEntityQueries == null) + m_RequiredEntityQueries = new EntityQuery[1] {query}; else - ArrayUtilityAdd(ref m_RequiredComponentGroups, group); + ArrayUtilityAdd(ref m_RequiredEntityQueries, query); } public void RequireSingletonForUpdate() { var type = ComponentType.ReadOnly(); - var group = GetComponentGroupInternal(&type, 1); - RequireForUpdate(group); + var query = GetEntityQueryInternal(&type, 1); + RequireForUpdate(query); } public bool HasSingleton() where T : struct, IComponentData { var type = ComponentType.ReadOnly(); - var group = GetComponentGroupInternal(&type, 1); - return !group.IsEmptyIgnoreFilter; + var query = GetEntityQueryInternal(&type, 1); + return !query.IsEmptyIgnoreFilter; } public T GetSingleton() where T : struct, IComponentData { var type = ComponentType.ReadOnly(); - var group = GetComponentGroupInternal(&type, 1); + var query = GetEntityQueryInternal(&type, 1); - return group.GetSingleton(); + return query.GetSingleton(); } public void SetSingleton(T value) where T : struct, IComponentData { var type = ComponentType.ReadWrite(); - var group = GetComponentGroupInternal(&type, 1); - group.SetSingleton(value); + var query = GetEntityQueryInternal(&type, 1); + query.SetSingleton(value); } public Entity GetSingletonEntity() where T : struct, IComponentData { var type = ComponentType.ReadOnly(); - var group = GetComponentGroupInternal(&type, 1); + var query = GetEntityQueryInternal(&type, 1); - return group.GetSingletonEntity(); + return query.GetSingletonEntity(); } internal void AddReaderWriter(ComponentType componentType) { - if (CalculateReaderWriterDependency.Add(componentType, ref JobDependencyForReadingManagersUnsafeList, ref JobDependencyForWritingManagersUnsafeList)) + if (CalculateReaderWriterDependency.Add(componentType, ref JobDependencyForReadingSystemsUnsafeList, + ref JobDependencyForWritingSystemsUnsafeList)) { CompleteDependencyInternal(); } } - internal void AddReaderWriters(ComponentGroup group) + internal void AddReaderWriters(EntityQuery query) { - if (group.AddReaderWritersToLists(ref JobDependencyForReadingManagersUnsafeList, ref JobDependencyForWritingManagersUnsafeList)) + if (query.AddReaderWritersToLists(ref JobDependencyForReadingSystemsUnsafeList, + ref JobDependencyForWritingSystemsUnsafeList)) { CompleteDependencyInternal(); } } - internal ComponentGroup GetComponentGroupInternal(ComponentType* componentTypes, int count) + internal EntityQuery GetEntityQueryInternal(ComponentType* componentTypes, int count) { - for (var i = 0; i != m_ComponentGroups.Length; i++) + for (var i = 0; i != m_EntityQueries.Length; i++) { - if (m_ComponentGroups[i].CompareComponents(componentTypes, count)) - return m_ComponentGroups[i]; + if (m_EntityQueries[i].CompareComponents(componentTypes, count)) + return m_EntityQueries[i]; } - var group = EntityManager.CreateComponentGroup(componentTypes, count); + var query = EntityManager.CreateEntityQuery(componentTypes, count); - AddReaderWriters(group); - AfterGroupCreated(group); + AddReaderWriters(query); + AfterQueryCreated(query); - return group; + return query; } - internal ComponentGroup GetComponentGroupInternal(ComponentType[] componentTypes) + internal EntityQuery GetEntityQueryInternal(ComponentType[] componentTypes) { fixed (ComponentType* componentTypesPtr = componentTypes) { - return GetComponentGroupInternal(componentTypesPtr, componentTypes.Length); + return GetEntityQueryInternal(componentTypesPtr, componentTypes.Length); } } - internal ComponentGroup GetComponentGroupInternal(EntityArchetypeQuery[] query) + internal EntityQuery GetEntityQueryInternal(EntityQueryDesc[] desc) { - for (var i = 0; i != m_ComponentGroups.Length; i++) + for (var i = 0; i != m_EntityQueries.Length; i++) { - if (m_ComponentGroups[i].CompareQuery(query)) - return m_ComponentGroups[i]; + if (m_EntityQueries[i].CompareQuery(desc)) + return m_EntityQueries[i]; } - var group = EntityManager.CreateComponentGroup(query); + var query = EntityManager.CreateEntityQuery(desc); - AddReaderWriters(group); - AfterGroupCreated(group); + AddReaderWriters(query); + AfterQueryCreated(query); - return group; + return query; } - void AfterGroupCreated(ComponentGroup group) + void AfterQueryCreated(EntityQuery query) { - group.SetFilterChangedRequiredVersion(m_LastSystemVersion); + query.SetFilterChangedRequiredVersion(m_LastSystemVersion); #if ENABLE_UNITY_COLLECTIONS_CHECKS - group.DisallowDisposing = "ComponentGroup.Dispose() may not be called on a ComponentGroup created with ComponentSystem.GetComponentGroup. The ComponentGroup will automatically be disposed by the ComponentSystem."; + query.DisallowDisposing = + "EntityQuery.Dispose() may not be called on a EntityQuery created with ComponentSystem.GetEntityQuery. The EntityQuery will automatically be disposed by the ComponentSystem."; #endif - ArrayUtilityAdd(ref m_ComponentGroups, group); + ArrayUtilityAdd(ref m_EntityQueries, query); } - protected internal ComponentGroup GetComponentGroup(params ComponentType[] componentTypes) + protected internal EntityQuery GetEntityQuery(params ComponentType[] componentTypes) { - return GetComponentGroupInternal(componentTypes); - } - protected ComponentGroup GetComponentGroup(NativeArray componentTypes) - { - return GetComponentGroupInternal((ComponentType*)componentTypes.GetUnsafeReadOnlyPtr(), componentTypes.Length); + return GetEntityQueryInternal(componentTypes); } - protected internal ComponentGroup GetComponentGroup(params EntityArchetypeQuery[] query) + protected EntityQuery GetEntityQuery(NativeArray componentTypes) { - return GetComponentGroupInternal(query); + return GetEntityQueryInternal((ComponentType*) componentTypes.GetUnsafeReadOnlyPtr(), + componentTypes.Length); } -#if !UNITY_CSHARP_TINY - [Obsolete("GetEntities has been deprecated. Use ComponentSystem.ForEach to access managed components.")] - protected ComponentGroupArray GetEntities() where T : struct + protected internal EntityQuery GetEntityQuery(params EntityQueryDesc[] queryDesc) { - for (var i = 0; i != m_CachedComponentGroupArrays.Length; i++) - { - if (m_CachedComponentGroupArrays[i].CachedType == typeof(T)) - return new ComponentGroupArray(m_CachedComponentGroupArrays[i]); - } - - var cache = new ComponentGroupArrayStaticCache(typeof(T), EntityManager, this); - ArrayUtilityAdd(ref m_CachedComponentGroupArrays, cache); - return new ComponentGroupArray(cache); - } -#endif - - protected void UpdateInjectedComponentGroups() - { -#if !UNITY_ZEROPLAYER - if (null == m_InjectedComponentGroups) - return; - - ulong gchandle; - var pinnedSystemPtr = (byte*)UnsafeUtility.PinGCObjectAndGetAddress(this, out gchandle); - - try - { - foreach (var group in m_InjectedComponentGroups) - group.UpdateInjection (pinnedSystemPtr); - - m_InjectFromEntityData.UpdateInjection(pinnedSystemPtr, EntityManager); - } - catch - { - UnsafeUtility.ReleaseGCObject(gchandle); - throw; - } - UnsafeUtility.ReleaseGCObject(gchandle); -#endif + return GetEntityQueryInternal(queryDesc); } internal void CompleteDependencyInternal() { - m_SafetyManager.CompleteDependenciesNoChecks(m_JobDependencyForReadingManagers.p, m_JobDependencyForReadingManagers.Count, m_JobDependencyForWritingManagers.p, m_JobDependencyForWritingManagers.Count); + m_SafetyManager->CompleteDependenciesNoChecks(m_JobDependencyForReadingSystems.p, + m_JobDependencyForReadingSystems.Count, m_JobDependencyForWritingSystems.p, + m_JobDependencyForWritingSystems.Count); } + + [EditorBrowsable(EditorBrowsableState.Never)] + [Obsolete("GetEntities has been deprecated. Use Entities.ForEach to access managed components. More information: https://forum.unity.com/threads/api-deprecation-faq-0-0-23.636994/", true)] + protected void GetEntities() where T : struct { } } public abstract partial class ComponentSystem : ComponentSystemBase @@ -443,6 +511,8 @@ public abstract partial class ComponentSystem : ComponentSystemBase protected internal void InitEntityQueryCache(int cacheSize) => m_EntityQueryCache = new EntityQueryCache(cacheSize); + internal EntityQueryCache EntityQueryCache => m_EntityQueryCache; + internal EntityQueryCache GetOrCreateEntityQueryCache() => m_EntityQueryCache ?? (m_EntityQueryCache = new EntityQueryCache()); @@ -452,7 +522,6 @@ unsafe void BeforeOnUpdate() { BeforeUpdateVersioning(); CompleteDependencyInternal(); - UpdateInjectedComponentGroups(); m_DeferredEntities = new EntityCommandBuffer(Allocator.TempJob, -1); } @@ -516,18 +585,20 @@ internal sealed override void InternalUpdate() } } - protected sealed override void OnBeforeCreateManagerInternal(World world) + internal sealed override void OnBeforeCreateInternal(World world) { - base.OnBeforeCreateManagerInternal(world); + base.OnBeforeCreateInternal(world); } - protected sealed override void OnBeforeDestroyManagerInternal() + internal sealed override void OnBeforeDestroyInternal() { - base.OnBeforeDestroyManagerInternal(); + base.OnBeforeDestroyInternal(); } /// - /// Called once per frame on the main thread. + /// Called once per frame on the main thread when any of this system's + /// EntityQueries would match, or if the system has the AlwaysUpdate + /// attribute. /// protected abstract void OnUpdate(); } @@ -535,14 +606,11 @@ protected sealed override void OnBeforeDestroyManagerInternal() public abstract class JobComponentSystem : ComponentSystemBase { JobHandle m_PreviousFrameDependency; - EntityCommandBufferSystem[] m_EntityCommandBufferSystemList; unsafe JobHandle BeforeOnUpdate() { BeforeUpdateVersioning(); - UpdateInjectedComponentGroups(); - // We need to wait on all previous frame dependencies, otherwise it is possible that we create infinitely long dependency chains // without anyone ever waiting on it m_PreviousFrameDependency.Complete(); @@ -558,13 +626,6 @@ unsafe void AfterOnUpdate(JobHandle outputJob, bool throwException) AddDependencyInternal(outputJob); - // Notify all injected barrier systems that they will need to sync on any jobs we spawned. - // This is conservative currently - the barriers will sync on too much if we use more than one. - for (int i = 0; i < m_EntityCommandBufferSystemList.Length; ++i) - { - m_EntityCommandBufferSystemList[i].AddJobHandleForProducer(outputJob); - } - #if ENABLE_UNITY_COLLECTIONS_CHECKS if (!JobsUtility.JobDebuggerEnabled) @@ -582,15 +643,15 @@ unsafe void AfterOnUpdate(JobHandle outputJob, bool throwException) // Which seems like debug code might have side-effects string dependencyError = null; - for (var index = 0; index < m_JobDependencyForReadingManagers.Count && dependencyError == null; index++) + for (var index = 0; index < m_JobDependencyForReadingSystems.Count && dependencyError == null; index++) { - var type = m_JobDependencyForReadingManagers.p[index]; + var type = m_JobDependencyForReadingSystems.p[index]; dependencyError = CheckJobDependencies(type); } - for (var index = 0; index < m_JobDependencyForWritingManagers.Count && dependencyError == null; index++) + for (var index = 0; index < m_JobDependencyForWritingSystems.Count && dependencyError == null; index++) { - var type = m_JobDependencyForWritingManagers.p[index]; + var type = m_JobDependencyForWritingSystems.p[index]; dependencyError = CheckJobDependencies(type); } @@ -644,17 +705,9 @@ internal sealed override void InternalUpdate() } } -#if !UNITY_ZEROPLAYER - protected sealed override void OnBeforeCreateManagerInternal(World world) - { - base.OnBeforeCreateManagerInternal(world); - - m_EntityCommandBufferSystemList = ComponentSystemInjection.GetAllInjectedManagers(this, world); - } -#endif - protected sealed override void OnBeforeDestroyManagerInternal() + internal sealed override void OnBeforeDestroyInternal() { - base.OnBeforeDestroyManagerInternal(); + base.OnBeforeDestroyInternal(); m_PreviousFrameDependency.Complete(); } @@ -669,7 +722,7 @@ public BufferFromEntity GetBufferFromEntity(bool isReadOnly = false) where #if ENABLE_UNITY_COLLECTIONS_CHECKS unsafe string CheckJobDependencies(int type) { - var h = m_SafetyManager.GetSafetyHandle(type, true); + var h = m_SafetyManager->GetSafetyHandle(type, true); var readerCount = AtomicSafetyHandle.GetReaderArray(h, 0, IntPtr.Zero); JobHandle* readers = stackalloc JobHandle[readerCount]; @@ -677,11 +730,11 @@ unsafe string CheckJobDependencies(int type) for (var i = 0; i < readerCount; ++i) { - if (!m_SafetyManager.HasReaderOrWriterDependency(type, readers[i])) + if (!m_SafetyManager->HasReaderOrWriterDependency(type, readers[i])) return $"The system {GetType()} reads {TypeManager.GetType(type)} via {AtomicSafetyHandle.GetReaderName(h, i)} but that type was not returned as a job dependency. To ensure correct behavior of other systems, the job or a dependency of it must be returned from the OnUpdate method."; } - if (!m_SafetyManager.HasReaderOrWriterDependency(type, AtomicSafetyHandle.GetWriter(h))) + if (!m_SafetyManager->HasReaderOrWriterDependency(type, AtomicSafetyHandle.GetWriter(h))) return $"The system {GetType()} writes {TypeManager.GetType(type)} via {AtomicSafetyHandle.GetWriterName(h)} but that was not returned as a job dependency. To ensure correct behavior of other systems, the job or a dependency of it must be returned from the OnUpdate method."; return null; @@ -689,28 +742,28 @@ unsafe string CheckJobDependencies(int type) unsafe void EmergencySyncAllJobs() { - for (int i = 0;i != m_JobDependencyForReadingManagers.Count;i++) + for (int i = 0;i != m_JobDependencyForReadingSystems.Count;i++) { - int type = m_JobDependencyForReadingManagers.p[i]; - AtomicSafetyHandle.EnforceAllBufferJobsHaveCompleted(m_SafetyManager.GetSafetyHandle(type, true)); + int type = m_JobDependencyForReadingSystems.p[i]; + AtomicSafetyHandle.EnforceAllBufferJobsHaveCompleted(m_SafetyManager->GetSafetyHandle(type, true)); } - for (int i = 0;i != m_JobDependencyForWritingManagers.Count;i++) + for (int i = 0;i != m_JobDependencyForWritingSystems.Count;i++) { - int type = m_JobDependencyForWritingManagers.p[i]; - AtomicSafetyHandle.EnforceAllBufferJobsHaveCompleted(m_SafetyManager.GetSafetyHandle(type, true)); + int type = m_JobDependencyForWritingSystems.p[i]; + AtomicSafetyHandle.EnforceAllBufferJobsHaveCompleted(m_SafetyManager->GetSafetyHandle(type, true)); } } #endif unsafe JobHandle GetDependency () { - return m_SafetyManager.GetDependency(m_JobDependencyForReadingManagers.p, m_JobDependencyForReadingManagers.Count, m_JobDependencyForWritingManagers.p, m_JobDependencyForWritingManagers.Count); + return m_SafetyManager->GetDependency(m_JobDependencyForReadingSystems.p, m_JobDependencyForReadingSystems.Count, m_JobDependencyForWritingSystems.p, m_JobDependencyForWritingSystems.Count); } unsafe void AddDependencyInternal(JobHandle dependency) { - m_PreviousFrameDependency = m_SafetyManager.AddDependency(m_JobDependencyForReadingManagers.p, m_JobDependencyForReadingManagers.Count, m_JobDependencyForWritingManagers.p, m_JobDependencyForWritingManagers.Count, dependency); + m_PreviousFrameDependency = m_SafetyManager->AddDependency(m_JobDependencyForReadingSystems.p, m_JobDependencyForReadingSystems.Count, m_JobDependencyForWritingSystems.p, m_JobDependencyForWritingSystems.Count, dependency); } } @@ -753,7 +806,7 @@ public EntityCommandBuffer CreateCommandBuffer() /// by this EntityCommandBufferSystem, to prevent the command buffer from being played back before /// it's complete. The general usage looks like: /// MyEntityCommandBufferSystem _barrier; - /// // in OnCreateManager(): + /// // in OnCreate(): /// _barrier = World.GetOrCreateManager(); /// // in OnUpdate(): /// EntityCommandBuffer cmd = _barrier.CreateCommandBuffer(); @@ -769,9 +822,9 @@ public void AddJobHandleForProducer(JobHandle producerJob) m_ProducerHandle = JobHandle.CombineDependencies(m_ProducerHandle, producerJob); } - protected override void OnCreateManager() + protected override void OnCreate() { - base.OnCreateManager(); + base.OnCreate(); #if ENABLE_UNITY_COLLECTIONS_CHECKS m_PendingBuffers = new List(); @@ -780,7 +833,7 @@ protected override void OnCreateManager() #endif } - protected override void OnDestroyManager() + protected override void OnDestroy() { FlushBuffers(false); @@ -788,7 +841,7 @@ protected override void OnDestroyManager() m_PendingBuffers.Dispose(); #endif - base.OnDestroyManager(); + base.OnDestroy(); } protected override void OnUpdate() @@ -861,7 +914,7 @@ private void FlushBuffers(bool playBack) #if ENABLE_UNITY_COLLECTIONS_CHECKS if (playbackErrorLog != null) { -#if !UNITY_CSHARP_TINY +#if !NET_DOTS string exceptionMessage = playbackErrorLog.Aggregate((str1, str2) => str1 + "\n" + str2); #else foreach (var err in playbackErrorLog) diff --git a/Unity.Entities/ComponentSystemGroup.cs b/Unity.Entities/ComponentSystemGroup.cs index eddd2780..55a2af9b 100644 --- a/Unity.Entities/ComponentSystemGroup.cs +++ b/Unity.Entities/ComponentSystemGroup.cs @@ -9,7 +9,7 @@ public class CircularSystemDependencyException : Exception public CircularSystemDependencyException(IEnumerable chain) { Chain = chain; -#if UNITY_CSHARP_TINY +#if NET_DOTS var lines = new List(); Console.WriteLine($"The following systems form a circular dependency cycle (check their [UpdateBefore]/[UpdateAfter] attributes):"); foreach (var s in Chain) @@ -23,7 +23,7 @@ public CircularSystemDependencyException(IEnumerable chain) public IEnumerable Chain { get; } -#if !UNITY_CSHARP_TINY +#if !NET_DOTS public override string Message { get @@ -147,7 +147,7 @@ public TypeHeapElement(int index, Type t) } public int CompareTo(TypeHeapElement other) { -#if UNITY_CSHARP_TINY +#if NET_DOTS // Workaround for missing string.CompareTo() in HPC#. This is not a fully compatible substitute, // but should be suitable for comparing system names. if (typeName.Length < other.typeName.Length) @@ -170,7 +170,7 @@ public int CompareTo(TypeHeapElement other) // Tiny doesn't have a data structure that can take Type as a key. // For now, this gives Tiny a linear search. Would like to do better. -#if !UNITY_CSHARP_TINY +#if !NET_DOTS private Dictionary lookupDictionary = null; private int LookupSysAndDep(Type t, SysAndDep[] array) { @@ -200,7 +200,7 @@ private int LookupSysAndDep(Type t, SysAndDep[] array) public virtual void SortSystemUpdateList() { -#if !UNITY_CSHARP_TINY +#if !NET_DOTS lookupDictionary = null; #endif // Populate dictionary mapping systemType to system-and-before/after-types. @@ -238,7 +238,7 @@ public virtual void SortSystemUpdateList() int depIndex = LookupSysAndDep(dep.SystemType, sysAndDep); if (depIndex < 0) { -#if !UNITY_CSHARP_TINY +#if !NET_DOTS Debug.LogWarning("Ignoring invalid [UpdateBefore] dependency for " + sys.GetType() + ": " + dep.SystemType + " must be a member of the same ComponentSystemGroup."); #else Debug.LogWarning("WARNING: invalid [UpdateBefore] dependency:"); @@ -257,7 +257,7 @@ public virtual void SortSystemUpdateList() int depIndex = LookupSysAndDep(dep.SystemType, sysAndDep); if (depIndex < 0) { -#if !UNITY_CSHARP_TINY +#if !NET_DOTS Debug.LogWarning("Ignoring invalid [UpdateAfter] dependency for " + sys.GetType() + ": " + dep.SystemType + " must be a member of the same ComponentSystemGroup."); #else Debug.LogWarning("WARNING: invalid [UpdateAfter] dependency:"); @@ -333,7 +333,7 @@ public virtual void SortSystemUpdateList() } -#if UNITY_CSHARP_TINY +#if NET_DOTS public void RecursiveLogToConsole() { foreach (var sys in m_systemsToUpdate) diff --git a/Unity.Entities/CopyUtility.cs b/Unity.Entities/CopyUtility.cs index 23192741..aadb4834 100644 --- a/Unity.Entities/CopyUtility.cs +++ b/Unity.Entities/CopyUtility.cs @@ -4,24 +4,6 @@ namespace Unity.Entities { - /// - /// Copy ComponentDataArray to NativeArray Job. - /// - /// Component data type stored in ComponentDataArray to be copied to NativeArray - [BurstCompile] - [System.Obsolete("CopyComponentData is deprecated. Use ComponentGroup.ToComponentDataArray instead.")] - public struct CopyComponentData : IJobParallelFor - where T : struct, IComponentData - { - [ReadOnly] public ComponentDataArray Source; - public NativeArray Results; - - public void Execute(int index) - { - Results[index] = Source[index]; - } - } - /// /// Assign Value to each element of NativeArray /// @@ -39,20 +21,4 @@ public void Execute(int index) Source[index] = Value; } } - - /// - /// Copy Entities from EntityArray to NativeArray - /// - [BurstCompile] - [System.Obsolete("CopyEntities is deprecated. Use ComponentGroup.ToEntityArray instead.")] - public struct CopyEntities : IJobParallelFor - { - [ReadOnly] public EntityArray Source; - public NativeArray Results; - - public void Execute(int index) - { - Results[index] = Source[index]; - } - } } diff --git a/Unity.Entities/DebugView.cs b/Unity.Entities/DebugView.cs index cf659784..a0cbb2c4 100644 --- a/Unity.Entities/DebugView.cs +++ b/Unity.Entities/DebugView.cs @@ -140,7 +140,7 @@ public unsafe EntityGroupData*[] Items } } -#if !UNITY_CSHARP_TINY +#if !NET_DOTS sealed unsafe class DebugViewUtility { [DebuggerDisplay("{name} {entity} Components: {components.Count}")] @@ -212,7 +212,7 @@ public static Components GetComponents(EntityManager m, Entity e) } #endif -#if !UNITY_CSHARP_TINY +#if !NET_DOTS sealed class EntityManagerDebugView { private EntityManager m_target; @@ -372,7 +372,7 @@ sealed class ArchetypeChunkDebugView } #endif -#if !UNITY_CSHARP_TINY +#if !NET_DOTS sealed class EntityArchetypeDebugView { private EntityArchetype m_EntityArchetype; diff --git a/Unity.Entities/Injection/DefaultTinyWorldInitialization.cs b/Unity.Entities/DefaultTinyWorldInitialization.cs similarity index 84% rename from Unity.Entities/Injection/DefaultTinyWorldInitialization.cs rename to Unity.Entities/DefaultTinyWorldInitialization.cs index 288434ea..e09f4800 100644 --- a/Unity.Entities/Injection/DefaultTinyWorldInitialization.cs +++ b/Unity.Entities/DefaultTinyWorldInitialization.cs @@ -29,8 +29,6 @@ public static World InitializeWorld(string worldName) { var world = new World(worldName); World.Active = world; - // Entity manager must be first so that other things can find it. - world.AddManager(new EntityManager()); return world; } @@ -46,13 +44,13 @@ public static void InitializeSystems(World world) // Create top level presentation system and simulation systems. InitializationSystemGroup initializationSystemGroup = new InitializationSystemGroup(); - world.AddManager(initializationSystemGroup); + world.AddSystem(initializationSystemGroup); SimulationSystemGroup simulationSystemGroup = new SimulationSystemGroup(); - world.AddManager(simulationSystemGroup); + world.AddSystem(simulationSystemGroup); PresentationSystemGroup presentationSystemGroup = new PresentationSystemGroup(); - world.AddManager(presentationSystemGroup); + world.AddSystem(presentationSystemGroup); // Create the working set of systems. int nSystems = 0; @@ -74,14 +72,14 @@ public static void InitializeSystems(World world) continue; } - if (world.GetExistingManager(allSystemTypes[i]) != null) + if (world.GetExistingSystem(allSystemTypes[i]) != null) continue; #if WRITE_LOG Console.WriteLine(allSystemNames[i]); #endif systemTypes[nSystems] = allSystemTypes[i]; systems[nSystems] = TypeManager.ConstructSystem(allSystemTypes[i]); - world.AddManager(systems[nSystems]); + world.AddSystem(systems[nSystems]); nSystems++; } #if WRITE_LOG @@ -102,7 +100,7 @@ public static void InitializeSystems(World world) for (int g = 0; g < groups.Length; ++g) { var groupType = groups[g] as UpdateInGroupAttribute; - var groupSystem = world.GetExistingManager(groupType.GroupType) as ComponentSystemGroup; + var groupSystem = world.GetExistingSystem(groupType.GroupType) as ComponentSystemGroup; if (groupSystem == null) throw new Exception("DefaultTinyWorldInitialization failed to find existing SystemGroup."); @@ -113,9 +111,9 @@ public static void InitializeSystems(World world) public static void SortSystems(World world) { - var initializationSystemGroup = world.GetExistingManager(); - var simulationSystemGroup = world.GetExistingManager(); - var presentationSystemGroup = world.GetExistingManager(); + var initializationSystemGroup = world.GetExistingSystem(); + var simulationSystemGroup = world.GetExistingSystem(); + var presentationSystemGroup = world.GetExistingSystem(); initializationSystemGroup.SortSystemUpdateList(); simulationSystemGroup.SortSystemUpdateList(); diff --git a/Unity.Entities/Injection/DefaultTinyWorldInitialization.cs.meta b/Unity.Entities/DefaultTinyWorldInitialization.cs.meta similarity index 100% rename from Unity.Entities/Injection/DefaultTinyWorldInitialization.cs.meta rename to Unity.Entities/DefaultTinyWorldInitialization.cs.meta diff --git a/Unity.Entities/DefaultWorld.cs b/Unity.Entities/DefaultWorld.cs index ba0eb855..eb796219 100644 --- a/Unity.Entities/DefaultWorld.cs +++ b/Unity.Entities/DefaultWorld.cs @@ -5,12 +5,10 @@ namespace Unity.Entities { [DisableAutoCreation] - [Preserve] [UnityEngine.ExecuteAlways] public class BeginInitializationEntityCommandBufferSystem : EntityCommandBufferSystem {} [DisableAutoCreation] - [Preserve] [UnityEngine.ExecuteAlways] public class EndInitializationEntityCommandBufferSystem : EntityCommandBufferSystem {} @@ -19,10 +17,10 @@ public class InitializationSystemGroup : ComponentSystemGroup private BeginInitializationEntityCommandBufferSystem m_BeginEntityCommandBufferSystem; private EndInitializationEntityCommandBufferSystem m_EndEntityCommandBufferSystem; - protected override void OnCreateManager() + protected override void OnCreate() { - m_BeginEntityCommandBufferSystem = World.GetOrCreateManager(); - m_EndEntityCommandBufferSystem = World.GetOrCreateManager(); + m_BeginEntityCommandBufferSystem = World.GetOrCreateSystem(); + m_EndEntityCommandBufferSystem = World.GetOrCreateSystem(); m_systemsToUpdate.Add(m_BeginEntityCommandBufferSystem); m_systemsToUpdate.Add(m_EndEntityCommandBufferSystem); } @@ -53,12 +51,10 @@ public override void SortSystemUpdateList() } [DisableAutoCreation] - [Preserve] [UnityEngine.ExecuteAlways] public class BeginSimulationEntityCommandBufferSystem : EntityCommandBufferSystem {} [DisableAutoCreation] - [Preserve] [UnityEngine.ExecuteAlways] public class EndSimulationEntityCommandBufferSystem : EntityCommandBufferSystem {} @@ -70,11 +66,11 @@ public class SimulationSystemGroup : ComponentSystemGroup private BeginSimulationEntityCommandBufferSystem m_BeginEntityCommandBufferSystem; private LateSimulationSystemGroup m_lateSimulationGroup; private EndSimulationEntityCommandBufferSystem m_EndEntityCommandBufferSystem; - protected override void OnCreateManager() + protected override void OnCreate() { - m_BeginEntityCommandBufferSystem = World.GetOrCreateManager(); - m_lateSimulationGroup = World.GetOrCreateManager(); - m_EndEntityCommandBufferSystem = World.GetOrCreateManager(); + m_BeginEntityCommandBufferSystem = World.GetOrCreateSystem(); + m_lateSimulationGroup = World.GetOrCreateSystem(); + m_EndEntityCommandBufferSystem = World.GetOrCreateSystem(); m_systemsToUpdate.Add(m_BeginEntityCommandBufferSystem); m_systemsToUpdate.Add(m_lateSimulationGroup); m_systemsToUpdate.Add(m_EndEntityCommandBufferSystem); @@ -109,12 +105,10 @@ s is LateSimulationSystemGroup || } [DisableAutoCreation] - [Preserve] [UnityEngine.ExecuteAlways] public class BeginPresentationEntityCommandBufferSystem : EntityCommandBufferSystem {} [DisableAutoCreation] - [Preserve] [UnityEngine.ExecuteAlways] public class EndPresentationEntityCommandBufferSystem : EntityCommandBufferSystem {} @@ -123,10 +117,10 @@ public class PresentationSystemGroup : ComponentSystemGroup private BeginPresentationEntityCommandBufferSystem m_BeginEntityCommandBufferSystem; private EndPresentationEntityCommandBufferSystem m_EndEntityCommandBufferSystem; - protected override void OnCreateManager() + protected override void OnCreate() { - m_BeginEntityCommandBufferSystem = World.GetOrCreateManager(); - m_EndEntityCommandBufferSystem = World.GetOrCreateManager(); + m_BeginEntityCommandBufferSystem = World.GetOrCreateSystem(); + m_EndEntityCommandBufferSystem = World.GetOrCreateSystem(); m_systemsToUpdate.Add(m_BeginEntityCommandBufferSystem); m_systemsToUpdate.Add(m_EndEntityCommandBufferSystem); } diff --git a/Unity.Entities/DeprecatedAPIStubs.cs b/Unity.Entities/DeprecatedAPIStubs.cs new file mode 100644 index 00000000..c8188fb9 --- /dev/null +++ b/Unity.Entities/DeprecatedAPIStubs.cs @@ -0,0 +1,54 @@ +using System; +using System.ComponentModel; + +namespace Unity.Entities +{ + [EditorBrowsable(EditorBrowsableState.Never)] + [Obsolete("ComponentDataArray is deprecated. Use IJobForEach or EntityQuery ToComponentDataArray/CopyFromComponentDataArray APIs instead. More information: https://forum.unity.com/threads/api-deprecation-faq-0-0-23.636994/", true)] + public unsafe struct ComponentDataArray where T : struct, IComponentData { } + + [EditorBrowsable(EditorBrowsableState.Never)] + [Obsolete("SharedComponentDataArray is deprecated. Use ArchetypeChunk.GetSharedComponentData. More information: https://forum.unity.com/threads/api-deprecation-faq-0-0-23.636994/", true)] + public struct SharedComponentDataArray where T : struct, ISharedComponentData { } + + [EditorBrowsable(EditorBrowsableState.Never)] + [Obsolete("BufferArray is deprecated. Use ArchetypeChunk.GetBufferAccessor() instead. More information: https://forum.unity.com/threads/api-deprecation-faq-0-0-23.636994/", true)] + public unsafe struct BufferArray where T : struct, IBufferElementData { } + + [EditorBrowsable(EditorBrowsableState.Never)] + [Obsolete("EntityArray is deprecated. Use IJobForEachWithEntity or EntityQuery.ToEntityArray(...) instead. More information: https://forum.unity.com/threads/api-deprecation-faq-0-0-23.636994/", true)] + public unsafe struct EntityArray { } + + [EditorBrowsable(EditorBrowsableState.Never)] + [Obsolete("ComponentGroupArray has been deprecated. Use Entities.ForEach instead to access managed components. More information: https://forum.unity.com/threads/api-deprecation-faq-0-0-23.636994/", true)] + public struct ComponentGroupArray where T : struct { } + + [EditorBrowsable(EditorBrowsableState.Never)] + [AttributeUsage(AttributeTargets.Field)] + [Obsolete("Injection API is deprecated. For struct injection, use the EntityQuery API instead. For ComponentDataFromEntity injection use (Job)ComponentSystem.GetComponentDataFromEntity. For system injection, use World.GetOrCreateManager(). More information: https://forum.unity.com/threads/api-deprecation-faq-0-0-23.636994/", true)] + public class InjectAttribute : Attribute { } + + [EditorBrowsable(EditorBrowsableState.Never)] + [Obsolete("CopyComponentData is deprecated. Use EntityQuery.ToComponentDataArray instead. More information: https://forum.unity.com/threads/api-deprecation-faq-0-0-23.636994/", true)] + public struct CopyComponentData + where T : struct, IComponentData { } + + [EditorBrowsable(EditorBrowsableState.Never)] + [Obsolete("CopyEntities is deprecated. Use EntityQuery.ToEntityArray instead. More information: https://forum.unity.com/threads/api-deprecation-faq-0-0-23.636994/", true)] + public struct CopyEntities { } + + [EditorBrowsable(EditorBrowsableState.Never)] + [Obsolete("GameObjectArray has been deprecated. Use ComponentSystem.ForEach or ToTransformArray()[index].gameObject to access entity GameObjects. More information: https://forum.unity.com/threads/api-deprecation-faq-0-0-23.636994/", true)] + public struct GameObjectArray { } + + [EditorBrowsable(EditorBrowsableState.Never)] + [Obsolete("ComponentArray has been deprecated. Use ComponentSystem.ForEach to access managed components. More information: https://forum.unity.com/threads/api-deprecation-faq-0-0-23.636994/", true)] + public struct ComponentArray where T: UnityEngine.Component { } + + public static class ComponentGroupExtensionsForGameObjectArray + { + [EditorBrowsable(EditorBrowsableState.Never)] + [Obsolete("GetGameObjectArray has been deprecated. Use ComponentSystem.ForEach or ToTransformArray()[index].gameObject to access entity GameObjects. More information: https://forum.unity.com/threads/api-deprecation-faq-0-0-23.636994/", true)] + public static void GetGameObjectArray(this ComponentGroup group) { } + } +} diff --git a/Unity.Entities.Tests/InjectComponentGroupTests.cs.meta b/Unity.Entities/DeprecatedAPIStubs.cs.meta similarity index 77% rename from Unity.Entities.Tests/InjectComponentGroupTests.cs.meta rename to Unity.Entities/DeprecatedAPIStubs.cs.meta index 111ecf8a..a3843d43 100644 --- a/Unity.Entities.Tests/InjectComponentGroupTests.cs.meta +++ b/Unity.Entities/DeprecatedAPIStubs.cs.meta @@ -1,6 +1,6 @@ fileFormatVersion: 2 -guid: 31ecd4b002f954cbfb4ca96c1a1b9d20 -timeCreated: 1505216351 +guid: bd6e9dca1b719403b9b6246585da9981 +timeCreated: 1504790404 licenseType: Pro MonoImporter: externalObjects: {} diff --git a/Unity.Entities/EntityCommandBuffer.cs b/Unity.Entities/EntityCommandBuffer.cs index 0bced056..8968f8a0 100644 --- a/Unity.Entities/EntityCommandBuffer.cs +++ b/Unity.Entities/EntityCommandBuffer.cs @@ -199,7 +199,9 @@ internal enum ECBCommand SetComponentWithEntityFixUp, AddBuffer, + AddBufferWithEntityFixUp, SetBuffer, + SetBufferWithEntityFixUp, AddSharedComponentData, SetSharedComponentData @@ -393,25 +395,36 @@ internal void AddEntityComponentCommand(EntityCommandBufferChain* chain, int } } - internal BufferHeader* AddEntityBufferCommand(EntityCommandBufferChain* chain, int jobIndex, ECBCommand op, Entity e) where T : struct, IBufferElementData + internal BufferHeader* AddEntityBufferCommand(EntityCommandBufferChain* chain, int jobIndex, ECBCommand op, + Entity e) where T : struct, IBufferElementData { var typeIndex = TypeManager.GetTypeIndex(); var type = TypeManager.GetTypeInfo(); var sizeNeeded = Align(sizeof(EntityBufferCommand) + type.SizeInChunk, 8); ResetCommandBatching(chain); - var data = (EntityBufferCommand*)Reserve(chain, jobIndex, sizeNeeded); + var cmd = (EntityBufferCommand*) Reserve(chain, jobIndex, sizeNeeded); - data->Header.Header.CommandType = (int)op; - data->Header.Header.TotalSize = sizeNeeded; - data->Header.Header.SortIndex = chain->m_LastSortIndex; - data->Header.Entity = e; - data->ComponentTypeIndex = typeIndex; - data->ComponentSize = type.SizeInChunk; + cmd->Header.Header.CommandType = (int) op; + cmd->Header.Header.TotalSize = sizeNeeded; + cmd->Header.Header.SortIndex = chain->m_LastSortIndex; + cmd->Header.Entity = e; + cmd->ComponentTypeIndex = typeIndex; + cmd->ComponentSize = type.SizeInChunk; - BufferHeader.Initialize(&data->TempBuffer, type.BufferCapacity); + BufferHeader* header = &cmd->TempBuffer; + BufferHeader.Initialize(header, type.BufferCapacity); - return &data->TempBuffer; + if (TypeManager.HasEntityReferences(typeIndex)) + { + + if (op == ECBCommand.AddBuffer) + cmd->Header.Header.CommandType = (int) ECBCommand.AddBufferWithEntityFixUp; + else if (op == ECBCommand.SetBuffer) + cmd->Header.Header.CommandType = (int) ECBCommand.SetBufferWithEntityFixUp; + } + + return header; } internal static int Align(int size, int alignmentPowerOfTwo) @@ -586,6 +599,8 @@ public bool ShouldPlayback private void EnforceSingleThreadOwnership() { #if ENABLE_UNITY_COLLECTIONS_CHECKS + if (m_Data == null) + throw new NullReferenceException("The EntityCommandBuffer has not been initialized!"); AtomicSafetyHandle.CheckWriteAndThrow(m_Safety0); #endif } @@ -808,7 +823,7 @@ private static bool IsDefaultObject(ref T component, out int hashCode) where { var defaultValue = default(T); - #if !UNITY_CSHARP_TINY + #if !NET_DOTS var typeIndex = TypeManager.GetTypeIndex(); var typeInfo = TypeManager.GetTypeInfo(typeIndex).FastEqualityTypeInfo; hashCode = FastEquality.GetHashCode(ref component, typeInfo); @@ -976,23 +991,30 @@ private static unsafe Entity SelectEntity(Entity cmdEntity, ECBSharedPlaybackSta } private static void FixupComponentData(byte* data, int typeIndex, ECBSharedPlaybackState playbackState) + { + FixupComponentData(data, 1, typeIndex, playbackState); + } + private static void FixupComponentData(byte* data, int count, int typeIndex, ECBSharedPlaybackState playbackState) { var componentTypeInfo = TypeManager.GetTypeInfo(typeIndex); Assert.IsTrue(componentTypeInfo.EntityOffsets != null); var offsets = componentTypeInfo.EntityOffsets; var offsetCount = componentTypeInfo.EntityOffsetCount; - - for (int i=0; i < offsetCount; i++) + for (var componentCount = 0; componentCount < count; componentCount++, data += componentTypeInfo.ElementSize) { - // Need fix ups - Entity* e = (Entity*) (data + offsets[i].Offset); - if (e->Index < 0) + for (int i=0; i < offsetCount; i++) { - var index = -e->Index - 1; - Entity real = *(playbackState.CreateEntityBatch + index); - *e = real; + // Need fix ups + Entity* e = (Entity*) (data + offsets[i].Offset); + if (e->Index < 0) + { + var index = -e->Index - 1; + Entity real = *(playbackState.CreateEntityBatch + index); + *e = real; + } } + } } @@ -1006,6 +1028,17 @@ private static unsafe void SetCommandDataWithFixup( playbackState); } + private static unsafe void SetBufferWithFixup( + EntityManager mgr, EntityBufferCommand* cmd, Entity entity, + ECBSharedPlaybackState playbackState) + { + byte* data = (byte*) BufferHeader.GetElementPointer(&cmd->TempBuffer); + FixupComponentData(data, cmd->TempBuffer.Length, + cmd->ComponentTypeIndex, playbackState); + + mgr.SetBufferRaw(entity, cmd->ComponentTypeIndex, &cmd->TempBuffer, cmd->ComponentSize); + } + private static unsafe void PlaybackChain(EntityManager mgr, ref ECBSharedPlaybackState playbackState, NativeArray chainStates, int currentChain, int nextChain) { int nextChainSortIndex = (nextChain != -1) ? chainStates[nextChain].NextSortIndex : -1; @@ -1123,6 +1156,15 @@ private static unsafe void PlaybackChain(EntityManager mgr, ref ECBSharedPlaybac } break; + case ECBCommand.AddBufferWithEntityFixUp: + { + var cmd = (EntityBufferCommand*) header; + var entity = SelectEntity(cmd->Header.Entity, playbackState); + mgr.AddComponent(entity, ComponentType.FromTypeIndex(cmd->ComponentTypeIndex)); + SetBufferWithFixup(mgr, cmd, entity, playbackState); + } + break; + case ECBCommand.SetBuffer: { var cmd = (EntityBufferCommand*)header; @@ -1131,6 +1173,14 @@ private static unsafe void PlaybackChain(EntityManager mgr, ref ECBSharedPlaybac } break; + case ECBCommand.SetBufferWithEntityFixUp: + { + var cmd = (EntityBufferCommand*)header; + var entity = SelectEntity(cmd->Header.Entity, playbackState); + SetBufferWithFixup(mgr, cmd, entity, playbackState); + } + break; + case ECBCommand.AddSharedComponentData: { var cmd = (EntitySharedComponentCommand*)header; @@ -1231,6 +1281,8 @@ unsafe public struct Concurrent private void CheckWriteAccess() { #if ENABLE_UNITY_COLLECTIONS_CHECKS + if (m_Data == null) + throw new NullReferenceException("The EntityCommandBuffer has not been initialized!"); AtomicSafetyHandle.CheckWriteAndThrow(m_Safety0); #endif } diff --git a/Unity.Entities/EntityDataManager.cs b/Unity.Entities/EntityDataManager.cs index abe1e146..406505e5 100644 --- a/Unity.Entities/EntityDataManager.cs +++ b/Unity.Entities/EntityDataManager.cs @@ -384,7 +384,7 @@ public void AssertCanAddComponent(Entity entity, ComponentType componentType) if (!Exists(entity)) throw new ArgumentException("The entity does not exist"); - if (HasComponent(entity, componentType)) + if (!componentType.IgnoreDuplicateAdd && HasComponent(entity, componentType)) throw new ArgumentException($"The component of type:{componentType} has already been added to the entity."); var chunk = GetComponentChunk(entity); @@ -1003,6 +1003,7 @@ public void AddComponents(Entity entity, ComponentTypes types, ArchetypeManager // zipper the two sorted arrays "type" and "componentTypeInArchetype" into "componentTypeInArchetype" // because this is done in-place, it must be done backwards so as not to disturb the existing contents. + var unusedIndices = 0; { var oldThings = oldArchetype->TypesCount; var newThings = types.Length; @@ -1018,30 +1019,27 @@ public void AddComponents(Entity entity, ComponentTypes types, ArchetypeManager } else { + if (oldThing.TypeIndex == newThing.TypeIndex && newThing.IgnoreDuplicateAdd) + --oldThings; + var componentTypeInArchetype = new ComponentTypeInArchetype(newThing); newTypes[--mixedThings] = componentTypeInArchetype; --newThings; indexOfNewTypeInNewArchetype[newThings] = mixedThings; // "this new thing ended up HERE" } } - while (newThings > 0) // if smallest new things < smallest old things, copy them here - { - var newThing = types.GetComponentType(newThings - 1); - var componentTypeInArchetype = new ComponentTypeInArchetype(newThing); - newTypes[--mixedThings] = componentTypeInArchetype; - --newThings; - indexOfNewTypeInNewArchetype[newThings] = mixedThings; // "this new thing ended up HERE" - } + + Assert.AreEqual(0, newThings); // must not be any new things to copy remaining, oldThings contain entity + while (oldThings > 0) // if there are remaining old things, copy them here { newTypes[--mixedThings] = oldTypes[--oldThings]; } - Assert.AreEqual(newThings, 0); // must not be any new things to copy remaining - Assert.AreEqual(oldThings, mixedThings); // all things we didn't copy must be old + unusedIndices = mixedThings; // In case we ignored duplicated types, this will be > 0 } - var newArchetype = archetypeManager.GetOrCreateArchetype(newTypes, newTypesCount, groupManager); + var newArchetype = archetypeManager.GetOrCreateArchetype(newTypes + unusedIndices, newTypesCount, groupManager); var sharedComponentValues = GetComponentChunk(entity)->SharedComponentValues; if (types.m_masks.m_SharedComponentMask != 0) @@ -1066,6 +1064,12 @@ public void AddComponent(Entity entity, ComponentType type, ArchetypeManager arc int indexInTypeArray=0; var newType = archetypeManager.GetArchetypeWithAddedComponentType(archetype, type, groupManager, &indexInTypeArray); + if (newType == null) + { + // This can happen if we are adding a tag component to an entity that already has it. + return; + } + var sharedComponentValues = GetComponentChunk(entity)->SharedComponentValues; if (type.IsSharedComponent) { diff --git a/Unity.Entities/EntityManager.cs b/Unity.Entities/EntityManager.cs index 97a6964c..81533673 100644 --- a/Unity.Entities/EntityManager.cs +++ b/Unity.Entities/EntityManager.cs @@ -244,7 +244,7 @@ public bool Equals(Entity entity) /// A string containing the entity index and generational version. public override string ToString() { - return $"Entity Index: {Index} Version: {Version}"; + return Equals(Entity.Null) ? "Entity.Null" : $"Entity({Index}:{Version})"; } } @@ -267,9 +267,10 @@ public override string ToString() /// [Preserve] [DebuggerTypeProxy(typeof(EntityManagerDebugView))] - public sealed unsafe class EntityManager : ScriptBehaviourManager + public sealed unsafe partial class EntityManager { EntityDataManager* m_Entities; + ComponentJobSafetyManager* m_ComponentJobSafetyManager; ArchetypeManager m_ArchetypeManager; EntityGroupManager m_GroupManager; @@ -279,11 +280,11 @@ public sealed unsafe class EntityManager : ScriptBehaviourManager ExclusiveEntityTransaction m_ExclusiveEntityTransaction; World m_World; - private ComponentGroup m_UniversalGroup; // matches all components + private EntityQuery m_UniversalQuery; // matches all components /// - /// A ComponentGroup instance that matches all components. + /// A EntityQuery instance that matches all components. /// - public ComponentGroup UniversalGroup => m_UniversalGroup; + public EntityQuery UniversalQuery => m_UniversalQuery; #if ENABLE_UNITY_COLLECTIONS_CHECKS int m_InsideForEach; @@ -308,6 +309,11 @@ internal EntityDataManager* Entities { get { return m_Entities; } } + + internal ComponentJobSafetyManager* ComponentJobSafetyManager + { + get { return m_ComponentJobSafetyManager; } + } internal EntityGroupManager GroupManager { @@ -364,16 +370,14 @@ public int EntityCapacity } } - internal ComponentJobSafetyManager ComponentJobSafetyManager { get; private set; } - /// /// The Job dependencies of the exclusive entity transaction. /// /// public JobHandle ExclusiveEntityTransactionDependency { - get { return ComponentJobSafetyManager.ExclusiveTransactionDependency; } - set { ComponentJobSafetyManager.ExclusiveTransactionDependency = value; } + get { return ComponentJobSafetyManager->ExclusiveTransactionDependency; } + set { ComponentJobSafetyManager->ExclusiveTransactionDependency = value; } } EntityManagerDebug m_Debug; @@ -383,55 +387,47 @@ public JobHandle ExclusiveEntityTransactionDependency /// public EntityManagerDebug Debug => m_Debug ?? (m_Debug = new EntityManagerDebug(this)); - protected override void OnBeforeCreateManagerInternal(World world) - { - m_World = world; - } - - protected override void OnBeforeDestroyManagerInternal() - { - } - - protected override void OnAfterDestroyManagerInternal() - { - } - - protected override void OnCreateManager() + internal EntityManager(World world) { TypeManager.Initialize(); + m_World = world; + m_Entities = (EntityDataManager*) UnsafeUtility.Malloc(sizeof(EntityDataManager), 64, Allocator.Persistent); - m_Entities->OnCreate(); + m_Entities->OnCreate(); m_SharedComponentManager = new SharedComponentDataManager(); + + m_ComponentJobSafetyManager = (ComponentJobSafetyManager*) UnsafeUtility.Malloc(sizeof(ComponentJobSafetyManager), 64, Allocator.Persistent); + m_ComponentJobSafetyManager->OnCreate(); - ComponentJobSafetyManager = new ComponentJobSafetyManager(); - m_GroupManager = new EntityGroupManager(ComponentJobSafetyManager); + m_GroupManager = new EntityGroupManager(m_ComponentJobSafetyManager); m_ArchetypeManager = new ArchetypeManager(m_SharedComponentManager, m_Entities, m_GroupManager); m_ExclusiveEntityTransaction = new ExclusiveEntityTransaction(ArchetypeManager, m_GroupManager, m_SharedComponentManager, Entities); - m_UniversalGroup = CreateComponentGroup( - new EntityArchetypeQuery + m_UniversalQuery = CreateEntityQuery( + new EntityQueryDesc { - Options = EntityArchetypeQueryOptions.IncludePrefab | EntityArchetypeQueryOptions.IncludeDisabled + Options = EntityQueryOptions.IncludePrefab | EntityQueryOptions.IncludeDisabled } ); } - protected override void OnDestroyManager() + internal void DestroyInstance() { EndExclusiveEntityTransaction(); - ComponentJobSafetyManager.PreDisposeCheck(); + m_ComponentJobSafetyManager->PreDisposeCheck(); - m_UniversalGroup.Dispose(); - m_UniversalGroup = null; + m_UniversalQuery.Dispose(); + m_UniversalQuery = null; - ComponentJobSafetyManager.Dispose(); - ComponentJobSafetyManager = null; + m_ComponentJobSafetyManager->Dispose(); + UnsafeUtility.Free(m_ComponentJobSafetyManager, Allocator.Persistent); + m_ComponentJobSafetyManager = null; m_Entities->OnDestroy(); UnsafeUtility.Free(m_Entities, Allocator.Persistent); @@ -440,16 +436,24 @@ protected override void OnDestroyManager() m_ArchetypeManager = null; m_GroupManager.Dispose(); m_GroupManager = null; - m_ExclusiveEntityTransaction.OnDestroyManager(); + m_ExclusiveEntityTransaction.OnDestroy(); m_SharedComponentManager.Dispose(); m_World = null; m_Debug = null; + + TypeManager.Shutdown(); } - internal override void InternalUpdate() + private EntityManager() { + // for tests only + } + + internal static EntityManager CreateEntityManagerInUninitializedState() + { + return new EntityManager(); } internal static int FillSortedArchetypeArray(ComponentTypeInArchetype* dst, ComponentType* requiredComponents, int count) @@ -466,30 +470,30 @@ internal static int FillSortedArchetypeArray(ComponentTypeInArchetype* dst, Comp } /// - /// Creates a ComponentGroup from an array of component types. + /// Creates a EntityQuery from an array of component types. /// /// An array containing the component types. - /// The ComponentGroup derived from the specified array of component types. - /// - public ComponentGroup CreateComponentGroup(params ComponentType[] requiredComponents) + /// The EntityQuery derived from the specified array of component types. + /// + public EntityQuery CreateEntityQuery(params ComponentType[] requiredComponents) { fixed (ComponentType* requiredComponentsPtr = requiredComponents) { return m_GroupManager.CreateEntityGroup(ArchetypeManager, Entities, requiredComponentsPtr, requiredComponents.Length); } } - internal ComponentGroup CreateComponentGroup(ComponentType* requiredComponents, int count) + internal EntityQuery CreateEntityQuery(ComponentType* requiredComponents, int count) { return m_GroupManager.CreateEntityGroup(ArchetypeManager, Entities, requiredComponents, count); } /// - /// Creates a ComponentGroup from an EntityArchetypeQuery. + /// Creates a EntityQuery from an EntityQueryDesc. /// - /// A query identifying a set of component types. - /// The ComponentGroup corresponding to the query. - public ComponentGroup CreateComponentGroup(params EntityArchetypeQuery[] queries) + /// A queryDesc identifying a set of component types. + /// The EntityQuery corresponding to the queryDesc. + public EntityQuery CreateEntityQuery(params EntityQueryDesc[] queriesDesc) { - return m_GroupManager.CreateEntityGroup(ArchetypeManager, Entities, queries); + return m_GroupManager.CreateEntityGroup(ArchetypeManager, Entities, queriesDesc); } internal EntityArchetype CreateArchetype(ComponentType* types, int count) @@ -597,28 +601,28 @@ public void UnlockChunk(NativeArray chunks) } - public void LockChunkOrder(ComponentGroup group) + public void LockChunkOrder(EntityQuery query) { - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { LockChunksInternal(chunks.GetUnsafePtr(), chunks.Length, ChunkFlags.LockedEntityOrder); } } - + public void LockChunkOrder(ArchetypeChunk chunk) { LockChunksInternal(&chunk, 1, ChunkFlags.LockedEntityOrder); } - - public void UnlockChunkOrder(ComponentGroup group) + + public void UnlockChunkOrder(EntityQuery query) { - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { UnlockChunksInternal(chunks.GetUnsafePtr(), chunks.Length, ChunkFlags.LockedEntityOrder); } } - + public void UnlockChunkOrder(ArchetypeChunk chunk) { UnlockChunksInternal(&chunk, 1, ChunkFlags.LockedEntityOrder); @@ -628,7 +632,7 @@ internal void UnlockChunksInternal(void* chunks, int count, ChunkFlags flags) { Entities->UnlockChunks((Chunk**)chunks, count, flags); } - + internal void LockChunksInternal(void* chunks, int count, ChunkFlags flags) { Entities->LockChunks((Chunk**)chunks, count, flags); @@ -711,7 +715,7 @@ public Entity CreateEntity() Entities->CreateEntities(ArchetypeManager, ArchetypeManager.GetEntityOnlyArchetype(GroupManager), &entity, 1); return entity; } - + internal void CreateEntityInternal(EntityArchetype archetype, Entity* entities, int count) { BeforeStructuralChange(); @@ -724,10 +728,10 @@ internal void CreateChunkInternal(EntityArchetype archetype, ArchetypeChunk* chu Entities->CreateChunks(ArchetypeManager, archetype.Archetype, chunks, entityCount); } - NativeArray GetTempEntityArray(ComponentGroup group) + NativeArray GetTempEntityArray(EntityQuery query) { //@TODO: Using Allocator.Temp here throws an exception... - var entityArray = group.ToEntityArray(Allocator.TempJob); + var entityArray = query.ToEntityArray(Allocator.TempJob); return entityArray; } @@ -735,19 +739,19 @@ NativeArray GetTempEntityArray(ComponentGroup group) /// Destroy all entities having a common set of component types. /// /// Since entities in the same chunk share the same component structure, this function effectively destroys - /// the chunks holding any entities identified by the `componentGroupFilter` parameter. - /// Defines the components an entity must have to qualify for destruction. - public void DestroyEntity(ComponentGroup componentGroupFilter) + /// the chunks holding any entities identified by the `entityQueryFilter` parameter. + /// Defines the components an entity must have to qualify for destruction. + public void DestroyEntity(EntityQuery entityQueryFilter) { - //@TODO: When destroying entities with componentGroupFilter we assume that any LinkedEntityGroup also get destroyed + //@TODO: When destroying entities with entityQueryFilter we assume that any LinkedEntityGroup also get destroyed // We should have some sort of validation that everything is included, and either give an error message or have a fast enough path to handle it... //@TODO: Locked checks... - - Profiler.BeginSample("DestroyEntity(ComponentGroup componentGroupFilter)"); + + Profiler.BeginSample("DestroyEntity(EntityQuery entityQueryFilter)"); Profiler.BeginSample("GetAllMatchingChunks"); - using (var chunks = componentGroupFilter.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = entityQueryFilter.CreateArchetypeChunkArray(Allocator.TempJob)) { Profiler.EndSample(); @@ -971,13 +975,16 @@ internal void InstantiateInternal(Entity srcEntity, Entity* outputEntities, int /// The type of component to add. public void AddComponent(Entity entity, ComponentType componentType) { + if (componentType.IgnoreDuplicateAdd && HasComponent(entity, componentType)) + return; + BeforeStructuralChange(); Entities->AssertCanAddComponent(entity, componentType); Entities->AddComponent(entity, componentType, ArchetypeManager, m_SharedComponentManager, m_GroupManager); } /// - /// Adds a component to a set of entities defined by a ComponentGroup. + /// Adds a component to a set of entities defined by a EntityQuery. /// /// /// Adding a component changes an entity's archetype and results in the entity being moved to a different @@ -990,11 +997,11 @@ public void AddComponent(Entity entity, ComponentType componentType) /// the function is finished. A sync point can cause a drop in performance because the ECS framework may not /// be able to make use of the processing power of all available cores. /// - /// The ComponentGroup defining the entities to modify. + /// The EntityQuery defining the entities to modify. /// The type of component to add. - public void AddComponent(ComponentGroup componentGroup, ComponentType componentType) + public void AddComponent(EntityQuery entityQuery, ComponentType componentType) { - using (var chunks = componentGroup.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = entityQuery.CreateArchetypeChunkArray(Allocator.TempJob)) { if (chunks.Length == 0) return; @@ -1077,7 +1084,7 @@ public void RemoveComponent(Entity entity, ComponentType type) } /// - /// Removes a component from a set of entities defined by a ComponentGroup. + /// Removes a component from a set of entities defined by a EntityQuery. /// /// /// Removing a component changes an entity's archetype and results in the entity being moved to a different @@ -1088,11 +1095,11 @@ public void RemoveComponent(Entity entity, ComponentType type) /// the function is finished. A sync point can cause a drop in performance because the ECS framework may not /// be able to make use of the processing power of all available cores. /// - /// The ComponentGroup defining the entities to modify. + /// The EntityQuery defining the entities to modify. /// The type of component to remove. - public void RemoveComponent(ComponentGroup componentGroup, ComponentType componentType) + public void RemoveComponent(EntityQuery entityQuery, ComponentType componentType) { - using (var chunks = componentGroup.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = entityQuery.CreateArchetypeChunkArray(Allocator.TempJob)) { if (chunks.Length == 0) return; @@ -1101,15 +1108,15 @@ public void RemoveComponent(ComponentGroup componentGroup, ComponentType compone Entities->RemoveComponent(chunks, componentType, ArchetypeManager, m_GroupManager, m_SharedComponentManager); } } - - public void RemoveComponent(ComponentGroup componentGroupFilter, ComponentTypes types) + + public void RemoveComponent(EntityQuery entityQueryFilter, ComponentTypes types) { - if (componentGroupFilter.CalculateLength() == 0) + if (entityQueryFilter.CalculateLength() == 0) return; // @TODO: Opportunity to do all components in batch on a per chunk basis. for (int i = 0; i != types.Length; i++) - RemoveComponent(componentGroupFilter, types.GetComponentType(i)); + RemoveComponent(entityQueryFilter, types.GetComponentType(i)); } //@TODO: optimize for batch @@ -1220,10 +1227,10 @@ public void AddChunkComponentData(Entity entity) where T : struct, IComponent } /// - /// Adds a component to each of the chunks identified by a ComponentGroup and set the component values. + /// Adds a component to each of the chunks identified by a EntityQuery and set the component values. /// /// - /// This function finds all chunks whose archetype satisfies the ComponentGroup and adds the specified + /// This function finds all chunks whose archetype satisfies the EntityQuery and adds the specified /// component to them. /// /// A chunk component is common to all entities in a chunk. You can access a chunk @@ -1234,12 +1241,12 @@ public void AddChunkComponentData(Entity entity) where T : struct, IComponent /// the function is finished. A sync point can cause a drop in performance because the ECS framework may not /// be able to make use of the processing power of all available cores. /// - /// The ComponentGroup identifying the chunks to modify. + /// The EntityQuery identifying the chunks to modify. /// The data to set. /// The type of component, which must implement IComponentData. - public void AddChunkComponentData(ComponentGroup componentGroup, T componentData) where T : struct, IComponentData + public void AddChunkComponentData(EntityQuery entityQuery, T componentData) where T : struct, IComponentData { - using (var chunks = componentGroup.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = entityQuery.CreateArchetypeChunkArray(Allocator.TempJob)) { if (chunks.Length == 0) return; @@ -1250,7 +1257,7 @@ public void AddChunkComponentData(ComponentGroup componentGroup, T componentD } /// - /// Removes a component from the chunks identified by a ComponentGroup. + /// Removes a component from the chunks identified by a EntityQuery. /// /// /// A chunk component is common to all entities in a chunk. You can access a chunk @@ -1261,11 +1268,11 @@ public void AddChunkComponentData(ComponentGroup componentGroup, T componentD /// the function is finished. A sync point can cause a drop in performance because the ECS framework may not /// be able to make use of the processing power of all available cores. /// - /// The ComponentGroup identifying the chunks to modify. + /// The EntityQuery identifying the chunks to modify. /// The type of component to remove. - public void RemoveChunkComponentData(ComponentGroup componentGroup) + public void RemoveChunkComponentData(EntityQuery entityQuery) { - using (var chunks = componentGroup.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = entityQuery.CreateArchetypeChunkArray(Allocator.TempJob)) { if (chunks.Length == 0) return; @@ -1312,7 +1319,7 @@ internal ComponentDataFromEntity GetComponentDataFromEntity(int typeIndex, { #if ENABLE_UNITY_COLLECTIONS_CHECKS return new ComponentDataFromEntity(typeIndex, Entities, - ComponentJobSafetyManager.GetSafetyHandle(typeIndex, isReadOnly)); + ComponentJobSafetyManager->GetSafetyHandle(typeIndex, isReadOnly)); #else return new ComponentDataFromEntity(typeIndex, m_Entities); #endif @@ -1329,8 +1336,8 @@ internal BufferFromEntity GetBufferFromEntity(int typeIndex, bool isReadOn { #if ENABLE_UNITY_COLLECTIONS_CHECKS return new BufferFromEntity(typeIndex, Entities, isReadOnly, - ComponentJobSafetyManager.GetSafetyHandle(typeIndex, isReadOnly), - ComponentJobSafetyManager.GetBufferSafetyHandle(typeIndex)); + ComponentJobSafetyManager->GetSafetyHandle(typeIndex, isReadOnly), + ComponentJobSafetyManager->GetBufferSafetyHandle(typeIndex)); #else return new BufferFromEntity(typeIndex, m_Entities, isReadOnly); #endif @@ -1354,7 +1361,7 @@ public T GetComponentData(Entity entity) where T : struct, IComponentData throw new System.ArgumentException($"GetComponentData<{typeof(T)}> can not be called with a zero sized component."); #endif - ComponentJobSafetyManager.CompleteWriteDependency(typeIndex); + ComponentJobSafetyManager->CompleteWriteDependency(typeIndex); var ptr = Entities->GetComponentDataWithTypeRO(entity, typeIndex); @@ -1381,7 +1388,7 @@ public void SetComponentData(Entity entity, T componentData) where T : struct throw new System.ArgumentException($"SetComponentData<{typeof(T)}> can not be called with a zero sized component."); #endif - ComponentJobSafetyManager.CompleteReadAndWriteDependency(typeIndex); + ComponentJobSafetyManager->CompleteReadAndWriteDependency(typeIndex); var ptr = Entities->GetComponentDataWithTypeRW(entity, typeIndex, Entities->GlobalSystemVersion); UnsafeUtility.CopyStructureToPtr(ref componentData, ptr); @@ -1617,7 +1624,7 @@ public void AddSharedComponentData(Entity entity, T componentData) where T : } /// - /// Adds a shared component to a set of entities defined by a ComponentGroup. + /// Adds a shared component to a set of entities defined by a EntityQuery. /// /// /// The fields of the `componentData` parameter are assigned to all of the added shared components. @@ -1631,13 +1638,13 @@ public void AddSharedComponentData(Entity entity, T componentData) where T : /// the function is finished. A sync point can cause a drop in performance because the ECS framework may not /// be able to make use of the processing power of all available cores. /// - /// The ComponentGroup defining a set of entities to modify. + /// The EntityQuery defining a set of entities to modify. /// The data to set. /// The data type of the shared component. - public void AddSharedComponentData(ComponentGroup componentGroup, T componentData) where T : struct, ISharedComponentData + public void AddSharedComponentData(EntityQuery entityQuery, T componentData) where T : struct, ISharedComponentData { var componentType = ComponentType.ReadWrite(); - using (var chunks = componentGroup.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = entityQuery.CreateArchetypeChunkArray(Allocator.TempJob)) { if (chunks.Length == 0) return; @@ -1687,7 +1694,7 @@ public void SetSharedComponentData(Entity entity, T componentData) where T : internal void SetSharedComponentDataBoxed(Entity entity, int typeIndex, object componentData) { - var hashCode = SharedComponentDataManager.GetHashCodeFast(componentData, typeIndex); + var hashCode = TypeManager.GetHashCode(componentData, typeIndex); SetSharedComponentDataBoxed(entity, typeIndex, hashCode, componentData); } @@ -1725,12 +1732,12 @@ public DynamicBuffer GetBuffer(Entity entity) where T : struct, IBufferEle $"GetBuffer<{typeof(T)}> may not be IComponentData or ISharedComponentData; currently {TypeManager.GetTypeInfo().Category}"); #endif - ComponentJobSafetyManager.CompleteReadAndWriteDependency(typeIndex); + ComponentJobSafetyManager->CompleteReadAndWriteDependency(typeIndex); BufferHeader* header = (BufferHeader*) Entities->GetComponentDataWithTypeRW(entity, typeIndex, Entities->GlobalSystemVersion); #if ENABLE_UNITY_COLLECTIONS_CHECKS - return new DynamicBuffer(header, ComponentJobSafetyManager.GetSafetyHandle(typeIndex, false), ComponentJobSafetyManager.GetBufferSafetyHandle(typeIndex), false); + return new DynamicBuffer(header, ComponentJobSafetyManager->GetSafetyHandle(typeIndex, false), ComponentJobSafetyManager->GetBufferSafetyHandle(typeIndex), false); #else return new DynamicBuffer(header); #endif @@ -1740,7 +1747,7 @@ public DynamicBuffer GetBuffer(Entity entity) where T : struct, IBufferEle { Entities->AssertEntityHasComponent(entity, typeIndex); - ComponentJobSafetyManager.CompleteReadAndWriteDependency(typeIndex); + ComponentJobSafetyManager->CompleteReadAndWriteDependency(typeIndex); BufferHeader* header = (BufferHeader*)Entities->GetComponentDataWithTypeRW(entity, typeIndex, Entities->GlobalSystemVersion); @@ -1751,7 +1758,7 @@ public DynamicBuffer GetBuffer(Entity entity) where T : struct, IBufferEle { Entities->AssertEntityHasComponent(entity, typeIndex); - ComponentJobSafetyManager.CompleteReadAndWriteDependency(typeIndex); + ComponentJobSafetyManager->CompleteReadAndWriteDependency(typeIndex); BufferHeader* header = (BufferHeader*)Entities->GetComponentDataWithTypeRO(entity, typeIndex); @@ -1762,7 +1769,7 @@ internal int GetBufferLength(Entity entity, int typeIndex) { Entities->AssertEntityHasComponent(entity, typeIndex); - ComponentJobSafetyManager.CompleteReadAndWriteDependency(typeIndex); + ComponentJobSafetyManager->CompleteReadAndWriteDependency(typeIndex); BufferHeader* header = (BufferHeader*)Entities->GetComponentDataWithTypeRO(entity, typeIndex); @@ -1832,7 +1839,7 @@ public NativeArray GetAllChunks(Allocator allocator = Allocator. { BeforeStructuralChange(); - return m_UniversalGroup.CreateArchetypeChunkArray(allocator); + return m_UniversalQuery.CreateArchetypeChunkArray(allocator); } /// @@ -1860,7 +1867,7 @@ internal void SetBufferRaw(Entity entity, int componentTypeIndex, BufferHeader* { Entities->AssertEntityHasComponent(entity, componentTypeIndex); - ComponentJobSafetyManager.CompleteReadAndWriteDependency(componentTypeIndex); + ComponentJobSafetyManager->CompleteReadAndWriteDependency(componentTypeIndex); var ptr = Entities->GetComponentDataWithTypeRW(entity, componentTypeIndex, Entities->GlobalSystemVersion); @@ -1895,7 +1902,7 @@ internal void SetComponentDataRaw(Entity entity, int typeIndex, void* data, int { Entities->AssertEntityHasComponent(entity, typeIndex); - ComponentJobSafetyManager.CompleteReadAndWriteDependency(typeIndex); + ComponentJobSafetyManager->CompleteReadAndWriteDependency(typeIndex); #if ENABLE_UNITY_COLLECTIONS_CHECKS if (TypeManager.GetTypeInfo(typeIndex).SizeInChunk != size) @@ -1910,7 +1917,7 @@ internal void SetComponentDataRaw(Entity entity, int typeIndex, void* data, int { Entities->AssertEntityHasComponent(entity, typeIndex); - ComponentJobSafetyManager.CompleteReadAndWriteDependency(typeIndex); + ComponentJobSafetyManager->CompleteReadAndWriteDependency(typeIndex); #if ENABLE_UNITY_COLLECTIONS_CHECKS if (TypeManager.GetTypeInfo(typeIndex).IsZeroSized) @@ -1926,7 +1933,7 @@ internal void SetComponentDataRaw(Entity entity, int typeIndex, void* data, int { Entities->AssertEntityHasComponent(entity, typeIndex); - ComponentJobSafetyManager.CompleteReadAndWriteDependency(typeIndex); + ComponentJobSafetyManager->CompleteReadAndWriteDependency(typeIndex); #if ENABLE_UNITY_COLLECTIONS_CHECKS if (TypeManager.GetTypeInfo(typeIndex).IsZeroSized) @@ -2006,9 +2013,9 @@ public int GetSharedComponentOrderVersion(T sharedComponent) where T : struct /// A transaction object that provides an functions for making structural changes. public ExclusiveEntityTransaction BeginExclusiveEntityTransaction() { - ComponentJobSafetyManager.BeginExclusiveTransaction(); + ComponentJobSafetyManager->BeginExclusiveTransaction(); #if ENABLE_UNITY_COLLECTIONS_CHECKS - m_ExclusiveEntityTransaction.SetAtomicSafetyHandle(ComponentJobSafetyManager.ExclusiveTransactionSafety); + m_ExclusiveEntityTransaction.SetAtomicSafetyHandle(ComponentJobSafetyManager->ExclusiveTransactionSafety); #endif return m_ExclusiveEntityTransaction; } @@ -2020,13 +2027,13 @@ public ExclusiveEntityTransaction BeginExclusiveEntityTransaction() /// public void EndExclusiveEntityTransaction() { - ComponentJobSafetyManager.EndExclusiveTransaction(); + ComponentJobSafetyManager->EndExclusiveTransaction(); } private void BeforeStructuralChange() { #if ENABLE_UNITY_COLLECTIONS_CHECKS - if (ComponentJobSafetyManager.IsInTransaction) + if (ComponentJobSafetyManager->IsInTransaction) throw new InvalidOperationException( "Access to EntityManager is not allowed after EntityManager.BeginExclusiveEntityTransaction(); has been called."); @@ -2034,7 +2041,7 @@ private void BeforeStructuralChange() throw new InvalidOperationException("EntityManager.AddComponent/RemoveComponent/CreateEntity/DestroyEntity are not allowed during Entities.ForEach. Please use PostUpdateCommandBuffer to delay applying those changes until after ForEach."); #endif - ComponentJobSafetyManager.CompleteAllJobsAndInvalidateArrays(); + ComponentJobSafetyManager->CompleteAllJobsAndInvalidateArrays(); } //@TODO: Not clear to me what this method is really for... @@ -2044,7 +2051,7 @@ private void BeforeStructuralChange() /// Calling CompleteAllJobs() blocks the main thread until all currently running Jobs finish. public void CompleteAllJobs() { - ComponentJobSafetyManager.CompleteAllJobsAndInvalidateArrays(); + ComponentJobSafetyManager->CompleteAllJobsAndInvalidateArrays(); } /// @@ -2099,8 +2106,11 @@ public void MoveEntitiesFrom(EntityManager srcEntities, NativeArrayCapacity) + throw new ArgumentException("entityRemapping.Length isn't large enough, use srcEntities.CreateEntityRemapArray"); + if (!srcEntities.m_SharedComponentManager.AllSharedComponentReferencesAreFromChunks(srcEntities.ArchetypeManager)) - throw new ArgumentException("EntityManager.MoveEntitiesFrom failed - All ISharedComponentData references must be from EntityManager. (For example ComponentGroup.SetFilter with a shared component type is not allowed during EntityManager.MoveEntitiesFrom)"); + throw new ArgumentException("EntityManager.MoveEntitiesFrom failed - All ISharedComponentData references must be from EntityManager. (For example EntityQuery.SetFilter with a shared component type is not allowed during EntityManager.MoveEntitiesFrom)"); #endif BeforeStructuralChange(); @@ -2136,12 +2146,12 @@ public void MoveEntitiesFrom(EntityManager srcEntities, NativeArray /// The EntityManager whose entities are appropriated. - /// A ComponentGroup that defines the entities to move. Must be part of the source + /// A EntityQuery that defines the entities to move. Must be part of the source /// World. /// A set of entity transformations to make during the transfer. - /// Thrown if the ComponentGroup object used as the `filter` comes + /// Thrown if the EntityQuery object used as the `filter` comes /// from a different world than the `srcEntities` EntityManager. - public void MoveEntitiesFrom(EntityManager srcEntities, ComponentGroup filter, NativeArray entityRemapping) + public void MoveEntitiesFrom(EntityManager srcEntities, EntityQuery filter, NativeArray entityRemapping) { #if ENABLE_UNITY_COLLECTIONS_CHECKS if(filter.ArchetypeManager != srcEntities.ArchetypeManager) @@ -2158,6 +2168,10 @@ internal void MoveEntitiesFrom(EntityManager srcEntities, NativeArrayCapacity) + throw new ArgumentException("entityRemapping.Length isn't large enough, use srcEntities.CreateEntityRemapArray"); + for(int i=0;iArchetype->HasChunkHeader) throw new ArgumentException("MoveEntitiesFrom can not move chunks that contain ChunkHeader components."); @@ -2226,7 +2240,7 @@ public void MoveEntitiesFrom(out NativeArray output, EntityManager srcEn throw new ArgumentException("srcEntities must not be the same as this EntityManager."); if (!srcEntities.m_SharedComponentManager.AllSharedComponentReferencesAreFromChunks(srcEntities.ArchetypeManager)) - throw new ArgumentException("EntityManager.MoveEntitiesFrom failed - All ISharedComponentData references must be from EntityManager. (For example ComponentGroup.SetFilter with a shared component type is not allowed during EntityManager.MoveEntitiesFrom)"); + throw new ArgumentException("EntityManager.MoveEntitiesFrom failed - All ISharedComponentData references must be from EntityManager. (For example EntityQuery.SetFilter with a shared component type is not allowed during EntityManager.MoveEntitiesFrom)"); #endif BeforeStructuralChange(); @@ -2255,11 +2269,11 @@ public void MoveEntitiesFrom(out NativeArray output, EntityManager srcEn /// /// An array to receive the Entity objects of the transferred entities. /// The EntityManager whose entities are appropriated. - /// A ComponentGroup that defines the entities to move. Must be part of the source + /// A EntityQuery that defines the entities to move. Must be part of the source /// World. /// A set of entity transformations to make during the transfer. /// - public void MoveEntitiesFrom(out NativeArray output, EntityManager srcEntities, ComponentGroup filter, NativeArray entityRemapping) + public void MoveEntitiesFrom(out NativeArray output, EntityManager srcEntities, EntityQuery filter, NativeArray entityRemapping) { #if ENABLE_UNITY_COLLECTIONS_CHECKS if(filter.ArchetypeManager != srcEntities.ArchetypeManager) @@ -2377,17 +2391,17 @@ private bool TestMatchingArchetypeAll(Archetype* archetype, ComponentType* allTy } [Obsolete("This function is deprecated and will be removed in a future release.")] - public void AddMatchingArchetypes(EntityArchetypeQuery query, NativeList foundArchetypes) + public void AddMatchingArchetypes(EntityQueryDesc queryDesc, NativeList foundArchetypes) { - var anyCount = query.Any.Length; - var noneCount = query.None.Length; - var allCount = query.All.Length; + var anyCount = queryDesc.Any.Length; + var noneCount = queryDesc.None.Length; + var allCount = queryDesc.All.Length; - fixed (ComponentType* any = query.Any) + fixed (ComponentType* any = queryDesc.Any) { - fixed (ComponentType* none = query.None) + fixed (ComponentType* none = queryDesc.None) { - fixed (ComponentType* all = query.All) + fixed (ComponentType* all = queryDesc.All) { for (var i = ArchetypeManager.m_Archetypes.Count - 1; i >= 0; --i) { @@ -2427,7 +2441,7 @@ public void GetAllArchetypes(NativeList allArchetypes) } } - [Obsolete("Please use ComponentGroup APIs instead.")] + [Obsolete("Please use EntityQuery APIs instead.")] public NativeArray CreateArchetypeChunkArray(NativeList archetypes, Allocator allocator) { @@ -2439,11 +2453,11 @@ public NativeArray CreateArchetypeChunkArray(NativeList CreateArchetypeChunkArray(EntityArchetypeQuery query, Allocator allocator) + [Obsolete("Please use EntityQuery APIs instead.")] + public NativeArray CreateArchetypeChunkArray(EntityQueryDesc queryDesc, Allocator allocator) { var foundArchetypes = new NativeList(Allocator.TempJob); - AddMatchingArchetypes(query, foundArchetypes); + AddMatchingArchetypes(queryDesc, foundArchetypes); var chunkStream = CreateArchetypeChunkArray(foundArchetypes, allocator); foundArchetypes.Dispose(); return chunkStream; @@ -2467,7 +2481,7 @@ public ArchetypeChunkComponentType GetArchetypeChunkComponentType(bool isR #if ENABLE_UNITY_COLLECTIONS_CHECKS var typeIndex = TypeManager.GetTypeIndex(); return new ArchetypeChunkComponentType( - ComponentJobSafetyManager.GetSafetyHandle(typeIndex, isReadOnly), isReadOnly, + ComponentJobSafetyManager->GetSafetyHandle(typeIndex, isReadOnly), isReadOnly, GlobalSystemVersion); #else return new ArchetypeChunkComponentType(isReadOnly, GlobalSystemVersion); @@ -2493,8 +2507,8 @@ public ArchetypeChunkBufferType GetArchetypeChunkBufferType(bool isReadOnl #if ENABLE_UNITY_COLLECTIONS_CHECKS var typeIndex = TypeManager.GetTypeIndex(); return new ArchetypeChunkBufferType( - ComponentJobSafetyManager.GetSafetyHandle(typeIndex, isReadOnly), - ComponentJobSafetyManager.GetBufferSafetyHandle(typeIndex), + ComponentJobSafetyManager->GetSafetyHandle(typeIndex, isReadOnly), + ComponentJobSafetyManager->GetBufferSafetyHandle(typeIndex), isReadOnly, GlobalSystemVersion); #else return new ArchetypeChunkBufferType(isReadOnly,GlobalSystemVersion); @@ -2517,7 +2531,7 @@ public ArchetypeChunkSharedComponentType GetArchetypeChunkSharedComponentType { #if ENABLE_UNITY_COLLECTIONS_CHECKS return new ArchetypeChunkSharedComponentType( - ComponentJobSafetyManager.GetEntityManagerSafetyHandle()); + ComponentJobSafetyManager->GetEntityManagerSafetyHandle()); #else return new ArchetypeChunkSharedComponentType(false); #endif @@ -2539,7 +2553,7 @@ public ArchetypeChunkEntityType GetArchetypeChunkEntityType() { #if ENABLE_UNITY_COLLECTIONS_CHECKS return new ArchetypeChunkEntityType( - ComponentJobSafetyManager.GetSafetyHandle(TypeManager.GetTypeIndex(), true)); + ComponentJobSafetyManager->GetSafetyHandle(TypeManager.GetTypeIndex(), true)); #else return new ArchetypeChunkEntityType(false); #endif @@ -2617,7 +2631,7 @@ public bool IsSharedComponentManagerEmpty() return m_Manager.m_SharedComponentManager.IsEmpty(); } -#if !UNITY_CSHARP_TINY +#if !NET_DOTS internal static string GetArchetypeDebugString(Archetype* a) { var buf = new System.Text.StringBuilder(); @@ -2660,12 +2674,16 @@ public void LogEntityInfo(Entity entity) public string GetEntityInfo(Entity entity) { var archetype = m_Manager.Entities->GetArchetype(entity); - #if !UNITY_CSHARP_TINY + #if !NET_DOTS var str = new System.Text.StringBuilder(); str.Append(entity.ToString()); #if UNITY_EDITOR - str.Append($" (name '{m_Manager.GetName(entity)}')"); - #endif + { + var name = m_Manager.GetName(entity); + if(!string.IsNullOrEmpty(name)) + str.Append($" (name '{name}')"); + } +#endif for (var i = 0; i < archetype->TypesCount; i++) { var componentTypeInArchetype = archetype->Types[i]; @@ -2686,7 +2704,7 @@ public string GetEntityInfo(Entity entity) #endif } -#if !UNITY_CSHARP_TINY +#if !NET_DOTS public object GetComponentBoxed(Entity entity, ComponentType type) { m_Manager.Entities->AssertEntityHasComponent(entity, type); @@ -2727,7 +2745,7 @@ public void CheckInternalConsistency() { #if ENABLE_UNITY_COLLECTIONS_CHECKS - //@TODO: Validate from perspective of componentgroup... + //@TODO: Validate from perspective of chunkquery... var entityCountEntityData = m_Manager.Entities->CheckInternalConsistency(); var entityCountArchetypeManager = m_Manager.ArchetypeManager.CheckInternalConsistency(m_Manager.Entities); Assert.AreEqual(entityCountEntityData, entityCountArchetypeManager); diff --git a/Unity.Entities/EntityQueryBuilder.cs b/Unity.Entities/EntityQueryBuilder.cs index 92a070b2..e9b99bee 100644 --- a/Unity.Entities/EntityQueryBuilder.cs +++ b/Unity.Entities/EntityQueryBuilder.cs @@ -11,7 +11,7 @@ public partial struct EntityQueryBuilder ComponentSystem m_System; ResizableArray64Byte m_Any, m_None, m_All; - ComponentGroup m_Group; + EntityQuery m_Query; internal EntityQueryBuilder(ComponentSystem system) { @@ -19,7 +19,7 @@ internal EntityQueryBuilder(ComponentSystem system) m_Any = new ResizableArray64Byte(); m_None = new ResizableArray64Byte(); m_All = new ResizableArray64Byte(); - m_Group = null; + m_Query = null; } // this is a specialized function intended only for validation that builders are hashing and getting cached @@ -36,7 +36,7 @@ internal bool ShallowEquals(ref EntityQueryBuilder other) m_Any .Equals(ref other.m_Any) && m_None.Equals(ref other.m_None) && m_All .Equals(ref other.m_All) && - ReferenceEquals(m_Group, other.m_Group); + ReferenceEquals(m_Query, other.m_Query); } public override int GetHashCode() => @@ -45,7 +45,7 @@ public override bool Equals(object obj) => throw new InvalidOperationException("Calling this function is a sign of inadvertent boxing"); [Conditional("ENABLE_UNITY_COLLECTIONS_CHECKS")] - void ValidateHasNoGroup() => ThrowIfInvalidMixing(m_Group != null); + void ValidateHasNoQuery() => ThrowIfInvalidMixing(m_Query != null); [Conditional("ENABLE_UNITY_COLLECTIONS_CHECKS")] void ValidateHasNoSpec() => ThrowIfInvalidMixing(m_Any.Length != 0 || m_None.Length != 0 || m_All.Length != 0); @@ -54,24 +54,24 @@ public override bool Equals(object obj) => void ThrowIfInvalidMixing(bool throwIfTrue) { if (throwIfTrue) - throw new InvalidOperationException($"Cannot mix {nameof(WithAny)}/{nameof(WithNone)}/{nameof(WithAll)} and {nameof(With)}({nameof(ComponentGroup)})"); + throw new InvalidOperationException($"Cannot mix {nameof(WithAny)}/{nameof(WithNone)}/{nameof(WithAll)} and {nameof(With)}({nameof(EntityQuery)})"); } - public EntityQueryBuilder With(ComponentGroup componentGroup) + public EntityQueryBuilder With(EntityQuery entityQuery) { #if ENABLE_UNITY_COLLECTIONS_CHECKS - if (componentGroup == null) - throw new ArgumentNullException(nameof(componentGroup)); - if (m_Group != null) - throw new InvalidOperationException($"{nameof(ComponentGroup)} has already been set"); + if (entityQuery == null) + throw new ArgumentNullException(nameof(entityQuery)); + if (m_Query != null) + throw new InvalidOperationException($"{nameof(EntityQuery)} has already been set"); ValidateHasNoSpec(); #endif - m_Group = componentGroup; + m_Query = entityQuery; return this; } - EntityArchetypeQuery ToEntityArchetypeQuery(int delegateTypeCount) + EntityQueryDesc ToEntityQueryDesc(int delegateTypeCount) { ComponentType[] ToComponentTypes(ref ResizableArray64Byte typeIndices, ComponentType.AccessMode mode, int extraCapacity = 0) { @@ -86,7 +86,7 @@ ComponentType[] ToComponentTypes(ref ResizableArray64Byte typeIndices, Comp return types; } - return new EntityArchetypeQuery + return new EntityQueryDesc { Any = ToComponentTypes(ref m_Any, ComponentType.AccessMode.ReadWrite), None = ToComponentTypes(ref m_None, ComponentType.AccessMode.ReadOnly), @@ -94,11 +94,11 @@ ComponentType[] ToComponentTypes(ref ResizableArray64Byte typeIndices, Comp }; } - public EntityArchetypeQuery ToEntityArchetypeQuery() => - ToEntityArchetypeQuery(0); + public EntityQueryDesc ToEntityQueryDesc() => + ToEntityQueryDesc(0); - public ComponentGroup ToComponentGroup() => - m_Group ?? (m_Group = m_System.GetComponentGroup(ToEntityArchetypeQuery())); + public EntityQuery ToEntityQuery() => + m_Query ?? (m_Query = m_System.GetEntityQuery(ToEntityQueryDesc())); // see EntityQueryBuilder.tt for the template that is converted into EntityQueryBuilder.gen.cs, // which contains ForEach and other generated methods. @@ -108,7 +108,7 @@ EntityManager.InsideForEach InsideForEach() => new EntityManager.InsideForEach(m_System.EntityManager); #endif - unsafe ComponentGroup ResolveComponentGroup(int* delegateTypeIndices, int delegateTypeCount) + unsafe EntityQuery ResolveEntityQuery(int* delegateTypeIndices, int delegateTypeCount) { var hash = (uint)m_Any .GetHashCode() * 0xEA928FF9 @@ -122,18 +122,18 @@ var hash if (found < 0) { // base query from builder spec, but reserve some extra room for the types detected from the delegate - var eaq = ToEntityArchetypeQuery(delegateTypeCount); + var eaq = ToEntityQueryDesc(delegateTypeCount); // now fill out the extra types for (var i = 0 ; i < delegateTypeCount; ++i) eaq.All[i + m_All.Length] = ComponentType.FromTypeIndex(delegateTypeIndices[i]); - var group = m_System.GetComponentGroup(eaq); + var query = m_System.GetEntityQuery(eaq); #if ENABLE_UNITY_COLLECTIONS_CHECKS - found = cache.CreateCachedQuery(hash, group, ref this, delegateTypeIndices, delegateTypeCount); + found = cache.CreateCachedQuery(hash, query, ref this, delegateTypeIndices, delegateTypeCount); #else - found = cache.CreateCachedQuery(hash, group); + found = cache.CreateCachedQuery(hash, query); #endif } #if ENABLE_UNITY_COLLECTIONS_CHECKS @@ -141,7 +141,7 @@ var hash { cache.ValidateMatchesCache(found, ref this, delegateTypeIndices, delegateTypeCount); - // TODO: also validate that m_Group spec matches m_Any/All/None and delegateTypeIndices + // TODO: also validate that m_Query spec matches m_Any/All/None and delegateTypeIndices } #endif diff --git a/Unity.Entities/EntityQueryBuilder.gen.cs.meta b/Unity.Entities/EntityQueryBuilder.gen.cs.meta deleted file mode 100644 index 5ee87395..00000000 --- a/Unity.Entities/EntityQueryBuilder.gen.cs.meta +++ /dev/null @@ -1,11 +0,0 @@ -fileFormatVersion: 2 -guid: 71b6bf888bcea364696d5cc546629b7a -MonoImporter: - externalObjects: {} - serializedVersion: 2 - defaultReferences: [] - executionOrder: 0 - icon: {instanceID: 0} - userData: - assetBundleName: - assetBundleVariant: diff --git a/Unity.Entities/EntityQueryBuilder.gen.cs b/Unity.Entities/EntityQueryBuilder_ForEach.gen.cs similarity index 73% rename from Unity.Entities/EntityQueryBuilder.gen.cs rename to Unity.Entities/EntityQueryBuilder_ForEach.gen.cs index 9bdc229e..cba956b0 100644 --- a/Unity.Entities/EntityQueryBuilder.gen.cs +++ b/Unity.Entities/EntityQueryBuilder_ForEach.gen.cs @@ -8,151 +8,16 @@ //------------------------------------------------------------------------------ // Generated by EntityQueryBuilder.tt (129 `foreach` combinations) +using System; +using System.ComponentModel; using Unity.Collections; using Unity.Collections.LowLevel.Unsafe; +using Unity.Jobs; namespace Unity.Entities { public partial struct EntityQueryBuilder { - // ** FLUENT QUERY ** - - public EntityQueryBuilder WithAny() - { - ValidateHasNoGroup(); - m_Any.Add(TypeManager.GetTypeIndex()); - return this; - } - - public EntityQueryBuilder WithAny() - { - ValidateHasNoGroup(); - m_Any.Add(TypeManager.GetTypeIndex()); - m_Any.Add(TypeManager.GetTypeIndex()); - return this; - } - - public EntityQueryBuilder WithAny() - { - ValidateHasNoGroup(); - m_Any.Add(TypeManager.GetTypeIndex()); - m_Any.Add(TypeManager.GetTypeIndex()); - m_Any.Add(TypeManager.GetTypeIndex()); - return this; - } - - public EntityQueryBuilder WithAny() - { - ValidateHasNoGroup(); - m_Any.Add(TypeManager.GetTypeIndex()); - m_Any.Add(TypeManager.GetTypeIndex()); - m_Any.Add(TypeManager.GetTypeIndex()); - m_Any.Add(TypeManager.GetTypeIndex()); - return this; - } - - public EntityQueryBuilder WithAny() - { - ValidateHasNoGroup(); - m_Any.Add(TypeManager.GetTypeIndex()); - m_Any.Add(TypeManager.GetTypeIndex()); - m_Any.Add(TypeManager.GetTypeIndex()); - m_Any.Add(TypeManager.GetTypeIndex()); - m_Any.Add(TypeManager.GetTypeIndex()); - return this; - } - - public EntityQueryBuilder WithNone() - { - ValidateHasNoGroup(); - m_None.Add(TypeManager.GetTypeIndex()); - return this; - } - - public EntityQueryBuilder WithNone() - { - ValidateHasNoGroup(); - m_None.Add(TypeManager.GetTypeIndex()); - m_None.Add(TypeManager.GetTypeIndex()); - return this; - } - - public EntityQueryBuilder WithNone() - { - ValidateHasNoGroup(); - m_None.Add(TypeManager.GetTypeIndex()); - m_None.Add(TypeManager.GetTypeIndex()); - m_None.Add(TypeManager.GetTypeIndex()); - return this; - } - - public EntityQueryBuilder WithNone() - { - ValidateHasNoGroup(); - m_None.Add(TypeManager.GetTypeIndex()); - m_None.Add(TypeManager.GetTypeIndex()); - m_None.Add(TypeManager.GetTypeIndex()); - m_None.Add(TypeManager.GetTypeIndex()); - return this; - } - - public EntityQueryBuilder WithNone() - { - ValidateHasNoGroup(); - m_None.Add(TypeManager.GetTypeIndex()); - m_None.Add(TypeManager.GetTypeIndex()); - m_None.Add(TypeManager.GetTypeIndex()); - m_None.Add(TypeManager.GetTypeIndex()); - m_None.Add(TypeManager.GetTypeIndex()); - return this; - } - - public EntityQueryBuilder WithAll() - { - ValidateHasNoGroup(); - m_All.Add(TypeManager.GetTypeIndex()); - return this; - } - - public EntityQueryBuilder WithAll() - { - ValidateHasNoGroup(); - m_All.Add(TypeManager.GetTypeIndex()); - m_All.Add(TypeManager.GetTypeIndex()); - return this; - } - - public EntityQueryBuilder WithAll() - { - ValidateHasNoGroup(); - m_All.Add(TypeManager.GetTypeIndex()); - m_All.Add(TypeManager.GetTypeIndex()); - m_All.Add(TypeManager.GetTypeIndex()); - return this; - } - - public EntityQueryBuilder WithAll() - { - ValidateHasNoGroup(); - m_All.Add(TypeManager.GetTypeIndex()); - m_All.Add(TypeManager.GetTypeIndex()); - m_All.Add(TypeManager.GetTypeIndex()); - m_All.Add(TypeManager.GetTypeIndex()); - return this; - } - - public EntityQueryBuilder WithAll() - { - ValidateHasNoGroup(); - m_All.Add(TypeManager.GetTypeIndex()); - m_All.Add(TypeManager.GetTypeIndex()); - m_All.Add(TypeManager.GetTypeIndex()); - m_All.Add(TypeManager.GetTypeIndex()); - m_All.Add(TypeManager.GetTypeIndex()); - return this; - } - - // ** FOREACH ** public delegate void F_E(Entity entity); @@ -162,15 +27,15 @@ public unsafe void ForEach(F_E action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { - group = ResolveComponentGroup(null, 0); + query = ResolveEntityQuery(null, 0); } var entityType = m_System.GetArchetypeChunkEntityType(); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -183,6 +48,7 @@ public unsafe void ForEach(F_E action) } } + public delegate void F_ED(Entity entity, ref T0 c0) where T0 : struct, IComponentData; @@ -193,19 +59,19 @@ public unsafe void ForEach(F_ED action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[1]; delegateTypes[0] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 1); + query = ResolveEntityQuery(delegateTypes, 1); } var entityType = m_System.GetArchetypeChunkEntityType(); var chunkComponentType0 = m_System.GetArchetypeChunkComponentType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -219,6 +85,7 @@ public unsafe void ForEach(F_ED action) } } + public delegate void F_D(ref T0 c0) where T0 : struct, IComponentData; @@ -229,18 +96,18 @@ public unsafe void ForEach(F_D action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[1]; delegateTypes[0] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 1); + query = ResolveEntityQuery(delegateTypes, 1); } var chunkComponentType0 = m_System.GetArchetypeChunkComponentType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -253,6 +120,7 @@ public unsafe void ForEach(F_D action) } } + public delegate void F_EC(Entity entity, T0 c0) where T0 : class; @@ -263,19 +131,19 @@ public unsafe void ForEach(F_EC action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[1]; delegateTypes[0] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 1); + query = ResolveEntityQuery(delegateTypes, 1); } var entityType = m_System.GetArchetypeChunkEntityType(); var chunkComponentType0 = m_System.GetArchetypeChunkComponentType(); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -289,6 +157,7 @@ public unsafe void ForEach(F_EC action) } } + public delegate void F_C(T0 c0) where T0 : class; @@ -299,18 +168,18 @@ public unsafe void ForEach(F_C action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[1]; delegateTypes[0] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 1); + query = ResolveEntityQuery(delegateTypes, 1); } var chunkComponentType0 = m_System.GetArchetypeChunkComponentType(); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -323,6 +192,7 @@ public unsafe void ForEach(F_C action) } } + public delegate void F_EB(Entity entity, DynamicBuffer c0) where T0 : struct, IBufferElementData; @@ -333,19 +203,19 @@ public unsafe void ForEach(F_EB action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[1]; delegateTypes[0] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 1); + query = ResolveEntityQuery(delegateTypes, 1); } var entityType = m_System.GetArchetypeChunkEntityType(); var chunkComponentType0 = m_System.GetArchetypeChunkBufferType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -359,6 +229,7 @@ public unsafe void ForEach(F_EB action) } } + public delegate void F_B(DynamicBuffer c0) where T0 : struct, IBufferElementData; @@ -369,18 +240,18 @@ public unsafe void ForEach(F_B action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[1]; delegateTypes[0] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 1); + query = ResolveEntityQuery(delegateTypes, 1); } var chunkComponentType0 = m_System.GetArchetypeChunkBufferType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -393,6 +264,7 @@ public unsafe void ForEach(F_B action) } } + public delegate void F_ES(Entity entity, T0 c0) where T0 : struct, ISharedComponentData; @@ -403,19 +275,19 @@ public unsafe void ForEach(F_ES action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[1]; delegateTypes[0] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 1); + query = ResolveEntityQuery(delegateTypes, 1); } var entityType = m_System.GetArchetypeChunkEntityType(); var chunkComponentType0 = m_System.GetArchetypeChunkSharedComponentType(); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -429,6 +301,7 @@ public unsafe void ForEach(F_ES action) } } + public delegate void F_S(T0 c0) where T0 : struct, ISharedComponentData; @@ -439,18 +312,18 @@ public unsafe void ForEach(F_S action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[1]; delegateTypes[0] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 1); + query = ResolveEntityQuery(delegateTypes, 1); } var chunkComponentType0 = m_System.GetArchetypeChunkSharedComponentType(); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -463,6 +336,7 @@ public unsafe void ForEach(F_S action) } } + public delegate void F_EDD(Entity entity, ref T0 c0, ref T1 c1) where T0 : struct, IComponentData where T1 : struct, IComponentData; @@ -475,21 +349,21 @@ public unsafe void ForEach(F_EDD action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[2]; delegateTypes[0] = TypeManager.GetTypeIndex(); delegateTypes[1] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 2); + query = ResolveEntityQuery(delegateTypes, 2); } var entityType = m_System.GetArchetypeChunkEntityType(); var chunkComponentType0 = m_System.GetArchetypeChunkComponentType(false); var chunkComponentType1 = m_System.GetArchetypeChunkComponentType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -504,6 +378,7 @@ public unsafe void ForEach(F_EDD action) } } + public delegate void F_DD(ref T0 c0, ref T1 c1) where T0 : struct, IComponentData where T1 : struct, IComponentData; @@ -516,20 +391,20 @@ public unsafe void ForEach(F_DD action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[2]; delegateTypes[0] = TypeManager.GetTypeIndex(); delegateTypes[1] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 2); + query = ResolveEntityQuery(delegateTypes, 2); } var chunkComponentType0 = m_System.GetArchetypeChunkComponentType(false); var chunkComponentType1 = m_System.GetArchetypeChunkComponentType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -543,6 +418,7 @@ public unsafe void ForEach(F_DD action) } } + public delegate void F_EDDD(Entity entity, ref T0 c0, ref T1 c1, ref T2 c2) where T0 : struct, IComponentData where T1 : struct, IComponentData @@ -557,15 +433,15 @@ public unsafe void ForEach(F_EDDD action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[3]; delegateTypes[0] = TypeManager.GetTypeIndex(); delegateTypes[1] = TypeManager.GetTypeIndex(); delegateTypes[2] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 3); + query = ResolveEntityQuery(delegateTypes, 3); } var entityType = m_System.GetArchetypeChunkEntityType(); @@ -573,7 +449,7 @@ public unsafe void ForEach(F_EDDD action) var chunkComponentType1 = m_System.GetArchetypeChunkComponentType(false); var chunkComponentType2 = m_System.GetArchetypeChunkComponentType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -589,6 +465,7 @@ public unsafe void ForEach(F_EDDD action) } } + public delegate void F_DDD(ref T0 c0, ref T1 c1, ref T2 c2) where T0 : struct, IComponentData where T1 : struct, IComponentData @@ -603,22 +480,22 @@ public unsafe void ForEach(F_DDD action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[3]; delegateTypes[0] = TypeManager.GetTypeIndex(); delegateTypes[1] = TypeManager.GetTypeIndex(); delegateTypes[2] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 3); + query = ResolveEntityQuery(delegateTypes, 3); } var chunkComponentType0 = m_System.GetArchetypeChunkComponentType(false); var chunkComponentType1 = m_System.GetArchetypeChunkComponentType(false); var chunkComponentType2 = m_System.GetArchetypeChunkComponentType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -633,6 +510,7 @@ public unsafe void ForEach(F_DDD action) } } + public delegate void F_EDDDD(Entity entity, ref T0 c0, ref T1 c1, ref T2 c2, ref T3 c3) where T0 : struct, IComponentData where T1 : struct, IComponentData @@ -649,8 +527,8 @@ public unsafe void ForEach(F_EDDDD action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[4]; delegateTypes[0] = TypeManager.GetTypeIndex(); @@ -658,7 +536,7 @@ public unsafe void ForEach(F_EDDDD action) delegateTypes[2] = TypeManager.GetTypeIndex(); delegateTypes[3] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 4); + query = ResolveEntityQuery(delegateTypes, 4); } var entityType = m_System.GetArchetypeChunkEntityType(); @@ -667,7 +545,7 @@ public unsafe void ForEach(F_EDDDD action) var chunkComponentType2 = m_System.GetArchetypeChunkComponentType(false); var chunkComponentType3 = m_System.GetArchetypeChunkComponentType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -684,6 +562,7 @@ public unsafe void ForEach(F_EDDDD action) } } + public delegate void F_DDDD(ref T0 c0, ref T1 c1, ref T2 c2, ref T3 c3) where T0 : struct, IComponentData where T1 : struct, IComponentData @@ -700,8 +579,8 @@ public unsafe void ForEach(F_DDDD action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[4]; delegateTypes[0] = TypeManager.GetTypeIndex(); @@ -709,7 +588,7 @@ public unsafe void ForEach(F_DDDD action) delegateTypes[2] = TypeManager.GetTypeIndex(); delegateTypes[3] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 4); + query = ResolveEntityQuery(delegateTypes, 4); } var chunkComponentType0 = m_System.GetArchetypeChunkComponentType(false); @@ -717,7 +596,7 @@ public unsafe void ForEach(F_DDDD action) var chunkComponentType2 = m_System.GetArchetypeChunkComponentType(false); var chunkComponentType3 = m_System.GetArchetypeChunkComponentType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -733,6 +612,7 @@ public unsafe void ForEach(F_DDDD action) } } + public delegate void F_EDDDDD(Entity entity, ref T0 c0, ref T1 c1, ref T2 c2, ref T3 c3, ref T4 c4) where T0 : struct, IComponentData where T1 : struct, IComponentData @@ -751,8 +631,8 @@ public unsafe void ForEach(F_EDDDDD acti using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[5]; delegateTypes[0] = TypeManager.GetTypeIndex(); @@ -761,7 +641,7 @@ public unsafe void ForEach(F_EDDDDD acti delegateTypes[3] = TypeManager.GetTypeIndex(); delegateTypes[4] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 5); + query = ResolveEntityQuery(delegateTypes, 5); } var entityType = m_System.GetArchetypeChunkEntityType(); @@ -771,7 +651,7 @@ public unsafe void ForEach(F_EDDDDD acti var chunkComponentType3 = m_System.GetArchetypeChunkComponentType(false); var chunkComponentType4 = m_System.GetArchetypeChunkComponentType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -789,6 +669,7 @@ public unsafe void ForEach(F_EDDDDD acti } } + public delegate void F_DDDDD(ref T0 c0, ref T1 c1, ref T2 c2, ref T3 c3, ref T4 c4) where T0 : struct, IComponentData where T1 : struct, IComponentData @@ -807,8 +688,8 @@ public unsafe void ForEach(F_DDDDD actio using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[5]; delegateTypes[0] = TypeManager.GetTypeIndex(); @@ -817,7 +698,7 @@ public unsafe void ForEach(F_DDDDD actio delegateTypes[3] = TypeManager.GetTypeIndex(); delegateTypes[4] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 5); + query = ResolveEntityQuery(delegateTypes, 5); } var chunkComponentType0 = m_System.GetArchetypeChunkComponentType(false); @@ -826,7 +707,7 @@ public unsafe void ForEach(F_DDDDD actio var chunkComponentType3 = m_System.GetArchetypeChunkComponentType(false); var chunkComponentType4 = m_System.GetArchetypeChunkComponentType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -843,6 +724,7 @@ public unsafe void ForEach(F_DDDDD actio } } + public delegate void F_EDDDDDD(Entity entity, ref T0 c0, ref T1 c1, ref T2 c2, ref T3 c3, ref T4 c4, ref T5 c5) where T0 : struct, IComponentData where T1 : struct, IComponentData @@ -863,8 +745,8 @@ public unsafe void ForEach(F_EDDDDDD(); @@ -874,7 +756,7 @@ public unsafe void ForEach(F_EDDDDDD(); delegateTypes[5] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 6); + query = ResolveEntityQuery(delegateTypes, 6); } var entityType = m_System.GetArchetypeChunkEntityType(); @@ -885,7 +767,7 @@ public unsafe void ForEach(F_EDDDDDD(false); var chunkComponentType5 = m_System.GetArchetypeChunkComponentType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -904,6 +786,7 @@ public unsafe void ForEach(F_EDDDDDD(ref T0 c0, ref T1 c1, ref T2 c2, ref T3 c3, ref T4 c4, ref T5 c5) where T0 : struct, IComponentData where T1 : struct, IComponentData @@ -924,8 +807,8 @@ public unsafe void ForEach(F_DDDDDD(); @@ -935,7 +818,7 @@ public unsafe void ForEach(F_DDDDDD(); delegateTypes[5] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 6); + query = ResolveEntityQuery(delegateTypes, 6); } var chunkComponentType0 = m_System.GetArchetypeChunkComponentType(false); @@ -945,7 +828,7 @@ public unsafe void ForEach(F_DDDDDD(false); var chunkComponentType5 = m_System.GetArchetypeChunkComponentType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -963,6 +846,7 @@ public unsafe void ForEach(F_DDDDDD(Entity entity, T0 c0, ref T1 c1) where T0 : class where T1 : struct, IComponentData; @@ -975,21 +859,21 @@ public unsafe void ForEach(F_ECD action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[2]; delegateTypes[0] = TypeManager.GetTypeIndex(); delegateTypes[1] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 2); + query = ResolveEntityQuery(delegateTypes, 2); } var entityType = m_System.GetArchetypeChunkEntityType(); var chunkComponentType0 = m_System.GetArchetypeChunkComponentType(); var chunkComponentType1 = m_System.GetArchetypeChunkComponentType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -1004,6 +888,7 @@ public unsafe void ForEach(F_ECD action) } } + public delegate void F_CD(T0 c0, ref T1 c1) where T0 : class where T1 : struct, IComponentData; @@ -1016,20 +901,20 @@ public unsafe void ForEach(F_CD action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[2]; delegateTypes[0] = TypeManager.GetTypeIndex(); delegateTypes[1] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 2); + query = ResolveEntityQuery(delegateTypes, 2); } var chunkComponentType0 = m_System.GetArchetypeChunkComponentType(); var chunkComponentType1 = m_System.GetArchetypeChunkComponentType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -1043,6 +928,7 @@ public unsafe void ForEach(F_CD action) } } + public delegate void F_ECDD(Entity entity, T0 c0, ref T1 c1, ref T2 c2) where T0 : class where T1 : struct, IComponentData @@ -1057,15 +943,15 @@ public unsafe void ForEach(F_ECDD action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[3]; delegateTypes[0] = TypeManager.GetTypeIndex(); delegateTypes[1] = TypeManager.GetTypeIndex(); delegateTypes[2] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 3); + query = ResolveEntityQuery(delegateTypes, 3); } var entityType = m_System.GetArchetypeChunkEntityType(); @@ -1073,7 +959,7 @@ public unsafe void ForEach(F_ECDD action) var chunkComponentType1 = m_System.GetArchetypeChunkComponentType(false); var chunkComponentType2 = m_System.GetArchetypeChunkComponentType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -1089,6 +975,7 @@ public unsafe void ForEach(F_ECDD action) } } + public delegate void F_CDD(T0 c0, ref T1 c1, ref T2 c2) where T0 : class where T1 : struct, IComponentData @@ -1103,22 +990,22 @@ public unsafe void ForEach(F_CDD action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[3]; delegateTypes[0] = TypeManager.GetTypeIndex(); delegateTypes[1] = TypeManager.GetTypeIndex(); delegateTypes[2] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 3); + query = ResolveEntityQuery(delegateTypes, 3); } var chunkComponentType0 = m_System.GetArchetypeChunkComponentType(); var chunkComponentType1 = m_System.GetArchetypeChunkComponentType(false); var chunkComponentType2 = m_System.GetArchetypeChunkComponentType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -1133,6 +1020,7 @@ public unsafe void ForEach(F_CDD action) } } + public delegate void F_ECDDD(Entity entity, T0 c0, ref T1 c1, ref T2 c2, ref T3 c3) where T0 : class where T1 : struct, IComponentData @@ -1149,8 +1037,8 @@ public unsafe void ForEach(F_ECDDD action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[4]; delegateTypes[0] = TypeManager.GetTypeIndex(); @@ -1158,7 +1046,7 @@ public unsafe void ForEach(F_ECDDD action) delegateTypes[2] = TypeManager.GetTypeIndex(); delegateTypes[3] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 4); + query = ResolveEntityQuery(delegateTypes, 4); } var entityType = m_System.GetArchetypeChunkEntityType(); @@ -1167,7 +1055,7 @@ public unsafe void ForEach(F_ECDDD action) var chunkComponentType2 = m_System.GetArchetypeChunkComponentType(false); var chunkComponentType3 = m_System.GetArchetypeChunkComponentType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -1184,6 +1072,7 @@ public unsafe void ForEach(F_ECDDD action) } } + public delegate void F_CDDD(T0 c0, ref T1 c1, ref T2 c2, ref T3 c3) where T0 : class where T1 : struct, IComponentData @@ -1200,8 +1089,8 @@ public unsafe void ForEach(F_CDDD action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[4]; delegateTypes[0] = TypeManager.GetTypeIndex(); @@ -1209,7 +1098,7 @@ public unsafe void ForEach(F_CDDD action) delegateTypes[2] = TypeManager.GetTypeIndex(); delegateTypes[3] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 4); + query = ResolveEntityQuery(delegateTypes, 4); } var chunkComponentType0 = m_System.GetArchetypeChunkComponentType(); @@ -1217,7 +1106,7 @@ public unsafe void ForEach(F_CDDD action) var chunkComponentType2 = m_System.GetArchetypeChunkComponentType(false); var chunkComponentType3 = m_System.GetArchetypeChunkComponentType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -1233,6 +1122,7 @@ public unsafe void ForEach(F_CDDD action) } } + public delegate void F_ECDDDD(Entity entity, T0 c0, ref T1 c1, ref T2 c2, ref T3 c3, ref T4 c4) where T0 : class where T1 : struct, IComponentData @@ -1251,8 +1141,8 @@ public unsafe void ForEach(F_ECDDDD acti using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[5]; delegateTypes[0] = TypeManager.GetTypeIndex(); @@ -1261,7 +1151,7 @@ public unsafe void ForEach(F_ECDDDD acti delegateTypes[3] = TypeManager.GetTypeIndex(); delegateTypes[4] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 5); + query = ResolveEntityQuery(delegateTypes, 5); } var entityType = m_System.GetArchetypeChunkEntityType(); @@ -1271,7 +1161,7 @@ public unsafe void ForEach(F_ECDDDD acti var chunkComponentType3 = m_System.GetArchetypeChunkComponentType(false); var chunkComponentType4 = m_System.GetArchetypeChunkComponentType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -1289,6 +1179,7 @@ public unsafe void ForEach(F_ECDDDD acti } } + public delegate void F_CDDDD(T0 c0, ref T1 c1, ref T2 c2, ref T3 c3, ref T4 c4) where T0 : class where T1 : struct, IComponentData @@ -1307,8 +1198,8 @@ public unsafe void ForEach(F_CDDDD actio using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[5]; delegateTypes[0] = TypeManager.GetTypeIndex(); @@ -1317,7 +1208,7 @@ public unsafe void ForEach(F_CDDDD actio delegateTypes[3] = TypeManager.GetTypeIndex(); delegateTypes[4] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 5); + query = ResolveEntityQuery(delegateTypes, 5); } var chunkComponentType0 = m_System.GetArchetypeChunkComponentType(); @@ -1326,7 +1217,7 @@ public unsafe void ForEach(F_CDDDD actio var chunkComponentType3 = m_System.GetArchetypeChunkComponentType(false); var chunkComponentType4 = m_System.GetArchetypeChunkComponentType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -1343,6 +1234,7 @@ public unsafe void ForEach(F_CDDDD actio } } + public delegate void F_ECDDDDD(Entity entity, T0 c0, ref T1 c1, ref T2 c2, ref T3 c3, ref T4 c4, ref T5 c5) where T0 : class where T1 : struct, IComponentData @@ -1363,8 +1255,8 @@ public unsafe void ForEach(F_ECDDDDD(); @@ -1374,7 +1266,7 @@ public unsafe void ForEach(F_ECDDDDD(); delegateTypes[5] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 6); + query = ResolveEntityQuery(delegateTypes, 6); } var entityType = m_System.GetArchetypeChunkEntityType(); @@ -1385,7 +1277,7 @@ public unsafe void ForEach(F_ECDDDDD(false); var chunkComponentType5 = m_System.GetArchetypeChunkComponentType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -1404,6 +1296,7 @@ public unsafe void ForEach(F_ECDDDDD(T0 c0, ref T1 c1, ref T2 c2, ref T3 c3, ref T4 c4, ref T5 c5) where T0 : class where T1 : struct, IComponentData @@ -1424,8 +1317,8 @@ public unsafe void ForEach(F_CDDDDD(); @@ -1435,7 +1328,7 @@ public unsafe void ForEach(F_CDDDDD(); delegateTypes[5] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 6); + query = ResolveEntityQuery(delegateTypes, 6); } var chunkComponentType0 = m_System.GetArchetypeChunkComponentType(); @@ -1445,7 +1338,7 @@ public unsafe void ForEach(F_CDDDDD(false); var chunkComponentType5 = m_System.GetArchetypeChunkComponentType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -1463,6 +1356,7 @@ public unsafe void ForEach(F_CDDDDD(Entity entity, DynamicBuffer c0, ref T1 c1) where T0 : struct, IBufferElementData where T1 : struct, IComponentData; @@ -1475,21 +1369,21 @@ public unsafe void ForEach(F_EBD action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[2]; delegateTypes[0] = TypeManager.GetTypeIndex(); delegateTypes[1] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 2); + query = ResolveEntityQuery(delegateTypes, 2); } var entityType = m_System.GetArchetypeChunkEntityType(); var chunkComponentType0 = m_System.GetArchetypeChunkBufferType(false); var chunkComponentType1 = m_System.GetArchetypeChunkComponentType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -1504,6 +1398,7 @@ public unsafe void ForEach(F_EBD action) } } + public delegate void F_BD(DynamicBuffer c0, ref T1 c1) where T0 : struct, IBufferElementData where T1 : struct, IComponentData; @@ -1516,20 +1411,20 @@ public unsafe void ForEach(F_BD action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[2]; delegateTypes[0] = TypeManager.GetTypeIndex(); delegateTypes[1] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 2); + query = ResolveEntityQuery(delegateTypes, 2); } var chunkComponentType0 = m_System.GetArchetypeChunkBufferType(false); var chunkComponentType1 = m_System.GetArchetypeChunkComponentType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -1543,6 +1438,7 @@ public unsafe void ForEach(F_BD action) } } + public delegate void F_EBDD(Entity entity, DynamicBuffer c0, ref T1 c1, ref T2 c2) where T0 : struct, IBufferElementData where T1 : struct, IComponentData @@ -1557,15 +1453,15 @@ public unsafe void ForEach(F_EBDD action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[3]; delegateTypes[0] = TypeManager.GetTypeIndex(); delegateTypes[1] = TypeManager.GetTypeIndex(); delegateTypes[2] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 3); + query = ResolveEntityQuery(delegateTypes, 3); } var entityType = m_System.GetArchetypeChunkEntityType(); @@ -1573,7 +1469,7 @@ public unsafe void ForEach(F_EBDD action) var chunkComponentType1 = m_System.GetArchetypeChunkComponentType(false); var chunkComponentType2 = m_System.GetArchetypeChunkComponentType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -1589,6 +1485,7 @@ public unsafe void ForEach(F_EBDD action) } } + public delegate void F_BDD(DynamicBuffer c0, ref T1 c1, ref T2 c2) where T0 : struct, IBufferElementData where T1 : struct, IComponentData @@ -1603,22 +1500,22 @@ public unsafe void ForEach(F_BDD action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[3]; delegateTypes[0] = TypeManager.GetTypeIndex(); delegateTypes[1] = TypeManager.GetTypeIndex(); delegateTypes[2] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 3); + query = ResolveEntityQuery(delegateTypes, 3); } var chunkComponentType0 = m_System.GetArchetypeChunkBufferType(false); var chunkComponentType1 = m_System.GetArchetypeChunkComponentType(false); var chunkComponentType2 = m_System.GetArchetypeChunkComponentType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -1633,6 +1530,7 @@ public unsafe void ForEach(F_BDD action) } } + public delegate void F_EBDDD(Entity entity, DynamicBuffer c0, ref T1 c1, ref T2 c2, ref T3 c3) where T0 : struct, IBufferElementData where T1 : struct, IComponentData @@ -1649,8 +1547,8 @@ public unsafe void ForEach(F_EBDDD action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[4]; delegateTypes[0] = TypeManager.GetTypeIndex(); @@ -1658,7 +1556,7 @@ public unsafe void ForEach(F_EBDDD action) delegateTypes[2] = TypeManager.GetTypeIndex(); delegateTypes[3] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 4); + query = ResolveEntityQuery(delegateTypes, 4); } var entityType = m_System.GetArchetypeChunkEntityType(); @@ -1667,7 +1565,7 @@ public unsafe void ForEach(F_EBDDD action) var chunkComponentType2 = m_System.GetArchetypeChunkComponentType(false); var chunkComponentType3 = m_System.GetArchetypeChunkComponentType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -1684,6 +1582,7 @@ public unsafe void ForEach(F_EBDDD action) } } + public delegate void F_BDDD(DynamicBuffer c0, ref T1 c1, ref T2 c2, ref T3 c3) where T0 : struct, IBufferElementData where T1 : struct, IComponentData @@ -1700,8 +1599,8 @@ public unsafe void ForEach(F_BDDD action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[4]; delegateTypes[0] = TypeManager.GetTypeIndex(); @@ -1709,7 +1608,7 @@ public unsafe void ForEach(F_BDDD action) delegateTypes[2] = TypeManager.GetTypeIndex(); delegateTypes[3] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 4); + query = ResolveEntityQuery(delegateTypes, 4); } var chunkComponentType0 = m_System.GetArchetypeChunkBufferType(false); @@ -1717,7 +1616,7 @@ public unsafe void ForEach(F_BDDD action) var chunkComponentType2 = m_System.GetArchetypeChunkComponentType(false); var chunkComponentType3 = m_System.GetArchetypeChunkComponentType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -1733,6 +1632,7 @@ public unsafe void ForEach(F_BDDD action) } } + public delegate void F_EBDDDD(Entity entity, DynamicBuffer c0, ref T1 c1, ref T2 c2, ref T3 c3, ref T4 c4) where T0 : struct, IBufferElementData where T1 : struct, IComponentData @@ -1751,8 +1651,8 @@ public unsafe void ForEach(F_EBDDDD acti using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[5]; delegateTypes[0] = TypeManager.GetTypeIndex(); @@ -1761,7 +1661,7 @@ public unsafe void ForEach(F_EBDDDD acti delegateTypes[3] = TypeManager.GetTypeIndex(); delegateTypes[4] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 5); + query = ResolveEntityQuery(delegateTypes, 5); } var entityType = m_System.GetArchetypeChunkEntityType(); @@ -1771,7 +1671,7 @@ public unsafe void ForEach(F_EBDDDD acti var chunkComponentType3 = m_System.GetArchetypeChunkComponentType(false); var chunkComponentType4 = m_System.GetArchetypeChunkComponentType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -1789,6 +1689,7 @@ public unsafe void ForEach(F_EBDDDD acti } } + public delegate void F_BDDDD(DynamicBuffer c0, ref T1 c1, ref T2 c2, ref T3 c3, ref T4 c4) where T0 : struct, IBufferElementData where T1 : struct, IComponentData @@ -1807,8 +1708,8 @@ public unsafe void ForEach(F_BDDDD actio using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[5]; delegateTypes[0] = TypeManager.GetTypeIndex(); @@ -1817,7 +1718,7 @@ public unsafe void ForEach(F_BDDDD actio delegateTypes[3] = TypeManager.GetTypeIndex(); delegateTypes[4] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 5); + query = ResolveEntityQuery(delegateTypes, 5); } var chunkComponentType0 = m_System.GetArchetypeChunkBufferType(false); @@ -1826,7 +1727,7 @@ public unsafe void ForEach(F_BDDDD actio var chunkComponentType3 = m_System.GetArchetypeChunkComponentType(false); var chunkComponentType4 = m_System.GetArchetypeChunkComponentType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -1843,6 +1744,7 @@ public unsafe void ForEach(F_BDDDD actio } } + public delegate void F_EBDDDDD(Entity entity, DynamicBuffer c0, ref T1 c1, ref T2 c2, ref T3 c3, ref T4 c4, ref T5 c5) where T0 : struct, IBufferElementData where T1 : struct, IComponentData @@ -1863,8 +1765,8 @@ public unsafe void ForEach(F_EBDDDDD(); @@ -1874,7 +1776,7 @@ public unsafe void ForEach(F_EBDDDDD(); delegateTypes[5] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 6); + query = ResolveEntityQuery(delegateTypes, 6); } var entityType = m_System.GetArchetypeChunkEntityType(); @@ -1885,7 +1787,7 @@ public unsafe void ForEach(F_EBDDDDD(false); var chunkComponentType5 = m_System.GetArchetypeChunkComponentType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -1904,6 +1806,7 @@ public unsafe void ForEach(F_EBDDDDD(DynamicBuffer c0, ref T1 c1, ref T2 c2, ref T3 c3, ref T4 c4, ref T5 c5) where T0 : struct, IBufferElementData where T1 : struct, IComponentData @@ -1924,8 +1827,8 @@ public unsafe void ForEach(F_BDDDDD(); @@ -1935,7 +1838,7 @@ public unsafe void ForEach(F_BDDDDD(); delegateTypes[5] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 6); + query = ResolveEntityQuery(delegateTypes, 6); } var chunkComponentType0 = m_System.GetArchetypeChunkBufferType(false); @@ -1945,7 +1848,7 @@ public unsafe void ForEach(F_BDDDDD(false); var chunkComponentType5 = m_System.GetArchetypeChunkComponentType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -1963,6 +1866,7 @@ public unsafe void ForEach(F_BDDDDD(Entity entity, T0 c0, ref T1 c1) where T0 : struct, ISharedComponentData where T1 : struct, IComponentData; @@ -1975,21 +1879,21 @@ public unsafe void ForEach(F_ESD action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[2]; delegateTypes[0] = TypeManager.GetTypeIndex(); delegateTypes[1] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 2); + query = ResolveEntityQuery(delegateTypes, 2); } var entityType = m_System.GetArchetypeChunkEntityType(); var chunkComponentType0 = m_System.GetArchetypeChunkSharedComponentType(); var chunkComponentType1 = m_System.GetArchetypeChunkComponentType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -2004,6 +1908,7 @@ public unsafe void ForEach(F_ESD action) } } + public delegate void F_SD(T0 c0, ref T1 c1) where T0 : struct, ISharedComponentData where T1 : struct, IComponentData; @@ -2016,20 +1921,20 @@ public unsafe void ForEach(F_SD action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[2]; delegateTypes[0] = TypeManager.GetTypeIndex(); delegateTypes[1] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 2); + query = ResolveEntityQuery(delegateTypes, 2); } var chunkComponentType0 = m_System.GetArchetypeChunkSharedComponentType(); var chunkComponentType1 = m_System.GetArchetypeChunkComponentType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -2043,6 +1948,7 @@ public unsafe void ForEach(F_SD action) } } + public delegate void F_ESDD(Entity entity, T0 c0, ref T1 c1, ref T2 c2) where T0 : struct, ISharedComponentData where T1 : struct, IComponentData @@ -2057,15 +1963,15 @@ public unsafe void ForEach(F_ESDD action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[3]; delegateTypes[0] = TypeManager.GetTypeIndex(); delegateTypes[1] = TypeManager.GetTypeIndex(); delegateTypes[2] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 3); + query = ResolveEntityQuery(delegateTypes, 3); } var entityType = m_System.GetArchetypeChunkEntityType(); @@ -2073,7 +1979,7 @@ public unsafe void ForEach(F_ESDD action) var chunkComponentType1 = m_System.GetArchetypeChunkComponentType(false); var chunkComponentType2 = m_System.GetArchetypeChunkComponentType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -2089,6 +1995,7 @@ public unsafe void ForEach(F_ESDD action) } } + public delegate void F_SDD(T0 c0, ref T1 c1, ref T2 c2) where T0 : struct, ISharedComponentData where T1 : struct, IComponentData @@ -2103,22 +2010,22 @@ public unsafe void ForEach(F_SDD action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[3]; delegateTypes[0] = TypeManager.GetTypeIndex(); delegateTypes[1] = TypeManager.GetTypeIndex(); delegateTypes[2] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 3); + query = ResolveEntityQuery(delegateTypes, 3); } var chunkComponentType0 = m_System.GetArchetypeChunkSharedComponentType(); var chunkComponentType1 = m_System.GetArchetypeChunkComponentType(false); var chunkComponentType2 = m_System.GetArchetypeChunkComponentType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -2133,6 +2040,7 @@ public unsafe void ForEach(F_SDD action) } } + public delegate void F_ESDDD(Entity entity, T0 c0, ref T1 c1, ref T2 c2, ref T3 c3) where T0 : struct, ISharedComponentData where T1 : struct, IComponentData @@ -2149,8 +2057,8 @@ public unsafe void ForEach(F_ESDDD action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[4]; delegateTypes[0] = TypeManager.GetTypeIndex(); @@ -2158,7 +2066,7 @@ public unsafe void ForEach(F_ESDDD action) delegateTypes[2] = TypeManager.GetTypeIndex(); delegateTypes[3] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 4); + query = ResolveEntityQuery(delegateTypes, 4); } var entityType = m_System.GetArchetypeChunkEntityType(); @@ -2167,7 +2075,7 @@ public unsafe void ForEach(F_ESDDD action) var chunkComponentType2 = m_System.GetArchetypeChunkComponentType(false); var chunkComponentType3 = m_System.GetArchetypeChunkComponentType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -2184,6 +2092,7 @@ public unsafe void ForEach(F_ESDDD action) } } + public delegate void F_SDDD(T0 c0, ref T1 c1, ref T2 c2, ref T3 c3) where T0 : struct, ISharedComponentData where T1 : struct, IComponentData @@ -2200,8 +2109,8 @@ public unsafe void ForEach(F_SDDD action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[4]; delegateTypes[0] = TypeManager.GetTypeIndex(); @@ -2209,7 +2118,7 @@ public unsafe void ForEach(F_SDDD action) delegateTypes[2] = TypeManager.GetTypeIndex(); delegateTypes[3] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 4); + query = ResolveEntityQuery(delegateTypes, 4); } var chunkComponentType0 = m_System.GetArchetypeChunkSharedComponentType(); @@ -2217,7 +2126,7 @@ public unsafe void ForEach(F_SDDD action) var chunkComponentType2 = m_System.GetArchetypeChunkComponentType(false); var chunkComponentType3 = m_System.GetArchetypeChunkComponentType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -2233,6 +2142,7 @@ public unsafe void ForEach(F_SDDD action) } } + public delegate void F_ESDDDD(Entity entity, T0 c0, ref T1 c1, ref T2 c2, ref T3 c3, ref T4 c4) where T0 : struct, ISharedComponentData where T1 : struct, IComponentData @@ -2251,8 +2161,8 @@ public unsafe void ForEach(F_ESDDDD acti using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[5]; delegateTypes[0] = TypeManager.GetTypeIndex(); @@ -2261,7 +2171,7 @@ public unsafe void ForEach(F_ESDDDD acti delegateTypes[3] = TypeManager.GetTypeIndex(); delegateTypes[4] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 5); + query = ResolveEntityQuery(delegateTypes, 5); } var entityType = m_System.GetArchetypeChunkEntityType(); @@ -2271,7 +2181,7 @@ public unsafe void ForEach(F_ESDDDD acti var chunkComponentType3 = m_System.GetArchetypeChunkComponentType(false); var chunkComponentType4 = m_System.GetArchetypeChunkComponentType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -2289,6 +2199,7 @@ public unsafe void ForEach(F_ESDDDD acti } } + public delegate void F_SDDDD(T0 c0, ref T1 c1, ref T2 c2, ref T3 c3, ref T4 c4) where T0 : struct, ISharedComponentData where T1 : struct, IComponentData @@ -2307,8 +2218,8 @@ public unsafe void ForEach(F_SDDDD actio using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[5]; delegateTypes[0] = TypeManager.GetTypeIndex(); @@ -2317,7 +2228,7 @@ public unsafe void ForEach(F_SDDDD actio delegateTypes[3] = TypeManager.GetTypeIndex(); delegateTypes[4] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 5); + query = ResolveEntityQuery(delegateTypes, 5); } var chunkComponentType0 = m_System.GetArchetypeChunkSharedComponentType(); @@ -2326,7 +2237,7 @@ public unsafe void ForEach(F_SDDDD actio var chunkComponentType3 = m_System.GetArchetypeChunkComponentType(false); var chunkComponentType4 = m_System.GetArchetypeChunkComponentType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -2343,6 +2254,7 @@ public unsafe void ForEach(F_SDDDD actio } } + public delegate void F_ESDDDDD(Entity entity, T0 c0, ref T1 c1, ref T2 c2, ref T3 c3, ref T4 c4, ref T5 c5) where T0 : struct, ISharedComponentData where T1 : struct, IComponentData @@ -2363,8 +2275,8 @@ public unsafe void ForEach(F_ESDDDDD(); @@ -2374,7 +2286,7 @@ public unsafe void ForEach(F_ESDDDDD(); delegateTypes[5] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 6); + query = ResolveEntityQuery(delegateTypes, 6); } var entityType = m_System.GetArchetypeChunkEntityType(); @@ -2385,7 +2297,7 @@ public unsafe void ForEach(F_ESDDDDD(false); var chunkComponentType5 = m_System.GetArchetypeChunkComponentType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -2404,6 +2316,7 @@ public unsafe void ForEach(F_ESDDDDD(T0 c0, ref T1 c1, ref T2 c2, ref T3 c3, ref T4 c4, ref T5 c5) where T0 : struct, ISharedComponentData where T1 : struct, IComponentData @@ -2424,8 +2337,8 @@ public unsafe void ForEach(F_SDDDDD(); @@ -2435,7 +2348,7 @@ public unsafe void ForEach(F_SDDDDD(); delegateTypes[5] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 6); + query = ResolveEntityQuery(delegateTypes, 6); } var chunkComponentType0 = m_System.GetArchetypeChunkSharedComponentType(); @@ -2445,7 +2358,7 @@ public unsafe void ForEach(F_SDDDDD(false); var chunkComponentType5 = m_System.GetArchetypeChunkComponentType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -2463,6 +2376,7 @@ public unsafe void ForEach(F_SDDDDD(Entity entity, ref T0 c0, T1 c1) where T0 : struct, IComponentData where T1 : class; @@ -2475,21 +2389,21 @@ public unsafe void ForEach(F_EDC action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[2]; delegateTypes[0] = TypeManager.GetTypeIndex(); delegateTypes[1] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 2); + query = ResolveEntityQuery(delegateTypes, 2); } var entityType = m_System.GetArchetypeChunkEntityType(); var chunkComponentType0 = m_System.GetArchetypeChunkComponentType(false); var chunkComponentType1 = m_System.GetArchetypeChunkComponentType(); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -2504,6 +2418,7 @@ public unsafe void ForEach(F_EDC action) } } + public delegate void F_DC(ref T0 c0, T1 c1) where T0 : struct, IComponentData where T1 : class; @@ -2516,20 +2431,20 @@ public unsafe void ForEach(F_DC action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[2]; delegateTypes[0] = TypeManager.GetTypeIndex(); delegateTypes[1] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 2); + query = ResolveEntityQuery(delegateTypes, 2); } var chunkComponentType0 = m_System.GetArchetypeChunkComponentType(false); var chunkComponentType1 = m_System.GetArchetypeChunkComponentType(); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -2543,6 +2458,7 @@ public unsafe void ForEach(F_DC action) } } + public delegate void F_EDCC(Entity entity, ref T0 c0, T1 c1, T2 c2) where T0 : struct, IComponentData where T1 : class @@ -2557,15 +2473,15 @@ public unsafe void ForEach(F_EDCC action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[3]; delegateTypes[0] = TypeManager.GetTypeIndex(); delegateTypes[1] = TypeManager.GetTypeIndex(); delegateTypes[2] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 3); + query = ResolveEntityQuery(delegateTypes, 3); } var entityType = m_System.GetArchetypeChunkEntityType(); @@ -2573,7 +2489,7 @@ public unsafe void ForEach(F_EDCC action) var chunkComponentType1 = m_System.GetArchetypeChunkComponentType(); var chunkComponentType2 = m_System.GetArchetypeChunkComponentType(); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -2589,6 +2505,7 @@ public unsafe void ForEach(F_EDCC action) } } + public delegate void F_DCC(ref T0 c0, T1 c1, T2 c2) where T0 : struct, IComponentData where T1 : class @@ -2603,22 +2520,22 @@ public unsafe void ForEach(F_DCC action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[3]; delegateTypes[0] = TypeManager.GetTypeIndex(); delegateTypes[1] = TypeManager.GetTypeIndex(); delegateTypes[2] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 3); + query = ResolveEntityQuery(delegateTypes, 3); } var chunkComponentType0 = m_System.GetArchetypeChunkComponentType(false); var chunkComponentType1 = m_System.GetArchetypeChunkComponentType(); var chunkComponentType2 = m_System.GetArchetypeChunkComponentType(); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -2633,6 +2550,7 @@ public unsafe void ForEach(F_DCC action) } } + public delegate void F_EDCCC(Entity entity, ref T0 c0, T1 c1, T2 c2, T3 c3) where T0 : struct, IComponentData where T1 : class @@ -2649,8 +2567,8 @@ public unsafe void ForEach(F_EDCCC action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[4]; delegateTypes[0] = TypeManager.GetTypeIndex(); @@ -2658,7 +2576,7 @@ public unsafe void ForEach(F_EDCCC action) delegateTypes[2] = TypeManager.GetTypeIndex(); delegateTypes[3] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 4); + query = ResolveEntityQuery(delegateTypes, 4); } var entityType = m_System.GetArchetypeChunkEntityType(); @@ -2667,7 +2585,7 @@ public unsafe void ForEach(F_EDCCC action) var chunkComponentType2 = m_System.GetArchetypeChunkComponentType(); var chunkComponentType3 = m_System.GetArchetypeChunkComponentType(); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -2684,6 +2602,7 @@ public unsafe void ForEach(F_EDCCC action) } } + public delegate void F_DCCC(ref T0 c0, T1 c1, T2 c2, T3 c3) where T0 : struct, IComponentData where T1 : class @@ -2700,8 +2619,8 @@ public unsafe void ForEach(F_DCCC action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[4]; delegateTypes[0] = TypeManager.GetTypeIndex(); @@ -2709,7 +2628,7 @@ public unsafe void ForEach(F_DCCC action) delegateTypes[2] = TypeManager.GetTypeIndex(); delegateTypes[3] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 4); + query = ResolveEntityQuery(delegateTypes, 4); } var chunkComponentType0 = m_System.GetArchetypeChunkComponentType(false); @@ -2717,7 +2636,7 @@ public unsafe void ForEach(F_DCCC action) var chunkComponentType2 = m_System.GetArchetypeChunkComponentType(); var chunkComponentType3 = m_System.GetArchetypeChunkComponentType(); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -2733,6 +2652,7 @@ public unsafe void ForEach(F_DCCC action) } } + public delegate void F_EDCCCC(Entity entity, ref T0 c0, T1 c1, T2 c2, T3 c3, T4 c4) where T0 : struct, IComponentData where T1 : class @@ -2751,8 +2671,8 @@ public unsafe void ForEach(F_EDCCCC acti using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[5]; delegateTypes[0] = TypeManager.GetTypeIndex(); @@ -2761,7 +2681,7 @@ public unsafe void ForEach(F_EDCCCC acti delegateTypes[3] = TypeManager.GetTypeIndex(); delegateTypes[4] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 5); + query = ResolveEntityQuery(delegateTypes, 5); } var entityType = m_System.GetArchetypeChunkEntityType(); @@ -2771,7 +2691,7 @@ public unsafe void ForEach(F_EDCCCC acti var chunkComponentType3 = m_System.GetArchetypeChunkComponentType(); var chunkComponentType4 = m_System.GetArchetypeChunkComponentType(); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -2789,6 +2709,7 @@ public unsafe void ForEach(F_EDCCCC acti } } + public delegate void F_DCCCC(ref T0 c0, T1 c1, T2 c2, T3 c3, T4 c4) where T0 : struct, IComponentData where T1 : class @@ -2807,8 +2728,8 @@ public unsafe void ForEach(F_DCCCC actio using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[5]; delegateTypes[0] = TypeManager.GetTypeIndex(); @@ -2817,7 +2738,7 @@ public unsafe void ForEach(F_DCCCC actio delegateTypes[3] = TypeManager.GetTypeIndex(); delegateTypes[4] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 5); + query = ResolveEntityQuery(delegateTypes, 5); } var chunkComponentType0 = m_System.GetArchetypeChunkComponentType(false); @@ -2826,7 +2747,7 @@ public unsafe void ForEach(F_DCCCC actio var chunkComponentType3 = m_System.GetArchetypeChunkComponentType(); var chunkComponentType4 = m_System.GetArchetypeChunkComponentType(); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -2843,6 +2764,7 @@ public unsafe void ForEach(F_DCCCC actio } } + public delegate void F_EDCCCCC(Entity entity, ref T0 c0, T1 c1, T2 c2, T3 c3, T4 c4, T5 c5) where T0 : struct, IComponentData where T1 : class @@ -2863,8 +2785,8 @@ public unsafe void ForEach(F_EDCCCCC(); @@ -2874,7 +2796,7 @@ public unsafe void ForEach(F_EDCCCCC(); delegateTypes[5] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 6); + query = ResolveEntityQuery(delegateTypes, 6); } var entityType = m_System.GetArchetypeChunkEntityType(); @@ -2885,7 +2807,7 @@ public unsafe void ForEach(F_EDCCCCC(); var chunkComponentType5 = m_System.GetArchetypeChunkComponentType(); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -2904,6 +2826,7 @@ public unsafe void ForEach(F_EDCCCCC(ref T0 c0, T1 c1, T2 c2, T3 c3, T4 c4, T5 c5) where T0 : struct, IComponentData where T1 : class @@ -2924,8 +2847,8 @@ public unsafe void ForEach(F_DCCCCC(); @@ -2935,7 +2858,7 @@ public unsafe void ForEach(F_DCCCCC(); delegateTypes[5] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 6); + query = ResolveEntityQuery(delegateTypes, 6); } var chunkComponentType0 = m_System.GetArchetypeChunkComponentType(false); @@ -2945,7 +2868,7 @@ public unsafe void ForEach(F_DCCCCC(); var chunkComponentType5 = m_System.GetArchetypeChunkComponentType(); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -2963,6 +2886,7 @@ public unsafe void ForEach(F_DCCCCC(Entity entity, T0 c0, T1 c1) where T0 : class where T1 : class; @@ -2975,21 +2899,21 @@ public unsafe void ForEach(F_ECC action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[2]; delegateTypes[0] = TypeManager.GetTypeIndex(); delegateTypes[1] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 2); + query = ResolveEntityQuery(delegateTypes, 2); } var entityType = m_System.GetArchetypeChunkEntityType(); var chunkComponentType0 = m_System.GetArchetypeChunkComponentType(); var chunkComponentType1 = m_System.GetArchetypeChunkComponentType(); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -3004,6 +2928,7 @@ public unsafe void ForEach(F_ECC action) } } + public delegate void F_CC(T0 c0, T1 c1) where T0 : class where T1 : class; @@ -3016,20 +2941,20 @@ public unsafe void ForEach(F_CC action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[2]; delegateTypes[0] = TypeManager.GetTypeIndex(); delegateTypes[1] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 2); + query = ResolveEntityQuery(delegateTypes, 2); } var chunkComponentType0 = m_System.GetArchetypeChunkComponentType(); var chunkComponentType1 = m_System.GetArchetypeChunkComponentType(); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -3043,6 +2968,7 @@ public unsafe void ForEach(F_CC action) } } + public delegate void F_ECCC(Entity entity, T0 c0, T1 c1, T2 c2) where T0 : class where T1 : class @@ -3057,15 +2983,15 @@ public unsafe void ForEach(F_ECCC action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[3]; delegateTypes[0] = TypeManager.GetTypeIndex(); delegateTypes[1] = TypeManager.GetTypeIndex(); delegateTypes[2] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 3); + query = ResolveEntityQuery(delegateTypes, 3); } var entityType = m_System.GetArchetypeChunkEntityType(); @@ -3073,7 +2999,7 @@ public unsafe void ForEach(F_ECCC action) var chunkComponentType1 = m_System.GetArchetypeChunkComponentType(); var chunkComponentType2 = m_System.GetArchetypeChunkComponentType(); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -3089,6 +3015,7 @@ public unsafe void ForEach(F_ECCC action) } } + public delegate void F_CCC(T0 c0, T1 c1, T2 c2) where T0 : class where T1 : class @@ -3103,22 +3030,22 @@ public unsafe void ForEach(F_CCC action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[3]; delegateTypes[0] = TypeManager.GetTypeIndex(); delegateTypes[1] = TypeManager.GetTypeIndex(); delegateTypes[2] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 3); + query = ResolveEntityQuery(delegateTypes, 3); } var chunkComponentType0 = m_System.GetArchetypeChunkComponentType(); var chunkComponentType1 = m_System.GetArchetypeChunkComponentType(); var chunkComponentType2 = m_System.GetArchetypeChunkComponentType(); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -3133,6 +3060,7 @@ public unsafe void ForEach(F_CCC action) } } + public delegate void F_ECCCC(Entity entity, T0 c0, T1 c1, T2 c2, T3 c3) where T0 : class where T1 : class @@ -3149,8 +3077,8 @@ public unsafe void ForEach(F_ECCCC action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[4]; delegateTypes[0] = TypeManager.GetTypeIndex(); @@ -3158,7 +3086,7 @@ public unsafe void ForEach(F_ECCCC action) delegateTypes[2] = TypeManager.GetTypeIndex(); delegateTypes[3] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 4); + query = ResolveEntityQuery(delegateTypes, 4); } var entityType = m_System.GetArchetypeChunkEntityType(); @@ -3167,7 +3095,7 @@ public unsafe void ForEach(F_ECCCC action) var chunkComponentType2 = m_System.GetArchetypeChunkComponentType(); var chunkComponentType3 = m_System.GetArchetypeChunkComponentType(); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -3184,6 +3112,7 @@ public unsafe void ForEach(F_ECCCC action) } } + public delegate void F_CCCC(T0 c0, T1 c1, T2 c2, T3 c3) where T0 : class where T1 : class @@ -3200,8 +3129,8 @@ public unsafe void ForEach(F_CCCC action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[4]; delegateTypes[0] = TypeManager.GetTypeIndex(); @@ -3209,7 +3138,7 @@ public unsafe void ForEach(F_CCCC action) delegateTypes[2] = TypeManager.GetTypeIndex(); delegateTypes[3] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 4); + query = ResolveEntityQuery(delegateTypes, 4); } var chunkComponentType0 = m_System.GetArchetypeChunkComponentType(); @@ -3217,7 +3146,7 @@ public unsafe void ForEach(F_CCCC action) var chunkComponentType2 = m_System.GetArchetypeChunkComponentType(); var chunkComponentType3 = m_System.GetArchetypeChunkComponentType(); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -3233,6 +3162,7 @@ public unsafe void ForEach(F_CCCC action) } } + public delegate void F_ECCCCC(Entity entity, T0 c0, T1 c1, T2 c2, T3 c3, T4 c4) where T0 : class where T1 : class @@ -3251,8 +3181,8 @@ public unsafe void ForEach(F_ECCCCC acti using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[5]; delegateTypes[0] = TypeManager.GetTypeIndex(); @@ -3261,7 +3191,7 @@ public unsafe void ForEach(F_ECCCCC acti delegateTypes[3] = TypeManager.GetTypeIndex(); delegateTypes[4] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 5); + query = ResolveEntityQuery(delegateTypes, 5); } var entityType = m_System.GetArchetypeChunkEntityType(); @@ -3271,7 +3201,7 @@ public unsafe void ForEach(F_ECCCCC acti var chunkComponentType3 = m_System.GetArchetypeChunkComponentType(); var chunkComponentType4 = m_System.GetArchetypeChunkComponentType(); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -3289,6 +3219,7 @@ public unsafe void ForEach(F_ECCCCC acti } } + public delegate void F_CCCCC(T0 c0, T1 c1, T2 c2, T3 c3, T4 c4) where T0 : class where T1 : class @@ -3307,8 +3238,8 @@ public unsafe void ForEach(F_CCCCC actio using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[5]; delegateTypes[0] = TypeManager.GetTypeIndex(); @@ -3317,7 +3248,7 @@ public unsafe void ForEach(F_CCCCC actio delegateTypes[3] = TypeManager.GetTypeIndex(); delegateTypes[4] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 5); + query = ResolveEntityQuery(delegateTypes, 5); } var chunkComponentType0 = m_System.GetArchetypeChunkComponentType(); @@ -3326,7 +3257,7 @@ public unsafe void ForEach(F_CCCCC actio var chunkComponentType3 = m_System.GetArchetypeChunkComponentType(); var chunkComponentType4 = m_System.GetArchetypeChunkComponentType(); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -3343,6 +3274,7 @@ public unsafe void ForEach(F_CCCCC actio } } + public delegate void F_ECCCCCC(Entity entity, T0 c0, T1 c1, T2 c2, T3 c3, T4 c4, T5 c5) where T0 : class where T1 : class @@ -3363,8 +3295,8 @@ public unsafe void ForEach(F_ECCCCCC(); @@ -3374,7 +3306,7 @@ public unsafe void ForEach(F_ECCCCCC(); delegateTypes[5] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 6); + query = ResolveEntityQuery(delegateTypes, 6); } var entityType = m_System.GetArchetypeChunkEntityType(); @@ -3385,7 +3317,7 @@ public unsafe void ForEach(F_ECCCCCC(); var chunkComponentType5 = m_System.GetArchetypeChunkComponentType(); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -3404,6 +3336,7 @@ public unsafe void ForEach(F_ECCCCCC(T0 c0, T1 c1, T2 c2, T3 c3, T4 c4, T5 c5) where T0 : class where T1 : class @@ -3424,8 +3357,8 @@ public unsafe void ForEach(F_CCCCCC(); @@ -3435,7 +3368,7 @@ public unsafe void ForEach(F_CCCCCC(); delegateTypes[5] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 6); + query = ResolveEntityQuery(delegateTypes, 6); } var chunkComponentType0 = m_System.GetArchetypeChunkComponentType(); @@ -3445,7 +3378,7 @@ public unsafe void ForEach(F_CCCCCC(); var chunkComponentType5 = m_System.GetArchetypeChunkComponentType(); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -3463,6 +3396,7 @@ public unsafe void ForEach(F_CCCCCC(Entity entity, DynamicBuffer c0, T1 c1) where T0 : struct, IBufferElementData where T1 : class; @@ -3475,21 +3409,21 @@ public unsafe void ForEach(F_EBC action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[2]; delegateTypes[0] = TypeManager.GetTypeIndex(); delegateTypes[1] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 2); + query = ResolveEntityQuery(delegateTypes, 2); } var entityType = m_System.GetArchetypeChunkEntityType(); var chunkComponentType0 = m_System.GetArchetypeChunkBufferType(false); var chunkComponentType1 = m_System.GetArchetypeChunkComponentType(); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -3504,6 +3438,7 @@ public unsafe void ForEach(F_EBC action) } } + public delegate void F_BC(DynamicBuffer c0, T1 c1) where T0 : struct, IBufferElementData where T1 : class; @@ -3516,20 +3451,20 @@ public unsafe void ForEach(F_BC action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[2]; delegateTypes[0] = TypeManager.GetTypeIndex(); delegateTypes[1] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 2); + query = ResolveEntityQuery(delegateTypes, 2); } var chunkComponentType0 = m_System.GetArchetypeChunkBufferType(false); var chunkComponentType1 = m_System.GetArchetypeChunkComponentType(); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -3543,6 +3478,7 @@ public unsafe void ForEach(F_BC action) } } + public delegate void F_EBCC(Entity entity, DynamicBuffer c0, T1 c1, T2 c2) where T0 : struct, IBufferElementData where T1 : class @@ -3557,15 +3493,15 @@ public unsafe void ForEach(F_EBCC action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[3]; delegateTypes[0] = TypeManager.GetTypeIndex(); delegateTypes[1] = TypeManager.GetTypeIndex(); delegateTypes[2] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 3); + query = ResolveEntityQuery(delegateTypes, 3); } var entityType = m_System.GetArchetypeChunkEntityType(); @@ -3573,7 +3509,7 @@ public unsafe void ForEach(F_EBCC action) var chunkComponentType1 = m_System.GetArchetypeChunkComponentType(); var chunkComponentType2 = m_System.GetArchetypeChunkComponentType(); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -3589,6 +3525,7 @@ public unsafe void ForEach(F_EBCC action) } } + public delegate void F_BCC(DynamicBuffer c0, T1 c1, T2 c2) where T0 : struct, IBufferElementData where T1 : class @@ -3603,22 +3540,22 @@ public unsafe void ForEach(F_BCC action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[3]; delegateTypes[0] = TypeManager.GetTypeIndex(); delegateTypes[1] = TypeManager.GetTypeIndex(); delegateTypes[2] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 3); + query = ResolveEntityQuery(delegateTypes, 3); } var chunkComponentType0 = m_System.GetArchetypeChunkBufferType(false); var chunkComponentType1 = m_System.GetArchetypeChunkComponentType(); var chunkComponentType2 = m_System.GetArchetypeChunkComponentType(); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -3633,6 +3570,7 @@ public unsafe void ForEach(F_BCC action) } } + public delegate void F_EBCCC(Entity entity, DynamicBuffer c0, T1 c1, T2 c2, T3 c3) where T0 : struct, IBufferElementData where T1 : class @@ -3649,8 +3587,8 @@ public unsafe void ForEach(F_EBCCC action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[4]; delegateTypes[0] = TypeManager.GetTypeIndex(); @@ -3658,7 +3596,7 @@ public unsafe void ForEach(F_EBCCC action) delegateTypes[2] = TypeManager.GetTypeIndex(); delegateTypes[3] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 4); + query = ResolveEntityQuery(delegateTypes, 4); } var entityType = m_System.GetArchetypeChunkEntityType(); @@ -3667,7 +3605,7 @@ public unsafe void ForEach(F_EBCCC action) var chunkComponentType2 = m_System.GetArchetypeChunkComponentType(); var chunkComponentType3 = m_System.GetArchetypeChunkComponentType(); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -3684,6 +3622,7 @@ public unsafe void ForEach(F_EBCCC action) } } + public delegate void F_BCCC(DynamicBuffer c0, T1 c1, T2 c2, T3 c3) where T0 : struct, IBufferElementData where T1 : class @@ -3700,8 +3639,8 @@ public unsafe void ForEach(F_BCCC action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[4]; delegateTypes[0] = TypeManager.GetTypeIndex(); @@ -3709,7 +3648,7 @@ public unsafe void ForEach(F_BCCC action) delegateTypes[2] = TypeManager.GetTypeIndex(); delegateTypes[3] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 4); + query = ResolveEntityQuery(delegateTypes, 4); } var chunkComponentType0 = m_System.GetArchetypeChunkBufferType(false); @@ -3717,7 +3656,7 @@ public unsafe void ForEach(F_BCCC action) var chunkComponentType2 = m_System.GetArchetypeChunkComponentType(); var chunkComponentType3 = m_System.GetArchetypeChunkComponentType(); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -3733,6 +3672,7 @@ public unsafe void ForEach(F_BCCC action) } } + public delegate void F_EBCCCC(Entity entity, DynamicBuffer c0, T1 c1, T2 c2, T3 c3, T4 c4) where T0 : struct, IBufferElementData where T1 : class @@ -3751,8 +3691,8 @@ public unsafe void ForEach(F_EBCCCC acti using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[5]; delegateTypes[0] = TypeManager.GetTypeIndex(); @@ -3761,7 +3701,7 @@ public unsafe void ForEach(F_EBCCCC acti delegateTypes[3] = TypeManager.GetTypeIndex(); delegateTypes[4] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 5); + query = ResolveEntityQuery(delegateTypes, 5); } var entityType = m_System.GetArchetypeChunkEntityType(); @@ -3771,7 +3711,7 @@ public unsafe void ForEach(F_EBCCCC acti var chunkComponentType3 = m_System.GetArchetypeChunkComponentType(); var chunkComponentType4 = m_System.GetArchetypeChunkComponentType(); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -3789,6 +3729,7 @@ public unsafe void ForEach(F_EBCCCC acti } } + public delegate void F_BCCCC(DynamicBuffer c0, T1 c1, T2 c2, T3 c3, T4 c4) where T0 : struct, IBufferElementData where T1 : class @@ -3807,8 +3748,8 @@ public unsafe void ForEach(F_BCCCC actio using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[5]; delegateTypes[0] = TypeManager.GetTypeIndex(); @@ -3817,7 +3758,7 @@ public unsafe void ForEach(F_BCCCC actio delegateTypes[3] = TypeManager.GetTypeIndex(); delegateTypes[4] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 5); + query = ResolveEntityQuery(delegateTypes, 5); } var chunkComponentType0 = m_System.GetArchetypeChunkBufferType(false); @@ -3826,7 +3767,7 @@ public unsafe void ForEach(F_BCCCC actio var chunkComponentType3 = m_System.GetArchetypeChunkComponentType(); var chunkComponentType4 = m_System.GetArchetypeChunkComponentType(); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -3843,6 +3784,7 @@ public unsafe void ForEach(F_BCCCC actio } } + public delegate void F_EBCCCCC(Entity entity, DynamicBuffer c0, T1 c1, T2 c2, T3 c3, T4 c4, T5 c5) where T0 : struct, IBufferElementData where T1 : class @@ -3863,8 +3805,8 @@ public unsafe void ForEach(F_EBCCCCC(); @@ -3874,7 +3816,7 @@ public unsafe void ForEach(F_EBCCCCC(); delegateTypes[5] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 6); + query = ResolveEntityQuery(delegateTypes, 6); } var entityType = m_System.GetArchetypeChunkEntityType(); @@ -3885,7 +3827,7 @@ public unsafe void ForEach(F_EBCCCCC(); var chunkComponentType5 = m_System.GetArchetypeChunkComponentType(); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -3904,6 +3846,7 @@ public unsafe void ForEach(F_EBCCCCC(DynamicBuffer c0, T1 c1, T2 c2, T3 c3, T4 c4, T5 c5) where T0 : struct, IBufferElementData where T1 : class @@ -3924,8 +3867,8 @@ public unsafe void ForEach(F_BCCCCC(); @@ -3935,7 +3878,7 @@ public unsafe void ForEach(F_BCCCCC(); delegateTypes[5] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 6); + query = ResolveEntityQuery(delegateTypes, 6); } var chunkComponentType0 = m_System.GetArchetypeChunkBufferType(false); @@ -3945,7 +3888,7 @@ public unsafe void ForEach(F_BCCCCC(); var chunkComponentType5 = m_System.GetArchetypeChunkComponentType(); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -3963,6 +3906,7 @@ public unsafe void ForEach(F_BCCCCC(Entity entity, T0 c0, T1 c1) where T0 : struct, ISharedComponentData where T1 : class; @@ -3975,21 +3919,21 @@ public unsafe void ForEach(F_ESC action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[2]; delegateTypes[0] = TypeManager.GetTypeIndex(); delegateTypes[1] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 2); + query = ResolveEntityQuery(delegateTypes, 2); } var entityType = m_System.GetArchetypeChunkEntityType(); var chunkComponentType0 = m_System.GetArchetypeChunkSharedComponentType(); var chunkComponentType1 = m_System.GetArchetypeChunkComponentType(); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -4004,6 +3948,7 @@ public unsafe void ForEach(F_ESC action) } } + public delegate void F_SC(T0 c0, T1 c1) where T0 : struct, ISharedComponentData where T1 : class; @@ -4016,20 +3961,20 @@ public unsafe void ForEach(F_SC action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[2]; delegateTypes[0] = TypeManager.GetTypeIndex(); delegateTypes[1] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 2); + query = ResolveEntityQuery(delegateTypes, 2); } var chunkComponentType0 = m_System.GetArchetypeChunkSharedComponentType(); var chunkComponentType1 = m_System.GetArchetypeChunkComponentType(); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -4043,6 +3988,7 @@ public unsafe void ForEach(F_SC action) } } + public delegate void F_ESCC(Entity entity, T0 c0, T1 c1, T2 c2) where T0 : struct, ISharedComponentData where T1 : class @@ -4057,15 +4003,15 @@ public unsafe void ForEach(F_ESCC action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[3]; delegateTypes[0] = TypeManager.GetTypeIndex(); delegateTypes[1] = TypeManager.GetTypeIndex(); delegateTypes[2] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 3); + query = ResolveEntityQuery(delegateTypes, 3); } var entityType = m_System.GetArchetypeChunkEntityType(); @@ -4073,7 +4019,7 @@ public unsafe void ForEach(F_ESCC action) var chunkComponentType1 = m_System.GetArchetypeChunkComponentType(); var chunkComponentType2 = m_System.GetArchetypeChunkComponentType(); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -4089,6 +4035,7 @@ public unsafe void ForEach(F_ESCC action) } } + public delegate void F_SCC(T0 c0, T1 c1, T2 c2) where T0 : struct, ISharedComponentData where T1 : class @@ -4103,22 +4050,22 @@ public unsafe void ForEach(F_SCC action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[3]; delegateTypes[0] = TypeManager.GetTypeIndex(); delegateTypes[1] = TypeManager.GetTypeIndex(); delegateTypes[2] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 3); + query = ResolveEntityQuery(delegateTypes, 3); } var chunkComponentType0 = m_System.GetArchetypeChunkSharedComponentType(); var chunkComponentType1 = m_System.GetArchetypeChunkComponentType(); var chunkComponentType2 = m_System.GetArchetypeChunkComponentType(); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -4133,6 +4080,7 @@ public unsafe void ForEach(F_SCC action) } } + public delegate void F_ESCCC(Entity entity, T0 c0, T1 c1, T2 c2, T3 c3) where T0 : struct, ISharedComponentData where T1 : class @@ -4149,8 +4097,8 @@ public unsafe void ForEach(F_ESCCC action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[4]; delegateTypes[0] = TypeManager.GetTypeIndex(); @@ -4158,7 +4106,7 @@ public unsafe void ForEach(F_ESCCC action) delegateTypes[2] = TypeManager.GetTypeIndex(); delegateTypes[3] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 4); + query = ResolveEntityQuery(delegateTypes, 4); } var entityType = m_System.GetArchetypeChunkEntityType(); @@ -4167,7 +4115,7 @@ public unsafe void ForEach(F_ESCCC action) var chunkComponentType2 = m_System.GetArchetypeChunkComponentType(); var chunkComponentType3 = m_System.GetArchetypeChunkComponentType(); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -4184,6 +4132,7 @@ public unsafe void ForEach(F_ESCCC action) } } + public delegate void F_SCCC(T0 c0, T1 c1, T2 c2, T3 c3) where T0 : struct, ISharedComponentData where T1 : class @@ -4200,8 +4149,8 @@ public unsafe void ForEach(F_SCCC action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[4]; delegateTypes[0] = TypeManager.GetTypeIndex(); @@ -4209,7 +4158,7 @@ public unsafe void ForEach(F_SCCC action) delegateTypes[2] = TypeManager.GetTypeIndex(); delegateTypes[3] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 4); + query = ResolveEntityQuery(delegateTypes, 4); } var chunkComponentType0 = m_System.GetArchetypeChunkSharedComponentType(); @@ -4217,7 +4166,7 @@ public unsafe void ForEach(F_SCCC action) var chunkComponentType2 = m_System.GetArchetypeChunkComponentType(); var chunkComponentType3 = m_System.GetArchetypeChunkComponentType(); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -4233,6 +4182,7 @@ public unsafe void ForEach(F_SCCC action) } } + public delegate void F_ESCCCC(Entity entity, T0 c0, T1 c1, T2 c2, T3 c3, T4 c4) where T0 : struct, ISharedComponentData where T1 : class @@ -4251,8 +4201,8 @@ public unsafe void ForEach(F_ESCCCC acti using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[5]; delegateTypes[0] = TypeManager.GetTypeIndex(); @@ -4261,7 +4211,7 @@ public unsafe void ForEach(F_ESCCCC acti delegateTypes[3] = TypeManager.GetTypeIndex(); delegateTypes[4] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 5); + query = ResolveEntityQuery(delegateTypes, 5); } var entityType = m_System.GetArchetypeChunkEntityType(); @@ -4271,7 +4221,7 @@ public unsafe void ForEach(F_ESCCCC acti var chunkComponentType3 = m_System.GetArchetypeChunkComponentType(); var chunkComponentType4 = m_System.GetArchetypeChunkComponentType(); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -4289,6 +4239,7 @@ public unsafe void ForEach(F_ESCCCC acti } } + public delegate void F_SCCCC(T0 c0, T1 c1, T2 c2, T3 c3, T4 c4) where T0 : struct, ISharedComponentData where T1 : class @@ -4307,8 +4258,8 @@ public unsafe void ForEach(F_SCCCC actio using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[5]; delegateTypes[0] = TypeManager.GetTypeIndex(); @@ -4317,7 +4268,7 @@ public unsafe void ForEach(F_SCCCC actio delegateTypes[3] = TypeManager.GetTypeIndex(); delegateTypes[4] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 5); + query = ResolveEntityQuery(delegateTypes, 5); } var chunkComponentType0 = m_System.GetArchetypeChunkSharedComponentType(); @@ -4326,7 +4277,7 @@ public unsafe void ForEach(F_SCCCC actio var chunkComponentType3 = m_System.GetArchetypeChunkComponentType(); var chunkComponentType4 = m_System.GetArchetypeChunkComponentType(); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -4343,6 +4294,7 @@ public unsafe void ForEach(F_SCCCC actio } } + public delegate void F_ESCCCCC(Entity entity, T0 c0, T1 c1, T2 c2, T3 c3, T4 c4, T5 c5) where T0 : struct, ISharedComponentData where T1 : class @@ -4363,8 +4315,8 @@ public unsafe void ForEach(F_ESCCCCC(); @@ -4374,7 +4326,7 @@ public unsafe void ForEach(F_ESCCCCC(); delegateTypes[5] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 6); + query = ResolveEntityQuery(delegateTypes, 6); } var entityType = m_System.GetArchetypeChunkEntityType(); @@ -4385,7 +4337,7 @@ public unsafe void ForEach(F_ESCCCCC(); var chunkComponentType5 = m_System.GetArchetypeChunkComponentType(); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -4404,6 +4356,7 @@ public unsafe void ForEach(F_ESCCCCC(T0 c0, T1 c1, T2 c2, T3 c3, T4 c4, T5 c5) where T0 : struct, ISharedComponentData where T1 : class @@ -4424,8 +4377,8 @@ public unsafe void ForEach(F_SCCCCC(); @@ -4435,7 +4388,7 @@ public unsafe void ForEach(F_SCCCCC(); delegateTypes[5] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 6); + query = ResolveEntityQuery(delegateTypes, 6); } var chunkComponentType0 = m_System.GetArchetypeChunkSharedComponentType(); @@ -4445,7 +4398,7 @@ public unsafe void ForEach(F_SCCCCC(); var chunkComponentType5 = m_System.GetArchetypeChunkComponentType(); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -4463,6 +4416,7 @@ public unsafe void ForEach(F_SCCCCC(Entity entity, ref T0 c0, DynamicBuffer c1) where T0 : struct, IComponentData where T1 : struct, IBufferElementData; @@ -4475,21 +4429,21 @@ public unsafe void ForEach(F_EDB action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[2]; delegateTypes[0] = TypeManager.GetTypeIndex(); delegateTypes[1] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 2); + query = ResolveEntityQuery(delegateTypes, 2); } var entityType = m_System.GetArchetypeChunkEntityType(); var chunkComponentType0 = m_System.GetArchetypeChunkComponentType(false); var chunkComponentType1 = m_System.GetArchetypeChunkBufferType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -4504,6 +4458,7 @@ public unsafe void ForEach(F_EDB action) } } + public delegate void F_DB(ref T0 c0, DynamicBuffer c1) where T0 : struct, IComponentData where T1 : struct, IBufferElementData; @@ -4516,20 +4471,20 @@ public unsafe void ForEach(F_DB action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[2]; delegateTypes[0] = TypeManager.GetTypeIndex(); delegateTypes[1] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 2); + query = ResolveEntityQuery(delegateTypes, 2); } var chunkComponentType0 = m_System.GetArchetypeChunkComponentType(false); var chunkComponentType1 = m_System.GetArchetypeChunkBufferType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -4543,6 +4498,7 @@ public unsafe void ForEach(F_DB action) } } + public delegate void F_EDBB(Entity entity, ref T0 c0, DynamicBuffer c1, DynamicBuffer c2) where T0 : struct, IComponentData where T1 : struct, IBufferElementData @@ -4557,15 +4513,15 @@ public unsafe void ForEach(F_EDBB action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[3]; delegateTypes[0] = TypeManager.GetTypeIndex(); delegateTypes[1] = TypeManager.GetTypeIndex(); delegateTypes[2] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 3); + query = ResolveEntityQuery(delegateTypes, 3); } var entityType = m_System.GetArchetypeChunkEntityType(); @@ -4573,7 +4529,7 @@ public unsafe void ForEach(F_EDBB action) var chunkComponentType1 = m_System.GetArchetypeChunkBufferType(false); var chunkComponentType2 = m_System.GetArchetypeChunkBufferType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -4589,6 +4545,7 @@ public unsafe void ForEach(F_EDBB action) } } + public delegate void F_DBB(ref T0 c0, DynamicBuffer c1, DynamicBuffer c2) where T0 : struct, IComponentData where T1 : struct, IBufferElementData @@ -4603,22 +4560,22 @@ public unsafe void ForEach(F_DBB action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[3]; delegateTypes[0] = TypeManager.GetTypeIndex(); delegateTypes[1] = TypeManager.GetTypeIndex(); delegateTypes[2] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 3); + query = ResolveEntityQuery(delegateTypes, 3); } var chunkComponentType0 = m_System.GetArchetypeChunkComponentType(false); var chunkComponentType1 = m_System.GetArchetypeChunkBufferType(false); var chunkComponentType2 = m_System.GetArchetypeChunkBufferType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -4633,6 +4590,7 @@ public unsafe void ForEach(F_DBB action) } } + public delegate void F_EDBBB(Entity entity, ref T0 c0, DynamicBuffer c1, DynamicBuffer c2, DynamicBuffer c3) where T0 : struct, IComponentData where T1 : struct, IBufferElementData @@ -4649,8 +4607,8 @@ public unsafe void ForEach(F_EDBBB action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[4]; delegateTypes[0] = TypeManager.GetTypeIndex(); @@ -4658,7 +4616,7 @@ public unsafe void ForEach(F_EDBBB action) delegateTypes[2] = TypeManager.GetTypeIndex(); delegateTypes[3] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 4); + query = ResolveEntityQuery(delegateTypes, 4); } var entityType = m_System.GetArchetypeChunkEntityType(); @@ -4667,7 +4625,7 @@ public unsafe void ForEach(F_EDBBB action) var chunkComponentType2 = m_System.GetArchetypeChunkBufferType(false); var chunkComponentType3 = m_System.GetArchetypeChunkBufferType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -4684,6 +4642,7 @@ public unsafe void ForEach(F_EDBBB action) } } + public delegate void F_DBBB(ref T0 c0, DynamicBuffer c1, DynamicBuffer c2, DynamicBuffer c3) where T0 : struct, IComponentData where T1 : struct, IBufferElementData @@ -4700,8 +4659,8 @@ public unsafe void ForEach(F_DBBB action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[4]; delegateTypes[0] = TypeManager.GetTypeIndex(); @@ -4709,7 +4668,7 @@ public unsafe void ForEach(F_DBBB action) delegateTypes[2] = TypeManager.GetTypeIndex(); delegateTypes[3] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 4); + query = ResolveEntityQuery(delegateTypes, 4); } var chunkComponentType0 = m_System.GetArchetypeChunkComponentType(false); @@ -4717,7 +4676,7 @@ public unsafe void ForEach(F_DBBB action) var chunkComponentType2 = m_System.GetArchetypeChunkBufferType(false); var chunkComponentType3 = m_System.GetArchetypeChunkBufferType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -4733,6 +4692,7 @@ public unsafe void ForEach(F_DBBB action) } } + public delegate void F_EDBBBB(Entity entity, ref T0 c0, DynamicBuffer c1, DynamicBuffer c2, DynamicBuffer c3, DynamicBuffer c4) where T0 : struct, IComponentData where T1 : struct, IBufferElementData @@ -4751,8 +4711,8 @@ public unsafe void ForEach(F_EDBBBB acti using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[5]; delegateTypes[0] = TypeManager.GetTypeIndex(); @@ -4761,7 +4721,7 @@ public unsafe void ForEach(F_EDBBBB acti delegateTypes[3] = TypeManager.GetTypeIndex(); delegateTypes[4] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 5); + query = ResolveEntityQuery(delegateTypes, 5); } var entityType = m_System.GetArchetypeChunkEntityType(); @@ -4771,7 +4731,7 @@ public unsafe void ForEach(F_EDBBBB acti var chunkComponentType3 = m_System.GetArchetypeChunkBufferType(false); var chunkComponentType4 = m_System.GetArchetypeChunkBufferType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -4789,6 +4749,7 @@ public unsafe void ForEach(F_EDBBBB acti } } + public delegate void F_DBBBB(ref T0 c0, DynamicBuffer c1, DynamicBuffer c2, DynamicBuffer c3, DynamicBuffer c4) where T0 : struct, IComponentData where T1 : struct, IBufferElementData @@ -4807,8 +4768,8 @@ public unsafe void ForEach(F_DBBBB actio using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[5]; delegateTypes[0] = TypeManager.GetTypeIndex(); @@ -4817,7 +4778,7 @@ public unsafe void ForEach(F_DBBBB actio delegateTypes[3] = TypeManager.GetTypeIndex(); delegateTypes[4] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 5); + query = ResolveEntityQuery(delegateTypes, 5); } var chunkComponentType0 = m_System.GetArchetypeChunkComponentType(false); @@ -4826,7 +4787,7 @@ public unsafe void ForEach(F_DBBBB actio var chunkComponentType3 = m_System.GetArchetypeChunkBufferType(false); var chunkComponentType4 = m_System.GetArchetypeChunkBufferType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -4843,6 +4804,7 @@ public unsafe void ForEach(F_DBBBB actio } } + public delegate void F_EDBBBBB(Entity entity, ref T0 c0, DynamicBuffer c1, DynamicBuffer c2, DynamicBuffer c3, DynamicBuffer c4, DynamicBuffer c5) where T0 : struct, IComponentData where T1 : struct, IBufferElementData @@ -4863,8 +4825,8 @@ public unsafe void ForEach(F_EDBBBBB(); @@ -4874,7 +4836,7 @@ public unsafe void ForEach(F_EDBBBBB(); delegateTypes[5] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 6); + query = ResolveEntityQuery(delegateTypes, 6); } var entityType = m_System.GetArchetypeChunkEntityType(); @@ -4885,7 +4847,7 @@ public unsafe void ForEach(F_EDBBBBB(false); var chunkComponentType5 = m_System.GetArchetypeChunkBufferType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -4904,6 +4866,7 @@ public unsafe void ForEach(F_EDBBBBB(ref T0 c0, DynamicBuffer c1, DynamicBuffer c2, DynamicBuffer c3, DynamicBuffer c4, DynamicBuffer c5) where T0 : struct, IComponentData where T1 : struct, IBufferElementData @@ -4924,8 +4887,8 @@ public unsafe void ForEach(F_DBBBBB(); @@ -4935,7 +4898,7 @@ public unsafe void ForEach(F_DBBBBB(); delegateTypes[5] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 6); + query = ResolveEntityQuery(delegateTypes, 6); } var chunkComponentType0 = m_System.GetArchetypeChunkComponentType(false); @@ -4945,7 +4908,7 @@ public unsafe void ForEach(F_DBBBBB(false); var chunkComponentType5 = m_System.GetArchetypeChunkBufferType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -4963,6 +4926,7 @@ public unsafe void ForEach(F_DBBBBB(Entity entity, T0 c0, DynamicBuffer c1) where T0 : class where T1 : struct, IBufferElementData; @@ -4975,21 +4939,21 @@ public unsafe void ForEach(F_ECB action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[2]; delegateTypes[0] = TypeManager.GetTypeIndex(); delegateTypes[1] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 2); + query = ResolveEntityQuery(delegateTypes, 2); } var entityType = m_System.GetArchetypeChunkEntityType(); var chunkComponentType0 = m_System.GetArchetypeChunkComponentType(); var chunkComponentType1 = m_System.GetArchetypeChunkBufferType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -5004,6 +4968,7 @@ public unsafe void ForEach(F_ECB action) } } + public delegate void F_CB(T0 c0, DynamicBuffer c1) where T0 : class where T1 : struct, IBufferElementData; @@ -5016,20 +4981,20 @@ public unsafe void ForEach(F_CB action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[2]; delegateTypes[0] = TypeManager.GetTypeIndex(); delegateTypes[1] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 2); + query = ResolveEntityQuery(delegateTypes, 2); } var chunkComponentType0 = m_System.GetArchetypeChunkComponentType(); var chunkComponentType1 = m_System.GetArchetypeChunkBufferType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -5043,6 +5008,7 @@ public unsafe void ForEach(F_CB action) } } + public delegate void F_ECBB(Entity entity, T0 c0, DynamicBuffer c1, DynamicBuffer c2) where T0 : class where T1 : struct, IBufferElementData @@ -5057,15 +5023,15 @@ public unsafe void ForEach(F_ECBB action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[3]; delegateTypes[0] = TypeManager.GetTypeIndex(); delegateTypes[1] = TypeManager.GetTypeIndex(); delegateTypes[2] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 3); + query = ResolveEntityQuery(delegateTypes, 3); } var entityType = m_System.GetArchetypeChunkEntityType(); @@ -5073,7 +5039,7 @@ public unsafe void ForEach(F_ECBB action) var chunkComponentType1 = m_System.GetArchetypeChunkBufferType(false); var chunkComponentType2 = m_System.GetArchetypeChunkBufferType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -5089,6 +5055,7 @@ public unsafe void ForEach(F_ECBB action) } } + public delegate void F_CBB(T0 c0, DynamicBuffer c1, DynamicBuffer c2) where T0 : class where T1 : struct, IBufferElementData @@ -5103,22 +5070,22 @@ public unsafe void ForEach(F_CBB action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[3]; delegateTypes[0] = TypeManager.GetTypeIndex(); delegateTypes[1] = TypeManager.GetTypeIndex(); delegateTypes[2] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 3); + query = ResolveEntityQuery(delegateTypes, 3); } var chunkComponentType0 = m_System.GetArchetypeChunkComponentType(); var chunkComponentType1 = m_System.GetArchetypeChunkBufferType(false); var chunkComponentType2 = m_System.GetArchetypeChunkBufferType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -5133,6 +5100,7 @@ public unsafe void ForEach(F_CBB action) } } + public delegate void F_ECBBB(Entity entity, T0 c0, DynamicBuffer c1, DynamicBuffer c2, DynamicBuffer c3) where T0 : class where T1 : struct, IBufferElementData @@ -5149,8 +5117,8 @@ public unsafe void ForEach(F_ECBBB action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[4]; delegateTypes[0] = TypeManager.GetTypeIndex(); @@ -5158,7 +5126,7 @@ public unsafe void ForEach(F_ECBBB action) delegateTypes[2] = TypeManager.GetTypeIndex(); delegateTypes[3] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 4); + query = ResolveEntityQuery(delegateTypes, 4); } var entityType = m_System.GetArchetypeChunkEntityType(); @@ -5167,7 +5135,7 @@ public unsafe void ForEach(F_ECBBB action) var chunkComponentType2 = m_System.GetArchetypeChunkBufferType(false); var chunkComponentType3 = m_System.GetArchetypeChunkBufferType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -5184,6 +5152,7 @@ public unsafe void ForEach(F_ECBBB action) } } + public delegate void F_CBBB(T0 c0, DynamicBuffer c1, DynamicBuffer c2, DynamicBuffer c3) where T0 : class where T1 : struct, IBufferElementData @@ -5200,8 +5169,8 @@ public unsafe void ForEach(F_CBBB action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[4]; delegateTypes[0] = TypeManager.GetTypeIndex(); @@ -5209,7 +5178,7 @@ public unsafe void ForEach(F_CBBB action) delegateTypes[2] = TypeManager.GetTypeIndex(); delegateTypes[3] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 4); + query = ResolveEntityQuery(delegateTypes, 4); } var chunkComponentType0 = m_System.GetArchetypeChunkComponentType(); @@ -5217,7 +5186,7 @@ public unsafe void ForEach(F_CBBB action) var chunkComponentType2 = m_System.GetArchetypeChunkBufferType(false); var chunkComponentType3 = m_System.GetArchetypeChunkBufferType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -5233,6 +5202,7 @@ public unsafe void ForEach(F_CBBB action) } } + public delegate void F_ECBBBB(Entity entity, T0 c0, DynamicBuffer c1, DynamicBuffer c2, DynamicBuffer c3, DynamicBuffer c4) where T0 : class where T1 : struct, IBufferElementData @@ -5251,8 +5221,8 @@ public unsafe void ForEach(F_ECBBBB acti using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[5]; delegateTypes[0] = TypeManager.GetTypeIndex(); @@ -5261,7 +5231,7 @@ public unsafe void ForEach(F_ECBBBB acti delegateTypes[3] = TypeManager.GetTypeIndex(); delegateTypes[4] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 5); + query = ResolveEntityQuery(delegateTypes, 5); } var entityType = m_System.GetArchetypeChunkEntityType(); @@ -5271,7 +5241,7 @@ public unsafe void ForEach(F_ECBBBB acti var chunkComponentType3 = m_System.GetArchetypeChunkBufferType(false); var chunkComponentType4 = m_System.GetArchetypeChunkBufferType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -5289,6 +5259,7 @@ public unsafe void ForEach(F_ECBBBB acti } } + public delegate void F_CBBBB(T0 c0, DynamicBuffer c1, DynamicBuffer c2, DynamicBuffer c3, DynamicBuffer c4) where T0 : class where T1 : struct, IBufferElementData @@ -5307,8 +5278,8 @@ public unsafe void ForEach(F_CBBBB actio using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[5]; delegateTypes[0] = TypeManager.GetTypeIndex(); @@ -5317,7 +5288,7 @@ public unsafe void ForEach(F_CBBBB actio delegateTypes[3] = TypeManager.GetTypeIndex(); delegateTypes[4] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 5); + query = ResolveEntityQuery(delegateTypes, 5); } var chunkComponentType0 = m_System.GetArchetypeChunkComponentType(); @@ -5326,7 +5297,7 @@ public unsafe void ForEach(F_CBBBB actio var chunkComponentType3 = m_System.GetArchetypeChunkBufferType(false); var chunkComponentType4 = m_System.GetArchetypeChunkBufferType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -5343,6 +5314,7 @@ public unsafe void ForEach(F_CBBBB actio } } + public delegate void F_ECBBBBB(Entity entity, T0 c0, DynamicBuffer c1, DynamicBuffer c2, DynamicBuffer c3, DynamicBuffer c4, DynamicBuffer c5) where T0 : class where T1 : struct, IBufferElementData @@ -5363,8 +5335,8 @@ public unsafe void ForEach(F_ECBBBBB(); @@ -5374,7 +5346,7 @@ public unsafe void ForEach(F_ECBBBBB(); delegateTypes[5] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 6); + query = ResolveEntityQuery(delegateTypes, 6); } var entityType = m_System.GetArchetypeChunkEntityType(); @@ -5385,7 +5357,7 @@ public unsafe void ForEach(F_ECBBBBB(false); var chunkComponentType5 = m_System.GetArchetypeChunkBufferType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -5404,6 +5376,7 @@ public unsafe void ForEach(F_ECBBBBB(T0 c0, DynamicBuffer c1, DynamicBuffer c2, DynamicBuffer c3, DynamicBuffer c4, DynamicBuffer c5) where T0 : class where T1 : struct, IBufferElementData @@ -5424,8 +5397,8 @@ public unsafe void ForEach(F_CBBBBB(); @@ -5435,7 +5408,7 @@ public unsafe void ForEach(F_CBBBBB(); delegateTypes[5] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 6); + query = ResolveEntityQuery(delegateTypes, 6); } var chunkComponentType0 = m_System.GetArchetypeChunkComponentType(); @@ -5445,7 +5418,7 @@ public unsafe void ForEach(F_CBBBBB(false); var chunkComponentType5 = m_System.GetArchetypeChunkBufferType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -5463,6 +5436,7 @@ public unsafe void ForEach(F_CBBBBB(Entity entity, DynamicBuffer c0, DynamicBuffer c1) where T0 : struct, IBufferElementData where T1 : struct, IBufferElementData; @@ -5475,21 +5449,21 @@ public unsafe void ForEach(F_EBB action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[2]; delegateTypes[0] = TypeManager.GetTypeIndex(); delegateTypes[1] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 2); + query = ResolveEntityQuery(delegateTypes, 2); } var entityType = m_System.GetArchetypeChunkEntityType(); var chunkComponentType0 = m_System.GetArchetypeChunkBufferType(false); var chunkComponentType1 = m_System.GetArchetypeChunkBufferType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -5504,6 +5478,7 @@ public unsafe void ForEach(F_EBB action) } } + public delegate void F_BB(DynamicBuffer c0, DynamicBuffer c1) where T0 : struct, IBufferElementData where T1 : struct, IBufferElementData; @@ -5516,20 +5491,20 @@ public unsafe void ForEach(F_BB action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[2]; delegateTypes[0] = TypeManager.GetTypeIndex(); delegateTypes[1] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 2); + query = ResolveEntityQuery(delegateTypes, 2); } var chunkComponentType0 = m_System.GetArchetypeChunkBufferType(false); var chunkComponentType1 = m_System.GetArchetypeChunkBufferType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -5543,6 +5518,7 @@ public unsafe void ForEach(F_BB action) } } + public delegate void F_EBBB(Entity entity, DynamicBuffer c0, DynamicBuffer c1, DynamicBuffer c2) where T0 : struct, IBufferElementData where T1 : struct, IBufferElementData @@ -5557,15 +5533,15 @@ public unsafe void ForEach(F_EBBB action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[3]; delegateTypes[0] = TypeManager.GetTypeIndex(); delegateTypes[1] = TypeManager.GetTypeIndex(); delegateTypes[2] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 3); + query = ResolveEntityQuery(delegateTypes, 3); } var entityType = m_System.GetArchetypeChunkEntityType(); @@ -5573,7 +5549,7 @@ public unsafe void ForEach(F_EBBB action) var chunkComponentType1 = m_System.GetArchetypeChunkBufferType(false); var chunkComponentType2 = m_System.GetArchetypeChunkBufferType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -5589,6 +5565,7 @@ public unsafe void ForEach(F_EBBB action) } } + public delegate void F_BBB(DynamicBuffer c0, DynamicBuffer c1, DynamicBuffer c2) where T0 : struct, IBufferElementData where T1 : struct, IBufferElementData @@ -5603,22 +5580,22 @@ public unsafe void ForEach(F_BBB action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[3]; delegateTypes[0] = TypeManager.GetTypeIndex(); delegateTypes[1] = TypeManager.GetTypeIndex(); delegateTypes[2] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 3); + query = ResolveEntityQuery(delegateTypes, 3); } var chunkComponentType0 = m_System.GetArchetypeChunkBufferType(false); var chunkComponentType1 = m_System.GetArchetypeChunkBufferType(false); var chunkComponentType2 = m_System.GetArchetypeChunkBufferType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -5633,6 +5610,7 @@ public unsafe void ForEach(F_BBB action) } } + public delegate void F_EBBBB(Entity entity, DynamicBuffer c0, DynamicBuffer c1, DynamicBuffer c2, DynamicBuffer c3) where T0 : struct, IBufferElementData where T1 : struct, IBufferElementData @@ -5649,8 +5627,8 @@ public unsafe void ForEach(F_EBBBB action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[4]; delegateTypes[0] = TypeManager.GetTypeIndex(); @@ -5658,7 +5636,7 @@ public unsafe void ForEach(F_EBBBB action) delegateTypes[2] = TypeManager.GetTypeIndex(); delegateTypes[3] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 4); + query = ResolveEntityQuery(delegateTypes, 4); } var entityType = m_System.GetArchetypeChunkEntityType(); @@ -5667,7 +5645,7 @@ public unsafe void ForEach(F_EBBBB action) var chunkComponentType2 = m_System.GetArchetypeChunkBufferType(false); var chunkComponentType3 = m_System.GetArchetypeChunkBufferType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -5684,6 +5662,7 @@ public unsafe void ForEach(F_EBBBB action) } } + public delegate void F_BBBB(DynamicBuffer c0, DynamicBuffer c1, DynamicBuffer c2, DynamicBuffer c3) where T0 : struct, IBufferElementData where T1 : struct, IBufferElementData @@ -5700,8 +5679,8 @@ public unsafe void ForEach(F_BBBB action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[4]; delegateTypes[0] = TypeManager.GetTypeIndex(); @@ -5709,7 +5688,7 @@ public unsafe void ForEach(F_BBBB action) delegateTypes[2] = TypeManager.GetTypeIndex(); delegateTypes[3] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 4); + query = ResolveEntityQuery(delegateTypes, 4); } var chunkComponentType0 = m_System.GetArchetypeChunkBufferType(false); @@ -5717,7 +5696,7 @@ public unsafe void ForEach(F_BBBB action) var chunkComponentType2 = m_System.GetArchetypeChunkBufferType(false); var chunkComponentType3 = m_System.GetArchetypeChunkBufferType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -5733,6 +5712,7 @@ public unsafe void ForEach(F_BBBB action) } } + public delegate void F_EBBBBB(Entity entity, DynamicBuffer c0, DynamicBuffer c1, DynamicBuffer c2, DynamicBuffer c3, DynamicBuffer c4) where T0 : struct, IBufferElementData where T1 : struct, IBufferElementData @@ -5751,8 +5731,8 @@ public unsafe void ForEach(F_EBBBBB acti using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[5]; delegateTypes[0] = TypeManager.GetTypeIndex(); @@ -5761,7 +5741,7 @@ public unsafe void ForEach(F_EBBBBB acti delegateTypes[3] = TypeManager.GetTypeIndex(); delegateTypes[4] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 5); + query = ResolveEntityQuery(delegateTypes, 5); } var entityType = m_System.GetArchetypeChunkEntityType(); @@ -5771,7 +5751,7 @@ public unsafe void ForEach(F_EBBBBB acti var chunkComponentType3 = m_System.GetArchetypeChunkBufferType(false); var chunkComponentType4 = m_System.GetArchetypeChunkBufferType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -5789,6 +5769,7 @@ public unsafe void ForEach(F_EBBBBB acti } } + public delegate void F_BBBBB(DynamicBuffer c0, DynamicBuffer c1, DynamicBuffer c2, DynamicBuffer c3, DynamicBuffer c4) where T0 : struct, IBufferElementData where T1 : struct, IBufferElementData @@ -5807,8 +5788,8 @@ public unsafe void ForEach(F_BBBBB actio using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[5]; delegateTypes[0] = TypeManager.GetTypeIndex(); @@ -5817,7 +5798,7 @@ public unsafe void ForEach(F_BBBBB actio delegateTypes[3] = TypeManager.GetTypeIndex(); delegateTypes[4] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 5); + query = ResolveEntityQuery(delegateTypes, 5); } var chunkComponentType0 = m_System.GetArchetypeChunkBufferType(false); @@ -5826,7 +5807,7 @@ public unsafe void ForEach(F_BBBBB actio var chunkComponentType3 = m_System.GetArchetypeChunkBufferType(false); var chunkComponentType4 = m_System.GetArchetypeChunkBufferType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -5843,6 +5824,7 @@ public unsafe void ForEach(F_BBBBB actio } } + public delegate void F_EBBBBBB(Entity entity, DynamicBuffer c0, DynamicBuffer c1, DynamicBuffer c2, DynamicBuffer c3, DynamicBuffer c4, DynamicBuffer c5) where T0 : struct, IBufferElementData where T1 : struct, IBufferElementData @@ -5863,8 +5845,8 @@ public unsafe void ForEach(F_EBBBBBB(); @@ -5874,7 +5856,7 @@ public unsafe void ForEach(F_EBBBBBB(); delegateTypes[5] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 6); + query = ResolveEntityQuery(delegateTypes, 6); } var entityType = m_System.GetArchetypeChunkEntityType(); @@ -5885,7 +5867,7 @@ public unsafe void ForEach(F_EBBBBBB(false); var chunkComponentType5 = m_System.GetArchetypeChunkBufferType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -5904,6 +5886,7 @@ public unsafe void ForEach(F_EBBBBBB(DynamicBuffer c0, DynamicBuffer c1, DynamicBuffer c2, DynamicBuffer c3, DynamicBuffer c4, DynamicBuffer c5) where T0 : struct, IBufferElementData where T1 : struct, IBufferElementData @@ -5924,8 +5907,8 @@ public unsafe void ForEach(F_BBBBBB(); @@ -5935,7 +5918,7 @@ public unsafe void ForEach(F_BBBBBB(); delegateTypes[5] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 6); + query = ResolveEntityQuery(delegateTypes, 6); } var chunkComponentType0 = m_System.GetArchetypeChunkBufferType(false); @@ -5945,7 +5928,7 @@ public unsafe void ForEach(F_BBBBBB(false); var chunkComponentType5 = m_System.GetArchetypeChunkBufferType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -5963,6 +5946,7 @@ public unsafe void ForEach(F_BBBBBB(Entity entity, T0 c0, DynamicBuffer c1) where T0 : struct, ISharedComponentData where T1 : struct, IBufferElementData; @@ -5975,21 +5959,21 @@ public unsafe void ForEach(F_ESB action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[2]; delegateTypes[0] = TypeManager.GetTypeIndex(); delegateTypes[1] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 2); + query = ResolveEntityQuery(delegateTypes, 2); } var entityType = m_System.GetArchetypeChunkEntityType(); var chunkComponentType0 = m_System.GetArchetypeChunkSharedComponentType(); var chunkComponentType1 = m_System.GetArchetypeChunkBufferType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -6004,6 +5988,7 @@ public unsafe void ForEach(F_ESB action) } } + public delegate void F_SB(T0 c0, DynamicBuffer c1) where T0 : struct, ISharedComponentData where T1 : struct, IBufferElementData; @@ -6016,20 +6001,20 @@ public unsafe void ForEach(F_SB action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[2]; delegateTypes[0] = TypeManager.GetTypeIndex(); delegateTypes[1] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 2); + query = ResolveEntityQuery(delegateTypes, 2); } var chunkComponentType0 = m_System.GetArchetypeChunkSharedComponentType(); var chunkComponentType1 = m_System.GetArchetypeChunkBufferType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -6043,6 +6028,7 @@ public unsafe void ForEach(F_SB action) } } + public delegate void F_ESBB(Entity entity, T0 c0, DynamicBuffer c1, DynamicBuffer c2) where T0 : struct, ISharedComponentData where T1 : struct, IBufferElementData @@ -6057,15 +6043,15 @@ public unsafe void ForEach(F_ESBB action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[3]; delegateTypes[0] = TypeManager.GetTypeIndex(); delegateTypes[1] = TypeManager.GetTypeIndex(); delegateTypes[2] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 3); + query = ResolveEntityQuery(delegateTypes, 3); } var entityType = m_System.GetArchetypeChunkEntityType(); @@ -6073,7 +6059,7 @@ public unsafe void ForEach(F_ESBB action) var chunkComponentType1 = m_System.GetArchetypeChunkBufferType(false); var chunkComponentType2 = m_System.GetArchetypeChunkBufferType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -6089,6 +6075,7 @@ public unsafe void ForEach(F_ESBB action) } } + public delegate void F_SBB(T0 c0, DynamicBuffer c1, DynamicBuffer c2) where T0 : struct, ISharedComponentData where T1 : struct, IBufferElementData @@ -6103,22 +6090,22 @@ public unsafe void ForEach(F_SBB action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[3]; delegateTypes[0] = TypeManager.GetTypeIndex(); delegateTypes[1] = TypeManager.GetTypeIndex(); delegateTypes[2] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 3); + query = ResolveEntityQuery(delegateTypes, 3); } var chunkComponentType0 = m_System.GetArchetypeChunkSharedComponentType(); var chunkComponentType1 = m_System.GetArchetypeChunkBufferType(false); var chunkComponentType2 = m_System.GetArchetypeChunkBufferType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -6133,6 +6120,7 @@ public unsafe void ForEach(F_SBB action) } } + public delegate void F_ESBBB(Entity entity, T0 c0, DynamicBuffer c1, DynamicBuffer c2, DynamicBuffer c3) where T0 : struct, ISharedComponentData where T1 : struct, IBufferElementData @@ -6149,8 +6137,8 @@ public unsafe void ForEach(F_ESBBB action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[4]; delegateTypes[0] = TypeManager.GetTypeIndex(); @@ -6158,7 +6146,7 @@ public unsafe void ForEach(F_ESBBB action) delegateTypes[2] = TypeManager.GetTypeIndex(); delegateTypes[3] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 4); + query = ResolveEntityQuery(delegateTypes, 4); } var entityType = m_System.GetArchetypeChunkEntityType(); @@ -6167,7 +6155,7 @@ public unsafe void ForEach(F_ESBBB action) var chunkComponentType2 = m_System.GetArchetypeChunkBufferType(false); var chunkComponentType3 = m_System.GetArchetypeChunkBufferType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -6184,6 +6172,7 @@ public unsafe void ForEach(F_ESBBB action) } } + public delegate void F_SBBB(T0 c0, DynamicBuffer c1, DynamicBuffer c2, DynamicBuffer c3) where T0 : struct, ISharedComponentData where T1 : struct, IBufferElementData @@ -6200,8 +6189,8 @@ public unsafe void ForEach(F_SBBB action) using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[4]; delegateTypes[0] = TypeManager.GetTypeIndex(); @@ -6209,7 +6198,7 @@ public unsafe void ForEach(F_SBBB action) delegateTypes[2] = TypeManager.GetTypeIndex(); delegateTypes[3] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 4); + query = ResolveEntityQuery(delegateTypes, 4); } var chunkComponentType0 = m_System.GetArchetypeChunkSharedComponentType(); @@ -6217,7 +6206,7 @@ public unsafe void ForEach(F_SBBB action) var chunkComponentType2 = m_System.GetArchetypeChunkBufferType(false); var chunkComponentType3 = m_System.GetArchetypeChunkBufferType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -6233,6 +6222,7 @@ public unsafe void ForEach(F_SBBB action) } } + public delegate void F_ESBBBB(Entity entity, T0 c0, DynamicBuffer c1, DynamicBuffer c2, DynamicBuffer c3, DynamicBuffer c4) where T0 : struct, ISharedComponentData where T1 : struct, IBufferElementData @@ -6251,8 +6241,8 @@ public unsafe void ForEach(F_ESBBBB acti using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[5]; delegateTypes[0] = TypeManager.GetTypeIndex(); @@ -6261,7 +6251,7 @@ public unsafe void ForEach(F_ESBBBB acti delegateTypes[3] = TypeManager.GetTypeIndex(); delegateTypes[4] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 5); + query = ResolveEntityQuery(delegateTypes, 5); } var entityType = m_System.GetArchetypeChunkEntityType(); @@ -6271,7 +6261,7 @@ public unsafe void ForEach(F_ESBBBB acti var chunkComponentType3 = m_System.GetArchetypeChunkBufferType(false); var chunkComponentType4 = m_System.GetArchetypeChunkBufferType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -6289,6 +6279,7 @@ public unsafe void ForEach(F_ESBBBB acti } } + public delegate void F_SBBBB(T0 c0, DynamicBuffer c1, DynamicBuffer c2, DynamicBuffer c3, DynamicBuffer c4) where T0 : struct, ISharedComponentData where T1 : struct, IBufferElementData @@ -6307,8 +6298,8 @@ public unsafe void ForEach(F_SBBBB actio using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { var delegateTypes = stackalloc int[5]; delegateTypes[0] = TypeManager.GetTypeIndex(); @@ -6317,7 +6308,7 @@ public unsafe void ForEach(F_SBBBB actio delegateTypes[3] = TypeManager.GetTypeIndex(); delegateTypes[4] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 5); + query = ResolveEntityQuery(delegateTypes, 5); } var chunkComponentType0 = m_System.GetArchetypeChunkSharedComponentType(); @@ -6326,7 +6317,7 @@ public unsafe void ForEach(F_SBBBB actio var chunkComponentType3 = m_System.GetArchetypeChunkBufferType(false); var chunkComponentType4 = m_System.GetArchetypeChunkBufferType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -6343,6 +6334,7 @@ public unsafe void ForEach(F_SBBBB actio } } + public delegate void F_ESBBBBB(Entity entity, T0 c0, DynamicBuffer c1, DynamicBuffer c2, DynamicBuffer c3, DynamicBuffer c4, DynamicBuffer c5) where T0 : struct, ISharedComponentData where T1 : struct, IBufferElementData @@ -6363,8 +6355,8 @@ public unsafe void ForEach(F_ESBBBBB(); @@ -6374,7 +6366,7 @@ public unsafe void ForEach(F_ESBBBBB(); delegateTypes[5] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 6); + query = ResolveEntityQuery(delegateTypes, 6); } var entityType = m_System.GetArchetypeChunkEntityType(); @@ -6385,7 +6377,7 @@ public unsafe void ForEach(F_ESBBBBB(false); var chunkComponentType5 = m_System.GetArchetypeChunkBufferType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -6404,6 +6396,7 @@ public unsafe void ForEach(F_ESBBBBB(T0 c0, DynamicBuffer c1, DynamicBuffer c2, DynamicBuffer c3, DynamicBuffer c4, DynamicBuffer c5) where T0 : struct, ISharedComponentData where T1 : struct, IBufferElementData @@ -6424,8 +6417,8 @@ public unsafe void ForEach(F_SBBBBB(); @@ -6435,7 +6428,7 @@ public unsafe void ForEach(F_SBBBBB(); delegateTypes[5] = TypeManager.GetTypeIndex(); - group = ResolveComponentGroup(delegateTypes, 6); + query = ResolveEntityQuery(delegateTypes, 6); } var chunkComponentType0 = m_System.GetArchetypeChunkSharedComponentType(); @@ -6445,7 +6438,7 @@ public unsafe void ForEach(F_SBBBBB(false); var chunkComponentType5 = m_System.GetArchetypeChunkBufferType(false); - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -6465,1429 +6458,539 @@ public unsafe void ForEach(F_SBBBBB(EntityQueryBuilder.F_ED action, ComponentGroup group = null) - where T0 : struct, IComponentData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_D action, ComponentGroup group = null) - where T0 : struct, IComponentData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_EC action, ComponentGroup group = null) - where T0 : class - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_C action, ComponentGroup group = null) - where T0 : class - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_EB action, ComponentGroup group = null) - where T0 : struct, IBufferElementData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_B action, ComponentGroup group = null) - where T0 : struct, IBufferElementData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_ES action, ComponentGroup group = null) - where T0 : struct, ISharedComponentData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_S action, ComponentGroup group = null) - where T0 : struct, ISharedComponentData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_EDD action, ComponentGroup group = null) - where T0 : struct, IComponentData where T1 : struct, IComponentData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_DD action, ComponentGroup group = null) - where T0 : struct, IComponentData where T1 : struct, IComponentData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_EDDD action, ComponentGroup group = null) - where T0 : struct, IComponentData where T1 : struct, IComponentData where T2 : struct, IComponentData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_DDD action, ComponentGroup group = null) - where T0 : struct, IComponentData where T1 : struct, IComponentData where T2 : struct, IComponentData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_EDDDD action, ComponentGroup group = null) - where T0 : struct, IComponentData where T1 : struct, IComponentData where T2 : struct, IComponentData where T3 : struct, IComponentData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_DDDD action, ComponentGroup group = null) - where T0 : struct, IComponentData where T1 : struct, IComponentData where T2 : struct, IComponentData where T3 : struct, IComponentData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_EDDDDD action, ComponentGroup group = null) - where T0 : struct, IComponentData where T1 : struct, IComponentData where T2 : struct, IComponentData where T3 : struct, IComponentData where T4 : struct, IComponentData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_DDDDD action, ComponentGroup group = null) - where T0 : struct, IComponentData where T1 : struct, IComponentData where T2 : struct, IComponentData where T3 : struct, IComponentData where T4 : struct, IComponentData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_EDDDDDD action, ComponentGroup group = null) - where T0 : struct, IComponentData where T1 : struct, IComponentData where T2 : struct, IComponentData where T3 : struct, IComponentData where T4 : struct, IComponentData where T5 : struct, IComponentData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_DDDDDD action, ComponentGroup group = null) - where T0 : struct, IComponentData where T1 : struct, IComponentData where T2 : struct, IComponentData where T3 : struct, IComponentData where T4 : struct, IComponentData where T5 : struct, IComponentData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_ECD action, ComponentGroup group = null) - where T0 : class where T1 : struct, IComponentData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_CD action, ComponentGroup group = null) - where T0 : class where T1 : struct, IComponentData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_ECDD action, ComponentGroup group = null) - where T0 : class where T1 : struct, IComponentData where T2 : struct, IComponentData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_CDD action, ComponentGroup group = null) - where T0 : class where T1 : struct, IComponentData where T2 : struct, IComponentData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_ECDDD action, ComponentGroup group = null) - where T0 : class where T1 : struct, IComponentData where T2 : struct, IComponentData where T3 : struct, IComponentData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_CDDD action, ComponentGroup group = null) - where T0 : class where T1 : struct, IComponentData where T2 : struct, IComponentData where T3 : struct, IComponentData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_ECDDDD action, ComponentGroup group = null) - where T0 : class where T1 : struct, IComponentData where T2 : struct, IComponentData where T3 : struct, IComponentData where T4 : struct, IComponentData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_CDDDD action, ComponentGroup group = null) - where T0 : class where T1 : struct, IComponentData where T2 : struct, IComponentData where T3 : struct, IComponentData where T4 : struct, IComponentData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_ECDDDDD action, ComponentGroup group = null) - where T0 : class where T1 : struct, IComponentData where T2 : struct, IComponentData where T3 : struct, IComponentData where T4 : struct, IComponentData where T5 : struct, IComponentData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_CDDDDD action, ComponentGroup group = null) - where T0 : class where T1 : struct, IComponentData where T2 : struct, IComponentData where T3 : struct, IComponentData where T4 : struct, IComponentData where T5 : struct, IComponentData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_EBD action, ComponentGroup group = null) - where T0 : struct, IBufferElementData where T1 : struct, IComponentData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_BD action, ComponentGroup group = null) - where T0 : struct, IBufferElementData where T1 : struct, IComponentData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_EBDD action, ComponentGroup group = null) - where T0 : struct, IBufferElementData where T1 : struct, IComponentData where T2 : struct, IComponentData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_BDD action, ComponentGroup group = null) - where T0 : struct, IBufferElementData where T1 : struct, IComponentData where T2 : struct, IComponentData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_EBDDD action, ComponentGroup group = null) - where T0 : struct, IBufferElementData where T1 : struct, IComponentData where T2 : struct, IComponentData where T3 : struct, IComponentData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_BDDD action, ComponentGroup group = null) - where T0 : struct, IBufferElementData where T1 : struct, IComponentData where T2 : struct, IComponentData where T3 : struct, IComponentData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_EBDDDD action, ComponentGroup group = null) - where T0 : struct, IBufferElementData where T1 : struct, IComponentData where T2 : struct, IComponentData where T3 : struct, IComponentData where T4 : struct, IComponentData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_BDDDD action, ComponentGroup group = null) - where T0 : struct, IBufferElementData where T1 : struct, IComponentData where T2 : struct, IComponentData where T3 : struct, IComponentData where T4 : struct, IComponentData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_EBDDDDD action, ComponentGroup group = null) - where T0 : struct, IBufferElementData where T1 : struct, IComponentData where T2 : struct, IComponentData where T3 : struct, IComponentData where T4 : struct, IComponentData where T5 : struct, IComponentData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_BDDDDD action, ComponentGroup group = null) - where T0 : struct, IBufferElementData where T1 : struct, IComponentData where T2 : struct, IComponentData where T3 : struct, IComponentData where T4 : struct, IComponentData where T5 : struct, IComponentData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_ESD action, ComponentGroup group = null) - where T0 : struct, ISharedComponentData where T1 : struct, IComponentData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_SD action, ComponentGroup group = null) - where T0 : struct, ISharedComponentData where T1 : struct, IComponentData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_ESDD action, ComponentGroup group = null) - where T0 : struct, ISharedComponentData where T1 : struct, IComponentData where T2 : struct, IComponentData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_SDD action, ComponentGroup group = null) - where T0 : struct, ISharedComponentData where T1 : struct, IComponentData where T2 : struct, IComponentData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_ESDDD action, ComponentGroup group = null) - where T0 : struct, ISharedComponentData where T1 : struct, IComponentData where T2 : struct, IComponentData where T3 : struct, IComponentData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_SDDD action, ComponentGroup group = null) - where T0 : struct, ISharedComponentData where T1 : struct, IComponentData where T2 : struct, IComponentData where T3 : struct, IComponentData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_ESDDDD action, ComponentGroup group = null) - where T0 : struct, ISharedComponentData where T1 : struct, IComponentData where T2 : struct, IComponentData where T3 : struct, IComponentData where T4 : struct, IComponentData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_SDDDD action, ComponentGroup group = null) - where T0 : struct, ISharedComponentData where T1 : struct, IComponentData where T2 : struct, IComponentData where T3 : struct, IComponentData where T4 : struct, IComponentData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_ESDDDDD action, ComponentGroup group = null) - where T0 : struct, ISharedComponentData where T1 : struct, IComponentData where T2 : struct, IComponentData where T3 : struct, IComponentData where T4 : struct, IComponentData where T5 : struct, IComponentData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_SDDDDD action, ComponentGroup group = null) - where T0 : struct, ISharedComponentData where T1 : struct, IComponentData where T2 : struct, IComponentData where T3 : struct, IComponentData where T4 : struct, IComponentData where T5 : struct, IComponentData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_EDC action, ComponentGroup group = null) - where T0 : struct, IComponentData where T1 : class - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_DC action, ComponentGroup group = null) - where T0 : struct, IComponentData where T1 : class - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_EDCC action, ComponentGroup group = null) - where T0 : struct, IComponentData where T1 : class where T2 : class - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_DCC action, ComponentGroup group = null) - where T0 : struct, IComponentData where T1 : class where T2 : class - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_EDCCC action, ComponentGroup group = null) - where T0 : struct, IComponentData where T1 : class where T2 : class where T3 : class - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_DCCC action, ComponentGroup group = null) - where T0 : struct, IComponentData where T1 : class where T2 : class where T3 : class - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_EDCCCC action, ComponentGroup group = null) - where T0 : struct, IComponentData where T1 : class where T2 : class where T3 : class where T4 : class - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_DCCCC action, ComponentGroup group = null) - where T0 : struct, IComponentData where T1 : class where T2 : class where T3 : class where T4 : class - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_EDCCCCC action, ComponentGroup group = null) - where T0 : struct, IComponentData where T1 : class where T2 : class where T3 : class where T4 : class where T5 : class - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_DCCCCC action, ComponentGroup group = null) - where T0 : struct, IComponentData where T1 : class where T2 : class where T3 : class where T4 : class where T5 : class - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_ECC action, ComponentGroup group = null) - where T0 : class where T1 : class - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_CC action, ComponentGroup group = null) - where T0 : class where T1 : class - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_ECCC action, ComponentGroup group = null) - where T0 : class where T1 : class where T2 : class - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_CCC action, ComponentGroup group = null) - where T0 : class where T1 : class where T2 : class - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_ECCCC action, ComponentGroup group = null) - where T0 : class where T1 : class where T2 : class where T3 : class - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_CCCC action, ComponentGroup group = null) - where T0 : class where T1 : class where T2 : class where T3 : class - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_ECCCCC action, ComponentGroup group = null) - where T0 : class where T1 : class where T2 : class where T3 : class where T4 : class - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_CCCCC action, ComponentGroup group = null) - where T0 : class where T1 : class where T2 : class where T3 : class where T4 : class - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_ECCCCCC action, ComponentGroup group = null) - where T0 : class where T1 : class where T2 : class where T3 : class where T4 : class where T5 : class - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_CCCCCC action, ComponentGroup group = null) - where T0 : class where T1 : class where T2 : class where T3 : class where T4 : class where T5 : class - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_EBC action, ComponentGroup group = null) - where T0 : struct, IBufferElementData where T1 : class - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_BC action, ComponentGroup group = null) - where T0 : struct, IBufferElementData where T1 : class - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_EBCC action, ComponentGroup group = null) - where T0 : struct, IBufferElementData where T1 : class where T2 : class - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_BCC action, ComponentGroup group = null) - where T0 : struct, IBufferElementData where T1 : class where T2 : class - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_EBCCC action, ComponentGroup group = null) - where T0 : struct, IBufferElementData where T1 : class where T2 : class where T3 : class - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_BCCC action, ComponentGroup group = null) - where T0 : struct, IBufferElementData where T1 : class where T2 : class where T3 : class - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_EBCCCC action, ComponentGroup group = null) - where T0 : struct, IBufferElementData where T1 : class where T2 : class where T3 : class where T4 : class - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_BCCCC action, ComponentGroup group = null) - where T0 : struct, IBufferElementData where T1 : class where T2 : class where T3 : class where T4 : class - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_EBCCCCC action, ComponentGroup group = null) - where T0 : struct, IBufferElementData where T1 : class where T2 : class where T3 : class where T4 : class where T5 : class - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_BCCCCC action, ComponentGroup group = null) - where T0 : struct, IBufferElementData where T1 : class where T2 : class where T3 : class where T4 : class where T5 : class - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_ESC action, ComponentGroup group = null) - where T0 : struct, ISharedComponentData where T1 : class - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_SC action, ComponentGroup group = null) - where T0 : struct, ISharedComponentData where T1 : class - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_ESCC action, ComponentGroup group = null) - where T0 : struct, ISharedComponentData where T1 : class where T2 : class - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_SCC action, ComponentGroup group = null) - where T0 : struct, ISharedComponentData where T1 : class where T2 : class - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_ESCCC action, ComponentGroup group = null) - where T0 : struct, ISharedComponentData where T1 : class where T2 : class where T3 : class - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_SCCC action, ComponentGroup group = null) - where T0 : struct, ISharedComponentData where T1 : class where T2 : class where T3 : class - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_ESCCCC action, ComponentGroup group = null) - where T0 : struct, ISharedComponentData where T1 : class where T2 : class where T3 : class where T4 : class - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_SCCCC action, ComponentGroup group = null) - where T0 : struct, ISharedComponentData where T1 : class where T2 : class where T3 : class where T4 : class - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_ESCCCCC action, ComponentGroup group = null) - where T0 : struct, ISharedComponentData where T1 : class where T2 : class where T3 : class where T4 : class where T5 : class - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_SCCCCC action, ComponentGroup group = null) - where T0 : struct, ISharedComponentData where T1 : class where T2 : class where T3 : class where T4 : class where T5 : class - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_EDB action, ComponentGroup group = null) - where T0 : struct, IComponentData where T1 : struct, IBufferElementData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_DB action, ComponentGroup group = null) - where T0 : struct, IComponentData where T1 : struct, IBufferElementData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_EDBB action, ComponentGroup group = null) - where T0 : struct, IComponentData where T1 : struct, IBufferElementData where T2 : struct, IBufferElementData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_DBB action, ComponentGroup group = null) - where T0 : struct, IComponentData where T1 : struct, IBufferElementData where T2 : struct, IBufferElementData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_EDBBB action, ComponentGroup group = null) - where T0 : struct, IComponentData where T1 : struct, IBufferElementData where T2 : struct, IBufferElementData where T3 : struct, IBufferElementData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_DBBB action, ComponentGroup group = null) - where T0 : struct, IComponentData where T1 : struct, IBufferElementData where T2 : struct, IBufferElementData where T3 : struct, IBufferElementData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_EDBBBB action, ComponentGroup group = null) - where T0 : struct, IComponentData where T1 : struct, IBufferElementData where T2 : struct, IBufferElementData where T3 : struct, IBufferElementData where T4 : struct, IBufferElementData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_DBBBB action, ComponentGroup group = null) - where T0 : struct, IComponentData where T1 : struct, IBufferElementData where T2 : struct, IBufferElementData where T3 : struct, IBufferElementData where T4 : struct, IBufferElementData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_EDBBBBB action, ComponentGroup group = null) - where T0 : struct, IComponentData where T1 : struct, IBufferElementData where T2 : struct, IBufferElementData where T3 : struct, IBufferElementData where T4 : struct, IBufferElementData where T5 : struct, IBufferElementData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_DBBBBB action, ComponentGroup group = null) - where T0 : struct, IComponentData where T1 : struct, IBufferElementData where T2 : struct, IBufferElementData where T3 : struct, IBufferElementData where T4 : struct, IBufferElementData where T5 : struct, IBufferElementData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_ECB action, ComponentGroup group = null) - where T0 : class where T1 : struct, IBufferElementData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_CB action, ComponentGroup group = null) - where T0 : class where T1 : struct, IBufferElementData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_ECBB action, ComponentGroup group = null) - where T0 : class where T1 : struct, IBufferElementData where T2 : struct, IBufferElementData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_CBB action, ComponentGroup group = null) - where T0 : class where T1 : struct, IBufferElementData where T2 : struct, IBufferElementData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_ECBBB action, ComponentGroup group = null) - where T0 : class where T1 : struct, IBufferElementData where T2 : struct, IBufferElementData where T3 : struct, IBufferElementData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_CBBB action, ComponentGroup group = null) - where T0 : class where T1 : struct, IBufferElementData where T2 : struct, IBufferElementData where T3 : struct, IBufferElementData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_ECBBBB action, ComponentGroup group = null) - where T0 : class where T1 : struct, IBufferElementData where T2 : struct, IBufferElementData where T3 : struct, IBufferElementData where T4 : struct, IBufferElementData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_CBBBB action, ComponentGroup group = null) - where T0 : class where T1 : struct, IBufferElementData where T2 : struct, IBufferElementData where T3 : struct, IBufferElementData where T4 : struct, IBufferElementData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_ECBBBBB action, ComponentGroup group = null) - where T0 : class where T1 : struct, IBufferElementData where T2 : struct, IBufferElementData where T3 : struct, IBufferElementData where T4 : struct, IBufferElementData where T5 : struct, IBufferElementData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_CBBBBB action, ComponentGroup group = null) - where T0 : class where T1 : struct, IBufferElementData where T2 : struct, IBufferElementData where T3 : struct, IBufferElementData where T4 : struct, IBufferElementData where T5 : struct, IBufferElementData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_EBB action, ComponentGroup group = null) - where T0 : struct, IBufferElementData where T1 : struct, IBufferElementData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_BB action, ComponentGroup group = null) - where T0 : struct, IBufferElementData where T1 : struct, IBufferElementData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_EBBB action, ComponentGroup group = null) - where T0 : struct, IBufferElementData where T1 : struct, IBufferElementData where T2 : struct, IBufferElementData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_BBB action, ComponentGroup group = null) - where T0 : struct, IBufferElementData where T1 : struct, IBufferElementData where T2 : struct, IBufferElementData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_EBBBB action, ComponentGroup group = null) - where T0 : struct, IBufferElementData where T1 : struct, IBufferElementData where T2 : struct, IBufferElementData where T3 : struct, IBufferElementData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_BBBB action, ComponentGroup group = null) - where T0 : struct, IBufferElementData where T1 : struct, IBufferElementData where T2 : struct, IBufferElementData where T3 : struct, IBufferElementData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_EBBBBB action, ComponentGroup group = null) - where T0 : struct, IBufferElementData where T1 : struct, IBufferElementData where T2 : struct, IBufferElementData where T3 : struct, IBufferElementData where T4 : struct, IBufferElementData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_BBBBB action, ComponentGroup group = null) - where T0 : struct, IBufferElementData where T1 : struct, IBufferElementData where T2 : struct, IBufferElementData where T3 : struct, IBufferElementData where T4 : struct, IBufferElementData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_EBBBBBB action, ComponentGroup group = null) - where T0 : struct, IBufferElementData where T1 : struct, IBufferElementData where T2 : struct, IBufferElementData where T3 : struct, IBufferElementData where T4 : struct, IBufferElementData where T5 : struct, IBufferElementData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_BBBBBB action, ComponentGroup group = null) - where T0 : struct, IBufferElementData where T1 : struct, IBufferElementData where T2 : struct, IBufferElementData where T3 : struct, IBufferElementData where T4 : struct, IBufferElementData where T5 : struct, IBufferElementData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_ESB action, ComponentGroup group = null) - where T0 : struct, ISharedComponentData where T1 : struct, IBufferElementData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_SB action, ComponentGroup group = null) - where T0 : struct, ISharedComponentData where T1 : struct, IBufferElementData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_ESBB action, ComponentGroup group = null) - where T0 : struct, ISharedComponentData where T1 : struct, IBufferElementData where T2 : struct, IBufferElementData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_SBB action, ComponentGroup group = null) - where T0 : struct, ISharedComponentData where T1 : struct, IBufferElementData where T2 : struct, IBufferElementData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_ESBBB action, ComponentGroup group = null) - where T0 : struct, ISharedComponentData where T1 : struct, IBufferElementData where T2 : struct, IBufferElementData where T3 : struct, IBufferElementData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_SBBB action, ComponentGroup group = null) - where T0 : struct, ISharedComponentData where T1 : struct, IBufferElementData where T2 : struct, IBufferElementData where T3 : struct, IBufferElementData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_ESBBBB action, ComponentGroup group = null) - where T0 : struct, ISharedComponentData where T1 : struct, IBufferElementData where T2 : struct, IBufferElementData where T3 : struct, IBufferElementData where T4 : struct, IBufferElementData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_SBBBB action, ComponentGroup group = null) - where T0 : struct, ISharedComponentData where T1 : struct, IBufferElementData where T2 : struct, IBufferElementData where T3 : struct, IBufferElementData where T4 : struct, IBufferElementData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_ESBBBBB action, ComponentGroup group = null) - where T0 : struct, ISharedComponentData where T1 : struct, IBufferElementData where T2 : struct, IBufferElementData where T3 : struct, IBufferElementData where T4 : struct, IBufferElementData where T5 : struct, IBufferElementData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach(EntityQueryBuilder.F_SBBBBB action, ComponentGroup group = null) - where T0 : struct, ISharedComponentData where T1 : struct, IBufferElementData where T2 : struct, IBufferElementData where T3 : struct, IBufferElementData where T4 : struct, IBufferElementData where T5 : struct, IBufferElementData - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } - - } -} + + // Schedule() implementation for DOTS-standalone. + // The IDE calls Schedule(), and then code-gen is used to + // replace the call to the correct Schedule_D method, With + // known types. + +#if UNITY_ZEROPLAYER + public static partial class JobForEachExtensions { + + // This method is used by the IDE, and is what will be replaced by one of the Schedule_D methods below. + public unsafe static JobHandle Schedule(this TJob job, ComponentSystemBase system, JobHandle dependsOn = default(JobHandle)) + where TJob : struct, IBaseJobForEach + { + throw new NotImplementedException("Schedule() should have been replaced by code-gen."); + } + + [EditorBrowsable(EditorBrowsableState.Never)] + public unsafe static JobHandle Schedule_D(TJob job, ComponentSystemBase system, JobHandle dependsOn = default(JobHandle)) + where TJob : struct, IJobForEach + where T0 : struct, IComponentData + { + { + EntityQuery query = null; + if (query== null) + { + var delegateTypes = stackalloc ComponentType[1]; + delegateTypes[0] = ComponentType.ReadWrite(); + + query = /*ResolveEntityQuery*/system.GetEntityQueryInternal(delegateTypes, 1); + } + + var chunkComponentType0 = system.GetArchetypeChunkComponentType(false); + + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) + { + foreach (var chunk in chunks) + { + var array0 = chunk.GetNativeArray(chunkComponentType0).GetUnsafePtr(); + + for (int i = 0, count = chunk.Count; i < count; ++i) + job.Execute(ref UnsafeUtilityEx.ArrayElementAsRef(array0, i)); + } + } + } + return new JobHandle(); + } + + + [EditorBrowsable(EditorBrowsableState.Never)] + public unsafe static JobHandle Schedule_ED(TJob job, ComponentSystemBase system, JobHandle dependsOn = default(JobHandle)) + where TJob : struct, IJobForEachWithEntity + where T0 : struct, IComponentData + { + { + EntityQuery query = null; + if (query== null) + { + var delegateTypes = stackalloc ComponentType[1]; + delegateTypes[0] = ComponentType.ReadWrite(); + + query = /*ResolveEntityQuery*/system.GetEntityQueryInternal(delegateTypes, 1); + } + + var entityType = system.GetArchetypeChunkEntityType(); + var chunkComponentType0 = system.GetArchetypeChunkComponentType(false); + + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) + { + foreach (var chunk in chunks) + { + var array0 = chunk.GetNativeArray(chunkComponentType0).GetUnsafePtr(); + var entityArray = (Entity*)chunk.GetNativeArray(entityType).GetUnsafeReadOnlyPtr(); + + for (int i = 0, count = chunk.Count; i < count; ++i) + job.Execute(entityArray[i], entityArray[i].Index, ref UnsafeUtilityEx.ArrayElementAsRef(array0, i)); + } + } + } + return new JobHandle(); + } + + + [EditorBrowsable(EditorBrowsableState.Never)] + public unsafe static JobHandle Schedule_DD(TJob job, ComponentSystemBase system, JobHandle dependsOn = default(JobHandle)) + where TJob : struct, IJobForEach + where T0 : struct, IComponentData + where T1 : struct, IComponentData + { + { + EntityQuery query = null; + if (query== null) + { + var delegateTypes = stackalloc ComponentType[2]; + delegateTypes[0] = ComponentType.ReadWrite(); + delegateTypes[1] = ComponentType.ReadWrite(); + + query = /*ResolveEntityQuery*/system.GetEntityQueryInternal(delegateTypes, 2); + } + + var chunkComponentType0 = system.GetArchetypeChunkComponentType(false); + var chunkComponentType1 = system.GetArchetypeChunkComponentType(false); + + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) + { + foreach (var chunk in chunks) + { + var array0 = chunk.GetNativeArray(chunkComponentType0).GetUnsafePtr(); + var array1 = chunk.GetNativeArray(chunkComponentType1).GetUnsafePtr(); + + for (int i = 0, count = chunk.Count; i < count; ++i) + job.Execute(ref UnsafeUtilityEx.ArrayElementAsRef(array0, i), ref UnsafeUtilityEx.ArrayElementAsRef(array1, i)); + } + } + } + return new JobHandle(); + } + + + [EditorBrowsable(EditorBrowsableState.Never)] + public unsafe static JobHandle Schedule_EDD(TJob job, ComponentSystemBase system, JobHandle dependsOn = default(JobHandle)) + where TJob : struct, IJobForEachWithEntity + where T0 : struct, IComponentData + where T1 : struct, IComponentData + { + { + EntityQuery query = null; + if (query== null) + { + var delegateTypes = stackalloc ComponentType[2]; + delegateTypes[0] = ComponentType.ReadWrite(); + delegateTypes[1] = ComponentType.ReadWrite(); + + query = /*ResolveEntityQuery*/system.GetEntityQueryInternal(delegateTypes, 2); + } + + var entityType = system.GetArchetypeChunkEntityType(); + var chunkComponentType0 = system.GetArchetypeChunkComponentType(false); + var chunkComponentType1 = system.GetArchetypeChunkComponentType(false); + + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) + { + foreach (var chunk in chunks) + { + var array0 = chunk.GetNativeArray(chunkComponentType0).GetUnsafePtr(); + var array1 = chunk.GetNativeArray(chunkComponentType1).GetUnsafePtr(); + var entityArray = (Entity*)chunk.GetNativeArray(entityType).GetUnsafeReadOnlyPtr(); + + for (int i = 0, count = chunk.Count; i < count; ++i) + job.Execute(entityArray[i], entityArray[i].Index, ref UnsafeUtilityEx.ArrayElementAsRef(array0, i), ref UnsafeUtilityEx.ArrayElementAsRef(array1, i)); + } + } + } + return new JobHandle(); + } + + + [EditorBrowsable(EditorBrowsableState.Never)] + public unsafe static JobHandle Schedule_DDD(TJob job, ComponentSystemBase system, JobHandle dependsOn = default(JobHandle)) + where TJob : struct, IJobForEach + where T0 : struct, IComponentData + where T1 : struct, IComponentData + where T2 : struct, IComponentData + { + { + EntityQuery query = null; + if (query== null) + { + var delegateTypes = stackalloc ComponentType[3]; + delegateTypes[0] = ComponentType.ReadWrite(); + delegateTypes[1] = ComponentType.ReadWrite(); + delegateTypes[2] = ComponentType.ReadWrite(); + + query = /*ResolveEntityQuery*/system.GetEntityQueryInternal(delegateTypes, 3); + } + + var chunkComponentType0 = system.GetArchetypeChunkComponentType(false); + var chunkComponentType1 = system.GetArchetypeChunkComponentType(false); + var chunkComponentType2 = system.GetArchetypeChunkComponentType(false); + + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) + { + foreach (var chunk in chunks) + { + var array0 = chunk.GetNativeArray(chunkComponentType0).GetUnsafePtr(); + var array1 = chunk.GetNativeArray(chunkComponentType1).GetUnsafePtr(); + var array2 = chunk.GetNativeArray(chunkComponentType2).GetUnsafePtr(); + + for (int i = 0, count = chunk.Count; i < count; ++i) + job.Execute(ref UnsafeUtilityEx.ArrayElementAsRef(array0, i), ref UnsafeUtilityEx.ArrayElementAsRef(array1, i), ref UnsafeUtilityEx.ArrayElementAsRef(array2, i)); + } + } + } + return new JobHandle(); + } + + + [EditorBrowsable(EditorBrowsableState.Never)] + public unsafe static JobHandle Schedule_EDDD(TJob job, ComponentSystemBase system, JobHandle dependsOn = default(JobHandle)) + where TJob : struct, IJobForEachWithEntity + where T0 : struct, IComponentData + where T1 : struct, IComponentData + where T2 : struct, IComponentData + { + { + EntityQuery query = null; + if (query== null) + { + var delegateTypes = stackalloc ComponentType[3]; + delegateTypes[0] = ComponentType.ReadWrite(); + delegateTypes[1] = ComponentType.ReadWrite(); + delegateTypes[2] = ComponentType.ReadWrite(); + + query = /*ResolveEntityQuery*/system.GetEntityQueryInternal(delegateTypes, 3); + } + + var entityType = system.GetArchetypeChunkEntityType(); + var chunkComponentType0 = system.GetArchetypeChunkComponentType(false); + var chunkComponentType1 = system.GetArchetypeChunkComponentType(false); + var chunkComponentType2 = system.GetArchetypeChunkComponentType(false); + + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) + { + foreach (var chunk in chunks) + { + var array0 = chunk.GetNativeArray(chunkComponentType0).GetUnsafePtr(); + var array1 = chunk.GetNativeArray(chunkComponentType1).GetUnsafePtr(); + var array2 = chunk.GetNativeArray(chunkComponentType2).GetUnsafePtr(); + var entityArray = (Entity*)chunk.GetNativeArray(entityType).GetUnsafeReadOnlyPtr(); + + for (int i = 0, count = chunk.Count; i < count; ++i) + job.Execute(entityArray[i], entityArray[i].Index, ref UnsafeUtilityEx.ArrayElementAsRef(array0, i), ref UnsafeUtilityEx.ArrayElementAsRef(array1, i), ref UnsafeUtilityEx.ArrayElementAsRef(array2, i)); + } + } + } + return new JobHandle(); + } + + + [EditorBrowsable(EditorBrowsableState.Never)] + public unsafe static JobHandle Schedule_DDDD(TJob job, ComponentSystemBase system, JobHandle dependsOn = default(JobHandle)) + where TJob : struct, IJobForEach + where T0 : struct, IComponentData + where T1 : struct, IComponentData + where T2 : struct, IComponentData + where T3 : struct, IComponentData + { + { + EntityQuery query = null; + if (query== null) + { + var delegateTypes = stackalloc ComponentType[4]; + delegateTypes[0] = ComponentType.ReadWrite(); + delegateTypes[1] = ComponentType.ReadWrite(); + delegateTypes[2] = ComponentType.ReadWrite(); + delegateTypes[3] = ComponentType.ReadWrite(); + + query = /*ResolveEntityQuery*/system.GetEntityQueryInternal(delegateTypes, 4); + } + + var chunkComponentType0 = system.GetArchetypeChunkComponentType(false); + var chunkComponentType1 = system.GetArchetypeChunkComponentType(false); + var chunkComponentType2 = system.GetArchetypeChunkComponentType(false); + var chunkComponentType3 = system.GetArchetypeChunkComponentType(false); + + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) + { + foreach (var chunk in chunks) + { + var array0 = chunk.GetNativeArray(chunkComponentType0).GetUnsafePtr(); + var array1 = chunk.GetNativeArray(chunkComponentType1).GetUnsafePtr(); + var array2 = chunk.GetNativeArray(chunkComponentType2).GetUnsafePtr(); + var array3 = chunk.GetNativeArray(chunkComponentType3).GetUnsafePtr(); + + for (int i = 0, count = chunk.Count; i < count; ++i) + job.Execute(ref UnsafeUtilityEx.ArrayElementAsRef(array0, i), ref UnsafeUtilityEx.ArrayElementAsRef(array1, i), ref UnsafeUtilityEx.ArrayElementAsRef(array2, i), ref UnsafeUtilityEx.ArrayElementAsRef(array3, i)); + } + } + } + return new JobHandle(); + } + + + [EditorBrowsable(EditorBrowsableState.Never)] + public unsafe static JobHandle Schedule_EDDDD(TJob job, ComponentSystemBase system, JobHandle dependsOn = default(JobHandle)) + where TJob : struct, IJobForEachWithEntity + where T0 : struct, IComponentData + where T1 : struct, IComponentData + where T2 : struct, IComponentData + where T3 : struct, IComponentData + { + { + EntityQuery query = null; + if (query== null) + { + var delegateTypes = stackalloc ComponentType[4]; + delegateTypes[0] = ComponentType.ReadWrite(); + delegateTypes[1] = ComponentType.ReadWrite(); + delegateTypes[2] = ComponentType.ReadWrite(); + delegateTypes[3] = ComponentType.ReadWrite(); + + query = /*ResolveEntityQuery*/system.GetEntityQueryInternal(delegateTypes, 4); + } + + var entityType = system.GetArchetypeChunkEntityType(); + var chunkComponentType0 = system.GetArchetypeChunkComponentType(false); + var chunkComponentType1 = system.GetArchetypeChunkComponentType(false); + var chunkComponentType2 = system.GetArchetypeChunkComponentType(false); + var chunkComponentType3 = system.GetArchetypeChunkComponentType(false); + + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) + { + foreach (var chunk in chunks) + { + var array0 = chunk.GetNativeArray(chunkComponentType0).GetUnsafePtr(); + var array1 = chunk.GetNativeArray(chunkComponentType1).GetUnsafePtr(); + var array2 = chunk.GetNativeArray(chunkComponentType2).GetUnsafePtr(); + var array3 = chunk.GetNativeArray(chunkComponentType3).GetUnsafePtr(); + var entityArray = (Entity*)chunk.GetNativeArray(entityType).GetUnsafeReadOnlyPtr(); + + for (int i = 0, count = chunk.Count; i < count; ++i) + job.Execute(entityArray[i], entityArray[i].Index, ref UnsafeUtilityEx.ArrayElementAsRef(array0, i), ref UnsafeUtilityEx.ArrayElementAsRef(array1, i), ref UnsafeUtilityEx.ArrayElementAsRef(array2, i), ref UnsafeUtilityEx.ArrayElementAsRef(array3, i)); + } + } + } + return new JobHandle(); + } + + + [EditorBrowsable(EditorBrowsableState.Never)] + public unsafe static JobHandle Schedule_DDDDD(TJob job, ComponentSystemBase system, JobHandle dependsOn = default(JobHandle)) + where TJob : struct, IJobForEach + where T0 : struct, IComponentData + where T1 : struct, IComponentData + where T2 : struct, IComponentData + where T3 : struct, IComponentData + where T4 : struct, IComponentData + { + { + EntityQuery query = null; + if (query== null) + { + var delegateTypes = stackalloc ComponentType[5]; + delegateTypes[0] = ComponentType.ReadWrite(); + delegateTypes[1] = ComponentType.ReadWrite(); + delegateTypes[2] = ComponentType.ReadWrite(); + delegateTypes[3] = ComponentType.ReadWrite(); + delegateTypes[4] = ComponentType.ReadWrite(); + + query = /*ResolveEntityQuery*/system.GetEntityQueryInternal(delegateTypes, 5); + } + + var chunkComponentType0 = system.GetArchetypeChunkComponentType(false); + var chunkComponentType1 = system.GetArchetypeChunkComponentType(false); + var chunkComponentType2 = system.GetArchetypeChunkComponentType(false); + var chunkComponentType3 = system.GetArchetypeChunkComponentType(false); + var chunkComponentType4 = system.GetArchetypeChunkComponentType(false); + + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) + { + foreach (var chunk in chunks) + { + var array0 = chunk.GetNativeArray(chunkComponentType0).GetUnsafePtr(); + var array1 = chunk.GetNativeArray(chunkComponentType1).GetUnsafePtr(); + var array2 = chunk.GetNativeArray(chunkComponentType2).GetUnsafePtr(); + var array3 = chunk.GetNativeArray(chunkComponentType3).GetUnsafePtr(); + var array4 = chunk.GetNativeArray(chunkComponentType4).GetUnsafePtr(); + + for (int i = 0, count = chunk.Count; i < count; ++i) + job.Execute(ref UnsafeUtilityEx.ArrayElementAsRef(array0, i), ref UnsafeUtilityEx.ArrayElementAsRef(array1, i), ref UnsafeUtilityEx.ArrayElementAsRef(array2, i), ref UnsafeUtilityEx.ArrayElementAsRef(array3, i), ref UnsafeUtilityEx.ArrayElementAsRef(array4, i)); + } + } + } + return new JobHandle(); + } + + + [EditorBrowsable(EditorBrowsableState.Never)] + public unsafe static JobHandle Schedule_EDDDDD(TJob job, ComponentSystemBase system, JobHandle dependsOn = default(JobHandle)) + where TJob : struct, IJobForEachWithEntity + where T0 : struct, IComponentData + where T1 : struct, IComponentData + where T2 : struct, IComponentData + where T3 : struct, IComponentData + where T4 : struct, IComponentData + { + { + EntityQuery query = null; + if (query== null) + { + var delegateTypes = stackalloc ComponentType[5]; + delegateTypes[0] = ComponentType.ReadWrite(); + delegateTypes[1] = ComponentType.ReadWrite(); + delegateTypes[2] = ComponentType.ReadWrite(); + delegateTypes[3] = ComponentType.ReadWrite(); + delegateTypes[4] = ComponentType.ReadWrite(); + + query = /*ResolveEntityQuery*/system.GetEntityQueryInternal(delegateTypes, 5); + } + + var entityType = system.GetArchetypeChunkEntityType(); + var chunkComponentType0 = system.GetArchetypeChunkComponentType(false); + var chunkComponentType1 = system.GetArchetypeChunkComponentType(false); + var chunkComponentType2 = system.GetArchetypeChunkComponentType(false); + var chunkComponentType3 = system.GetArchetypeChunkComponentType(false); + var chunkComponentType4 = system.GetArchetypeChunkComponentType(false); + + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) + { + foreach (var chunk in chunks) + { + var array0 = chunk.GetNativeArray(chunkComponentType0).GetUnsafePtr(); + var array1 = chunk.GetNativeArray(chunkComponentType1).GetUnsafePtr(); + var array2 = chunk.GetNativeArray(chunkComponentType2).GetUnsafePtr(); + var array3 = chunk.GetNativeArray(chunkComponentType3).GetUnsafePtr(); + var array4 = chunk.GetNativeArray(chunkComponentType4).GetUnsafePtr(); + var entityArray = (Entity*)chunk.GetNativeArray(entityType).GetUnsafeReadOnlyPtr(); + + for (int i = 0, count = chunk.Count; i < count; ++i) + job.Execute(entityArray[i], entityArray[i].Index, ref UnsafeUtilityEx.ArrayElementAsRef(array0, i), ref UnsafeUtilityEx.ArrayElementAsRef(array1, i), ref UnsafeUtilityEx.ArrayElementAsRef(array2, i), ref UnsafeUtilityEx.ArrayElementAsRef(array3, i), ref UnsafeUtilityEx.ArrayElementAsRef(array4, i)); + } + } + } + return new JobHandle(); + } + + + [EditorBrowsable(EditorBrowsableState.Never)] + public unsafe static JobHandle Schedule_DDDDDD(TJob job, ComponentSystemBase system, JobHandle dependsOn = default(JobHandle)) + where TJob : struct, IJobForEach + where T0 : struct, IComponentData + where T1 : struct, IComponentData + where T2 : struct, IComponentData + where T3 : struct, IComponentData + where T4 : struct, IComponentData + where T5 : struct, IComponentData + { + { + EntityQuery query = null; + if (query== null) + { + var delegateTypes = stackalloc ComponentType[6]; + delegateTypes[0] = ComponentType.ReadWrite(); + delegateTypes[1] = ComponentType.ReadWrite(); + delegateTypes[2] = ComponentType.ReadWrite(); + delegateTypes[3] = ComponentType.ReadWrite(); + delegateTypes[4] = ComponentType.ReadWrite(); + delegateTypes[5] = ComponentType.ReadWrite(); + + query = /*ResolveEntityQuery*/system.GetEntityQueryInternal(delegateTypes, 6); + } + + var chunkComponentType0 = system.GetArchetypeChunkComponentType(false); + var chunkComponentType1 = system.GetArchetypeChunkComponentType(false); + var chunkComponentType2 = system.GetArchetypeChunkComponentType(false); + var chunkComponentType3 = system.GetArchetypeChunkComponentType(false); + var chunkComponentType4 = system.GetArchetypeChunkComponentType(false); + var chunkComponentType5 = system.GetArchetypeChunkComponentType(false); + + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) + { + foreach (var chunk in chunks) + { + var array0 = chunk.GetNativeArray(chunkComponentType0).GetUnsafePtr(); + var array1 = chunk.GetNativeArray(chunkComponentType1).GetUnsafePtr(); + var array2 = chunk.GetNativeArray(chunkComponentType2).GetUnsafePtr(); + var array3 = chunk.GetNativeArray(chunkComponentType3).GetUnsafePtr(); + var array4 = chunk.GetNativeArray(chunkComponentType4).GetUnsafePtr(); + var array5 = chunk.GetNativeArray(chunkComponentType5).GetUnsafePtr(); + + for (int i = 0, count = chunk.Count; i < count; ++i) + job.Execute(ref UnsafeUtilityEx.ArrayElementAsRef(array0, i), ref UnsafeUtilityEx.ArrayElementAsRef(array1, i), ref UnsafeUtilityEx.ArrayElementAsRef(array2, i), ref UnsafeUtilityEx.ArrayElementAsRef(array3, i), ref UnsafeUtilityEx.ArrayElementAsRef(array4, i), ref UnsafeUtilityEx.ArrayElementAsRef(array5, i)); + } + } + } + return new JobHandle(); + } + + + [EditorBrowsable(EditorBrowsableState.Never)] + public unsafe static JobHandle Schedule_EDDDDDD(TJob job, ComponentSystemBase system, JobHandle dependsOn = default(JobHandle)) + where TJob : struct, IJobForEachWithEntity + where T0 : struct, IComponentData + where T1 : struct, IComponentData + where T2 : struct, IComponentData + where T3 : struct, IComponentData + where T4 : struct, IComponentData + where T5 : struct, IComponentData + { + { + EntityQuery query = null; + if (query== null) + { + var delegateTypes = stackalloc ComponentType[6]; + delegateTypes[0] = ComponentType.ReadWrite(); + delegateTypes[1] = ComponentType.ReadWrite(); + delegateTypes[2] = ComponentType.ReadWrite(); + delegateTypes[3] = ComponentType.ReadWrite(); + delegateTypes[4] = ComponentType.ReadWrite(); + delegateTypes[5] = ComponentType.ReadWrite(); + + query = /*ResolveEntityQuery*/system.GetEntityQueryInternal(delegateTypes, 6); + } + + var entityType = system.GetArchetypeChunkEntityType(); + var chunkComponentType0 = system.GetArchetypeChunkComponentType(false); + var chunkComponentType1 = system.GetArchetypeChunkComponentType(false); + var chunkComponentType2 = system.GetArchetypeChunkComponentType(false); + var chunkComponentType3 = system.GetArchetypeChunkComponentType(false); + var chunkComponentType4 = system.GetArchetypeChunkComponentType(false); + var chunkComponentType5 = system.GetArchetypeChunkComponentType(false); + + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) + { + foreach (var chunk in chunks) + { + var array0 = chunk.GetNativeArray(chunkComponentType0).GetUnsafePtr(); + var array1 = chunk.GetNativeArray(chunkComponentType1).GetUnsafePtr(); + var array2 = chunk.GetNativeArray(chunkComponentType2).GetUnsafePtr(); + var array3 = chunk.GetNativeArray(chunkComponentType3).GetUnsafePtr(); + var array4 = chunk.GetNativeArray(chunkComponentType4).GetUnsafePtr(); + var array5 = chunk.GetNativeArray(chunkComponentType5).GetUnsafePtr(); + var entityArray = (Entity*)chunk.GetNativeArray(entityType).GetUnsafeReadOnlyPtr(); + + for (int i = 0, count = chunk.Count; i < count; ++i) + job.Execute(entityArray[i], entityArray[i].Index, ref UnsafeUtilityEx.ArrayElementAsRef(array0, i), ref UnsafeUtilityEx.ArrayElementAsRef(array1, i), ref UnsafeUtilityEx.ArrayElementAsRef(array2, i), ref UnsafeUtilityEx.ArrayElementAsRef(array3, i), ref UnsafeUtilityEx.ArrayElementAsRef(array4, i), ref UnsafeUtilityEx.ArrayElementAsRef(array5, i)); + } + } + } + return new JobHandle(); + } + + } +#endif // UNITY_ZEROPLAYER + +} // namespace Unity.Entities diff --git a/Unity.Entities/EntityQueryBuilder_ForEach.gen.cs.meta b/Unity.Entities/EntityQueryBuilder_ForEach.gen.cs.meta new file mode 100644 index 00000000..0413c1b2 --- /dev/null +++ b/Unity.Entities/EntityQueryBuilder_ForEach.gen.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 118db93182504784ea2d31aeef050909 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/Unity.Entities/EntityQueryBuilder.tt b/Unity.Entities/EntityQueryBuilder_ForEach.tt similarity index 69% rename from Unity.Entities/EntityQueryBuilder.tt rename to Unity.Entities/EntityQueryBuilder_ForEach.tt index f203b40e..c5ea83b8 100644 --- a/Unity.Entities/EntityQueryBuilder.tt +++ b/Unity.Entities/EntityQueryBuilder_ForEach.tt @@ -16,31 +16,16 @@ <# var combinations = InitCombinations(); #> // Generated by EntityQueryBuilder.tt (<#=combinations.Count * 2 - 1#> `foreach` combinations) +using System; +using System.ComponentModel; using Unity.Collections; using Unity.Collections.LowLevel.Unsafe; +using Unity.Jobs; namespace Unity.Entities { public partial struct EntityQueryBuilder { - // ** FLUENT QUERY ** - -<# -foreach (var (name, count) in new[] { ("Any", 5), ( "None", 5), ("All", 5) }) { - for (var it = 1; it <= count; ++it) { -#> - public EntityQueryBuilder With<#=name#><<#=Series("T{0}", it, ", ")#>>() - { - ValidateHasNoGroup(); -<# for (var ia = 0; ia < it; ++ia) {#> - m_<#=name#>.Add(TypeManager.GetTypeIndex>()); -<# }#> - return this; - } - -<#}}#> - // ** FOREACH ** - <# foreach (var categories in combinations) { foreach (var hasEntity in new[] { true, false }) { @@ -50,9 +35,11 @@ foreach (var categories in combinations) { var delegateName = GetDelegateName(categories, hasEntity); var delegateParameters = GetDelegateParameters(categories, hasEntity); var genericTypes = categories.Any() ? ("<" + Series("T{0}", categories.Length, ", ") + ">") : ""; + var genericTypesWithJob = ""; var genericConstraints = GetGenericConstraints(categories); var actionParams = GetActionParams(categories, hasEntity); #> + public delegate void <#=delegateName#><#=genericTypes#>(<#=delegateParameters#>)<#=categories.Any() ? "" : ";"#> <# foreach (var constraint in Smart(genericConstraints)) {#> <#=constraint.Value#><#=constraint.IfLast(";")#> @@ -67,8 +54,8 @@ foreach (var categories in combinations) { using (InsideForEach()) #endif { - var group = m_Group; - if (group == null) + var query = m_Query; + if (query == null) { <# if (categories.Any()) {#> var delegateTypes = stackalloc int[<#=categories.Length#>]; @@ -77,7 +64,7 @@ foreach (var categories in combinations) { <# }#> <# }#> - group = ResolveComponentGroup(<#=categories.Any() ? "delegateTypes" : "null"#>, <#=categories.Length#>); + query = ResolveEntityQuery(<#=categories.Any() ? "delegateTypes" : "null"#>, <#=categories.Length#>); } <# if (hasEntity) {#> @@ -87,7 +74,7 @@ foreach (var categories in combinations) { var chunkComponentType<#=i#> = m_System.<#=AccessFunction[c]#>>(<#=IsReadOnly[c]#>); <# }#> - using (var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) { foreach (var chunk in chunks) { @@ -108,33 +95,102 @@ foreach (var categories in combinations) { <#}}#> } - // BACK-COMPAT - TO BE REMOVED - public partial class ComponentSystem - { + // Schedule() implementation for DOTS-standalone. + // The IDE calls Schedule(), and then code-gen is used to + // replace the call to the correct Schedule_D method, With + // known types. + +#if UNITY_ZEROPLAYER + public static partial class JobForEachExtensions { + + // This method is used by the IDE, and is what will be replaced by one of the Schedule_D methods below. + public unsafe static JobHandle Schedule(this TJob job, ComponentSystemBase system, JobHandle dependsOn = default(JobHandle)) + where TJob : struct, IBaseJobForEach + { + throw new NotImplementedException("Schedule() should have been replaced by code-gen."); + } <# + +string GetScheduleParams(Category[] categories, bool hasEntity) +{ + var parts = categories.Select((c, i) => string.Format(ArrayAccess[(int)c], i)); + if (hasEntity) { + parts = parts.Prepend("entityArray[i].Index"); + parts = parts.Prepend("entityArray[i]"); + } + return string.Join(", ", parts); +} + foreach (var categories in combinations) { - foreach (var hasEntity in new[] { true, false }) { - if (!categories.Any() && !hasEntity) + foreach (var hasEntity in new[] { false, true }) { + if (!categories.Any()) continue; + var mappedCategories = categories.Select((c, i) => (c: (int)c, i: i)); var delegateName = GetDelegateName(categories, hasEntity); + var delegateParameters = GetDelegateParameters(categories, hasEntity); var genericTypes = categories.Any() ? ("<" + Series("T{0}", categories.Length, ", ") + ">") : ""; + var genericTypesWithJob = ""; var genericConstraints = GetGenericConstraints(categories); + var actionParams = GetScheduleParams(categories, hasEntity); + var dName = (hasEntity ? "E" : "") + Series("D", categories.Length, ""); + var jobInterface = hasEntity ? "IJobForEachWithEntity" : "IJobForEach"; #> - [System.Obsolete("Call Entities.ForEach() or Entities.With(group).ForEach() instead")] - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public unsafe void ForEach<#=genericTypes#>(EntityQueryBuilder.<#=delegateName#><#=genericTypes#> action, ComponentGroup group = null) - <#=string.Join(" ", genericConstraints)#> - { - var q = Entities; - if (group != null) - q = q.With(group); - q.ForEach(action); - } +<# if (categories.Any() && !categories.Any(c => c != Category.D)) { #> + [EditorBrowsable(EditorBrowsableState.Never)] + public unsafe static JobHandle Schedule_<#=dName#><#=genericTypesWithJob#>(TJob job, ComponentSystemBase system, JobHandle dependsOn = default(JobHandle)) + where TJob : struct, <#=jobInterface#><#=genericTypes#> +<# foreach (var constraint in genericConstraints) {#> + <#=constraint#> +<# }#> + { + { + EntityQuery query = null; + if (query== null) + { +<# if (categories.Any()) {#> + var delegateTypes = stackalloc ComponentType[<#=categories.Length#>]; +<# foreach (var (c, i) in mappedCategories) { /*TODO: when tiny on c# 7.3 switch to initializer syntax */#> + delegateTypes[<#=i#>] = ComponentType.ReadWrite>(); +<# }#> + +<# }#> + query = /*ResolveEntityQuery*/system.GetEntityQueryInternal(<#=categories.Any() ? "delegateTypes" : "null"#>, <#=categories.Length#>); + } + +<# if (hasEntity) {#> + var entityType = system.GetArchetypeChunkEntityType(); +<# }#> +<# foreach (var (c, i) in mappedCategories) {#> + var chunkComponentType<#=i#> = system.<#=AccessFunction[c]#>>(<#=IsReadOnly[c]#>); +<# }#> + + using (var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob)) + { + foreach (var chunk in chunks) + { +<# foreach (var (c, i) in mappedCategories) {#> + var array<#=i#> = chunk.<#=string.Format(ChunkGetArray[c], i)#>; +<# }#> +<# if (hasEntity) {#> + var entityArray = (Entity*)chunk.GetNativeArray(entityType).GetUnsafeReadOnlyPtr(); +<# }#> + + for (int i = 0, count = chunk.Count; i < count; ++i) + job.Execute(<#=actionParams#>); + } + } + } + return new JobHandle(); + } + +<# } #> <#}}#> - } -} + } +#endif // UNITY_ZEROPLAYER + +} // namespace Unity.Entities <#+ enum Category diff --git a/Unity.Entities/EntityQueryBuilder_ForEach.tt.meta b/Unity.Entities/EntityQueryBuilder_ForEach.tt.meta new file mode 100644 index 00000000..ff7f7df3 --- /dev/null +++ b/Unity.Entities/EntityQueryBuilder_ForEach.tt.meta @@ -0,0 +1,7 @@ +fileFormatVersion: 2 +guid: 389ff16d82330834188334e010bd0923 +DefaultImporter: + externalObjects: {} + userData: + assetBundleName: + assetBundleVariant: diff --git a/Unity.Entities/EntityQueryBuilder_With.gen.cs b/Unity.Entities/EntityQueryBuilder_With.gen.cs new file mode 100644 index 00000000..95a0e751 --- /dev/null +++ b/Unity.Entities/EntityQueryBuilder_With.gen.cs @@ -0,0 +1,156 @@ +//------------------------------------------------------------------------------ +// +// This code was generated by a tool. +// +// Changes to this file may cause incorrect behavior and will be lost if +// the code is regenerated. +// +//------------------------------------------------------------------------------ +// Generated by EntityQueryBuilder_With.tt + +using Unity.Collections; +using Unity.Collections.LowLevel.Unsafe; + +namespace Unity.Entities +{ + public partial struct EntityQueryBuilder + { + // ** FLUENT QUERY ** + + public EntityQueryBuilder WithAny() + { + ValidateHasNoQuery(); + m_Any.Add(TypeManager.GetTypeIndex()); + return this; + } + + public EntityQueryBuilder WithAny() + { + ValidateHasNoQuery(); + m_Any.Add(TypeManager.GetTypeIndex()); + m_Any.Add(TypeManager.GetTypeIndex()); + return this; + } + + public EntityQueryBuilder WithAny() + { + ValidateHasNoQuery(); + m_Any.Add(TypeManager.GetTypeIndex()); + m_Any.Add(TypeManager.GetTypeIndex()); + m_Any.Add(TypeManager.GetTypeIndex()); + return this; + } + + public EntityQueryBuilder WithAny() + { + ValidateHasNoQuery(); + m_Any.Add(TypeManager.GetTypeIndex()); + m_Any.Add(TypeManager.GetTypeIndex()); + m_Any.Add(TypeManager.GetTypeIndex()); + m_Any.Add(TypeManager.GetTypeIndex()); + return this; + } + + public EntityQueryBuilder WithAny() + { + ValidateHasNoQuery(); + m_Any.Add(TypeManager.GetTypeIndex()); + m_Any.Add(TypeManager.GetTypeIndex()); + m_Any.Add(TypeManager.GetTypeIndex()); + m_Any.Add(TypeManager.GetTypeIndex()); + m_Any.Add(TypeManager.GetTypeIndex()); + return this; + } + + public EntityQueryBuilder WithNone() + { + ValidateHasNoQuery(); + m_None.Add(TypeManager.GetTypeIndex()); + return this; + } + + public EntityQueryBuilder WithNone() + { + ValidateHasNoQuery(); + m_None.Add(TypeManager.GetTypeIndex()); + m_None.Add(TypeManager.GetTypeIndex()); + return this; + } + + public EntityQueryBuilder WithNone() + { + ValidateHasNoQuery(); + m_None.Add(TypeManager.GetTypeIndex()); + m_None.Add(TypeManager.GetTypeIndex()); + m_None.Add(TypeManager.GetTypeIndex()); + return this; + } + + public EntityQueryBuilder WithNone() + { + ValidateHasNoQuery(); + m_None.Add(TypeManager.GetTypeIndex()); + m_None.Add(TypeManager.GetTypeIndex()); + m_None.Add(TypeManager.GetTypeIndex()); + m_None.Add(TypeManager.GetTypeIndex()); + return this; + } + + public EntityQueryBuilder WithNone() + { + ValidateHasNoQuery(); + m_None.Add(TypeManager.GetTypeIndex()); + m_None.Add(TypeManager.GetTypeIndex()); + m_None.Add(TypeManager.GetTypeIndex()); + m_None.Add(TypeManager.GetTypeIndex()); + m_None.Add(TypeManager.GetTypeIndex()); + return this; + } + + public EntityQueryBuilder WithAll() + { + ValidateHasNoQuery(); + m_All.Add(TypeManager.GetTypeIndex()); + return this; + } + + public EntityQueryBuilder WithAll() + { + ValidateHasNoQuery(); + m_All.Add(TypeManager.GetTypeIndex()); + m_All.Add(TypeManager.GetTypeIndex()); + return this; + } + + public EntityQueryBuilder WithAll() + { + ValidateHasNoQuery(); + m_All.Add(TypeManager.GetTypeIndex()); + m_All.Add(TypeManager.GetTypeIndex()); + m_All.Add(TypeManager.GetTypeIndex()); + return this; + } + + public EntityQueryBuilder WithAll() + { + ValidateHasNoQuery(); + m_All.Add(TypeManager.GetTypeIndex()); + m_All.Add(TypeManager.GetTypeIndex()); + m_All.Add(TypeManager.GetTypeIndex()); + m_All.Add(TypeManager.GetTypeIndex()); + return this; + } + + public EntityQueryBuilder WithAll() + { + ValidateHasNoQuery(); + m_All.Add(TypeManager.GetTypeIndex()); + m_All.Add(TypeManager.GetTypeIndex()); + m_All.Add(TypeManager.GetTypeIndex()); + m_All.Add(TypeManager.GetTypeIndex()); + m_All.Add(TypeManager.GetTypeIndex()); + return this; + } + + } +} diff --git a/Unity.Entities/EntityQueryBuilder_With.gen.cs.meta b/Unity.Entities/EntityQueryBuilder_With.gen.cs.meta new file mode 100644 index 00000000..05b4d4b4 --- /dev/null +++ b/Unity.Entities/EntityQueryBuilder_With.gen.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 35b8234d1cdce984490dca58bf2c083c +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/Unity.Entities/EntityQueryBuilder_With.tt b/Unity.Entities/EntityQueryBuilder_With.tt new file mode 100644 index 00000000..0e494265 --- /dev/null +++ b/Unity.Entities/EntityQueryBuilder_With.tt @@ -0,0 +1,49 @@ +<#/*THIS IS A T4 FILE - see HACKING.md for what it is and how to run codegen*/#> +<#@ assembly name="System.Collections" #> +<#@ assembly name="System.Core" #> +<#@ assembly name="System.Linq" #> +<#@ import namespace="System.Collections.Generic" #> +<#@ import namespace="System.Linq" #> +<#@ output extension=".gen.cs" #> +//------------------------------------------------------------------------------ +// +// This code was generated by a tool. +// +// Changes to this file may cause incorrect behavior and will be lost if +// the code is regenerated. +// +//------------------------------------------------------------------------------ +// Generated by EntityQueryBuilder_With.tt + +using Unity.Collections; +using Unity.Collections.LowLevel.Unsafe; + +namespace Unity.Entities +{ + public partial struct EntityQueryBuilder + { + // ** FLUENT QUERY ** + +<# +foreach (var (name, count) in new[] { ("Any", 5), ( "None", 5), ("All", 5) }) { + for (var it = 1; it <= count; ++it) { +#> + public EntityQueryBuilder With<#=name#><<#=Series("T{0}", it, ", ")#>>() + { + ValidateHasNoQuery(); +<# for (var ia = 0; ia < it; ++ia) {#> + m_<#=name#>.Add(TypeManager.GetTypeIndex>()); +<# }#> + return this; + } + +<#}}#> + } +} +<#+ + +// misc support utils + +static string Series(string formatString, int count, string separator) => + string.Join(separator, Enumerable.Range(0, count).Select(i => string.Format(formatString, i))); +#> diff --git a/Unity.Entities/EntityQueryBuilder_With.tt.meta b/Unity.Entities/EntityQueryBuilder_With.tt.meta new file mode 100644 index 00000000..9dfe94d4 --- /dev/null +++ b/Unity.Entities/EntityQueryBuilder_With.tt.meta @@ -0,0 +1,7 @@ +fileFormatVersion: 2 +guid: d43583fb8d51c0e479ff1df8ae774905 +DefaultImporter: + externalObjects: {} + userData: + assetBundleName: + assetBundleVariant: diff --git a/Unity.Entities/EntityQueryCache.cs b/Unity.Entities/EntityQueryCache.cs index 4a1e501a..f28cd24c 100644 --- a/Unity.Entities/EntityQueryCache.cs +++ b/Unity.Entities/EntityQueryCache.cs @@ -4,11 +4,11 @@ namespace Unity.Entities { class EntityQueryCache { - uint[] m_CacheHashes; // combined hash of QueryComponentBuilder and delegate types - ComponentGroup[] m_CacheComponentGroups; + uint[] m_CacheHashes; // combined hash of QueryComponentBuilder and delegate types + EntityQuery[] m_CachedEntityQueries; #if ENABLE_UNITY_COLLECTIONS_CHECKS - EntityQueryBuilder[] m_CacheCheckQueryBuilders; - int[][] m_CacheCheckDelegateTypeIndices; + EntityQueryBuilder[] m_CacheCheckQueryBuilders; + int[][] m_CacheCheckDelegateTypeIndices; #endif public EntityQueryCache(int cacheSize = 10) @@ -19,7 +19,7 @@ public EntityQueryCache(int cacheSize = 10) #endif m_CacheHashes = new uint[cacheSize]; - m_CacheComponentGroups = new ComponentGroup[cacheSize]; + m_CachedEntityQueries = new EntityQuery[cacheSize]; // we use these for an additional equality check to avoid accidental hash collisions #if ENABLE_UNITY_COLLECTIONS_CHECKS @@ -28,9 +28,18 @@ public EntityQueryCache(int cacheSize = 10) #endif } + public int CalcUsedCacheCount() + { + var used = 0; + while (used < m_CacheHashes.Length && m_CachedEntityQueries[used] != null) + ++used; + + return used; + } + public int FindQueryInCache(uint hash) { - for (var i = 0; i < m_CacheHashes.Length && m_CacheComponentGroups[i] != null; ++i) + for (var i = 0; i < m_CacheHashes.Length && m_CachedEntityQueries[i] != null; ++i) { if (m_CacheHashes[i] == hash) return i; @@ -39,15 +48,15 @@ public int FindQueryInCache(uint hash) return -1; } - public unsafe int CreateCachedQuery(uint hash, ComponentGroup group + public unsafe int CreateCachedQuery(uint hash, EntityQuery query #if ENABLE_UNITY_COLLECTIONS_CHECKS , ref EntityQueryBuilder builder, int* delegateTypeIndices, int delegateTypeCount #endif ) { #if ENABLE_UNITY_COLLECTIONS_CHECKS - if (group == null) - throw new ArgumentNullException(nameof(group)); + if (query == null) + throw new ArgumentNullException(nameof(query)); #endif var index = 0; @@ -65,7 +74,7 @@ public unsafe int CreateCachedQuery(uint hash, ComponentGroup group $"but this may cause a GC. Set cache size at init time via {nameof(ComponentSystem.InitEntityQueryCache)}() to a large enough number to ensure no allocations are required at run time."); Array.Resize(ref m_CacheHashes, newSize); - Array.Resize(ref m_CacheComponentGroups, newSize); + Array.Resize(ref m_CachedEntityQueries, newSize); #if ENABLE_UNITY_COLLECTIONS_CHECKS Array.Resize(ref m_CacheCheckQueryBuilders, newSize); @@ -74,7 +83,7 @@ public unsafe int CreateCachedQuery(uint hash, ComponentGroup group break; } - if (m_CacheComponentGroups[index] == null) + if (m_CachedEntityQueries[index] == null) break; #if ENABLE_UNITY_COLLECTIONS_CHECKS @@ -88,7 +97,7 @@ public unsafe int CreateCachedQuery(uint hash, ComponentGroup group // store in cache m_CacheHashes[index] = hash; - m_CacheComponentGroups[index] = group; + m_CachedEntityQueries[index] = query; #if ENABLE_UNITY_COLLECTIONS_CHECKS m_CacheCheckQueryBuilders[index] = builder; @@ -101,8 +110,8 @@ public unsafe int CreateCachedQuery(uint hash, ComponentGroup group return index; } - public ComponentGroup GetCachedQuery(int cacheIndex) - => m_CacheComponentGroups[cacheIndex]; + public EntityQuery GetCachedQuery(int cacheIndex) + => m_CachedEntityQueries[cacheIndex]; #if ENABLE_UNITY_COLLECTIONS_CHECKS public unsafe void ValidateMatchesCache(int foundCacheIndex, ref EntityQueryBuilder builder, int* delegateTypeIndices, int delegateTypeCount) diff --git a/Unity.Entities/EntityRemapUtility.cs b/Unity.Entities/EntityRemapUtility.cs index d6533a8b..0aa9565e 100644 --- a/Unity.Entities/EntityRemapUtility.cs +++ b/Unity.Entities/EntityRemapUtility.cs @@ -82,7 +82,7 @@ public struct BufferEntityPatchInfo public int ElementStride; } -#if UNITY_CSHARP_TINY +#if NET_DOTS // @TODO TINY -- Need to use UnsafeArray to provide a view of the data in sEntityOffsetArray in the static type manager public static EntityOffsetInfo[] CalculateEntityOffsets() { diff --git a/Unity.Entities/ExclusiveEntityTransaction.cs b/Unity.Entities/ExclusiveEntityTransaction.cs index 46a3c13f..808ead66 100644 --- a/Unity.Entities/ExclusiveEntityTransaction.cs +++ b/Unity.Entities/ExclusiveEntityTransaction.cs @@ -39,7 +39,7 @@ internal ExclusiveEntityTransaction(ArchetypeManager archetypes, EntityGroupMana m_SharedComponentDataManager = GCHandle.Alloc(sharedComponentDataManager, GCHandleType.Weak); } - internal void OnDestroyManager() + internal void OnDestroy() { m_ArchetypeManager.Free(); m_EntityGroupManager.Free(); diff --git a/Unity.Entities/IJobChunk.cs b/Unity.Entities/IJobChunk.cs index 79aaaba4..09884a7a 100644 --- a/Unity.Entities/IJobChunk.cs +++ b/Unity.Entities/IJobChunk.cs @@ -11,7 +11,7 @@ namespace Unity.Entities #endif public interface IJobChunk { - // firstEntityIndex refers to the index of the first entity in the current chunk within the ComponentGroup the job was scheduled with + // firstEntityIndex refers to the index of the first entity in the current chunk within the EntityQuery the job was scheduled with // For example, if the job operates on 3 chunks with 20 entities each, then the firstEntityIndices will be [0, 20, 40] respectively void Execute(ArchetypeChunk chunk, int chunkIndex, int firstEntityIndex); } @@ -40,25 +40,25 @@ internal struct JobChunkData where T : struct public NativeArray PrefilterData; } - public static unsafe JobHandle Schedule(this T jobData, ComponentGroup group, JobHandle dependsOn = default(JobHandle)) + public static unsafe JobHandle Schedule(this T jobData, EntityQuery query, JobHandle dependsOn = default(JobHandle)) where T : struct, IJobChunk { - return ScheduleInternal(ref jobData, group, dependsOn, ScheduleMode.Batched); + return ScheduleInternal(ref jobData, query, dependsOn, ScheduleMode.Batched); } - public static void Run(this T jobData, ComponentGroup group) + public static void Run(this T jobData, EntityQuery query) where T : struct, IJobChunk { - ScheduleInternal(ref jobData, group, default(JobHandle), ScheduleMode.Run); + ScheduleInternal(ref jobData, query, default(JobHandle), ScheduleMode.Run); } #if !UNITY_ZEROPLAYER - internal static unsafe JobHandle ScheduleInternal(ref T jobData, ComponentGroup group, JobHandle dependsOn, ScheduleMode mode) + internal static unsafe JobHandle ScheduleInternal(ref T jobData, EntityQuery query, JobHandle dependsOn, ScheduleMode mode) where T : struct, IJobChunk { - ComponentChunkIterator iterator = group.GetComponentChunkIterator(); + ComponentChunkIterator iterator = query.GetComponentChunkIterator(); - var unfilteredChunkCount = group.CalculateNumberOfChunksWithoutFiltering(); + var unfilteredChunkCount = query.CalculateNumberOfChunksWithoutFiltering(); var prefilterHandle = ComponentChunkIterator.PreparePrefilteredChunkLists(unfilteredChunkCount, iterator.m_MatchingArchetypeList, iterator.m_Filter, dependsOn, mode, out var prefilterData, @@ -69,7 +69,7 @@ internal static unsafe JobHandle ScheduleInternal(ref T jobData, ComponentGro #if ENABLE_UNITY_COLLECTIONS_CHECKS // All IJobChunk jobs have a EntityManager safety handle to ensure that BeforeStructuralChange throws an error if // jobs without any other safety handles are still running (haven't been synced). - safety = new EntitySafetyHandle{m_Safety = group.SafetyManager.GetEntityManagerSafetyHandle()}, + safety = new EntitySafetyHandle{m_Safety = query.SafetyManager->GetEntityManagerSafetyHandle()}, #endif Data = jobData, PrefilterData = prefilterData, @@ -124,8 +124,7 @@ public unsafe static void Execute(ref JobChunkData jobData, System.IntPtr add internal unsafe static void ExecuteInternal(ref JobChunkData jobData, ref JobRanges ranges, int jobIndex) { - var filteredChunks = (ArchetypeChunk*)NativeArrayUnsafeUtility.GetUnsafePtr(jobData.PrefilterData); - var entityIndices = (int*) (filteredChunks + ranges.TotalIterationCount); + ComponentChunkIterator.UnpackPrefilterData(jobData.PrefilterData, out var filteredChunks, out var entityIndices, out var chunkCount); int chunkIndex, end; while (JobsUtility.GetWorkStealingRange(ref ranges, jobIndex, out chunkIndex, out end)) @@ -138,10 +137,10 @@ internal unsafe static void ExecuteInternal(ref JobChunkData jobData, ref Job } } #else - internal static unsafe JobHandle ScheduleInternal(ref T jobData, ComponentGroup group, JobHandle dependsOn, ScheduleMode mode) + internal static unsafe JobHandle ScheduleInternal(ref T jobData, EntityQuery query, JobHandle dependsOn, ScheduleMode mode) where T : struct, IJobChunk { - using (var chunks = group.CreateArchetypeChunkArray(Allocator.Temp)) + using (var chunks = query.CreateArchetypeChunkArray(Allocator.Temp)) { int currentChunk = 0; int currentEntity = 0; @@ -153,8 +152,15 @@ internal static unsafe JobHandle ScheduleInternal(ref T jobData, ComponentGro } } + DoDeallocateOnJobCompletion(jobData); + return new JobHandle(); } + + static internal void DoDeallocateOnJobCompletion(object jobData) + { + throw new NotImplementedException("This function should have been replaced by codegen"); + } #endif } } diff --git a/Unity.Entities/IJobProcessComponentData.cs b/Unity.Entities/IJobForEach.cs similarity index 79% rename from Unity.Entities/IJobProcessComponentData.cs rename to Unity.Entities/IJobForEach.cs index 3e3f6323..fc42f2bd 100644 --- a/Unity.Entities/IJobProcessComponentData.cs +++ b/Unity.Entities/IJobForEach.cs @@ -10,8 +10,6 @@ using Unity.Jobs.LowLevel.Unsafe; using ReadOnlyAttribute = Unity.Collections.ReadOnlyAttribute; -#if !UNITY_ZEROPLAYER - namespace Unity.Entities { //@TODO: What about change or add? @@ -19,7 +17,7 @@ namespace Unity.Entities public class ChangedFilterAttribute : Attribute { } - + [AttributeUsage(AttributeTargets.Struct)] public class RequireComponentTagAttribute : Attribute { @@ -42,11 +40,17 @@ public ExcludeComponentAttribute(params Type[] subtractiveComponents) } } - public static partial class JobProcessComponentDataExtensions + public static partial class JobForEachExtensions { + [EditorBrowsable(EditorBrowsableState.Never)] + public interface IBaseJobForEach + { + } + +#if !UNITY_ZEROPLAYER static ComponentType[] GetComponentTypes(Type jobType) { - var interfaceType = GetIJobProcessComponentDataInterface(jobType); + var interfaceType = GetIJobForEachInterface(jobType); if (interfaceType != null) { int temp; @@ -63,19 +67,19 @@ static ComponentType[] GetComponentTypes(Type jobType, Type interfaceType, out i var genericArgs = interfaceType.GetGenericArguments(); var executeMethodParameters = jobType.GetMethod("Execute").GetParameters(); - + var componentTypes = new List(); var changedFilterTypes = new List(); - + // void Execute(Entity entity, int index, ref T0 data0, ref T1 data1, ref T2 data2); // First two parameters are optional, depending on the interface name used. var methodParameterOffset = genericArgs.Length != executeMethodParameters.Length ? 2 : 0; - + for (var i = 0; i < genericArgs.Length; i++) { var isReadonly = executeMethodParameters[i + methodParameterOffset].GetCustomAttribute(typeof(ReadOnlyAttribute)) != null; - + var type = new ComponentType(genericArgs[i], isReadonly ? ComponentType.AccessMode.ReadOnly : ComponentType.AccessMode.ReadWrite); componentTypes.Add(type); @@ -99,16 +103,16 @@ static ComponentType[] GetComponentTypes(Type jobType, Type interfaceType, out i changedFilter = changedFilterTypes.ToArray(); return componentTypes.ToArray(); } - + static int CalculateEntityCount(ComponentSystemBase system, Type jobType) { - var componentGroup = GetComponentGroupForIJobProcessComponentData(system, jobType); + var query = GetEntityQueryForIJobForEach(system, jobType); - int entityCount = componentGroup.CalculateLength(); + int entityCount = query.CalculateLength(); return entityCount; } - + static IntPtr GetJobReflection(Type jobType, Type wrapperJobType, Type interfaceType, bool isIJobParallelFor) { @@ -127,34 +131,34 @@ static IntPtr GetJobReflection(Type jobType, Type wrapperJobType, Type interface return (IntPtr) reflectionDataRes; } - static Type GetIJobProcessComponentDataInterface(Type jobType) + static Type GetIJobForEachInterface(Type jobType) { foreach (var iType in jobType.GetInterfaces()) - if (iType.Assembly == typeof(IBaseJobProcessComponentData).Assembly && - iType.Name.StartsWith("IJobProcessComponentData")) + if (iType.Assembly == typeof(IBaseJobForEach).Assembly && + iType.Name.StartsWith("IJobForEach")) return iType; return null; } - static void PrepareComponentGroup(ComponentSystemBase system, Type jobType) + static void PrepareEntityQuery(ComponentSystemBase system, Type jobType) { - var iType = GetIJobProcessComponentDataInterface(jobType); - + var iType = GetIJobForEachInterface(jobType); + ComponentType[] filterChanged; int processTypesCount; var types = GetComponentTypes(jobType, iType, out processTypesCount, out filterChanged); - system.GetComponentGroupInternal(types); + system.GetEntityQueryInternal(types); } - static unsafe void Initialize(ComponentSystemBase system, ComponentGroup componentGroup, Type jobType, Type wrapperJobType, - bool isParallelFor, ref JobProcessComponentDataCache cache, out ProcessIterationData iterator) + static unsafe void Initialize(ComponentSystemBase system, EntityQuery entityQuery, Type jobType, Type wrapperJobType, + bool isParallelFor, ref JobForEachCache cache, out ProcessIterationData iterator) { // Get the job reflection data and cache it if we don't already have it cached. if (isParallelFor && cache.JobReflectionDataParallelFor == IntPtr.Zero || !isParallelFor && cache.JobReflectionData == IntPtr.Zero) { - var iType = GetIJobProcessComponentDataInterface(jobType); + var iType = GetIJobForEachInterface(jobType); if (cache.Types == null) cache.Types = GetComponentTypes(jobType, iType, out cache.ProcessTypesCount, out cache.FilterChanged); @@ -167,43 +171,43 @@ static unsafe void Initialize(ComponentSystemBase system, ComponentGroup compone cache.JobReflectionData = res; } - // Update cached ComponentGroup and ComponentSystem data. + // Update cached EntityQuery and ComponentSystem data. if (system != null) { if (cache.ComponentSystem != system) { - cache.ComponentGroup = system.GetComponentGroupInternal(cache.Types); + cache.EntityQuery = system.GetEntityQueryInternal(cache.Types); - // If the cached filter has changed, update the newly cached ComponentGroup with those changes. + // If the cached filter has changed, update the newly cached EntityQuery with those changes. if (cache.FilterChanged.Length != 0) - cache.ComponentGroup.SetFilterChanged(cache.FilterChanged); + cache.EntityQuery.SetFilterChanged(cache.FilterChanged); - // Otherwise, just reset our newly cached ComponentGroup's filter. + // Otherwise, just reset our newly cached EntityQuery's filter. else - cache.ComponentGroup.ResetFilter(); + cache.EntityQuery.ResetFilter(); cache.ComponentSystem = system; } } - else if (componentGroup != null) + else if (entityQuery != null) { - if (cache.ComponentGroup != componentGroup) + if (cache.EntityQuery != entityQuery) { - // Cache the new ComponentGroup and cache that our system is null. - cache.ComponentGroup = componentGroup; + // Cache the new EntityQuery and cache that our system is null. + cache.EntityQuery = entityQuery; cache.ComponentSystem = null; } } - var group = cache.ComponentGroup; - + var query = cache.EntityQuery; + iterator.IsReadOnly0 = iterator.IsReadOnly1 = iterator.IsReadOnly2 = iterator.IsReadOnly3 = iterator.IsReadOnly4 = iterator.IsReadOnly5= 0; fixed (int* isReadOnly = &iterator.IsReadOnly0) { for (var i = 0; i != cache.ProcessTypesCount; i++) isReadOnly[i] = cache.Types[i].AccessModeType == ComponentType.AccessMode.ReadOnly ? 1 : 0; } - + iterator.TypeIndex0 = iterator.TypeIndex1 = iterator.TypeIndex2 = iterator.TypeIndex3 = iterator.TypeIndex4 = iterator.TypeIndex5 = -1; fixed (int* typeIndices = &iterator.TypeIndex0) { @@ -212,14 +216,14 @@ static unsafe void Initialize(ComponentSystemBase system, ComponentGroup compone } iterator.m_IsParallelFor = isParallelFor; - iterator.m_Length = group.CalculateNumberOfChunksWithoutFiltering(); + iterator.m_Length = query.CalculateNumberOfChunksWithoutFiltering(); - iterator.GlobalSystemVersion = group.GetComponentChunkIterator().m_GlobalSystemVersion; + iterator.GlobalSystemVersion = query.GetComponentChunkIterator().m_GlobalSystemVersion; #if ENABLE_UNITY_COLLECTIONS_CHECKS iterator.m_MaxIndex = iterator.m_Length - 1; iterator.m_MinIndex = 0; - + iterator.m_Safety0 = iterator.m_Safety1 = iterator.m_Safety2 = iterator.m_Safety3 = iterator.m_Safety4 = iterator.m_Safety5 = default(AtomicSafetyHandle); @@ -230,7 +234,7 @@ static unsafe void Initialize(ComponentSystemBase system, ComponentGroup compone if (cache.Types[i].AccessModeType == ComponentType.AccessMode.ReadOnly) { safety[iterator.m_SafetyReadOnlyCount] = - group.GetSafetyHandle(group.GetIndexInComponentGroup(cache.Types[i].TypeIndex)); + query.GetSafetyHandle(query.GetIndexInEntityQuery(cache.Types[i].TypeIndex)); iterator.m_SafetyReadOnlyCount++; } } @@ -242,7 +246,7 @@ static unsafe void Initialize(ComponentSystemBase system, ComponentGroup compone if (cache.Types[i].AccessModeType == ComponentType.AccessMode.ReadWrite) { safety[iterator.m_SafetyReadOnlyCount + iterator.m_SafetyReadWriteCount] = - group.GetSafetyHandle(group.GetIndexInComponentGroup(cache.Types[i].TypeIndex)); + query.GetSafetyHandle(query.GetIndexInEntityQuery(cache.Types[i].TypeIndex)); iterator.m_SafetyReadWriteCount++; } } @@ -250,13 +254,8 @@ static unsafe void Initialize(ComponentSystemBase system, ComponentGroup compone Assert.AreEqual(cache.ProcessTypesCount, iterator.m_SafetyReadWriteCount + iterator.m_SafetyReadOnlyCount); #endif } - - [EditorBrowsable(EditorBrowsableState.Never)] - public interface IBaseJobProcessComponentData - { - } - internal struct JobProcessComponentDataCache + internal struct JobForEachCache { public IntPtr JobReflectionData; public IntPtr JobReflectionDataParallelFor; @@ -265,7 +264,7 @@ internal struct JobProcessComponentDataCache public int ProcessTypesCount; - public ComponentGroup ComponentGroup; + public EntityQuery EntityQuery; public ComponentSystemBase ComponentSystem; } @@ -275,7 +274,7 @@ internal struct JobProcessComponentDataCache internal struct ProcessIterationData { public uint GlobalSystemVersion; - + public int TypeIndex0; public int TypeIndex1; public int TypeIndex2; @@ -289,7 +288,7 @@ internal struct ProcessIterationData public int IsReadOnly3; public int IsReadOnly4; public int IsReadOnly5; - + public bool m_IsParallelFor; public int m_Length; @@ -311,35 +310,36 @@ internal struct ProcessIterationData #pragma warning restore #endif } - public static ComponentGroup GetComponentGroupForIJobProcessComponentData(this ComponentSystemBase system, + + public static EntityQuery GetEntityQueryForIJobForEach(this ComponentSystemBase system, Type jobType) { var types = GetComponentTypes(jobType); if (types != null) - return system.GetComponentGroupInternal(types); + return system.GetEntityQueryInternal(types); else return null; } - + //NOTE: It would be much better if C# could resolve the branch with generic resolving, // but apparently the interface constraint is not enough.. - public static void PrepareComponentGroup(this T jobData, ComponentSystemBase system) - where T : struct, IBaseJobProcessComponentData + public static void PrepareEntityQuery(this T jobData, ComponentSystemBase system) + where T : struct, IBaseJobForEach { - PrepareComponentGroup(system, typeof(T)); + PrepareEntityQuery(system, typeof(T)); } - + public static int CalculateEntityCount(this T jobData, ComponentSystemBase system) - where T : struct, IBaseJobProcessComponentData + where T : struct, IBaseJobForEach { return CalculateEntityCount(system, typeof(T)); } - static unsafe JobHandle Schedule(void* fullData, NativeArray prefilterData, int unfilteredLength, int innerloopBatchCount, - bool isParallelFor, bool isFiltered, ref JobProcessComponentDataCache cache, void* deferredCountData, JobHandle dependsOn, ScheduleMode mode) + static unsafe JobHandle Schedule(void* fullData, NativeArray prefilterData, int unfilteredLength, int innerloopBatchCount, + bool isParallelFor, bool isFiltered, ref JobForEachCache cache, void* deferredCountData, JobHandle dependsOn, ScheduleMode mode) { -#if ENABLE_UNITY_COLLECTIONS_CHECKS +#if ENABLE_UNITY_COLLECTIONS_CHECKS try { #endif @@ -356,15 +356,15 @@ static unsafe JobHandle Schedule(void* fullData, NativeArray prefilterData var scheduleParams = new JobsUtility.JobScheduleParameters(fullData, cache.JobReflectionData, dependsOn, mode); return JobsUtility.Schedule(ref scheduleParams); } -#if ENABLE_UNITY_COLLECTIONS_CHECKS +#if ENABLE_UNITY_COLLECTIONS_CHECKS } catch (InvalidOperationException e) { prefilterData.Dispose(); throw e; } -#endif +#endif } +#endif } } -#endif \ No newline at end of file diff --git a/Unity.Entities/IJobProcessComponentData.cs.meta b/Unity.Entities/IJobForEach.cs.meta similarity index 100% rename from Unity.Entities/IJobProcessComponentData.cs.meta rename to Unity.Entities/IJobForEach.cs.meta diff --git a/Unity.Entities/IJobForEach.gen.cs b/Unity.Entities/IJobForEach.gen.cs new file mode 100644 index 00000000..a3392406 --- /dev/null +++ b/Unity.Entities/IJobForEach.gen.cs @@ -0,0 +1,1649 @@ +//------------------------------------------------------------------------------ +// +// This code was generated by a tool. +// +// Changes to this file may cause incorrect behavior and will be lost if +// the code is regenerated. +// +//------------------------------------------------------------------------------ +// Generated by T4 (TextTransform.exe) from the file IJobForEach.tt +// + +using Unity.Collections; +using Unity.Collections.LowLevel.Unsafe; +using Unity.Jobs; +using Unity.Jobs.LowLevel.Unsafe; +using System.Runtime.InteropServices; +using UnityEngine.Scripting; +using System; + +namespace Unity.Entities +{ + +#if !UNITY_ZEROPLAYER + [JobProducerType(typeof(JobForEachExtensions.JobStruct_Process_D<,>))] +#endif + public interface IJobForEach : JobForEachExtensions.IBaseJobForEach_D + where U0 : struct, IComponentData + { + void Execute(ref U0 c0); + } + +#if !UNITY_ZEROPLAYER + [JobProducerType(typeof(JobForEachExtensions.JobStruct_Process_ED<,>))] +#endif + public interface IJobForEachWithEntity : JobForEachExtensions.IBaseJobForEach_ED + where U0 : struct, IComponentData + { + void Execute(Entity entity, int index, ref U0 c0); + } + +#if !UNITY_ZEROPLAYER + [JobProducerType(typeof(JobForEachExtensions.JobStruct_Process_DD<,,>))] +#endif + public interface IJobForEach : JobForEachExtensions.IBaseJobForEach_DD + where U0 : struct, IComponentData + where U1 : struct, IComponentData + { + void Execute(ref U0 c0, ref U1 c1); + } + +#if !UNITY_ZEROPLAYER + [JobProducerType(typeof(JobForEachExtensions.JobStruct_Process_EDD<,,>))] +#endif + public interface IJobForEachWithEntity : JobForEachExtensions.IBaseJobForEach_EDD + where U0 : struct, IComponentData + where U1 : struct, IComponentData + { + void Execute(Entity entity, int index, ref U0 c0, ref U1 c1); + } + +#if !UNITY_ZEROPLAYER + [JobProducerType(typeof(JobForEachExtensions.JobStruct_Process_DDD<,,,>))] +#endif + public interface IJobForEach : JobForEachExtensions.IBaseJobForEach_DDD + where U0 : struct, IComponentData + where U1 : struct, IComponentData + where U2 : struct, IComponentData + { + void Execute(ref U0 c0, ref U1 c1, ref U2 c2); + } + +#if !UNITY_ZEROPLAYER + [JobProducerType(typeof(JobForEachExtensions.JobStruct_Process_EDDD<,,,>))] +#endif + public interface IJobForEachWithEntity : JobForEachExtensions.IBaseJobForEach_EDDD + where U0 : struct, IComponentData + where U1 : struct, IComponentData + where U2 : struct, IComponentData + { + void Execute(Entity entity, int index, ref U0 c0, ref U1 c1, ref U2 c2); + } + +#if !UNITY_ZEROPLAYER + [JobProducerType(typeof(JobForEachExtensions.JobStruct_Process_DDDD<,,,,>))] +#endif + public interface IJobForEach : JobForEachExtensions.IBaseJobForEach_DDDD + where U0 : struct, IComponentData + where U1 : struct, IComponentData + where U2 : struct, IComponentData + where U3 : struct, IComponentData + { + void Execute(ref U0 c0, ref U1 c1, ref U2 c2, ref U3 c3); + } + +#if !UNITY_ZEROPLAYER + [JobProducerType(typeof(JobForEachExtensions.JobStruct_Process_EDDDD<,,,,>))] +#endif + public interface IJobForEachWithEntity : JobForEachExtensions.IBaseJobForEach_EDDDD + where U0 : struct, IComponentData + where U1 : struct, IComponentData + where U2 : struct, IComponentData + where U3 : struct, IComponentData + { + void Execute(Entity entity, int index, ref U0 c0, ref U1 c1, ref U2 c2, ref U3 c3); + } + +#if !UNITY_ZEROPLAYER + [JobProducerType(typeof(JobForEachExtensions.JobStruct_Process_DDDDD<,,,,,>))] +#endif + public interface IJobForEach : JobForEachExtensions.IBaseJobForEach_DDDDD + where U0 : struct, IComponentData + where U1 : struct, IComponentData + where U2 : struct, IComponentData + where U3 : struct, IComponentData + where U4 : struct, IComponentData + { + void Execute(ref U0 c0, ref U1 c1, ref U2 c2, ref U3 c3, ref U4 c4); + } + +#if !UNITY_ZEROPLAYER + [JobProducerType(typeof(JobForEachExtensions.JobStruct_Process_EDDDDD<,,,,,>))] +#endif + public interface IJobForEachWithEntity : JobForEachExtensions.IBaseJobForEach_EDDDDD + where U0 : struct, IComponentData + where U1 : struct, IComponentData + where U2 : struct, IComponentData + where U3 : struct, IComponentData + where U4 : struct, IComponentData + { + void Execute(Entity entity, int index, ref U0 c0, ref U1 c1, ref U2 c2, ref U3 c3, ref U4 c4); + } + +#if !UNITY_ZEROPLAYER + [JobProducerType(typeof(JobForEachExtensions.JobStruct_Process_DDDDDD<,,,,,,>))] +#endif + public interface IJobForEach : JobForEachExtensions.IBaseJobForEach_DDDDDD + where U0 : struct, IComponentData + where U1 : struct, IComponentData + where U2 : struct, IComponentData + where U3 : struct, IComponentData + where U4 : struct, IComponentData + where U5 : struct, IComponentData + { + void Execute(ref U0 c0, ref U1 c1, ref U2 c2, ref U3 c3, ref U4 c4, ref U5 c5); + } + +#if !UNITY_ZEROPLAYER + [JobProducerType(typeof(JobForEachExtensions.JobStruct_Process_EDDDDDD<,,,,,,>))] +#endif + public interface IJobForEachWithEntity : JobForEachExtensions.IBaseJobForEach_EDDDDDD + where U0 : struct, IComponentData + where U1 : struct, IComponentData + where U2 : struct, IComponentData + where U3 : struct, IComponentData + where U4 : struct, IComponentData + where U5 : struct, IComponentData + { + void Execute(Entity entity, int index, ref U0 c0, ref U1 c1, ref U2 c2, ref U3 c3, ref U4 c4, ref U5 c5); + } + + public static partial class JobForEachExtensions + { +#if !UNITY_ZEROPLAYER + public static JobHandle Schedule(this T jobData, ComponentSystemBase system, JobHandle dependsOn = default(JobHandle)) + where T : struct, IBaseJobForEach + { + var typeT = typeof(T); + if (typeof(IBaseJobForEach_D).IsAssignableFrom(typeT)) + return ScheduleInternal_D(ref jobData, system, null, 1, dependsOn, ScheduleMode.Batched); + if (typeof(IBaseJobForEach_ED).IsAssignableFrom(typeT)) + return ScheduleInternal_ED(ref jobData, system, null, 1, dependsOn, ScheduleMode.Batched); + if (typeof(IBaseJobForEach_DD).IsAssignableFrom(typeT)) + return ScheduleInternal_DD(ref jobData, system, null, 1, dependsOn, ScheduleMode.Batched); + if (typeof(IBaseJobForEach_EDD).IsAssignableFrom(typeT)) + return ScheduleInternal_EDD(ref jobData, system, null, 1, dependsOn, ScheduleMode.Batched); + if (typeof(IBaseJobForEach_DDD).IsAssignableFrom(typeT)) + return ScheduleInternal_DDD(ref jobData, system, null, 1, dependsOn, ScheduleMode.Batched); + if (typeof(IBaseJobForEach_EDDD).IsAssignableFrom(typeT)) + return ScheduleInternal_EDDD(ref jobData, system, null, 1, dependsOn, ScheduleMode.Batched); + if (typeof(IBaseJobForEach_DDDD).IsAssignableFrom(typeT)) + return ScheduleInternal_DDDD(ref jobData, system, null, 1, dependsOn, ScheduleMode.Batched); + if (typeof(IBaseJobForEach_EDDDD).IsAssignableFrom(typeT)) + return ScheduleInternal_EDDDD(ref jobData, system, null, 1, dependsOn, ScheduleMode.Batched); + if (typeof(IBaseJobForEach_DDDDD).IsAssignableFrom(typeT)) + return ScheduleInternal_DDDDD(ref jobData, system, null, 1, dependsOn, ScheduleMode.Batched); + if (typeof(IBaseJobForEach_EDDDDD).IsAssignableFrom(typeT)) + return ScheduleInternal_EDDDDD(ref jobData, system, null, 1, dependsOn, ScheduleMode.Batched); + if (typeof(IBaseJobForEach_DDDDDD).IsAssignableFrom(typeT)) + return ScheduleInternal_DDDDDD(ref jobData, system, null, 1, dependsOn, ScheduleMode.Batched); + if (typeof(IBaseJobForEach_EDDDDDD).IsAssignableFrom(typeT)) + return ScheduleInternal_EDDDDDD(ref jobData, system, null, 1, dependsOn, ScheduleMode.Batched); + throw new System.ArgumentException("Not supported"); + } +#endif +#if !UNITY_ZEROPLAYER + public static JobHandle ScheduleSingle(this T jobData, ComponentSystemBase system, JobHandle dependsOn = default(JobHandle)) + where T : struct, IBaseJobForEach + { + var typeT = typeof(T); + if (typeof(IBaseJobForEach_D).IsAssignableFrom(typeT)) + return ScheduleInternal_D(ref jobData, system, null, -1, dependsOn, ScheduleMode.Batched); + if (typeof(IBaseJobForEach_ED).IsAssignableFrom(typeT)) + return ScheduleInternal_ED(ref jobData, system, null, -1, dependsOn, ScheduleMode.Batched); + if (typeof(IBaseJobForEach_DD).IsAssignableFrom(typeT)) + return ScheduleInternal_DD(ref jobData, system, null, -1, dependsOn, ScheduleMode.Batched); + if (typeof(IBaseJobForEach_EDD).IsAssignableFrom(typeT)) + return ScheduleInternal_EDD(ref jobData, system, null, -1, dependsOn, ScheduleMode.Batched); + if (typeof(IBaseJobForEach_DDD).IsAssignableFrom(typeT)) + return ScheduleInternal_DDD(ref jobData, system, null, -1, dependsOn, ScheduleMode.Batched); + if (typeof(IBaseJobForEach_EDDD).IsAssignableFrom(typeT)) + return ScheduleInternal_EDDD(ref jobData, system, null, -1, dependsOn, ScheduleMode.Batched); + if (typeof(IBaseJobForEach_DDDD).IsAssignableFrom(typeT)) + return ScheduleInternal_DDDD(ref jobData, system, null, -1, dependsOn, ScheduleMode.Batched); + if (typeof(IBaseJobForEach_EDDDD).IsAssignableFrom(typeT)) + return ScheduleInternal_EDDDD(ref jobData, system, null, -1, dependsOn, ScheduleMode.Batched); + if (typeof(IBaseJobForEach_DDDDD).IsAssignableFrom(typeT)) + return ScheduleInternal_DDDDD(ref jobData, system, null, -1, dependsOn, ScheduleMode.Batched); + if (typeof(IBaseJobForEach_EDDDDD).IsAssignableFrom(typeT)) + return ScheduleInternal_EDDDDD(ref jobData, system, null, -1, dependsOn, ScheduleMode.Batched); + if (typeof(IBaseJobForEach_DDDDDD).IsAssignableFrom(typeT)) + return ScheduleInternal_DDDDDD(ref jobData, system, null, -1, dependsOn, ScheduleMode.Batched); + if (typeof(IBaseJobForEach_EDDDDDD).IsAssignableFrom(typeT)) + return ScheduleInternal_EDDDDDD(ref jobData, system, null, -1, dependsOn, ScheduleMode.Batched); + throw new System.ArgumentException("Not supported"); + } +#endif +#if !UNITY_ZEROPLAYER + public static JobHandle Run(this T jobData, ComponentSystemBase system, JobHandle dependsOn = default(JobHandle)) + where T : struct, IBaseJobForEach + { + var typeT = typeof(T); + if (typeof(IBaseJobForEach_D).IsAssignableFrom(typeT)) + return ScheduleInternal_D(ref jobData, system, null, -1, dependsOn, ScheduleMode.Run); + if (typeof(IBaseJobForEach_ED).IsAssignableFrom(typeT)) + return ScheduleInternal_ED(ref jobData, system, null, -1, dependsOn, ScheduleMode.Run); + if (typeof(IBaseJobForEach_DD).IsAssignableFrom(typeT)) + return ScheduleInternal_DD(ref jobData, system, null, -1, dependsOn, ScheduleMode.Run); + if (typeof(IBaseJobForEach_EDD).IsAssignableFrom(typeT)) + return ScheduleInternal_EDD(ref jobData, system, null, -1, dependsOn, ScheduleMode.Run); + if (typeof(IBaseJobForEach_DDD).IsAssignableFrom(typeT)) + return ScheduleInternal_DDD(ref jobData, system, null, -1, dependsOn, ScheduleMode.Run); + if (typeof(IBaseJobForEach_EDDD).IsAssignableFrom(typeT)) + return ScheduleInternal_EDDD(ref jobData, system, null, -1, dependsOn, ScheduleMode.Run); + if (typeof(IBaseJobForEach_DDDD).IsAssignableFrom(typeT)) + return ScheduleInternal_DDDD(ref jobData, system, null, -1, dependsOn, ScheduleMode.Run); + if (typeof(IBaseJobForEach_EDDDD).IsAssignableFrom(typeT)) + return ScheduleInternal_EDDDD(ref jobData, system, null, -1, dependsOn, ScheduleMode.Run); + if (typeof(IBaseJobForEach_DDDDD).IsAssignableFrom(typeT)) + return ScheduleInternal_DDDDD(ref jobData, system, null, -1, dependsOn, ScheduleMode.Run); + if (typeof(IBaseJobForEach_EDDDDD).IsAssignableFrom(typeT)) + return ScheduleInternal_EDDDDD(ref jobData, system, null, -1, dependsOn, ScheduleMode.Run); + if (typeof(IBaseJobForEach_DDDDDD).IsAssignableFrom(typeT)) + return ScheduleInternal_DDDDDD(ref jobData, system, null, -1, dependsOn, ScheduleMode.Run); + if (typeof(IBaseJobForEach_EDDDDDD).IsAssignableFrom(typeT)) + return ScheduleInternal_EDDDDDD(ref jobData, system, null, -1, dependsOn, ScheduleMode.Run); + throw new System.ArgumentException("Not supported"); + } +#endif + +#if !UNITY_ZEROPLAYER + public static JobHandle Schedule(this T jobData, EntityQuery query, JobHandle dependsOn = default(JobHandle)) + where T : struct, IBaseJobForEach + { + var typeT = typeof(T); + if (typeof(IBaseJobForEach_D).IsAssignableFrom(typeT)) + return ScheduleInternal_D(ref jobData, null, query, 1, dependsOn, ScheduleMode.Batched); + if (typeof(IBaseJobForEach_ED).IsAssignableFrom(typeT)) + return ScheduleInternal_ED(ref jobData, null, query, 1, dependsOn, ScheduleMode.Batched); + if (typeof(IBaseJobForEach_DD).IsAssignableFrom(typeT)) + return ScheduleInternal_DD(ref jobData, null, query, 1, dependsOn, ScheduleMode.Batched); + if (typeof(IBaseJobForEach_EDD).IsAssignableFrom(typeT)) + return ScheduleInternal_EDD(ref jobData, null, query, 1, dependsOn, ScheduleMode.Batched); + if (typeof(IBaseJobForEach_DDD).IsAssignableFrom(typeT)) + return ScheduleInternal_DDD(ref jobData, null, query, 1, dependsOn, ScheduleMode.Batched); + if (typeof(IBaseJobForEach_EDDD).IsAssignableFrom(typeT)) + return ScheduleInternal_EDDD(ref jobData, null, query, 1, dependsOn, ScheduleMode.Batched); + if (typeof(IBaseJobForEach_DDDD).IsAssignableFrom(typeT)) + return ScheduleInternal_DDDD(ref jobData, null, query, 1, dependsOn, ScheduleMode.Batched); + if (typeof(IBaseJobForEach_EDDDD).IsAssignableFrom(typeT)) + return ScheduleInternal_EDDDD(ref jobData, null, query, 1, dependsOn, ScheduleMode.Batched); + if (typeof(IBaseJobForEach_DDDDD).IsAssignableFrom(typeT)) + return ScheduleInternal_DDDDD(ref jobData, null, query, 1, dependsOn, ScheduleMode.Batched); + if (typeof(IBaseJobForEach_EDDDDD).IsAssignableFrom(typeT)) + return ScheduleInternal_EDDDDD(ref jobData, null, query, 1, dependsOn, ScheduleMode.Batched); + if (typeof(IBaseJobForEach_DDDDDD).IsAssignableFrom(typeT)) + return ScheduleInternal_DDDDDD(ref jobData, null, query, 1, dependsOn, ScheduleMode.Batched); + if (typeof(IBaseJobForEach_EDDDDDD).IsAssignableFrom(typeT)) + return ScheduleInternal_EDDDDDD(ref jobData, null, query, 1, dependsOn, ScheduleMode.Batched); + throw new System.ArgumentException("Not supported"); + } +#endif +#if !UNITY_ZEROPLAYER + public static JobHandle ScheduleSingle(this T jobData, EntityQuery query, JobHandle dependsOn = default(JobHandle)) + where T : struct, IBaseJobForEach + { + var typeT = typeof(T); + if (typeof(IBaseJobForEach_D).IsAssignableFrom(typeT)) + return ScheduleInternal_D(ref jobData, null, query, -1, dependsOn, ScheduleMode.Batched); + if (typeof(IBaseJobForEach_ED).IsAssignableFrom(typeT)) + return ScheduleInternal_ED(ref jobData, null, query, -1, dependsOn, ScheduleMode.Batched); + if (typeof(IBaseJobForEach_DD).IsAssignableFrom(typeT)) + return ScheduleInternal_DD(ref jobData, null, query, -1, dependsOn, ScheduleMode.Batched); + if (typeof(IBaseJobForEach_EDD).IsAssignableFrom(typeT)) + return ScheduleInternal_EDD(ref jobData, null, query, -1, dependsOn, ScheduleMode.Batched); + if (typeof(IBaseJobForEach_DDD).IsAssignableFrom(typeT)) + return ScheduleInternal_DDD(ref jobData, null, query, -1, dependsOn, ScheduleMode.Batched); + if (typeof(IBaseJobForEach_EDDD).IsAssignableFrom(typeT)) + return ScheduleInternal_EDDD(ref jobData, null, query, -1, dependsOn, ScheduleMode.Batched); + if (typeof(IBaseJobForEach_DDDD).IsAssignableFrom(typeT)) + return ScheduleInternal_DDDD(ref jobData, null, query, -1, dependsOn, ScheduleMode.Batched); + if (typeof(IBaseJobForEach_EDDDD).IsAssignableFrom(typeT)) + return ScheduleInternal_EDDDD(ref jobData, null, query, -1, dependsOn, ScheduleMode.Batched); + if (typeof(IBaseJobForEach_DDDDD).IsAssignableFrom(typeT)) + return ScheduleInternal_DDDDD(ref jobData, null, query, -1, dependsOn, ScheduleMode.Batched); + if (typeof(IBaseJobForEach_EDDDDD).IsAssignableFrom(typeT)) + return ScheduleInternal_EDDDDD(ref jobData, null, query, -1, dependsOn, ScheduleMode.Batched); + if (typeof(IBaseJobForEach_DDDDDD).IsAssignableFrom(typeT)) + return ScheduleInternal_DDDDDD(ref jobData, null, query, -1, dependsOn, ScheduleMode.Batched); + if (typeof(IBaseJobForEach_EDDDDDD).IsAssignableFrom(typeT)) + return ScheduleInternal_EDDDDDD(ref jobData, null, query, -1, dependsOn, ScheduleMode.Batched); + throw new System.ArgumentException("Not supported"); + } +#endif +#if !UNITY_ZEROPLAYER + public static JobHandle Run(this T jobData, EntityQuery query, JobHandle dependsOn = default(JobHandle)) + where T : struct, IBaseJobForEach + { + var typeT = typeof(T); + if (typeof(IBaseJobForEach_D).IsAssignableFrom(typeT)) + return ScheduleInternal_D(ref jobData, null, query, -1, dependsOn, ScheduleMode.Run); + if (typeof(IBaseJobForEach_ED).IsAssignableFrom(typeT)) + return ScheduleInternal_ED(ref jobData, null, query, -1, dependsOn, ScheduleMode.Run); + if (typeof(IBaseJobForEach_DD).IsAssignableFrom(typeT)) + return ScheduleInternal_DD(ref jobData, null, query, -1, dependsOn, ScheduleMode.Run); + if (typeof(IBaseJobForEach_EDD).IsAssignableFrom(typeT)) + return ScheduleInternal_EDD(ref jobData, null, query, -1, dependsOn, ScheduleMode.Run); + if (typeof(IBaseJobForEach_DDD).IsAssignableFrom(typeT)) + return ScheduleInternal_DDD(ref jobData, null, query, -1, dependsOn, ScheduleMode.Run); + if (typeof(IBaseJobForEach_EDDD).IsAssignableFrom(typeT)) + return ScheduleInternal_EDDD(ref jobData, null, query, -1, dependsOn, ScheduleMode.Run); + if (typeof(IBaseJobForEach_DDDD).IsAssignableFrom(typeT)) + return ScheduleInternal_DDDD(ref jobData, null, query, -1, dependsOn, ScheduleMode.Run); + if (typeof(IBaseJobForEach_EDDDD).IsAssignableFrom(typeT)) + return ScheduleInternal_EDDDD(ref jobData, null, query, -1, dependsOn, ScheduleMode.Run); + if (typeof(IBaseJobForEach_DDDDD).IsAssignableFrom(typeT)) + return ScheduleInternal_DDDDD(ref jobData, null, query, -1, dependsOn, ScheduleMode.Run); + if (typeof(IBaseJobForEach_EDDDDD).IsAssignableFrom(typeT)) + return ScheduleInternal_EDDDDD(ref jobData, null, query, -1, dependsOn, ScheduleMode.Run); + if (typeof(IBaseJobForEach_DDDDDD).IsAssignableFrom(typeT)) + return ScheduleInternal_DDDDDD(ref jobData, null, query, -1, dependsOn, ScheduleMode.Run); + if (typeof(IBaseJobForEach_EDDDDDD).IsAssignableFrom(typeT)) + return ScheduleInternal_EDDDDDD(ref jobData, null, query, -1, dependsOn, ScheduleMode.Run); + throw new System.ArgumentException("Not supported"); + } +#endif + +#if !UNITY_ZEROPLAYER + internal static unsafe JobHandle ScheduleInternal_D(ref T jobData, ComponentSystemBase system, EntityQuery query, int innerloopBatchCount, JobHandle dependsOn, ScheduleMode mode) + where T : struct + { + JobStruct_ProcessInfer_D fullData; + fullData.Data = jobData; + + var isParallelFor = innerloopBatchCount != -1; + Initialize(system, query, typeof(T), typeof(JobStruct_Process_D<,>), isParallelFor, ref JobStruct_ProcessInfer_D.Cache, out fullData.Iterator); + + var unfilteredChunkCount = fullData.Iterator.m_Length; + var iterator = JobStruct_ProcessInfer_D.Cache.EntityQuery.GetComponentChunkIterator(); + + var prefilterHandle = ComponentChunkIterator.PreparePrefilteredChunkLists(unfilteredChunkCount, iterator.m_MatchingArchetypeList, iterator.m_Filter, dependsOn, mode, out fullData.PrefilterData, out var deferredCountData); + + return Schedule(UnsafeUtility.AddressOf(ref fullData), fullData.PrefilterData, fullData.Iterator.m_Length, innerloopBatchCount, isParallelFor, iterator.RequiresFilter(), ref JobStruct_ProcessInfer_D.Cache, deferredCountData, prefilterHandle, mode); + } +#endif + + [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] + public interface IBaseJobForEach_D : IBaseJobForEach {} + +#if !UNITY_ZEROPLAYER + [StructLayout(LayoutKind.Sequential)] + private struct JobStruct_ProcessInfer_D where T : struct + { + public static JobForEachCache Cache; + + public ProcessIterationData Iterator; + public T Data; + + [DeallocateOnJobCompletion] + [NativeDisableContainerSafetyRestriction] + public NativeArray PrefilterData; + } + + [StructLayout(LayoutKind.Sequential)] + internal struct JobStruct_Process_D + where T : struct, IJobForEach + where U0 : struct, IComponentData + { + public ProcessIterationData Iterator; + public T Data; + + [DeallocateOnJobCompletion] + [NativeDisableContainerSafetyRestriction] + public NativeArray PrefilterData; + + [Preserve] + public static IntPtr Initialize(JobType jobType) + { + return JobsUtility.CreateJobReflectionData(typeof(JobStruct_Process_D), typeof(T), jobType, (ExecuteJobFunction) Execute); + } + + delegate void ExecuteJobFunction(ref JobStruct_Process_D data, IntPtr additionalPtr, IntPtr bufferRangePatchData, ref JobRanges ranges, int jobIndex); + + public static unsafe void Execute(ref JobStruct_Process_D jobData, IntPtr additionalPtr, IntPtr bufferRangePatchData, ref JobRanges ranges, int jobIndex) + { + ComponentChunkIterator.UnpackPrefilterData(jobData.PrefilterData, out var chunks, out var entityIndices, out var chunkCount); + + if (jobData.Iterator.m_IsParallelFor) + { + int begin, end; + while (JobsUtility.GetWorkStealingRange(ref ranges, jobIndex, out begin, out end)) + ExecuteChunk(ref jobData, bufferRangePatchData, begin, end, chunks, entityIndices); + } + else + { + ExecuteChunk(ref jobData, bufferRangePatchData, 0, chunkCount, chunks, entityIndices); + } + } + + static unsafe void ExecuteChunk(ref JobStruct_Process_D jobData, IntPtr bufferRangePatchData, int begin, int end, ArchetypeChunk* chunks, int* entityIndices) + { + var typeLookupCache0 = 0; + + for (var blockIndex = begin; blockIndex != end; ++blockIndex) + { + var chunk = chunks[blockIndex]; + int beginIndex = entityIndices[blockIndex]; + var count = chunk.Count; +#if ENABLE_UNITY_COLLECTIONS_CHECKS + JobsUtility.PatchBufferMinMaxRanges(bufferRangePatchData, UnsafeUtility.AddressOf(ref jobData), beginIndex, beginIndex + count); +#endif + ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex0, ref typeLookupCache0); + var ptr0 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly0 == 0, typeLookupCache0, jobData.Iterator.GlobalSystemVersion)); + + + for (var i = 0; i != count; i++) + { + jobData.Data.Execute(ref UnsafeUtilityEx.ArrayElementAsRef(ptr0, i)); + } + } + } + } +#endif + +#if !UNITY_ZEROPLAYER + internal static unsafe JobHandle ScheduleInternal_ED(ref T jobData, ComponentSystemBase system, EntityQuery query, int innerloopBatchCount, JobHandle dependsOn, ScheduleMode mode) + where T : struct + { + JobStruct_ProcessInfer_ED fullData; + fullData.Data = jobData; + + var isParallelFor = innerloopBatchCount != -1; + Initialize(system, query, typeof(T), typeof(JobStruct_Process_ED<,>), isParallelFor, ref JobStruct_ProcessInfer_ED.Cache, out fullData.Iterator); + + var unfilteredChunkCount = fullData.Iterator.m_Length; + var iterator = JobStruct_ProcessInfer_ED.Cache.EntityQuery.GetComponentChunkIterator(); + + var prefilterHandle = ComponentChunkIterator.PreparePrefilteredChunkLists(unfilteredChunkCount, iterator.m_MatchingArchetypeList, iterator.m_Filter, dependsOn, mode, out fullData.PrefilterData, out var deferredCountData); + + return Schedule(UnsafeUtility.AddressOf(ref fullData), fullData.PrefilterData, fullData.Iterator.m_Length, innerloopBatchCount, isParallelFor, iterator.RequiresFilter(), ref JobStruct_ProcessInfer_ED.Cache, deferredCountData, prefilterHandle, mode); + } +#endif + + [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] + public interface IBaseJobForEach_ED : IBaseJobForEach {} + +#if !UNITY_ZEROPLAYER + [StructLayout(LayoutKind.Sequential)] + private struct JobStruct_ProcessInfer_ED where T : struct + { + public static JobForEachCache Cache; + + public ProcessIterationData Iterator; + public T Data; + + [DeallocateOnJobCompletion] + [NativeDisableContainerSafetyRestriction] + public NativeArray PrefilterData; + } + + [StructLayout(LayoutKind.Sequential)] + internal struct JobStruct_Process_ED + where T : struct, IJobForEachWithEntity + where U0 : struct, IComponentData + { + public ProcessIterationData Iterator; + public T Data; + + [DeallocateOnJobCompletion] + [NativeDisableContainerSafetyRestriction] + public NativeArray PrefilterData; + + [Preserve] + public static IntPtr Initialize(JobType jobType) + { + return JobsUtility.CreateJobReflectionData(typeof(JobStruct_Process_ED), typeof(T), jobType, (ExecuteJobFunction) Execute); + } + + delegate void ExecuteJobFunction(ref JobStruct_Process_ED data, IntPtr additionalPtr, IntPtr bufferRangePatchData, ref JobRanges ranges, int jobIndex); + + public static unsafe void Execute(ref JobStruct_Process_ED jobData, IntPtr additionalPtr, IntPtr bufferRangePatchData, ref JobRanges ranges, int jobIndex) + { + ComponentChunkIterator.UnpackPrefilterData(jobData.PrefilterData, out var chunks, out var entityIndices, out var chunkCount); + + if (jobData.Iterator.m_IsParallelFor) + { + int begin, end; + while (JobsUtility.GetWorkStealingRange(ref ranges, jobIndex, out begin, out end)) + ExecuteChunk(ref jobData, bufferRangePatchData, begin, end, chunks, entityIndices); + } + else + { + ExecuteChunk(ref jobData, bufferRangePatchData, 0, chunkCount, chunks, entityIndices); + } + } + + static unsafe void ExecuteChunk(ref JobStruct_Process_ED jobData, IntPtr bufferRangePatchData, int begin, int end, ArchetypeChunk* chunks, int* entityIndices) + { + var typeLookupCache0 = 0; + + for (var blockIndex = begin; blockIndex != end; ++blockIndex) + { + var chunk = chunks[blockIndex]; + int beginIndex = entityIndices[blockIndex]; + var count = chunk.Count; +#if ENABLE_UNITY_COLLECTIONS_CHECKS + JobsUtility.PatchBufferMinMaxRanges(bufferRangePatchData, UnsafeUtility.AddressOf(ref jobData), beginIndex, beginIndex + count); +#endif + var ptrE = (Entity*)UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, false, 0, jobData.Iterator.GlobalSystemVersion)); + ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex0, ref typeLookupCache0); + var ptr0 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly0 == 0, typeLookupCache0, jobData.Iterator.GlobalSystemVersion)); + + + for (var i = 0; i != count; i++) + { + jobData.Data.Execute(ptrE[i], i + beginIndex, ref UnsafeUtilityEx.ArrayElementAsRef(ptr0, i)); + } + } + } + } +#endif + +#if !UNITY_ZEROPLAYER + internal static unsafe JobHandle ScheduleInternal_DD(ref T jobData, ComponentSystemBase system, EntityQuery query, int innerloopBatchCount, JobHandle dependsOn, ScheduleMode mode) + where T : struct + { + JobStruct_ProcessInfer_DD fullData; + fullData.Data = jobData; + + var isParallelFor = innerloopBatchCount != -1; + Initialize(system, query, typeof(T), typeof(JobStruct_Process_DD<,,>), isParallelFor, ref JobStruct_ProcessInfer_DD.Cache, out fullData.Iterator); + + var unfilteredChunkCount = fullData.Iterator.m_Length; + var iterator = JobStruct_ProcessInfer_DD.Cache.EntityQuery.GetComponentChunkIterator(); + + var prefilterHandle = ComponentChunkIterator.PreparePrefilteredChunkLists(unfilteredChunkCount, iterator.m_MatchingArchetypeList, iterator.m_Filter, dependsOn, mode, out fullData.PrefilterData, out var deferredCountData); + + return Schedule(UnsafeUtility.AddressOf(ref fullData), fullData.PrefilterData, fullData.Iterator.m_Length, innerloopBatchCount, isParallelFor, iterator.RequiresFilter(), ref JobStruct_ProcessInfer_DD.Cache, deferredCountData, prefilterHandle, mode); + } +#endif + + [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] + public interface IBaseJobForEach_DD : IBaseJobForEach {} + +#if !UNITY_ZEROPLAYER + [StructLayout(LayoutKind.Sequential)] + private struct JobStruct_ProcessInfer_DD where T : struct + { + public static JobForEachCache Cache; + + public ProcessIterationData Iterator; + public T Data; + + [DeallocateOnJobCompletion] + [NativeDisableContainerSafetyRestriction] + public NativeArray PrefilterData; + } + + [StructLayout(LayoutKind.Sequential)] + internal struct JobStruct_Process_DD + where T : struct, IJobForEach + where U0 : struct, IComponentData + where U1 : struct, IComponentData + { + public ProcessIterationData Iterator; + public T Data; + + [DeallocateOnJobCompletion] + [NativeDisableContainerSafetyRestriction] + public NativeArray PrefilterData; + + [Preserve] + public static IntPtr Initialize(JobType jobType) + { + return JobsUtility.CreateJobReflectionData(typeof(JobStruct_Process_DD), typeof(T), jobType, (ExecuteJobFunction) Execute); + } + + delegate void ExecuteJobFunction(ref JobStruct_Process_DD data, IntPtr additionalPtr, IntPtr bufferRangePatchData, ref JobRanges ranges, int jobIndex); + + public static unsafe void Execute(ref JobStruct_Process_DD jobData, IntPtr additionalPtr, IntPtr bufferRangePatchData, ref JobRanges ranges, int jobIndex) + { + ComponentChunkIterator.UnpackPrefilterData(jobData.PrefilterData, out var chunks, out var entityIndices, out var chunkCount); + + if (jobData.Iterator.m_IsParallelFor) + { + int begin, end; + while (JobsUtility.GetWorkStealingRange(ref ranges, jobIndex, out begin, out end)) + ExecuteChunk(ref jobData, bufferRangePatchData, begin, end, chunks, entityIndices); + } + else + { + ExecuteChunk(ref jobData, bufferRangePatchData, 0, chunkCount, chunks, entityIndices); + } + } + + static unsafe void ExecuteChunk(ref JobStruct_Process_DD jobData, IntPtr bufferRangePatchData, int begin, int end, ArchetypeChunk* chunks, int* entityIndices) + { + var typeLookupCache0 = 0; + var typeLookupCache1 = 0; + + for (var blockIndex = begin; blockIndex != end; ++blockIndex) + { + var chunk = chunks[blockIndex]; + int beginIndex = entityIndices[blockIndex]; + var count = chunk.Count; +#if ENABLE_UNITY_COLLECTIONS_CHECKS + JobsUtility.PatchBufferMinMaxRanges(bufferRangePatchData, UnsafeUtility.AddressOf(ref jobData), beginIndex, beginIndex + count); +#endif + ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex0, ref typeLookupCache0); + var ptr0 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly0 == 0, typeLookupCache0, jobData.Iterator.GlobalSystemVersion)); + ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex1, ref typeLookupCache1); + var ptr1 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly1 == 0, typeLookupCache1, jobData.Iterator.GlobalSystemVersion)); + + + for (var i = 0; i != count; i++) + { + jobData.Data.Execute(ref UnsafeUtilityEx.ArrayElementAsRef(ptr0, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr1, i)); + } + } + } + } +#endif + +#if !UNITY_ZEROPLAYER + internal static unsafe JobHandle ScheduleInternal_EDD(ref T jobData, ComponentSystemBase system, EntityQuery query, int innerloopBatchCount, JobHandle dependsOn, ScheduleMode mode) + where T : struct + { + JobStruct_ProcessInfer_EDD fullData; + fullData.Data = jobData; + + var isParallelFor = innerloopBatchCount != -1; + Initialize(system, query, typeof(T), typeof(JobStruct_Process_EDD<,,>), isParallelFor, ref JobStruct_ProcessInfer_EDD.Cache, out fullData.Iterator); + + var unfilteredChunkCount = fullData.Iterator.m_Length; + var iterator = JobStruct_ProcessInfer_EDD.Cache.EntityQuery.GetComponentChunkIterator(); + + var prefilterHandle = ComponentChunkIterator.PreparePrefilteredChunkLists(unfilteredChunkCount, iterator.m_MatchingArchetypeList, iterator.m_Filter, dependsOn, mode, out fullData.PrefilterData, out var deferredCountData); + + return Schedule(UnsafeUtility.AddressOf(ref fullData), fullData.PrefilterData, fullData.Iterator.m_Length, innerloopBatchCount, isParallelFor, iterator.RequiresFilter(), ref JobStruct_ProcessInfer_EDD.Cache, deferredCountData, prefilterHandle, mode); + } +#endif + + [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] + public interface IBaseJobForEach_EDD : IBaseJobForEach {} + +#if !UNITY_ZEROPLAYER + [StructLayout(LayoutKind.Sequential)] + private struct JobStruct_ProcessInfer_EDD where T : struct + { + public static JobForEachCache Cache; + + public ProcessIterationData Iterator; + public T Data; + + [DeallocateOnJobCompletion] + [NativeDisableContainerSafetyRestriction] + public NativeArray PrefilterData; + } + + [StructLayout(LayoutKind.Sequential)] + internal struct JobStruct_Process_EDD + where T : struct, IJobForEachWithEntity + where U0 : struct, IComponentData + where U1 : struct, IComponentData + { + public ProcessIterationData Iterator; + public T Data; + + [DeallocateOnJobCompletion] + [NativeDisableContainerSafetyRestriction] + public NativeArray PrefilterData; + + [Preserve] + public static IntPtr Initialize(JobType jobType) + { + return JobsUtility.CreateJobReflectionData(typeof(JobStruct_Process_EDD), typeof(T), jobType, (ExecuteJobFunction) Execute); + } + + delegate void ExecuteJobFunction(ref JobStruct_Process_EDD data, IntPtr additionalPtr, IntPtr bufferRangePatchData, ref JobRanges ranges, int jobIndex); + + public static unsafe void Execute(ref JobStruct_Process_EDD jobData, IntPtr additionalPtr, IntPtr bufferRangePatchData, ref JobRanges ranges, int jobIndex) + { + ComponentChunkIterator.UnpackPrefilterData(jobData.PrefilterData, out var chunks, out var entityIndices, out var chunkCount); + + if (jobData.Iterator.m_IsParallelFor) + { + int begin, end; + while (JobsUtility.GetWorkStealingRange(ref ranges, jobIndex, out begin, out end)) + ExecuteChunk(ref jobData, bufferRangePatchData, begin, end, chunks, entityIndices); + } + else + { + ExecuteChunk(ref jobData, bufferRangePatchData, 0, chunkCount, chunks, entityIndices); + } + } + + static unsafe void ExecuteChunk(ref JobStruct_Process_EDD jobData, IntPtr bufferRangePatchData, int begin, int end, ArchetypeChunk* chunks, int* entityIndices) + { + var typeLookupCache0 = 0; + var typeLookupCache1 = 0; + + for (var blockIndex = begin; blockIndex != end; ++blockIndex) + { + var chunk = chunks[blockIndex]; + int beginIndex = entityIndices[blockIndex]; + var count = chunk.Count; +#if ENABLE_UNITY_COLLECTIONS_CHECKS + JobsUtility.PatchBufferMinMaxRanges(bufferRangePatchData, UnsafeUtility.AddressOf(ref jobData), beginIndex, beginIndex + count); +#endif + var ptrE = (Entity*)UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, false, 0, jobData.Iterator.GlobalSystemVersion)); + ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex0, ref typeLookupCache0); + var ptr0 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly0 == 0, typeLookupCache0, jobData.Iterator.GlobalSystemVersion)); + ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex1, ref typeLookupCache1); + var ptr1 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly1 == 0, typeLookupCache1, jobData.Iterator.GlobalSystemVersion)); + + + for (var i = 0; i != count; i++) + { + jobData.Data.Execute(ptrE[i], i + beginIndex, ref UnsafeUtilityEx.ArrayElementAsRef(ptr0, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr1, i)); + } + } + } + } +#endif + +#if !UNITY_ZEROPLAYER + internal static unsafe JobHandle ScheduleInternal_DDD(ref T jobData, ComponentSystemBase system, EntityQuery query, int innerloopBatchCount, JobHandle dependsOn, ScheduleMode mode) + where T : struct + { + JobStruct_ProcessInfer_DDD fullData; + fullData.Data = jobData; + + var isParallelFor = innerloopBatchCount != -1; + Initialize(system, query, typeof(T), typeof(JobStruct_Process_DDD<,,,>), isParallelFor, ref JobStruct_ProcessInfer_DDD.Cache, out fullData.Iterator); + + var unfilteredChunkCount = fullData.Iterator.m_Length; + var iterator = JobStruct_ProcessInfer_DDD.Cache.EntityQuery.GetComponentChunkIterator(); + + var prefilterHandle = ComponentChunkIterator.PreparePrefilteredChunkLists(unfilteredChunkCount, iterator.m_MatchingArchetypeList, iterator.m_Filter, dependsOn, mode, out fullData.PrefilterData, out var deferredCountData); + + return Schedule(UnsafeUtility.AddressOf(ref fullData), fullData.PrefilterData, fullData.Iterator.m_Length, innerloopBatchCount, isParallelFor, iterator.RequiresFilter(), ref JobStruct_ProcessInfer_DDD.Cache, deferredCountData, prefilterHandle, mode); + } +#endif + + [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] + public interface IBaseJobForEach_DDD : IBaseJobForEach {} + +#if !UNITY_ZEROPLAYER + [StructLayout(LayoutKind.Sequential)] + private struct JobStruct_ProcessInfer_DDD where T : struct + { + public static JobForEachCache Cache; + + public ProcessIterationData Iterator; + public T Data; + + [DeallocateOnJobCompletion] + [NativeDisableContainerSafetyRestriction] + public NativeArray PrefilterData; + } + + [StructLayout(LayoutKind.Sequential)] + internal struct JobStruct_Process_DDD + where T : struct, IJobForEach + where U0 : struct, IComponentData + where U1 : struct, IComponentData + where U2 : struct, IComponentData + { + public ProcessIterationData Iterator; + public T Data; + + [DeallocateOnJobCompletion] + [NativeDisableContainerSafetyRestriction] + public NativeArray PrefilterData; + + [Preserve] + public static IntPtr Initialize(JobType jobType) + { + return JobsUtility.CreateJobReflectionData(typeof(JobStruct_Process_DDD), typeof(T), jobType, (ExecuteJobFunction) Execute); + } + + delegate void ExecuteJobFunction(ref JobStruct_Process_DDD data, IntPtr additionalPtr, IntPtr bufferRangePatchData, ref JobRanges ranges, int jobIndex); + + public static unsafe void Execute(ref JobStruct_Process_DDD jobData, IntPtr additionalPtr, IntPtr bufferRangePatchData, ref JobRanges ranges, int jobIndex) + { + ComponentChunkIterator.UnpackPrefilterData(jobData.PrefilterData, out var chunks, out var entityIndices, out var chunkCount); + + if (jobData.Iterator.m_IsParallelFor) + { + int begin, end; + while (JobsUtility.GetWorkStealingRange(ref ranges, jobIndex, out begin, out end)) + ExecuteChunk(ref jobData, bufferRangePatchData, begin, end, chunks, entityIndices); + } + else + { + ExecuteChunk(ref jobData, bufferRangePatchData, 0, chunkCount, chunks, entityIndices); + } + } + + static unsafe void ExecuteChunk(ref JobStruct_Process_DDD jobData, IntPtr bufferRangePatchData, int begin, int end, ArchetypeChunk* chunks, int* entityIndices) + { + var typeLookupCache0 = 0; + var typeLookupCache1 = 0; + var typeLookupCache2 = 0; + + for (var blockIndex = begin; blockIndex != end; ++blockIndex) + { + var chunk = chunks[blockIndex]; + int beginIndex = entityIndices[blockIndex]; + var count = chunk.Count; +#if ENABLE_UNITY_COLLECTIONS_CHECKS + JobsUtility.PatchBufferMinMaxRanges(bufferRangePatchData, UnsafeUtility.AddressOf(ref jobData), beginIndex, beginIndex + count); +#endif + ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex0, ref typeLookupCache0); + var ptr0 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly0 == 0, typeLookupCache0, jobData.Iterator.GlobalSystemVersion)); + ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex1, ref typeLookupCache1); + var ptr1 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly1 == 0, typeLookupCache1, jobData.Iterator.GlobalSystemVersion)); + ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex2, ref typeLookupCache2); + var ptr2 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly2 == 0, typeLookupCache2, jobData.Iterator.GlobalSystemVersion)); + + + for (var i = 0; i != count; i++) + { + jobData.Data.Execute(ref UnsafeUtilityEx.ArrayElementAsRef(ptr0, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr1, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr2, i)); + } + } + } + } +#endif + +#if !UNITY_ZEROPLAYER + internal static unsafe JobHandle ScheduleInternal_EDDD(ref T jobData, ComponentSystemBase system, EntityQuery query, int innerloopBatchCount, JobHandle dependsOn, ScheduleMode mode) + where T : struct + { + JobStruct_ProcessInfer_EDDD fullData; + fullData.Data = jobData; + + var isParallelFor = innerloopBatchCount != -1; + Initialize(system, query, typeof(T), typeof(JobStruct_Process_EDDD<,,,>), isParallelFor, ref JobStruct_ProcessInfer_EDDD.Cache, out fullData.Iterator); + + var unfilteredChunkCount = fullData.Iterator.m_Length; + var iterator = JobStruct_ProcessInfer_EDDD.Cache.EntityQuery.GetComponentChunkIterator(); + + var prefilterHandle = ComponentChunkIterator.PreparePrefilteredChunkLists(unfilteredChunkCount, iterator.m_MatchingArchetypeList, iterator.m_Filter, dependsOn, mode, out fullData.PrefilterData, out var deferredCountData); + + return Schedule(UnsafeUtility.AddressOf(ref fullData), fullData.PrefilterData, fullData.Iterator.m_Length, innerloopBatchCount, isParallelFor, iterator.RequiresFilter(), ref JobStruct_ProcessInfer_EDDD.Cache, deferredCountData, prefilterHandle, mode); + } +#endif + + [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] + public interface IBaseJobForEach_EDDD : IBaseJobForEach {} + +#if !UNITY_ZEROPLAYER + [StructLayout(LayoutKind.Sequential)] + private struct JobStruct_ProcessInfer_EDDD where T : struct + { + public static JobForEachCache Cache; + + public ProcessIterationData Iterator; + public T Data; + + [DeallocateOnJobCompletion] + [NativeDisableContainerSafetyRestriction] + public NativeArray PrefilterData; + } + + [StructLayout(LayoutKind.Sequential)] + internal struct JobStruct_Process_EDDD + where T : struct, IJobForEachWithEntity + where U0 : struct, IComponentData + where U1 : struct, IComponentData + where U2 : struct, IComponentData + { + public ProcessIterationData Iterator; + public T Data; + + [DeallocateOnJobCompletion] + [NativeDisableContainerSafetyRestriction] + public NativeArray PrefilterData; + + [Preserve] + public static IntPtr Initialize(JobType jobType) + { + return JobsUtility.CreateJobReflectionData(typeof(JobStruct_Process_EDDD), typeof(T), jobType, (ExecuteJobFunction) Execute); + } + + delegate void ExecuteJobFunction(ref JobStruct_Process_EDDD data, IntPtr additionalPtr, IntPtr bufferRangePatchData, ref JobRanges ranges, int jobIndex); + + public static unsafe void Execute(ref JobStruct_Process_EDDD jobData, IntPtr additionalPtr, IntPtr bufferRangePatchData, ref JobRanges ranges, int jobIndex) + { + ComponentChunkIterator.UnpackPrefilterData(jobData.PrefilterData, out var chunks, out var entityIndices, out var chunkCount); + + if (jobData.Iterator.m_IsParallelFor) + { + int begin, end; + while (JobsUtility.GetWorkStealingRange(ref ranges, jobIndex, out begin, out end)) + ExecuteChunk(ref jobData, bufferRangePatchData, begin, end, chunks, entityIndices); + } + else + { + ExecuteChunk(ref jobData, bufferRangePatchData, 0, chunkCount, chunks, entityIndices); + } + } + + static unsafe void ExecuteChunk(ref JobStruct_Process_EDDD jobData, IntPtr bufferRangePatchData, int begin, int end, ArchetypeChunk* chunks, int* entityIndices) + { + var typeLookupCache0 = 0; + var typeLookupCache1 = 0; + var typeLookupCache2 = 0; + + for (var blockIndex = begin; blockIndex != end; ++blockIndex) + { + var chunk = chunks[blockIndex]; + int beginIndex = entityIndices[blockIndex]; + var count = chunk.Count; +#if ENABLE_UNITY_COLLECTIONS_CHECKS + JobsUtility.PatchBufferMinMaxRanges(bufferRangePatchData, UnsafeUtility.AddressOf(ref jobData), beginIndex, beginIndex + count); +#endif + var ptrE = (Entity*)UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, false, 0, jobData.Iterator.GlobalSystemVersion)); + ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex0, ref typeLookupCache0); + var ptr0 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly0 == 0, typeLookupCache0, jobData.Iterator.GlobalSystemVersion)); + ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex1, ref typeLookupCache1); + var ptr1 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly1 == 0, typeLookupCache1, jobData.Iterator.GlobalSystemVersion)); + ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex2, ref typeLookupCache2); + var ptr2 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly2 == 0, typeLookupCache2, jobData.Iterator.GlobalSystemVersion)); + + + for (var i = 0; i != count; i++) + { + jobData.Data.Execute(ptrE[i], i + beginIndex, ref UnsafeUtilityEx.ArrayElementAsRef(ptr0, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr1, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr2, i)); + } + } + } + } +#endif + +#if !UNITY_ZEROPLAYER + internal static unsafe JobHandle ScheduleInternal_DDDD(ref T jobData, ComponentSystemBase system, EntityQuery query, int innerloopBatchCount, JobHandle dependsOn, ScheduleMode mode) + where T : struct + { + JobStruct_ProcessInfer_DDDD fullData; + fullData.Data = jobData; + + var isParallelFor = innerloopBatchCount != -1; + Initialize(system, query, typeof(T), typeof(JobStruct_Process_DDDD<,,,,>), isParallelFor, ref JobStruct_ProcessInfer_DDDD.Cache, out fullData.Iterator); + + var unfilteredChunkCount = fullData.Iterator.m_Length; + var iterator = JobStruct_ProcessInfer_DDDD.Cache.EntityQuery.GetComponentChunkIterator(); + + var prefilterHandle = ComponentChunkIterator.PreparePrefilteredChunkLists(unfilteredChunkCount, iterator.m_MatchingArchetypeList, iterator.m_Filter, dependsOn, mode, out fullData.PrefilterData, out var deferredCountData); + + return Schedule(UnsafeUtility.AddressOf(ref fullData), fullData.PrefilterData, fullData.Iterator.m_Length, innerloopBatchCount, isParallelFor, iterator.RequiresFilter(), ref JobStruct_ProcessInfer_DDDD.Cache, deferredCountData, prefilterHandle, mode); + } +#endif + + [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] + public interface IBaseJobForEach_DDDD : IBaseJobForEach {} + +#if !UNITY_ZEROPLAYER + [StructLayout(LayoutKind.Sequential)] + private struct JobStruct_ProcessInfer_DDDD where T : struct + { + public static JobForEachCache Cache; + + public ProcessIterationData Iterator; + public T Data; + + [DeallocateOnJobCompletion] + [NativeDisableContainerSafetyRestriction] + public NativeArray PrefilterData; + } + + [StructLayout(LayoutKind.Sequential)] + internal struct JobStruct_Process_DDDD + where T : struct, IJobForEach + where U0 : struct, IComponentData + where U1 : struct, IComponentData + where U2 : struct, IComponentData + where U3 : struct, IComponentData + { + public ProcessIterationData Iterator; + public T Data; + + [DeallocateOnJobCompletion] + [NativeDisableContainerSafetyRestriction] + public NativeArray PrefilterData; + + [Preserve] + public static IntPtr Initialize(JobType jobType) + { + return JobsUtility.CreateJobReflectionData(typeof(JobStruct_Process_DDDD), typeof(T), jobType, (ExecuteJobFunction) Execute); + } + + delegate void ExecuteJobFunction(ref JobStruct_Process_DDDD data, IntPtr additionalPtr, IntPtr bufferRangePatchData, ref JobRanges ranges, int jobIndex); + + public static unsafe void Execute(ref JobStruct_Process_DDDD jobData, IntPtr additionalPtr, IntPtr bufferRangePatchData, ref JobRanges ranges, int jobIndex) + { + ComponentChunkIterator.UnpackPrefilterData(jobData.PrefilterData, out var chunks, out var entityIndices, out var chunkCount); + + if (jobData.Iterator.m_IsParallelFor) + { + int begin, end; + while (JobsUtility.GetWorkStealingRange(ref ranges, jobIndex, out begin, out end)) + ExecuteChunk(ref jobData, bufferRangePatchData, begin, end, chunks, entityIndices); + } + else + { + ExecuteChunk(ref jobData, bufferRangePatchData, 0, chunkCount, chunks, entityIndices); + } + } + + static unsafe void ExecuteChunk(ref JobStruct_Process_DDDD jobData, IntPtr bufferRangePatchData, int begin, int end, ArchetypeChunk* chunks, int* entityIndices) + { + var typeLookupCache0 = 0; + var typeLookupCache1 = 0; + var typeLookupCache2 = 0; + var typeLookupCache3 = 0; + + for (var blockIndex = begin; blockIndex != end; ++blockIndex) + { + var chunk = chunks[blockIndex]; + int beginIndex = entityIndices[blockIndex]; + var count = chunk.Count; +#if ENABLE_UNITY_COLLECTIONS_CHECKS + JobsUtility.PatchBufferMinMaxRanges(bufferRangePatchData, UnsafeUtility.AddressOf(ref jobData), beginIndex, beginIndex + count); +#endif + ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex0, ref typeLookupCache0); + var ptr0 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly0 == 0, typeLookupCache0, jobData.Iterator.GlobalSystemVersion)); + ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex1, ref typeLookupCache1); + var ptr1 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly1 == 0, typeLookupCache1, jobData.Iterator.GlobalSystemVersion)); + ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex2, ref typeLookupCache2); + var ptr2 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly2 == 0, typeLookupCache2, jobData.Iterator.GlobalSystemVersion)); + ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex3, ref typeLookupCache3); + var ptr3 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly3 == 0, typeLookupCache3, jobData.Iterator.GlobalSystemVersion)); + + + for (var i = 0; i != count; i++) + { + jobData.Data.Execute(ref UnsafeUtilityEx.ArrayElementAsRef(ptr0, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr1, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr2, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr3, i)); + } + } + } + } +#endif + +#if !UNITY_ZEROPLAYER + internal static unsafe JobHandle ScheduleInternal_EDDDD(ref T jobData, ComponentSystemBase system, EntityQuery query, int innerloopBatchCount, JobHandle dependsOn, ScheduleMode mode) + where T : struct + { + JobStruct_ProcessInfer_EDDDD fullData; + fullData.Data = jobData; + + var isParallelFor = innerloopBatchCount != -1; + Initialize(system, query, typeof(T), typeof(JobStruct_Process_EDDDD<,,,,>), isParallelFor, ref JobStruct_ProcessInfer_EDDDD.Cache, out fullData.Iterator); + + var unfilteredChunkCount = fullData.Iterator.m_Length; + var iterator = JobStruct_ProcessInfer_EDDDD.Cache.EntityQuery.GetComponentChunkIterator(); + + var prefilterHandle = ComponentChunkIterator.PreparePrefilteredChunkLists(unfilteredChunkCount, iterator.m_MatchingArchetypeList, iterator.m_Filter, dependsOn, mode, out fullData.PrefilterData, out var deferredCountData); + + return Schedule(UnsafeUtility.AddressOf(ref fullData), fullData.PrefilterData, fullData.Iterator.m_Length, innerloopBatchCount, isParallelFor, iterator.RequiresFilter(), ref JobStruct_ProcessInfer_EDDDD.Cache, deferredCountData, prefilterHandle, mode); + } +#endif + + [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] + public interface IBaseJobForEach_EDDDD : IBaseJobForEach {} + +#if !UNITY_ZEROPLAYER + [StructLayout(LayoutKind.Sequential)] + private struct JobStruct_ProcessInfer_EDDDD where T : struct + { + public static JobForEachCache Cache; + + public ProcessIterationData Iterator; + public T Data; + + [DeallocateOnJobCompletion] + [NativeDisableContainerSafetyRestriction] + public NativeArray PrefilterData; + } + + [StructLayout(LayoutKind.Sequential)] + internal struct JobStruct_Process_EDDDD + where T : struct, IJobForEachWithEntity + where U0 : struct, IComponentData + where U1 : struct, IComponentData + where U2 : struct, IComponentData + where U3 : struct, IComponentData + { + public ProcessIterationData Iterator; + public T Data; + + [DeallocateOnJobCompletion] + [NativeDisableContainerSafetyRestriction] + public NativeArray PrefilterData; + + [Preserve] + public static IntPtr Initialize(JobType jobType) + { + return JobsUtility.CreateJobReflectionData(typeof(JobStruct_Process_EDDDD), typeof(T), jobType, (ExecuteJobFunction) Execute); + } + + delegate void ExecuteJobFunction(ref JobStruct_Process_EDDDD data, IntPtr additionalPtr, IntPtr bufferRangePatchData, ref JobRanges ranges, int jobIndex); + + public static unsafe void Execute(ref JobStruct_Process_EDDDD jobData, IntPtr additionalPtr, IntPtr bufferRangePatchData, ref JobRanges ranges, int jobIndex) + { + ComponentChunkIterator.UnpackPrefilterData(jobData.PrefilterData, out var chunks, out var entityIndices, out var chunkCount); + + if (jobData.Iterator.m_IsParallelFor) + { + int begin, end; + while (JobsUtility.GetWorkStealingRange(ref ranges, jobIndex, out begin, out end)) + ExecuteChunk(ref jobData, bufferRangePatchData, begin, end, chunks, entityIndices); + } + else + { + ExecuteChunk(ref jobData, bufferRangePatchData, 0, chunkCount, chunks, entityIndices); + } + } + + static unsafe void ExecuteChunk(ref JobStruct_Process_EDDDD jobData, IntPtr bufferRangePatchData, int begin, int end, ArchetypeChunk* chunks, int* entityIndices) + { + var typeLookupCache0 = 0; + var typeLookupCache1 = 0; + var typeLookupCache2 = 0; + var typeLookupCache3 = 0; + + for (var blockIndex = begin; blockIndex != end; ++blockIndex) + { + var chunk = chunks[blockIndex]; + int beginIndex = entityIndices[blockIndex]; + var count = chunk.Count; +#if ENABLE_UNITY_COLLECTIONS_CHECKS + JobsUtility.PatchBufferMinMaxRanges(bufferRangePatchData, UnsafeUtility.AddressOf(ref jobData), beginIndex, beginIndex + count); +#endif + var ptrE = (Entity*)UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, false, 0, jobData.Iterator.GlobalSystemVersion)); + ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex0, ref typeLookupCache0); + var ptr0 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly0 == 0, typeLookupCache0, jobData.Iterator.GlobalSystemVersion)); + ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex1, ref typeLookupCache1); + var ptr1 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly1 == 0, typeLookupCache1, jobData.Iterator.GlobalSystemVersion)); + ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex2, ref typeLookupCache2); + var ptr2 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly2 == 0, typeLookupCache2, jobData.Iterator.GlobalSystemVersion)); + ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex3, ref typeLookupCache3); + var ptr3 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly3 == 0, typeLookupCache3, jobData.Iterator.GlobalSystemVersion)); + + + for (var i = 0; i != count; i++) + { + jobData.Data.Execute(ptrE[i], i + beginIndex, ref UnsafeUtilityEx.ArrayElementAsRef(ptr0, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr1, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr2, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr3, i)); + } + } + } + } +#endif + +#if !UNITY_ZEROPLAYER + internal static unsafe JobHandle ScheduleInternal_DDDDD(ref T jobData, ComponentSystemBase system, EntityQuery query, int innerloopBatchCount, JobHandle dependsOn, ScheduleMode mode) + where T : struct + { + JobStruct_ProcessInfer_DDDDD fullData; + fullData.Data = jobData; + + var isParallelFor = innerloopBatchCount != -1; + Initialize(system, query, typeof(T), typeof(JobStruct_Process_DDDDD<,,,,,>), isParallelFor, ref JobStruct_ProcessInfer_DDDDD.Cache, out fullData.Iterator); + + var unfilteredChunkCount = fullData.Iterator.m_Length; + var iterator = JobStruct_ProcessInfer_DDDDD.Cache.EntityQuery.GetComponentChunkIterator(); + + var prefilterHandle = ComponentChunkIterator.PreparePrefilteredChunkLists(unfilteredChunkCount, iterator.m_MatchingArchetypeList, iterator.m_Filter, dependsOn, mode, out fullData.PrefilterData, out var deferredCountData); + + return Schedule(UnsafeUtility.AddressOf(ref fullData), fullData.PrefilterData, fullData.Iterator.m_Length, innerloopBatchCount, isParallelFor, iterator.RequiresFilter(), ref JobStruct_ProcessInfer_DDDDD.Cache, deferredCountData, prefilterHandle, mode); + } +#endif + + [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] + public interface IBaseJobForEach_DDDDD : IBaseJobForEach {} + +#if !UNITY_ZEROPLAYER + [StructLayout(LayoutKind.Sequential)] + private struct JobStruct_ProcessInfer_DDDDD where T : struct + { + public static JobForEachCache Cache; + + public ProcessIterationData Iterator; + public T Data; + + [DeallocateOnJobCompletion] + [NativeDisableContainerSafetyRestriction] + public NativeArray PrefilterData; + } + + [StructLayout(LayoutKind.Sequential)] + internal struct JobStruct_Process_DDDDD + where T : struct, IJobForEach + where U0 : struct, IComponentData + where U1 : struct, IComponentData + where U2 : struct, IComponentData + where U3 : struct, IComponentData + where U4 : struct, IComponentData + { + public ProcessIterationData Iterator; + public T Data; + + [DeallocateOnJobCompletion] + [NativeDisableContainerSafetyRestriction] + public NativeArray PrefilterData; + + [Preserve] + public static IntPtr Initialize(JobType jobType) + { + return JobsUtility.CreateJobReflectionData(typeof(JobStruct_Process_DDDDD), typeof(T), jobType, (ExecuteJobFunction) Execute); + } + + delegate void ExecuteJobFunction(ref JobStruct_Process_DDDDD data, IntPtr additionalPtr, IntPtr bufferRangePatchData, ref JobRanges ranges, int jobIndex); + + public static unsafe void Execute(ref JobStruct_Process_DDDDD jobData, IntPtr additionalPtr, IntPtr bufferRangePatchData, ref JobRanges ranges, int jobIndex) + { + ComponentChunkIterator.UnpackPrefilterData(jobData.PrefilterData, out var chunks, out var entityIndices, out var chunkCount); + + if (jobData.Iterator.m_IsParallelFor) + { + int begin, end; + while (JobsUtility.GetWorkStealingRange(ref ranges, jobIndex, out begin, out end)) + ExecuteChunk(ref jobData, bufferRangePatchData, begin, end, chunks, entityIndices); + } + else + { + ExecuteChunk(ref jobData, bufferRangePatchData, 0, chunkCount, chunks, entityIndices); + } + } + + static unsafe void ExecuteChunk(ref JobStruct_Process_DDDDD jobData, IntPtr bufferRangePatchData, int begin, int end, ArchetypeChunk* chunks, int* entityIndices) + { + var typeLookupCache0 = 0; + var typeLookupCache1 = 0; + var typeLookupCache2 = 0; + var typeLookupCache3 = 0; + var typeLookupCache4 = 0; + + for (var blockIndex = begin; blockIndex != end; ++blockIndex) + { + var chunk = chunks[blockIndex]; + int beginIndex = entityIndices[blockIndex]; + var count = chunk.Count; +#if ENABLE_UNITY_COLLECTIONS_CHECKS + JobsUtility.PatchBufferMinMaxRanges(bufferRangePatchData, UnsafeUtility.AddressOf(ref jobData), beginIndex, beginIndex + count); +#endif + ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex0, ref typeLookupCache0); + var ptr0 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly0 == 0, typeLookupCache0, jobData.Iterator.GlobalSystemVersion)); + ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex1, ref typeLookupCache1); + var ptr1 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly1 == 0, typeLookupCache1, jobData.Iterator.GlobalSystemVersion)); + ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex2, ref typeLookupCache2); + var ptr2 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly2 == 0, typeLookupCache2, jobData.Iterator.GlobalSystemVersion)); + ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex3, ref typeLookupCache3); + var ptr3 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly3 == 0, typeLookupCache3, jobData.Iterator.GlobalSystemVersion)); + ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex4, ref typeLookupCache4); + var ptr4 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly4 == 0, typeLookupCache4, jobData.Iterator.GlobalSystemVersion)); + + + for (var i = 0; i != count; i++) + { + jobData.Data.Execute(ref UnsafeUtilityEx.ArrayElementAsRef(ptr0, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr1, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr2, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr3, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr4, i)); + } + } + } + } +#endif + +#if !UNITY_ZEROPLAYER + internal static unsafe JobHandle ScheduleInternal_EDDDDD(ref T jobData, ComponentSystemBase system, EntityQuery query, int innerloopBatchCount, JobHandle dependsOn, ScheduleMode mode) + where T : struct + { + JobStruct_ProcessInfer_EDDDDD fullData; + fullData.Data = jobData; + + var isParallelFor = innerloopBatchCount != -1; + Initialize(system, query, typeof(T), typeof(JobStruct_Process_EDDDDD<,,,,,>), isParallelFor, ref JobStruct_ProcessInfer_EDDDDD.Cache, out fullData.Iterator); + + var unfilteredChunkCount = fullData.Iterator.m_Length; + var iterator = JobStruct_ProcessInfer_EDDDDD.Cache.EntityQuery.GetComponentChunkIterator(); + + var prefilterHandle = ComponentChunkIterator.PreparePrefilteredChunkLists(unfilteredChunkCount, iterator.m_MatchingArchetypeList, iterator.m_Filter, dependsOn, mode, out fullData.PrefilterData, out var deferredCountData); + + return Schedule(UnsafeUtility.AddressOf(ref fullData), fullData.PrefilterData, fullData.Iterator.m_Length, innerloopBatchCount, isParallelFor, iterator.RequiresFilter(), ref JobStruct_ProcessInfer_EDDDDD.Cache, deferredCountData, prefilterHandle, mode); + } +#endif + + [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] + public interface IBaseJobForEach_EDDDDD : IBaseJobForEach {} + +#if !UNITY_ZEROPLAYER + [StructLayout(LayoutKind.Sequential)] + private struct JobStruct_ProcessInfer_EDDDDD where T : struct + { + public static JobForEachCache Cache; + + public ProcessIterationData Iterator; + public T Data; + + [DeallocateOnJobCompletion] + [NativeDisableContainerSafetyRestriction] + public NativeArray PrefilterData; + } + + [StructLayout(LayoutKind.Sequential)] + internal struct JobStruct_Process_EDDDDD + where T : struct, IJobForEachWithEntity + where U0 : struct, IComponentData + where U1 : struct, IComponentData + where U2 : struct, IComponentData + where U3 : struct, IComponentData + where U4 : struct, IComponentData + { + public ProcessIterationData Iterator; + public T Data; + + [DeallocateOnJobCompletion] + [NativeDisableContainerSafetyRestriction] + public NativeArray PrefilterData; + + [Preserve] + public static IntPtr Initialize(JobType jobType) + { + return JobsUtility.CreateJobReflectionData(typeof(JobStruct_Process_EDDDDD), typeof(T), jobType, (ExecuteJobFunction) Execute); + } + + delegate void ExecuteJobFunction(ref JobStruct_Process_EDDDDD data, IntPtr additionalPtr, IntPtr bufferRangePatchData, ref JobRanges ranges, int jobIndex); + + public static unsafe void Execute(ref JobStruct_Process_EDDDDD jobData, IntPtr additionalPtr, IntPtr bufferRangePatchData, ref JobRanges ranges, int jobIndex) + { + ComponentChunkIterator.UnpackPrefilterData(jobData.PrefilterData, out var chunks, out var entityIndices, out var chunkCount); + + if (jobData.Iterator.m_IsParallelFor) + { + int begin, end; + while (JobsUtility.GetWorkStealingRange(ref ranges, jobIndex, out begin, out end)) + ExecuteChunk(ref jobData, bufferRangePatchData, begin, end, chunks, entityIndices); + } + else + { + ExecuteChunk(ref jobData, bufferRangePatchData, 0, chunkCount, chunks, entityIndices); + } + } + + static unsafe void ExecuteChunk(ref JobStruct_Process_EDDDDD jobData, IntPtr bufferRangePatchData, int begin, int end, ArchetypeChunk* chunks, int* entityIndices) + { + var typeLookupCache0 = 0; + var typeLookupCache1 = 0; + var typeLookupCache2 = 0; + var typeLookupCache3 = 0; + var typeLookupCache4 = 0; + + for (var blockIndex = begin; blockIndex != end; ++blockIndex) + { + var chunk = chunks[blockIndex]; + int beginIndex = entityIndices[blockIndex]; + var count = chunk.Count; +#if ENABLE_UNITY_COLLECTIONS_CHECKS + JobsUtility.PatchBufferMinMaxRanges(bufferRangePatchData, UnsafeUtility.AddressOf(ref jobData), beginIndex, beginIndex + count); +#endif + var ptrE = (Entity*)UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, false, 0, jobData.Iterator.GlobalSystemVersion)); + ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex0, ref typeLookupCache0); + var ptr0 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly0 == 0, typeLookupCache0, jobData.Iterator.GlobalSystemVersion)); + ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex1, ref typeLookupCache1); + var ptr1 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly1 == 0, typeLookupCache1, jobData.Iterator.GlobalSystemVersion)); + ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex2, ref typeLookupCache2); + var ptr2 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly2 == 0, typeLookupCache2, jobData.Iterator.GlobalSystemVersion)); + ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex3, ref typeLookupCache3); + var ptr3 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly3 == 0, typeLookupCache3, jobData.Iterator.GlobalSystemVersion)); + ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex4, ref typeLookupCache4); + var ptr4 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly4 == 0, typeLookupCache4, jobData.Iterator.GlobalSystemVersion)); + + + for (var i = 0; i != count; i++) + { + jobData.Data.Execute(ptrE[i], i + beginIndex, ref UnsafeUtilityEx.ArrayElementAsRef(ptr0, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr1, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr2, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr3, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr4, i)); + } + } + } + } +#endif + +#if !UNITY_ZEROPLAYER + internal static unsafe JobHandle ScheduleInternal_DDDDDD(ref T jobData, ComponentSystemBase system, EntityQuery query, int innerloopBatchCount, JobHandle dependsOn, ScheduleMode mode) + where T : struct + { + JobStruct_ProcessInfer_DDDDDD fullData; + fullData.Data = jobData; + + var isParallelFor = innerloopBatchCount != -1; + Initialize(system, query, typeof(T), typeof(JobStruct_Process_DDDDDD<,,,,,,>), isParallelFor, ref JobStruct_ProcessInfer_DDDDDD.Cache, out fullData.Iterator); + + var unfilteredChunkCount = fullData.Iterator.m_Length; + var iterator = JobStruct_ProcessInfer_DDDDDD.Cache.EntityQuery.GetComponentChunkIterator(); + + var prefilterHandle = ComponentChunkIterator.PreparePrefilteredChunkLists(unfilteredChunkCount, iterator.m_MatchingArchetypeList, iterator.m_Filter, dependsOn, mode, out fullData.PrefilterData, out var deferredCountData); + + return Schedule(UnsafeUtility.AddressOf(ref fullData), fullData.PrefilterData, fullData.Iterator.m_Length, innerloopBatchCount, isParallelFor, iterator.RequiresFilter(), ref JobStruct_ProcessInfer_DDDDDD.Cache, deferredCountData, prefilterHandle, mode); + } +#endif + + [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] + public interface IBaseJobForEach_DDDDDD : IBaseJobForEach {} + +#if !UNITY_ZEROPLAYER + [StructLayout(LayoutKind.Sequential)] + private struct JobStruct_ProcessInfer_DDDDDD where T : struct + { + public static JobForEachCache Cache; + + public ProcessIterationData Iterator; + public T Data; + + [DeallocateOnJobCompletion] + [NativeDisableContainerSafetyRestriction] + public NativeArray PrefilterData; + } + + [StructLayout(LayoutKind.Sequential)] + internal struct JobStruct_Process_DDDDDD + where T : struct, IJobForEach + where U0 : struct, IComponentData + where U1 : struct, IComponentData + where U2 : struct, IComponentData + where U3 : struct, IComponentData + where U4 : struct, IComponentData + where U5 : struct, IComponentData + { + public ProcessIterationData Iterator; + public T Data; + + [DeallocateOnJobCompletion] + [NativeDisableContainerSafetyRestriction] + public NativeArray PrefilterData; + + [Preserve] + public static IntPtr Initialize(JobType jobType) + { + return JobsUtility.CreateJobReflectionData(typeof(JobStruct_Process_DDDDDD), typeof(T), jobType, (ExecuteJobFunction) Execute); + } + + delegate void ExecuteJobFunction(ref JobStruct_Process_DDDDDD data, IntPtr additionalPtr, IntPtr bufferRangePatchData, ref JobRanges ranges, int jobIndex); + + public static unsafe void Execute(ref JobStruct_Process_DDDDDD jobData, IntPtr additionalPtr, IntPtr bufferRangePatchData, ref JobRanges ranges, int jobIndex) + { + ComponentChunkIterator.UnpackPrefilterData(jobData.PrefilterData, out var chunks, out var entityIndices, out var chunkCount); + + if (jobData.Iterator.m_IsParallelFor) + { + int begin, end; + while (JobsUtility.GetWorkStealingRange(ref ranges, jobIndex, out begin, out end)) + ExecuteChunk(ref jobData, bufferRangePatchData, begin, end, chunks, entityIndices); + } + else + { + ExecuteChunk(ref jobData, bufferRangePatchData, 0, chunkCount, chunks, entityIndices); + } + } + + static unsafe void ExecuteChunk(ref JobStruct_Process_DDDDDD jobData, IntPtr bufferRangePatchData, int begin, int end, ArchetypeChunk* chunks, int* entityIndices) + { + var typeLookupCache0 = 0; + var typeLookupCache1 = 0; + var typeLookupCache2 = 0; + var typeLookupCache3 = 0; + var typeLookupCache4 = 0; + var typeLookupCache5 = 0; + + for (var blockIndex = begin; blockIndex != end; ++blockIndex) + { + var chunk = chunks[blockIndex]; + int beginIndex = entityIndices[blockIndex]; + var count = chunk.Count; +#if ENABLE_UNITY_COLLECTIONS_CHECKS + JobsUtility.PatchBufferMinMaxRanges(bufferRangePatchData, UnsafeUtility.AddressOf(ref jobData), beginIndex, beginIndex + count); +#endif + ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex0, ref typeLookupCache0); + var ptr0 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly0 == 0, typeLookupCache0, jobData.Iterator.GlobalSystemVersion)); + ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex1, ref typeLookupCache1); + var ptr1 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly1 == 0, typeLookupCache1, jobData.Iterator.GlobalSystemVersion)); + ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex2, ref typeLookupCache2); + var ptr2 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly2 == 0, typeLookupCache2, jobData.Iterator.GlobalSystemVersion)); + ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex3, ref typeLookupCache3); + var ptr3 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly3 == 0, typeLookupCache3, jobData.Iterator.GlobalSystemVersion)); + ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex4, ref typeLookupCache4); + var ptr4 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly4 == 0, typeLookupCache4, jobData.Iterator.GlobalSystemVersion)); + ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex5, ref typeLookupCache5); + var ptr5 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly5 == 0, typeLookupCache5, jobData.Iterator.GlobalSystemVersion)); + + + for (var i = 0; i != count; i++) + { + jobData.Data.Execute(ref UnsafeUtilityEx.ArrayElementAsRef(ptr0, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr1, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr2, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr3, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr4, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr5, i)); + } + } + } + } +#endif + +#if !UNITY_ZEROPLAYER + internal static unsafe JobHandle ScheduleInternal_EDDDDDD(ref T jobData, ComponentSystemBase system, EntityQuery query, int innerloopBatchCount, JobHandle dependsOn, ScheduleMode mode) + where T : struct + { + JobStruct_ProcessInfer_EDDDDDD fullData; + fullData.Data = jobData; + + var isParallelFor = innerloopBatchCount != -1; + Initialize(system, query, typeof(T), typeof(JobStruct_Process_EDDDDDD<,,,,,,>), isParallelFor, ref JobStruct_ProcessInfer_EDDDDDD.Cache, out fullData.Iterator); + + var unfilteredChunkCount = fullData.Iterator.m_Length; + var iterator = JobStruct_ProcessInfer_EDDDDDD.Cache.EntityQuery.GetComponentChunkIterator(); + + var prefilterHandle = ComponentChunkIterator.PreparePrefilteredChunkLists(unfilteredChunkCount, iterator.m_MatchingArchetypeList, iterator.m_Filter, dependsOn, mode, out fullData.PrefilterData, out var deferredCountData); + + return Schedule(UnsafeUtility.AddressOf(ref fullData), fullData.PrefilterData, fullData.Iterator.m_Length, innerloopBatchCount, isParallelFor, iterator.RequiresFilter(), ref JobStruct_ProcessInfer_EDDDDDD.Cache, deferredCountData, prefilterHandle, mode); + } +#endif + + [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] + public interface IBaseJobForEach_EDDDDDD : IBaseJobForEach {} + +#if !UNITY_ZEROPLAYER + [StructLayout(LayoutKind.Sequential)] + private struct JobStruct_ProcessInfer_EDDDDDD where T : struct + { + public static JobForEachCache Cache; + + public ProcessIterationData Iterator; + public T Data; + + [DeallocateOnJobCompletion] + [NativeDisableContainerSafetyRestriction] + public NativeArray PrefilterData; + } + + [StructLayout(LayoutKind.Sequential)] + internal struct JobStruct_Process_EDDDDDD + where T : struct, IJobForEachWithEntity + where U0 : struct, IComponentData + where U1 : struct, IComponentData + where U2 : struct, IComponentData + where U3 : struct, IComponentData + where U4 : struct, IComponentData + where U5 : struct, IComponentData + { + public ProcessIterationData Iterator; + public T Data; + + [DeallocateOnJobCompletion] + [NativeDisableContainerSafetyRestriction] + public NativeArray PrefilterData; + + [Preserve] + public static IntPtr Initialize(JobType jobType) + { + return JobsUtility.CreateJobReflectionData(typeof(JobStruct_Process_EDDDDDD), typeof(T), jobType, (ExecuteJobFunction) Execute); + } + + delegate void ExecuteJobFunction(ref JobStruct_Process_EDDDDDD data, IntPtr additionalPtr, IntPtr bufferRangePatchData, ref JobRanges ranges, int jobIndex); + + public static unsafe void Execute(ref JobStruct_Process_EDDDDDD jobData, IntPtr additionalPtr, IntPtr bufferRangePatchData, ref JobRanges ranges, int jobIndex) + { + ComponentChunkIterator.UnpackPrefilterData(jobData.PrefilterData, out var chunks, out var entityIndices, out var chunkCount); + + if (jobData.Iterator.m_IsParallelFor) + { + int begin, end; + while (JobsUtility.GetWorkStealingRange(ref ranges, jobIndex, out begin, out end)) + ExecuteChunk(ref jobData, bufferRangePatchData, begin, end, chunks, entityIndices); + } + else + { + ExecuteChunk(ref jobData, bufferRangePatchData, 0, chunkCount, chunks, entityIndices); + } + } + + static unsafe void ExecuteChunk(ref JobStruct_Process_EDDDDDD jobData, IntPtr bufferRangePatchData, int begin, int end, ArchetypeChunk* chunks, int* entityIndices) + { + var typeLookupCache0 = 0; + var typeLookupCache1 = 0; + var typeLookupCache2 = 0; + var typeLookupCache3 = 0; + var typeLookupCache4 = 0; + var typeLookupCache5 = 0; + + for (var blockIndex = begin; blockIndex != end; ++blockIndex) + { + var chunk = chunks[blockIndex]; + int beginIndex = entityIndices[blockIndex]; + var count = chunk.Count; +#if ENABLE_UNITY_COLLECTIONS_CHECKS + JobsUtility.PatchBufferMinMaxRanges(bufferRangePatchData, UnsafeUtility.AddressOf(ref jobData), beginIndex, beginIndex + count); +#endif + var ptrE = (Entity*)UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, false, 0, jobData.Iterator.GlobalSystemVersion)); + ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex0, ref typeLookupCache0); + var ptr0 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly0 == 0, typeLookupCache0, jobData.Iterator.GlobalSystemVersion)); + ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex1, ref typeLookupCache1); + var ptr1 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly1 == 0, typeLookupCache1, jobData.Iterator.GlobalSystemVersion)); + ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex2, ref typeLookupCache2); + var ptr2 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly2 == 0, typeLookupCache2, jobData.Iterator.GlobalSystemVersion)); + ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex3, ref typeLookupCache3); + var ptr3 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly3 == 0, typeLookupCache3, jobData.Iterator.GlobalSystemVersion)); + ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex4, ref typeLookupCache4); + var ptr4 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly4 == 0, typeLookupCache4, jobData.Iterator.GlobalSystemVersion)); + ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex5, ref typeLookupCache5); + var ptr5 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly5 == 0, typeLookupCache5, jobData.Iterator.GlobalSystemVersion)); + + + for (var i = 0; i != count; i++) + { + jobData.Data.Execute(ptrE[i], i + beginIndex, ref UnsafeUtilityEx.ArrayElementAsRef(ptr0, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr1, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr2, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr3, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr4, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr5, i)); + } + } + } + } +#endif + + } +} + diff --git a/Unity.Entities/IJobForEach.gen.cs.meta b/Unity.Entities/IJobForEach.gen.cs.meta new file mode 100644 index 00000000..ce849e68 --- /dev/null +++ b/Unity.Entities/IJobForEach.gen.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: c6e655e914344af458054023f1d180eb +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/Unity.Entities/IJobForEach.tt b/Unity.Entities/IJobForEach.tt new file mode 100644 index 00000000..5c92fa7f --- /dev/null +++ b/Unity.Entities/IJobForEach.tt @@ -0,0 +1,337 @@ +<#/*THIS IS A T4 FILE - see HACKING.md for what it is and how to run codegen*/#> +<#@ assembly name="System.Collections" #> +<#@ assembly name="System.Core" #> +<#@ assembly name="System.Linq" #> +<#@ import namespace="System.Collections.Generic" #> +<#@ import namespace="System.Linq" #> +<#@ import namespace="System.Text" #> +<#@ output extension=".gen.cs" #> +//------------------------------------------------------------------------------ +// +// This code was generated by a tool. +// +// Changes to this file may cause incorrect behavior and will be lost if +// the code is regenerated. +// +//------------------------------------------------------------------------------ +// Generated by T4 (TextTransform.exe) from the file IJobForEach.tt +// + +using Unity.Collections; +using Unity.Collections.LowLevel.Unsafe; +using Unity.Jobs; +using Unity.Jobs.LowLevel.Unsafe; +using System.Runtime.InteropServices; +using UnityEngine.Scripting; +using System; + +namespace Unity.Entities +{ +<# +var combinations = GenCombinations(); +foreach(var combination in combinations) { + foreach (var withEntity in new[] { false, true }) { + var comboString = GetComboString(withEntity, combination); + var name = withEntity ? "IJobForEachWithEntity" : "IJobForEach"; +#> + +#if !UNITY_ZEROPLAYER + [JobProducerType(typeof(JobForEachExtensions.JobStruct_Process_<#=comboString#><<#=GetUntypedGenericParams(combination)#>>))] +#endif + public interface <#=name#><<#=GenericParams(combination)#>> : JobForEachExtensions.IBaseJobForEach_<#=GetComboString(withEntity, combination)#> +<#=GenericConstraints(combination)#> + { + void Execute(<#=ExecuteParams(withEntity, combination)#>); + } +<# + } +} +#> + + public static partial class JobForEachExtensions + { +<# +string[] funcNames = new string[]{"Schedule", "ScheduleSingle", "Run"}; +int [] forEachVals = new int[]{1, -1, -1}; +string[] scheduleMode = new string[]{"ScheduleMode.Batched", "ScheduleMode.Batched", "ScheduleMode.Run"}; +for(int i=0; i +#if !UNITY_ZEROPLAYER + public static JobHandle <#=funcNames[i]#>(this T jobData, ComponentSystemBase system, JobHandle dependsOn = default(JobHandle)) + where T : struct, IBaseJobForEach + { + var typeT = typeof(T); +<# + foreach(var combination in combinations) { + var comboStringD = GetComboString(false, combination); + var comboStringE = GetComboString(true, combination); +#> + if (typeof(IBaseJobForEach_<#=comboStringD#>).IsAssignableFrom(typeT)) + return ScheduleInternal_<#=comboStringD#>(ref jobData, system, null, <#=forEachVals[i]#>, dependsOn, <#=scheduleMode[i]#>); + if (typeof(IBaseJobForEach_<#=comboStringE#>).IsAssignableFrom(typeT)) + return ScheduleInternal_<#=comboStringE#>(ref jobData, system, null, <#=forEachVals[i]#>, dependsOn, <#=scheduleMode[i]#>); +<# + } +#> + throw new System.ArgumentException("Not supported"); + } +#endif +<# +} +#> + +<# +string[] groupFuncNames = new string[]{"Schedule", "ScheduleSingle", "Run"}; +for(int i=0; i +#if !UNITY_ZEROPLAYER + public static JobHandle <#=groupFuncNames[i]#>(this T jobData, EntityQuery query, JobHandle dependsOn = default(JobHandle)) + where T : struct, IBaseJobForEach + { + var typeT = typeof(T); +<# + foreach(var combination in combinations) { + var comboStringD = GetComboString(false, combination); + var comboStringE = GetComboString(true, combination); +#> + if (typeof(IBaseJobForEach_<#=comboStringD#>).IsAssignableFrom(typeT)) + return ScheduleInternal_<#=comboStringD#>(ref jobData, null, query, <#=forEachVals[i]#>, dependsOn, <#=scheduleMode[i]#>); + if (typeof(IBaseJobForEach_<#=comboStringE#>).IsAssignableFrom(typeT)) + return ScheduleInternal_<#=comboStringE#>(ref jobData, null, query, <#=forEachVals[i]#>, dependsOn, <#=scheduleMode[i]#>); +<# + } +#> + throw new System.ArgumentException("Not supported"); + } +#endif +<# +} +#> + +<# +foreach(var combination in combinations) { + foreach (var withEntity in new[] { false, true }) { + var comboString = GetComboString(withEntity, combination); + var untypedGenericParams = new StringBuilder(); + var genericParams = new StringBuilder(); + var genericConstraints = new StringBuilder(); + + var executeParams = new StringBuilder(); + var executeCallParams = new StringBuilder(); + var ptrs = new StringBuilder(); + var typeLookupCache = new StringBuilder(); + + var interfaceName = withEntity ? "IJobForEachWithEntity" : "IJobForEach"; + if (withEntity) + { + executeCallParams.Append("ptrE[i], i + beginIndex, "); + executeParams.Append("Entity entity, int index, "); + ptrs.AppendLine + ( +$" var ptrE = (Entity*)UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, false, 0, jobData.Iterator.GlobalSystemVersion));" + ); + } + for (int i = 0; i != combination.Length; i++) + { + ptrs.AppendLine + ( + +$@" ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex{i}, ref typeLookupCache{i}); + var ptr{i} = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly{i} == 0, typeLookupCache{i}, jobData.Iterator.GlobalSystemVersion));" + ); + + typeLookupCache.AppendLine + ( + +$@" var typeLookupCache{i} = 0; " + ); + + genericConstraints.Append($" where U{i} : struct, IComponentData"); + if (i != combination.Length - 1) + genericConstraints.AppendLine(); + untypedGenericParams.Append(","); + + genericParams.Append($"U{i}"); + if (i != combination.Length - 1) + genericParams.Append(", "); + + executeCallParams.Append($"ref UnsafeUtilityEx.ArrayElementAsRef(ptr{i}, i)"); + if (i != combination.Length - 1) + executeCallParams.Append(", "); + + executeParams.Append($"ref U{i} c{i}"); + if (i != combination.Length - 1) + executeParams.Append(", "); + } +#> +#if !UNITY_ZEROPLAYER + internal static unsafe JobHandle ScheduleInternal_<#=comboString#>(ref T jobData, ComponentSystemBase system, EntityQuery query, int innerloopBatchCount, JobHandle dependsOn, ScheduleMode mode) + where T : struct + { + JobStruct_ProcessInfer_<#=comboString#> fullData; + fullData.Data = jobData; + + var isParallelFor = innerloopBatchCount != -1; + Initialize(system, query, typeof(T), typeof(JobStruct_Process_<#=comboString#><<#=untypedGenericParams#>>), isParallelFor, ref JobStruct_ProcessInfer_<#=comboString#>.Cache, out fullData.Iterator); + + var unfilteredChunkCount = fullData.Iterator.m_Length; + var iterator = JobStruct_ProcessInfer_<#=comboString#>.Cache.EntityQuery.GetComponentChunkIterator(); + + var prefilterHandle = ComponentChunkIterator.PreparePrefilteredChunkLists(unfilteredChunkCount, iterator.m_MatchingArchetypeList, iterator.m_Filter, dependsOn, mode, out fullData.PrefilterData, out var deferredCountData); + + return Schedule(UnsafeUtility.AddressOf(ref fullData), fullData.PrefilterData, fullData.Iterator.m_Length, innerloopBatchCount, isParallelFor, iterator.RequiresFilter(), ref JobStruct_ProcessInfer_<#=comboString#>.Cache, deferredCountData, prefilterHandle, mode); + } +#endif + + [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] + public interface IBaseJobForEach_<#=comboString#> : IBaseJobForEach {} + +#if !UNITY_ZEROPLAYER + [StructLayout(LayoutKind.Sequential)] + private struct JobStruct_ProcessInfer_<#=comboString#> where T : struct + { + public static JobForEachCache Cache; + + public ProcessIterationData Iterator; + public T Data; + + [DeallocateOnJobCompletion] + [NativeDisableContainerSafetyRestriction] + public NativeArray PrefilterData; + } + + [StructLayout(LayoutKind.Sequential)] + internal struct JobStruct_Process_<#=comboString#>> + where T : struct, <#=interfaceName#><<#=genericParams#>> +<#=genericConstraints#> + { + public ProcessIterationData Iterator; + public T Data; + + [DeallocateOnJobCompletion] + [NativeDisableContainerSafetyRestriction] + public NativeArray PrefilterData; + + [Preserve] + public static IntPtr Initialize(JobType jobType) + { + return JobsUtility.CreateJobReflectionData(typeof(JobStruct_Process_<#=comboString#>>), typeof(T), jobType, (ExecuteJobFunction) Execute); + } + + delegate void ExecuteJobFunction(ref JobStruct_Process_<#=comboString#>> data, IntPtr additionalPtr, IntPtr bufferRangePatchData, ref JobRanges ranges, int jobIndex); + + public static unsafe void Execute(ref JobStruct_Process_<#=comboString#>> jobData, IntPtr additionalPtr, IntPtr bufferRangePatchData, ref JobRanges ranges, int jobIndex) + { + ComponentChunkIterator.UnpackPrefilterData(jobData.PrefilterData, out var chunks, out var entityIndices, out var chunkCount); + + if (jobData.Iterator.m_IsParallelFor) + { + int begin, end; + while (JobsUtility.GetWorkStealingRange(ref ranges, jobIndex, out begin, out end)) + ExecuteChunk(ref jobData, bufferRangePatchData, begin, end, chunks, entityIndices); + } + else + { + ExecuteChunk(ref jobData, bufferRangePatchData, 0, chunkCount, chunks, entityIndices); + } + } + + static unsafe void ExecuteChunk(ref JobStruct_Process_<#=comboString#>> jobData, IntPtr bufferRangePatchData, int begin, int end, ArchetypeChunk* chunks, int* entityIndices) + { +<#=typeLookupCache#> + for (var blockIndex = begin; blockIndex != end; ++blockIndex) + { + var chunk = chunks[blockIndex]; + int beginIndex = entityIndices[blockIndex]; + var count = chunk.Count; +#if ENABLE_UNITY_COLLECTIONS_CHECKS + JobsUtility.PatchBufferMinMaxRanges(bufferRangePatchData, UnsafeUtility.AddressOf(ref jobData), beginIndex, beginIndex + count); +#endif +<#=ptrs#> + + for (var i = 0; i != count; i++) + { + jobData.Data.Execute(<#=executeCallParams#>); + } + } + } + } +#endif + +<# + } +} +#> + } +} + +<#+ +enum Combination +{ + D +} + +List GenCombinations() +{ + var combinations = new List(); + for (int count = 1; count <= 6; count++) + { + var array = new Combination[count]; + for (var c = 0; c != array.Length; c++) + array[c] = Combination.D; + + combinations.Add(array); + } + return combinations; +} + +string GetComboString(bool withEntity, Combination[] combination) +{ + var baseType = new StringBuilder(); + + if (withEntity) + baseType.Append("E"); + foreach (var c in combination) + baseType.Append(Enum.GetName(typeof(Combination), c)); + return baseType.ToString(); +} + +string GetUntypedGenericParams(Combination[] combination) { + return new String(',', combination.Length); +} + +string GenericParams(Combination[] combination) { + StringBuilder genericParams = new StringBuilder(); + for(int i=0; i \ No newline at end of file diff --git a/Unity.Entities/IJobForEach.tt.meta b/Unity.Entities/IJobForEach.tt.meta new file mode 100644 index 00000000..35e42f6d --- /dev/null +++ b/Unity.Entities/IJobForEach.tt.meta @@ -0,0 +1,7 @@ +fileFormatVersion: 2 +guid: dd1f79449e827b643bc786b747ca5fa3 +DefaultImporter: + externalObjects: {} + userData: + assetBundleName: + assetBundleVariant: diff --git a/Unity.Entities/IJobProcessComponentData.generated.cs b/Unity.Entities/IJobProcessComponentData.generated.cs deleted file mode 100644 index ff16e4d1..00000000 --- a/Unity.Entities/IJobProcessComponentData.generated.cs +++ /dev/null @@ -1,1598 +0,0 @@ -// Generated by IJobProcessComponentDataGenerator.cs -//------------------------------------------------------------------------------ -// -// This code was generated by a tool. -// -// Changes to this file may cause incorrect behavior and will be lost if -// the code is regenerated. -// -//------------------------------------------------------------------------------ -#if !UNITY_ZEROPLAYER - -using Unity.Collections; -using Unity.Collections.LowLevel.Unsafe; -using Unity.Jobs; -using Unity.Jobs.LowLevel.Unsafe; -using System.Runtime.InteropServices; -using UnityEngine.Scripting; -using System; - - -namespace Unity.Entities -{ - - - - [JobProducerType(typeof(JobProcessComponentDataExtensions.JobStruct_Process_D<,>))] - public interface IJobProcessComponentData : JobProcessComponentDataExtensions.IBaseJobProcessComponentData_D - where U0 : struct, IComponentData - { - void Execute(ref U0 c0); - } - - [JobProducerType(typeof(JobProcessComponentDataExtensions.JobStruct_Process_ED<,>))] - public interface IJobProcessComponentDataWithEntity : JobProcessComponentDataExtensions.IBaseJobProcessComponentData_ED - where U0 : struct, IComponentData - { - void Execute(Entity entity, int index, ref U0 c0); - } - - [JobProducerType(typeof(JobProcessComponentDataExtensions.JobStruct_Process_DD<,,>))] - public interface IJobProcessComponentData : JobProcessComponentDataExtensions.IBaseJobProcessComponentData_DD - where U0 : struct, IComponentData - where U1 : struct, IComponentData - { - void Execute(ref U0 c0, ref U1 c1); - } - - [JobProducerType(typeof(JobProcessComponentDataExtensions.JobStruct_Process_EDD<,,>))] - public interface IJobProcessComponentDataWithEntity : JobProcessComponentDataExtensions.IBaseJobProcessComponentData_EDD - where U0 : struct, IComponentData - where U1 : struct, IComponentData - { - void Execute(Entity entity, int index, ref U0 c0, ref U1 c1); - } - - [JobProducerType(typeof(JobProcessComponentDataExtensions.JobStruct_Process_DDD<,,,>))] - public interface IJobProcessComponentData : JobProcessComponentDataExtensions.IBaseJobProcessComponentData_DDD - where U0 : struct, IComponentData - where U1 : struct, IComponentData - where U2 : struct, IComponentData - { - void Execute(ref U0 c0, ref U1 c1, ref U2 c2); - } - - [JobProducerType(typeof(JobProcessComponentDataExtensions.JobStruct_Process_EDDD<,,,>))] - public interface IJobProcessComponentDataWithEntity : JobProcessComponentDataExtensions.IBaseJobProcessComponentData_EDDD - where U0 : struct, IComponentData - where U1 : struct, IComponentData - where U2 : struct, IComponentData - { - void Execute(Entity entity, int index, ref U0 c0, ref U1 c1, ref U2 c2); - } - - [JobProducerType(typeof(JobProcessComponentDataExtensions.JobStruct_Process_DDDD<,,,,>))] - public interface IJobProcessComponentData : JobProcessComponentDataExtensions.IBaseJobProcessComponentData_DDDD - where U0 : struct, IComponentData - where U1 : struct, IComponentData - where U2 : struct, IComponentData - where U3 : struct, IComponentData - { - void Execute(ref U0 c0, ref U1 c1, ref U2 c2, ref U3 c3); - } - - [JobProducerType(typeof(JobProcessComponentDataExtensions.JobStruct_Process_EDDDD<,,,,>))] - public interface IJobProcessComponentDataWithEntity : JobProcessComponentDataExtensions.IBaseJobProcessComponentData_EDDDD - where U0 : struct, IComponentData - where U1 : struct, IComponentData - where U2 : struct, IComponentData - where U3 : struct, IComponentData - { - void Execute(Entity entity, int index, ref U0 c0, ref U1 c1, ref U2 c2, ref U3 c3); - } - - [JobProducerType(typeof(JobProcessComponentDataExtensions.JobStruct_Process_DDDDD<,,,,,>))] - public interface IJobProcessComponentData : JobProcessComponentDataExtensions.IBaseJobProcessComponentData_DDDDD - where U0 : struct, IComponentData - where U1 : struct, IComponentData - where U2 : struct, IComponentData - where U3 : struct, IComponentData - where U4 : struct, IComponentData - { - void Execute(ref U0 c0, ref U1 c1, ref U2 c2, ref U3 c3, ref U4 c4); - } - - [JobProducerType(typeof(JobProcessComponentDataExtensions.JobStruct_Process_EDDDDD<,,,,,>))] - public interface IJobProcessComponentDataWithEntity : JobProcessComponentDataExtensions.IBaseJobProcessComponentData_EDDDDD - where U0 : struct, IComponentData - where U1 : struct, IComponentData - where U2 : struct, IComponentData - where U3 : struct, IComponentData - where U4 : struct, IComponentData - { - void Execute(Entity entity, int index, ref U0 c0, ref U1 c1, ref U2 c2, ref U3 c3, ref U4 c4); - } - - [JobProducerType(typeof(JobProcessComponentDataExtensions.JobStruct_Process_DDDDDD<,,,,,,>))] - public interface IJobProcessComponentData : JobProcessComponentDataExtensions.IBaseJobProcessComponentData_DDDDDD - where U0 : struct, IComponentData - where U1 : struct, IComponentData - where U2 : struct, IComponentData - where U3 : struct, IComponentData - where U4 : struct, IComponentData - where U5 : struct, IComponentData - { - void Execute(ref U0 c0, ref U1 c1, ref U2 c2, ref U3 c3, ref U4 c4, ref U5 c5); - } - - [JobProducerType(typeof(JobProcessComponentDataExtensions.JobStruct_Process_EDDDDDD<,,,,,,>))] - public interface IJobProcessComponentDataWithEntity : JobProcessComponentDataExtensions.IBaseJobProcessComponentData_EDDDDDD - where U0 : struct, IComponentData - where U1 : struct, IComponentData - where U2 : struct, IComponentData - where U3 : struct, IComponentData - where U4 : struct, IComponentData - where U5 : struct, IComponentData - { - void Execute(Entity entity, int index, ref U0 c0, ref U1 c1, ref U2 c2, ref U3 c3, ref U4 c4, ref U5 c5); - } - - public static partial class JobProcessComponentDataExtensions - { - - public static JobHandle Schedule(this T jobData, ComponentSystemBase system, JobHandle dependsOn = default(JobHandle)) - where T : struct, IBaseJobProcessComponentData - { - var typeT = typeof(T); - if (typeof(IBaseJobProcessComponentData_D).IsAssignableFrom(typeT)) - return ScheduleInternal_D(ref jobData, system, null, 1, dependsOn, ScheduleMode.Batched); - if (typeof(IBaseJobProcessComponentData_ED).IsAssignableFrom(typeT)) - return ScheduleInternal_ED(ref jobData, system, null, 1, dependsOn, ScheduleMode.Batched); - if (typeof(IBaseJobProcessComponentData_DD).IsAssignableFrom(typeT)) - return ScheduleInternal_DD(ref jobData, system, null, 1, dependsOn, ScheduleMode.Batched); - if (typeof(IBaseJobProcessComponentData_EDD).IsAssignableFrom(typeT)) - return ScheduleInternal_EDD(ref jobData, system, null, 1, dependsOn, ScheduleMode.Batched); - if (typeof(IBaseJobProcessComponentData_DDD).IsAssignableFrom(typeT)) - return ScheduleInternal_DDD(ref jobData, system, null, 1, dependsOn, ScheduleMode.Batched); - if (typeof(IBaseJobProcessComponentData_EDDD).IsAssignableFrom(typeT)) - return ScheduleInternal_EDDD(ref jobData, system, null, 1, dependsOn, ScheduleMode.Batched); - if (typeof(IBaseJobProcessComponentData_DDDD).IsAssignableFrom(typeT)) - return ScheduleInternal_DDDD(ref jobData, system, null, 1, dependsOn, ScheduleMode.Batched); - if (typeof(IBaseJobProcessComponentData_EDDDD).IsAssignableFrom(typeT)) - return ScheduleInternal_EDDDD(ref jobData, system, null, 1, dependsOn, ScheduleMode.Batched); - if (typeof(IBaseJobProcessComponentData_DDDDD).IsAssignableFrom(typeT)) - return ScheduleInternal_DDDDD(ref jobData, system, null, 1, dependsOn, ScheduleMode.Batched); - if (typeof(IBaseJobProcessComponentData_EDDDDD).IsAssignableFrom(typeT)) - return ScheduleInternal_EDDDDD(ref jobData, system, null, 1, dependsOn, ScheduleMode.Batched); - if (typeof(IBaseJobProcessComponentData_DDDDDD).IsAssignableFrom(typeT)) - return ScheduleInternal_DDDDDD(ref jobData, system, null, 1, dependsOn, ScheduleMode.Batched); - if (typeof(IBaseJobProcessComponentData_EDDDDDD).IsAssignableFrom(typeT)) - return ScheduleInternal_EDDDDDD(ref jobData, system, null, 1, dependsOn, ScheduleMode.Batched); - throw new System.ArgumentException("Not supported"); - } - - public static JobHandle ScheduleSingle(this T jobData, ComponentSystemBase system, JobHandle dependsOn = default(JobHandle)) - where T : struct, IBaseJobProcessComponentData - { - var typeT = typeof(T); - if (typeof(IBaseJobProcessComponentData_D).IsAssignableFrom(typeT)) - return ScheduleInternal_D(ref jobData, system, null, -1, dependsOn, ScheduleMode.Batched); - if (typeof(IBaseJobProcessComponentData_ED).IsAssignableFrom(typeT)) - return ScheduleInternal_ED(ref jobData, system, null, -1, dependsOn, ScheduleMode.Batched); - if (typeof(IBaseJobProcessComponentData_DD).IsAssignableFrom(typeT)) - return ScheduleInternal_DD(ref jobData, system, null, -1, dependsOn, ScheduleMode.Batched); - if (typeof(IBaseJobProcessComponentData_EDD).IsAssignableFrom(typeT)) - return ScheduleInternal_EDD(ref jobData, system, null, -1, dependsOn, ScheduleMode.Batched); - if (typeof(IBaseJobProcessComponentData_DDD).IsAssignableFrom(typeT)) - return ScheduleInternal_DDD(ref jobData, system, null, -1, dependsOn, ScheduleMode.Batched); - if (typeof(IBaseJobProcessComponentData_EDDD).IsAssignableFrom(typeT)) - return ScheduleInternal_EDDD(ref jobData, system, null, -1, dependsOn, ScheduleMode.Batched); - if (typeof(IBaseJobProcessComponentData_DDDD).IsAssignableFrom(typeT)) - return ScheduleInternal_DDDD(ref jobData, system, null, -1, dependsOn, ScheduleMode.Batched); - if (typeof(IBaseJobProcessComponentData_EDDDD).IsAssignableFrom(typeT)) - return ScheduleInternal_EDDDD(ref jobData, system, null, -1, dependsOn, ScheduleMode.Batched); - if (typeof(IBaseJobProcessComponentData_DDDDD).IsAssignableFrom(typeT)) - return ScheduleInternal_DDDDD(ref jobData, system, null, -1, dependsOn, ScheduleMode.Batched); - if (typeof(IBaseJobProcessComponentData_EDDDDD).IsAssignableFrom(typeT)) - return ScheduleInternal_EDDDDD(ref jobData, system, null, -1, dependsOn, ScheduleMode.Batched); - if (typeof(IBaseJobProcessComponentData_DDDDDD).IsAssignableFrom(typeT)) - return ScheduleInternal_DDDDDD(ref jobData, system, null, -1, dependsOn, ScheduleMode.Batched); - if (typeof(IBaseJobProcessComponentData_EDDDDDD).IsAssignableFrom(typeT)) - return ScheduleInternal_EDDDDDD(ref jobData, system, null, -1, dependsOn, ScheduleMode.Batched); - throw new System.ArgumentException("Not supported"); - } - - public static JobHandle Run(this T jobData, ComponentSystemBase system, JobHandle dependsOn = default(JobHandle)) - where T : struct, IBaseJobProcessComponentData - { - var typeT = typeof(T); - if (typeof(IBaseJobProcessComponentData_D).IsAssignableFrom(typeT)) - return ScheduleInternal_D(ref jobData, system, null, -1, dependsOn, ScheduleMode.Run); - if (typeof(IBaseJobProcessComponentData_ED).IsAssignableFrom(typeT)) - return ScheduleInternal_ED(ref jobData, system, null, -1, dependsOn, ScheduleMode.Run); - if (typeof(IBaseJobProcessComponentData_DD).IsAssignableFrom(typeT)) - return ScheduleInternal_DD(ref jobData, system, null, -1, dependsOn, ScheduleMode.Run); - if (typeof(IBaseJobProcessComponentData_EDD).IsAssignableFrom(typeT)) - return ScheduleInternal_EDD(ref jobData, system, null, -1, dependsOn, ScheduleMode.Run); - if (typeof(IBaseJobProcessComponentData_DDD).IsAssignableFrom(typeT)) - return ScheduleInternal_DDD(ref jobData, system, null, -1, dependsOn, ScheduleMode.Run); - if (typeof(IBaseJobProcessComponentData_EDDD).IsAssignableFrom(typeT)) - return ScheduleInternal_EDDD(ref jobData, system, null, -1, dependsOn, ScheduleMode.Run); - if (typeof(IBaseJobProcessComponentData_DDDD).IsAssignableFrom(typeT)) - return ScheduleInternal_DDDD(ref jobData, system, null, -1, dependsOn, ScheduleMode.Run); - if (typeof(IBaseJobProcessComponentData_EDDDD).IsAssignableFrom(typeT)) - return ScheduleInternal_EDDDD(ref jobData, system, null, -1, dependsOn, ScheduleMode.Run); - if (typeof(IBaseJobProcessComponentData_DDDDD).IsAssignableFrom(typeT)) - return ScheduleInternal_DDDDD(ref jobData, system, null, -1, dependsOn, ScheduleMode.Run); - if (typeof(IBaseJobProcessComponentData_EDDDDD).IsAssignableFrom(typeT)) - return ScheduleInternal_EDDDDD(ref jobData, system, null, -1, dependsOn, ScheduleMode.Run); - if (typeof(IBaseJobProcessComponentData_DDDDDD).IsAssignableFrom(typeT)) - return ScheduleInternal_DDDDDD(ref jobData, system, null, -1, dependsOn, ScheduleMode.Run); - if (typeof(IBaseJobProcessComponentData_EDDDDDD).IsAssignableFrom(typeT)) - return ScheduleInternal_EDDDDDD(ref jobData, system, null, -1, dependsOn, ScheduleMode.Run); - throw new System.ArgumentException("Not supported"); - } - - public static JobHandle ScheduleGroup(this T jobData, ComponentGroup componentGroup, JobHandle dependsOn = default(JobHandle)) - where T : struct, IBaseJobProcessComponentData - { - var typeT = typeof(T); - if (typeof(IBaseJobProcessComponentData_D).IsAssignableFrom(typeT)) - return ScheduleInternal_D(ref jobData, null, componentGroup, 1, dependsOn, ScheduleMode.Batched); - if (typeof(IBaseJobProcessComponentData_ED).IsAssignableFrom(typeT)) - return ScheduleInternal_ED(ref jobData, null, componentGroup, 1, dependsOn, ScheduleMode.Batched); - if (typeof(IBaseJobProcessComponentData_DD).IsAssignableFrom(typeT)) - return ScheduleInternal_DD(ref jobData, null, componentGroup, 1, dependsOn, ScheduleMode.Batched); - if (typeof(IBaseJobProcessComponentData_EDD).IsAssignableFrom(typeT)) - return ScheduleInternal_EDD(ref jobData, null, componentGroup, 1, dependsOn, ScheduleMode.Batched); - if (typeof(IBaseJobProcessComponentData_DDD).IsAssignableFrom(typeT)) - return ScheduleInternal_DDD(ref jobData, null, componentGroup, 1, dependsOn, ScheduleMode.Batched); - if (typeof(IBaseJobProcessComponentData_EDDD).IsAssignableFrom(typeT)) - return ScheduleInternal_EDDD(ref jobData, null, componentGroup, 1, dependsOn, ScheduleMode.Batched); - if (typeof(IBaseJobProcessComponentData_DDDD).IsAssignableFrom(typeT)) - return ScheduleInternal_DDDD(ref jobData, null, componentGroup, 1, dependsOn, ScheduleMode.Batched); - if (typeof(IBaseJobProcessComponentData_EDDDD).IsAssignableFrom(typeT)) - return ScheduleInternal_EDDDD(ref jobData, null, componentGroup, 1, dependsOn, ScheduleMode.Batched); - if (typeof(IBaseJobProcessComponentData_DDDDD).IsAssignableFrom(typeT)) - return ScheduleInternal_DDDDD(ref jobData, null, componentGroup, 1, dependsOn, ScheduleMode.Batched); - if (typeof(IBaseJobProcessComponentData_EDDDDD).IsAssignableFrom(typeT)) - return ScheduleInternal_EDDDDD(ref jobData, null, componentGroup, 1, dependsOn, ScheduleMode.Batched); - if (typeof(IBaseJobProcessComponentData_DDDDDD).IsAssignableFrom(typeT)) - return ScheduleInternal_DDDDDD(ref jobData, null, componentGroup, 1, dependsOn, ScheduleMode.Batched); - if (typeof(IBaseJobProcessComponentData_EDDDDDD).IsAssignableFrom(typeT)) - return ScheduleInternal_EDDDDDD(ref jobData, null, componentGroup, 1, dependsOn, ScheduleMode.Batched); - throw new System.ArgumentException("Not supported"); - } - - public static JobHandle ScheduleGroupSingle(this T jobData, ComponentGroup componentGroup, JobHandle dependsOn = default(JobHandle)) - where T : struct, IBaseJobProcessComponentData - { - var typeT = typeof(T); - if (typeof(IBaseJobProcessComponentData_D).IsAssignableFrom(typeT)) - return ScheduleInternal_D(ref jobData, null, componentGroup, -1, dependsOn, ScheduleMode.Batched); - if (typeof(IBaseJobProcessComponentData_ED).IsAssignableFrom(typeT)) - return ScheduleInternal_ED(ref jobData, null, componentGroup, -1, dependsOn, ScheduleMode.Batched); - if (typeof(IBaseJobProcessComponentData_DD).IsAssignableFrom(typeT)) - return ScheduleInternal_DD(ref jobData, null, componentGroup, -1, dependsOn, ScheduleMode.Batched); - if (typeof(IBaseJobProcessComponentData_EDD).IsAssignableFrom(typeT)) - return ScheduleInternal_EDD(ref jobData, null, componentGroup, -1, dependsOn, ScheduleMode.Batched); - if (typeof(IBaseJobProcessComponentData_DDD).IsAssignableFrom(typeT)) - return ScheduleInternal_DDD(ref jobData, null, componentGroup, -1, dependsOn, ScheduleMode.Batched); - if (typeof(IBaseJobProcessComponentData_EDDD).IsAssignableFrom(typeT)) - return ScheduleInternal_EDDD(ref jobData, null, componentGroup, -1, dependsOn, ScheduleMode.Batched); - if (typeof(IBaseJobProcessComponentData_DDDD).IsAssignableFrom(typeT)) - return ScheduleInternal_DDDD(ref jobData, null, componentGroup, -1, dependsOn, ScheduleMode.Batched); - if (typeof(IBaseJobProcessComponentData_EDDDD).IsAssignableFrom(typeT)) - return ScheduleInternal_EDDDD(ref jobData, null, componentGroup, -1, dependsOn, ScheduleMode.Batched); - if (typeof(IBaseJobProcessComponentData_DDDDD).IsAssignableFrom(typeT)) - return ScheduleInternal_DDDDD(ref jobData, null, componentGroup, -1, dependsOn, ScheduleMode.Batched); - if (typeof(IBaseJobProcessComponentData_EDDDDD).IsAssignableFrom(typeT)) - return ScheduleInternal_EDDDDD(ref jobData, null, componentGroup, -1, dependsOn, ScheduleMode.Batched); - if (typeof(IBaseJobProcessComponentData_DDDDDD).IsAssignableFrom(typeT)) - return ScheduleInternal_DDDDDD(ref jobData, null, componentGroup, -1, dependsOn, ScheduleMode.Batched); - if (typeof(IBaseJobProcessComponentData_EDDDDDD).IsAssignableFrom(typeT)) - return ScheduleInternal_EDDDDDD(ref jobData, null, componentGroup, -1, dependsOn, ScheduleMode.Batched); - throw new System.ArgumentException("Not supported"); - } - - public static JobHandle RunGroup(this T jobData, ComponentGroup componentGroup, JobHandle dependsOn = default(JobHandle)) - where T : struct, IBaseJobProcessComponentData - { - var typeT = typeof(T); - if (typeof(IBaseJobProcessComponentData_D).IsAssignableFrom(typeT)) - return ScheduleInternal_D(ref jobData, null, componentGroup, -1, dependsOn, ScheduleMode.Run); - if (typeof(IBaseJobProcessComponentData_ED).IsAssignableFrom(typeT)) - return ScheduleInternal_ED(ref jobData, null, componentGroup, -1, dependsOn, ScheduleMode.Run); - if (typeof(IBaseJobProcessComponentData_DD).IsAssignableFrom(typeT)) - return ScheduleInternal_DD(ref jobData, null, componentGroup, -1, dependsOn, ScheduleMode.Run); - if (typeof(IBaseJobProcessComponentData_EDD).IsAssignableFrom(typeT)) - return ScheduleInternal_EDD(ref jobData, null, componentGroup, -1, dependsOn, ScheduleMode.Run); - if (typeof(IBaseJobProcessComponentData_DDD).IsAssignableFrom(typeT)) - return ScheduleInternal_DDD(ref jobData, null, componentGroup, -1, dependsOn, ScheduleMode.Run); - if (typeof(IBaseJobProcessComponentData_EDDD).IsAssignableFrom(typeT)) - return ScheduleInternal_EDDD(ref jobData, null, componentGroup, -1, dependsOn, ScheduleMode.Run); - if (typeof(IBaseJobProcessComponentData_DDDD).IsAssignableFrom(typeT)) - return ScheduleInternal_DDDD(ref jobData, null, componentGroup, -1, dependsOn, ScheduleMode.Run); - if (typeof(IBaseJobProcessComponentData_EDDDD).IsAssignableFrom(typeT)) - return ScheduleInternal_EDDDD(ref jobData, null, componentGroup, -1, dependsOn, ScheduleMode.Run); - if (typeof(IBaseJobProcessComponentData_DDDDD).IsAssignableFrom(typeT)) - return ScheduleInternal_DDDDD(ref jobData, null, componentGroup, -1, dependsOn, ScheduleMode.Run); - if (typeof(IBaseJobProcessComponentData_EDDDDD).IsAssignableFrom(typeT)) - return ScheduleInternal_EDDDDD(ref jobData, null, componentGroup, -1, dependsOn, ScheduleMode.Run); - if (typeof(IBaseJobProcessComponentData_DDDDDD).IsAssignableFrom(typeT)) - return ScheduleInternal_DDDDDD(ref jobData, null, componentGroup, -1, dependsOn, ScheduleMode.Run); - if (typeof(IBaseJobProcessComponentData_EDDDDDD).IsAssignableFrom(typeT)) - return ScheduleInternal_EDDDDDD(ref jobData, null, componentGroup, -1, dependsOn, ScheduleMode.Run); - throw new System.ArgumentException("Not supported"); - } - - internal static unsafe JobHandle ScheduleInternal_D(ref T jobData, ComponentSystemBase system, ComponentGroup componentGroup, int innerloopBatchCount, JobHandle dependsOn, ScheduleMode mode) - where T : struct - { - JobStruct_ProcessInfer_D fullData; - fullData.Data = jobData; - - var isParallelFor = innerloopBatchCount != -1; - Initialize(system, componentGroup, typeof(T), typeof(JobStruct_Process_D<,>), isParallelFor, ref JobStruct_ProcessInfer_D.Cache, out fullData.Iterator); - - var unfilteredChunkCount = fullData.Iterator.m_Length; - var iterator = JobStruct_ProcessInfer_D.Cache.ComponentGroup.GetComponentChunkIterator(); - - var prefilterHandle = ComponentChunkIterator.PreparePrefilteredChunkLists(unfilteredChunkCount, iterator.m_MatchingArchetypeList, iterator.m_Filter, dependsOn, mode, out fullData.PrefilterData, out var deferredCountData); - - return Schedule(UnsafeUtility.AddressOf(ref fullData), fullData.PrefilterData, fullData.Iterator.m_Length, innerloopBatchCount, isParallelFor, iterator.RequiresFilter(), ref JobStruct_ProcessInfer_D.Cache, deferredCountData, prefilterHandle, mode); - } - - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public interface IBaseJobProcessComponentData_D : IBaseJobProcessComponentData { } - - [StructLayout(LayoutKind.Sequential)] - private struct JobStruct_ProcessInfer_D where T : struct - { - public static JobProcessComponentDataCache Cache; - - public ProcessIterationData Iterator; - public T Data; - - [DeallocateOnJobCompletion] - [NativeDisableContainerSafetyRestriction] - public NativeArray PrefilterData; - } - - [StructLayout(LayoutKind.Sequential)] - internal struct JobStruct_Process_D - where T : struct, IJobProcessComponentData - where U0 : struct, IComponentData - { - public ProcessIterationData Iterator; - public T Data; - - [DeallocateOnJobCompletion] - [NativeDisableContainerSafetyRestriction] - public NativeArray PrefilterData; - - [Preserve] - public static IntPtr Initialize(JobType jobType) - { - return JobsUtility.CreateJobReflectionData(typeof(JobStruct_Process_D), typeof(T), jobType, (ExecuteJobFunction) Execute); - } - - delegate void ExecuteJobFunction(ref JobStruct_Process_D data, IntPtr additionalPtr, IntPtr bufferRangePatchData, ref JobRanges ranges, int jobIndex); - - public static unsafe void Execute(ref JobStruct_Process_D jobData, IntPtr additionalPtr, IntPtr bufferRangePatchData, ref JobRanges ranges, int jobIndex) - { - var chunks = (ArchetypeChunk*)NativeArrayUnsafeUtility.GetUnsafePtr(jobData.PrefilterData); - var entityIndices = (int*) (chunks + jobData.Iterator.m_Length); - var chunkCount = *(entityIndices + jobData.Iterator.m_Length); - - if (jobData.Iterator.m_IsParallelFor) - { - int begin, end; - while (JobsUtility.GetWorkStealingRange(ref ranges, jobIndex, out begin, out end)) - ExecuteChunk(ref jobData, bufferRangePatchData, begin, end, chunks, entityIndices); - } - else - { - ExecuteChunk(ref jobData, bufferRangePatchData, 0, chunkCount, chunks, entityIndices); - } - } - - static unsafe void ExecuteChunk(ref JobStruct_Process_D jobData, IntPtr bufferRangePatchData, int begin, int end, ArchetypeChunk* chunks, int* entityIndices) - { - var typeLookupCache0 = 0; - - for (var blockIndex = begin; blockIndex != end; ++blockIndex) - { - var chunk = chunks[blockIndex]; - int beginIndex = entityIndices[blockIndex]; - var count = chunk.Count; - #if ENABLE_UNITY_COLLECTIONS_CHECKS - JobsUtility.PatchBufferMinMaxRanges(bufferRangePatchData, UnsafeUtility.AddressOf(ref jobData), beginIndex, beginIndex + count); - #endif - ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex0, ref typeLookupCache0); - var ptr0 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly0 == 0, typeLookupCache0, jobData.Iterator.GlobalSystemVersion)); - - - for (var i = 0; i != count; i++) - { - jobData.Data.Execute(ref UnsafeUtilityEx.ArrayElementAsRef(ptr0, i)); - } - } - } - } - - internal static unsafe JobHandle ScheduleInternal_ED(ref T jobData, ComponentSystemBase system, ComponentGroup componentGroup, int innerloopBatchCount, JobHandle dependsOn, ScheduleMode mode) - where T : struct - { - JobStruct_ProcessInfer_ED fullData; - fullData.Data = jobData; - - var isParallelFor = innerloopBatchCount != -1; - Initialize(system, componentGroup, typeof(T), typeof(JobStruct_Process_ED<,>), isParallelFor, ref JobStruct_ProcessInfer_ED.Cache, out fullData.Iterator); - - var unfilteredChunkCount = fullData.Iterator.m_Length; - var iterator = JobStruct_ProcessInfer_ED.Cache.ComponentGroup.GetComponentChunkIterator(); - - var prefilterHandle = ComponentChunkIterator.PreparePrefilteredChunkLists(unfilteredChunkCount, iterator.m_MatchingArchetypeList, iterator.m_Filter, dependsOn, mode, out fullData.PrefilterData, out var deferredCountData); - - return Schedule(UnsafeUtility.AddressOf(ref fullData), fullData.PrefilterData, fullData.Iterator.m_Length, innerloopBatchCount, isParallelFor, iterator.RequiresFilter(), ref JobStruct_ProcessInfer_ED.Cache, deferredCountData, prefilterHandle, mode); - } - - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public interface IBaseJobProcessComponentData_ED : IBaseJobProcessComponentData { } - - [StructLayout(LayoutKind.Sequential)] - private struct JobStruct_ProcessInfer_ED where T : struct - { - public static JobProcessComponentDataCache Cache; - - public ProcessIterationData Iterator; - public T Data; - - [DeallocateOnJobCompletion] - [NativeDisableContainerSafetyRestriction] - public NativeArray PrefilterData; - } - - [StructLayout(LayoutKind.Sequential)] - internal struct JobStruct_Process_ED - where T : struct, IJobProcessComponentDataWithEntity - where U0 : struct, IComponentData - { - public ProcessIterationData Iterator; - public T Data; - - [DeallocateOnJobCompletion] - [NativeDisableContainerSafetyRestriction] - public NativeArray PrefilterData; - - [Preserve] - public static IntPtr Initialize(JobType jobType) - { - return JobsUtility.CreateJobReflectionData(typeof(JobStruct_Process_ED), typeof(T), jobType, (ExecuteJobFunction) Execute); - } - - delegate void ExecuteJobFunction(ref JobStruct_Process_ED data, IntPtr additionalPtr, IntPtr bufferRangePatchData, ref JobRanges ranges, int jobIndex); - - public static unsafe void Execute(ref JobStruct_Process_ED jobData, IntPtr additionalPtr, IntPtr bufferRangePatchData, ref JobRanges ranges, int jobIndex) - { - var chunks = (ArchetypeChunk*)NativeArrayUnsafeUtility.GetUnsafePtr(jobData.PrefilterData); - var entityIndices = (int*) (chunks + jobData.Iterator.m_Length); - var chunkCount = *(entityIndices + jobData.Iterator.m_Length); - - if (jobData.Iterator.m_IsParallelFor) - { - int begin, end; - while (JobsUtility.GetWorkStealingRange(ref ranges, jobIndex, out begin, out end)) - ExecuteChunk(ref jobData, bufferRangePatchData, begin, end, chunks, entityIndices); - } - else - { - ExecuteChunk(ref jobData, bufferRangePatchData, 0, chunkCount, chunks, entityIndices); - } - } - - static unsafe void ExecuteChunk(ref JobStruct_Process_ED jobData, IntPtr bufferRangePatchData, int begin, int end, ArchetypeChunk* chunks, int* entityIndices) - { - var typeLookupCache0 = 0; - - for (var blockIndex = begin; blockIndex != end; ++blockIndex) - { - var chunk = chunks[blockIndex]; - int beginIndex = entityIndices[blockIndex]; - var count = chunk.Count; - #if ENABLE_UNITY_COLLECTIONS_CHECKS - JobsUtility.PatchBufferMinMaxRanges(bufferRangePatchData, UnsafeUtility.AddressOf(ref jobData), beginIndex, beginIndex + count); - #endif - var ptrE = (Entity*)UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, false, 0, jobData.Iterator.GlobalSystemVersion)); - ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex0, ref typeLookupCache0); - var ptr0 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly0 == 0, typeLookupCache0, jobData.Iterator.GlobalSystemVersion)); - - - for (var i = 0; i != count; i++) - { - jobData.Data.Execute(ptrE[i], i + beginIndex, ref UnsafeUtilityEx.ArrayElementAsRef(ptr0, i)); - } - } - } - } - - internal static unsafe JobHandle ScheduleInternal_DD(ref T jobData, ComponentSystemBase system, ComponentGroup componentGroup, int innerloopBatchCount, JobHandle dependsOn, ScheduleMode mode) - where T : struct - { - JobStruct_ProcessInfer_DD fullData; - fullData.Data = jobData; - - var isParallelFor = innerloopBatchCount != -1; - Initialize(system, componentGroup, typeof(T), typeof(JobStruct_Process_DD<,,>), isParallelFor, ref JobStruct_ProcessInfer_DD.Cache, out fullData.Iterator); - - var unfilteredChunkCount = fullData.Iterator.m_Length; - var iterator = JobStruct_ProcessInfer_DD.Cache.ComponentGroup.GetComponentChunkIterator(); - - var prefilterHandle = ComponentChunkIterator.PreparePrefilteredChunkLists(unfilteredChunkCount, iterator.m_MatchingArchetypeList, iterator.m_Filter, dependsOn, mode, out fullData.PrefilterData, out var deferredCountData); - - return Schedule(UnsafeUtility.AddressOf(ref fullData), fullData.PrefilterData, fullData.Iterator.m_Length, innerloopBatchCount, isParallelFor, iterator.RequiresFilter(), ref JobStruct_ProcessInfer_DD.Cache, deferredCountData, prefilterHandle, mode); - } - - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public interface IBaseJobProcessComponentData_DD : IBaseJobProcessComponentData { } - - [StructLayout(LayoutKind.Sequential)] - private struct JobStruct_ProcessInfer_DD where T : struct - { - public static JobProcessComponentDataCache Cache; - - public ProcessIterationData Iterator; - public T Data; - - [DeallocateOnJobCompletion] - [NativeDisableContainerSafetyRestriction] - public NativeArray PrefilterData; - } - - [StructLayout(LayoutKind.Sequential)] - internal struct JobStruct_Process_DD - where T : struct, IJobProcessComponentData - where U0 : struct, IComponentData - where U1 : struct, IComponentData - { - public ProcessIterationData Iterator; - public T Data; - - [DeallocateOnJobCompletion] - [NativeDisableContainerSafetyRestriction] - public NativeArray PrefilterData; - - [Preserve] - public static IntPtr Initialize(JobType jobType) - { - return JobsUtility.CreateJobReflectionData(typeof(JobStruct_Process_DD), typeof(T), jobType, (ExecuteJobFunction) Execute); - } - - delegate void ExecuteJobFunction(ref JobStruct_Process_DD data, IntPtr additionalPtr, IntPtr bufferRangePatchData, ref JobRanges ranges, int jobIndex); - - public static unsafe void Execute(ref JobStruct_Process_DD jobData, IntPtr additionalPtr, IntPtr bufferRangePatchData, ref JobRanges ranges, int jobIndex) - { - var chunks = (ArchetypeChunk*)NativeArrayUnsafeUtility.GetUnsafePtr(jobData.PrefilterData); - var entityIndices = (int*) (chunks + jobData.Iterator.m_Length); - var chunkCount = *(entityIndices + jobData.Iterator.m_Length); - - if (jobData.Iterator.m_IsParallelFor) - { - int begin, end; - while (JobsUtility.GetWorkStealingRange(ref ranges, jobIndex, out begin, out end)) - ExecuteChunk(ref jobData, bufferRangePatchData, begin, end, chunks, entityIndices); - } - else - { - ExecuteChunk(ref jobData, bufferRangePatchData, 0, chunkCount, chunks, entityIndices); - } - } - - static unsafe void ExecuteChunk(ref JobStruct_Process_DD jobData, IntPtr bufferRangePatchData, int begin, int end, ArchetypeChunk* chunks, int* entityIndices) - { - var typeLookupCache0 = 0; - var typeLookupCache1 = 0; - - for (var blockIndex = begin; blockIndex != end; ++blockIndex) - { - var chunk = chunks[blockIndex]; - int beginIndex = entityIndices[blockIndex]; - var count = chunk.Count; - #if ENABLE_UNITY_COLLECTIONS_CHECKS - JobsUtility.PatchBufferMinMaxRanges(bufferRangePatchData, UnsafeUtility.AddressOf(ref jobData), beginIndex, beginIndex + count); - #endif - ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex0, ref typeLookupCache0); - var ptr0 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly0 == 0, typeLookupCache0, jobData.Iterator.GlobalSystemVersion)); - ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex1, ref typeLookupCache1); - var ptr1 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly1 == 0, typeLookupCache1, jobData.Iterator.GlobalSystemVersion)); - - - for (var i = 0; i != count; i++) - { - jobData.Data.Execute(ref UnsafeUtilityEx.ArrayElementAsRef(ptr0, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr1, i)); - } - } - } - } - - internal static unsafe JobHandle ScheduleInternal_EDD(ref T jobData, ComponentSystemBase system, ComponentGroup componentGroup, int innerloopBatchCount, JobHandle dependsOn, ScheduleMode mode) - where T : struct - { - JobStruct_ProcessInfer_EDD fullData; - fullData.Data = jobData; - - var isParallelFor = innerloopBatchCount != -1; - Initialize(system, componentGroup, typeof(T), typeof(JobStruct_Process_EDD<,,>), isParallelFor, ref JobStruct_ProcessInfer_EDD.Cache, out fullData.Iterator); - - var unfilteredChunkCount = fullData.Iterator.m_Length; - var iterator = JobStruct_ProcessInfer_EDD.Cache.ComponentGroup.GetComponentChunkIterator(); - - var prefilterHandle = ComponentChunkIterator.PreparePrefilteredChunkLists(unfilteredChunkCount, iterator.m_MatchingArchetypeList, iterator.m_Filter, dependsOn, mode, out fullData.PrefilterData, out var deferredCountData); - - return Schedule(UnsafeUtility.AddressOf(ref fullData), fullData.PrefilterData, fullData.Iterator.m_Length, innerloopBatchCount, isParallelFor, iterator.RequiresFilter(), ref JobStruct_ProcessInfer_EDD.Cache, deferredCountData, prefilterHandle, mode); - } - - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public interface IBaseJobProcessComponentData_EDD : IBaseJobProcessComponentData { } - - [StructLayout(LayoutKind.Sequential)] - private struct JobStruct_ProcessInfer_EDD where T : struct - { - public static JobProcessComponentDataCache Cache; - - public ProcessIterationData Iterator; - public T Data; - - [DeallocateOnJobCompletion] - [NativeDisableContainerSafetyRestriction] - public NativeArray PrefilterData; - } - - [StructLayout(LayoutKind.Sequential)] - internal struct JobStruct_Process_EDD - where T : struct, IJobProcessComponentDataWithEntity - where U0 : struct, IComponentData - where U1 : struct, IComponentData - { - public ProcessIterationData Iterator; - public T Data; - - [DeallocateOnJobCompletion] - [NativeDisableContainerSafetyRestriction] - public NativeArray PrefilterData; - - [Preserve] - public static IntPtr Initialize(JobType jobType) - { - return JobsUtility.CreateJobReflectionData(typeof(JobStruct_Process_EDD), typeof(T), jobType, (ExecuteJobFunction) Execute); - } - - delegate void ExecuteJobFunction(ref JobStruct_Process_EDD data, IntPtr additionalPtr, IntPtr bufferRangePatchData, ref JobRanges ranges, int jobIndex); - - public static unsafe void Execute(ref JobStruct_Process_EDD jobData, IntPtr additionalPtr, IntPtr bufferRangePatchData, ref JobRanges ranges, int jobIndex) - { - var chunks = (ArchetypeChunk*)NativeArrayUnsafeUtility.GetUnsafePtr(jobData.PrefilterData); - var entityIndices = (int*) (chunks + jobData.Iterator.m_Length); - var chunkCount = *(entityIndices + jobData.Iterator.m_Length); - - if (jobData.Iterator.m_IsParallelFor) - { - int begin, end; - while (JobsUtility.GetWorkStealingRange(ref ranges, jobIndex, out begin, out end)) - ExecuteChunk(ref jobData, bufferRangePatchData, begin, end, chunks, entityIndices); - } - else - { - ExecuteChunk(ref jobData, bufferRangePatchData, 0, chunkCount, chunks, entityIndices); - } - } - - static unsafe void ExecuteChunk(ref JobStruct_Process_EDD jobData, IntPtr bufferRangePatchData, int begin, int end, ArchetypeChunk* chunks, int* entityIndices) - { - var typeLookupCache0 = 0; - var typeLookupCache1 = 0; - - for (var blockIndex = begin; blockIndex != end; ++blockIndex) - { - var chunk = chunks[blockIndex]; - int beginIndex = entityIndices[blockIndex]; - var count = chunk.Count; - #if ENABLE_UNITY_COLLECTIONS_CHECKS - JobsUtility.PatchBufferMinMaxRanges(bufferRangePatchData, UnsafeUtility.AddressOf(ref jobData), beginIndex, beginIndex + count); - #endif - var ptrE = (Entity*)UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, false, 0, jobData.Iterator.GlobalSystemVersion)); - ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex0, ref typeLookupCache0); - var ptr0 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly0 == 0, typeLookupCache0, jobData.Iterator.GlobalSystemVersion)); - ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex1, ref typeLookupCache1); - var ptr1 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly1 == 0, typeLookupCache1, jobData.Iterator.GlobalSystemVersion)); - - - for (var i = 0; i != count; i++) - { - jobData.Data.Execute(ptrE[i], i + beginIndex, ref UnsafeUtilityEx.ArrayElementAsRef(ptr0, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr1, i)); - } - } - } - } - - internal static unsafe JobHandle ScheduleInternal_DDD(ref T jobData, ComponentSystemBase system, ComponentGroup componentGroup, int innerloopBatchCount, JobHandle dependsOn, ScheduleMode mode) - where T : struct - { - JobStruct_ProcessInfer_DDD fullData; - fullData.Data = jobData; - - var isParallelFor = innerloopBatchCount != -1; - Initialize(system, componentGroup, typeof(T), typeof(JobStruct_Process_DDD<,,,>), isParallelFor, ref JobStruct_ProcessInfer_DDD.Cache, out fullData.Iterator); - - var unfilteredChunkCount = fullData.Iterator.m_Length; - var iterator = JobStruct_ProcessInfer_DDD.Cache.ComponentGroup.GetComponentChunkIterator(); - - var prefilterHandle = ComponentChunkIterator.PreparePrefilteredChunkLists(unfilteredChunkCount, iterator.m_MatchingArchetypeList, iterator.m_Filter, dependsOn, mode, out fullData.PrefilterData, out var deferredCountData); - - return Schedule(UnsafeUtility.AddressOf(ref fullData), fullData.PrefilterData, fullData.Iterator.m_Length, innerloopBatchCount, isParallelFor, iterator.RequiresFilter(), ref JobStruct_ProcessInfer_DDD.Cache, deferredCountData, prefilterHandle, mode); - } - - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public interface IBaseJobProcessComponentData_DDD : IBaseJobProcessComponentData { } - - [StructLayout(LayoutKind.Sequential)] - private struct JobStruct_ProcessInfer_DDD where T : struct - { - public static JobProcessComponentDataCache Cache; - - public ProcessIterationData Iterator; - public T Data; - - [DeallocateOnJobCompletion] - [NativeDisableContainerSafetyRestriction] - public NativeArray PrefilterData; - } - - [StructLayout(LayoutKind.Sequential)] - internal struct JobStruct_Process_DDD - where T : struct, IJobProcessComponentData - where U0 : struct, IComponentData - where U1 : struct, IComponentData - where U2 : struct, IComponentData - { - public ProcessIterationData Iterator; - public T Data; - - [DeallocateOnJobCompletion] - [NativeDisableContainerSafetyRestriction] - public NativeArray PrefilterData; - - [Preserve] - public static IntPtr Initialize(JobType jobType) - { - return JobsUtility.CreateJobReflectionData(typeof(JobStruct_Process_DDD), typeof(T), jobType, (ExecuteJobFunction) Execute); - } - - delegate void ExecuteJobFunction(ref JobStruct_Process_DDD data, IntPtr additionalPtr, IntPtr bufferRangePatchData, ref JobRanges ranges, int jobIndex); - - public static unsafe void Execute(ref JobStruct_Process_DDD jobData, IntPtr additionalPtr, IntPtr bufferRangePatchData, ref JobRanges ranges, int jobIndex) - { - var chunks = (ArchetypeChunk*)NativeArrayUnsafeUtility.GetUnsafePtr(jobData.PrefilterData); - var entityIndices = (int*) (chunks + jobData.Iterator.m_Length); - var chunkCount = *(entityIndices + jobData.Iterator.m_Length); - - if (jobData.Iterator.m_IsParallelFor) - { - int begin, end; - while (JobsUtility.GetWorkStealingRange(ref ranges, jobIndex, out begin, out end)) - ExecuteChunk(ref jobData, bufferRangePatchData, begin, end, chunks, entityIndices); - } - else - { - ExecuteChunk(ref jobData, bufferRangePatchData, 0, chunkCount, chunks, entityIndices); - } - } - - static unsafe void ExecuteChunk(ref JobStruct_Process_DDD jobData, IntPtr bufferRangePatchData, int begin, int end, ArchetypeChunk* chunks, int* entityIndices) - { - var typeLookupCache0 = 0; - var typeLookupCache1 = 0; - var typeLookupCache2 = 0; - - for (var blockIndex = begin; blockIndex != end; ++blockIndex) - { - var chunk = chunks[blockIndex]; - int beginIndex = entityIndices[blockIndex]; - var count = chunk.Count; - #if ENABLE_UNITY_COLLECTIONS_CHECKS - JobsUtility.PatchBufferMinMaxRanges(bufferRangePatchData, UnsafeUtility.AddressOf(ref jobData), beginIndex, beginIndex + count); - #endif - ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex0, ref typeLookupCache0); - var ptr0 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly0 == 0, typeLookupCache0, jobData.Iterator.GlobalSystemVersion)); - ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex1, ref typeLookupCache1); - var ptr1 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly1 == 0, typeLookupCache1, jobData.Iterator.GlobalSystemVersion)); - ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex2, ref typeLookupCache2); - var ptr2 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly2 == 0, typeLookupCache2, jobData.Iterator.GlobalSystemVersion)); - - - for (var i = 0; i != count; i++) - { - jobData.Data.Execute(ref UnsafeUtilityEx.ArrayElementAsRef(ptr0, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr1, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr2, i)); - } - } - } - } - - internal static unsafe JobHandle ScheduleInternal_EDDD(ref T jobData, ComponentSystemBase system, ComponentGroup componentGroup, int innerloopBatchCount, JobHandle dependsOn, ScheduleMode mode) - where T : struct - { - JobStruct_ProcessInfer_EDDD fullData; - fullData.Data = jobData; - - var isParallelFor = innerloopBatchCount != -1; - Initialize(system, componentGroup, typeof(T), typeof(JobStruct_Process_EDDD<,,,>), isParallelFor, ref JobStruct_ProcessInfer_EDDD.Cache, out fullData.Iterator); - - var unfilteredChunkCount = fullData.Iterator.m_Length; - var iterator = JobStruct_ProcessInfer_EDDD.Cache.ComponentGroup.GetComponentChunkIterator(); - - var prefilterHandle = ComponentChunkIterator.PreparePrefilteredChunkLists(unfilteredChunkCount, iterator.m_MatchingArchetypeList, iterator.m_Filter, dependsOn, mode, out fullData.PrefilterData, out var deferredCountData); - - return Schedule(UnsafeUtility.AddressOf(ref fullData), fullData.PrefilterData, fullData.Iterator.m_Length, innerloopBatchCount, isParallelFor, iterator.RequiresFilter(), ref JobStruct_ProcessInfer_EDDD.Cache, deferredCountData, prefilterHandle, mode); - } - - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public interface IBaseJobProcessComponentData_EDDD : IBaseJobProcessComponentData { } - - [StructLayout(LayoutKind.Sequential)] - private struct JobStruct_ProcessInfer_EDDD where T : struct - { - public static JobProcessComponentDataCache Cache; - - public ProcessIterationData Iterator; - public T Data; - - [DeallocateOnJobCompletion] - [NativeDisableContainerSafetyRestriction] - public NativeArray PrefilterData; - } - - [StructLayout(LayoutKind.Sequential)] - internal struct JobStruct_Process_EDDD - where T : struct, IJobProcessComponentDataWithEntity - where U0 : struct, IComponentData - where U1 : struct, IComponentData - where U2 : struct, IComponentData - { - public ProcessIterationData Iterator; - public T Data; - - [DeallocateOnJobCompletion] - [NativeDisableContainerSafetyRestriction] - public NativeArray PrefilterData; - - [Preserve] - public static IntPtr Initialize(JobType jobType) - { - return JobsUtility.CreateJobReflectionData(typeof(JobStruct_Process_EDDD), typeof(T), jobType, (ExecuteJobFunction) Execute); - } - - delegate void ExecuteJobFunction(ref JobStruct_Process_EDDD data, IntPtr additionalPtr, IntPtr bufferRangePatchData, ref JobRanges ranges, int jobIndex); - - public static unsafe void Execute(ref JobStruct_Process_EDDD jobData, IntPtr additionalPtr, IntPtr bufferRangePatchData, ref JobRanges ranges, int jobIndex) - { - var chunks = (ArchetypeChunk*)NativeArrayUnsafeUtility.GetUnsafePtr(jobData.PrefilterData); - var entityIndices = (int*) (chunks + jobData.Iterator.m_Length); - var chunkCount = *(entityIndices + jobData.Iterator.m_Length); - - if (jobData.Iterator.m_IsParallelFor) - { - int begin, end; - while (JobsUtility.GetWorkStealingRange(ref ranges, jobIndex, out begin, out end)) - ExecuteChunk(ref jobData, bufferRangePatchData, begin, end, chunks, entityIndices); - } - else - { - ExecuteChunk(ref jobData, bufferRangePatchData, 0, chunkCount, chunks, entityIndices); - } - } - - static unsafe void ExecuteChunk(ref JobStruct_Process_EDDD jobData, IntPtr bufferRangePatchData, int begin, int end, ArchetypeChunk* chunks, int* entityIndices) - { - var typeLookupCache0 = 0; - var typeLookupCache1 = 0; - var typeLookupCache2 = 0; - - for (var blockIndex = begin; blockIndex != end; ++blockIndex) - { - var chunk = chunks[blockIndex]; - int beginIndex = entityIndices[blockIndex]; - var count = chunk.Count; - #if ENABLE_UNITY_COLLECTIONS_CHECKS - JobsUtility.PatchBufferMinMaxRanges(bufferRangePatchData, UnsafeUtility.AddressOf(ref jobData), beginIndex, beginIndex + count); - #endif - var ptrE = (Entity*)UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, false, 0, jobData.Iterator.GlobalSystemVersion)); - ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex0, ref typeLookupCache0); - var ptr0 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly0 == 0, typeLookupCache0, jobData.Iterator.GlobalSystemVersion)); - ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex1, ref typeLookupCache1); - var ptr1 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly1 == 0, typeLookupCache1, jobData.Iterator.GlobalSystemVersion)); - ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex2, ref typeLookupCache2); - var ptr2 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly2 == 0, typeLookupCache2, jobData.Iterator.GlobalSystemVersion)); - - - for (var i = 0; i != count; i++) - { - jobData.Data.Execute(ptrE[i], i + beginIndex, ref UnsafeUtilityEx.ArrayElementAsRef(ptr0, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr1, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr2, i)); - } - } - } - } - - internal static unsafe JobHandle ScheduleInternal_DDDD(ref T jobData, ComponentSystemBase system, ComponentGroup componentGroup, int innerloopBatchCount, JobHandle dependsOn, ScheduleMode mode) - where T : struct - { - JobStruct_ProcessInfer_DDDD fullData; - fullData.Data = jobData; - - var isParallelFor = innerloopBatchCount != -1; - Initialize(system, componentGroup, typeof(T), typeof(JobStruct_Process_DDDD<,,,,>), isParallelFor, ref JobStruct_ProcessInfer_DDDD.Cache, out fullData.Iterator); - - var unfilteredChunkCount = fullData.Iterator.m_Length; - var iterator = JobStruct_ProcessInfer_DDDD.Cache.ComponentGroup.GetComponentChunkIterator(); - - var prefilterHandle = ComponentChunkIterator.PreparePrefilteredChunkLists(unfilteredChunkCount, iterator.m_MatchingArchetypeList, iterator.m_Filter, dependsOn, mode, out fullData.PrefilterData, out var deferredCountData); - - return Schedule(UnsafeUtility.AddressOf(ref fullData), fullData.PrefilterData, fullData.Iterator.m_Length, innerloopBatchCount, isParallelFor, iterator.RequiresFilter(), ref JobStruct_ProcessInfer_DDDD.Cache, deferredCountData, prefilterHandle, mode); - } - - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public interface IBaseJobProcessComponentData_DDDD : IBaseJobProcessComponentData { } - - [StructLayout(LayoutKind.Sequential)] - private struct JobStruct_ProcessInfer_DDDD where T : struct - { - public static JobProcessComponentDataCache Cache; - - public ProcessIterationData Iterator; - public T Data; - - [DeallocateOnJobCompletion] - [NativeDisableContainerSafetyRestriction] - public NativeArray PrefilterData; - } - - [StructLayout(LayoutKind.Sequential)] - internal struct JobStruct_Process_DDDD - where T : struct, IJobProcessComponentData - where U0 : struct, IComponentData - where U1 : struct, IComponentData - where U2 : struct, IComponentData - where U3 : struct, IComponentData - { - public ProcessIterationData Iterator; - public T Data; - - [DeallocateOnJobCompletion] - [NativeDisableContainerSafetyRestriction] - public NativeArray PrefilterData; - - [Preserve] - public static IntPtr Initialize(JobType jobType) - { - return JobsUtility.CreateJobReflectionData(typeof(JobStruct_Process_DDDD), typeof(T), jobType, (ExecuteJobFunction) Execute); - } - - delegate void ExecuteJobFunction(ref JobStruct_Process_DDDD data, IntPtr additionalPtr, IntPtr bufferRangePatchData, ref JobRanges ranges, int jobIndex); - - public static unsafe void Execute(ref JobStruct_Process_DDDD jobData, IntPtr additionalPtr, IntPtr bufferRangePatchData, ref JobRanges ranges, int jobIndex) - { - var chunks = (ArchetypeChunk*)NativeArrayUnsafeUtility.GetUnsafePtr(jobData.PrefilterData); - var entityIndices = (int*) (chunks + jobData.Iterator.m_Length); - var chunkCount = *(entityIndices + jobData.Iterator.m_Length); - - if (jobData.Iterator.m_IsParallelFor) - { - int begin, end; - while (JobsUtility.GetWorkStealingRange(ref ranges, jobIndex, out begin, out end)) - ExecuteChunk(ref jobData, bufferRangePatchData, begin, end, chunks, entityIndices); - } - else - { - ExecuteChunk(ref jobData, bufferRangePatchData, 0, chunkCount, chunks, entityIndices); - } - } - - static unsafe void ExecuteChunk(ref JobStruct_Process_DDDD jobData, IntPtr bufferRangePatchData, int begin, int end, ArchetypeChunk* chunks, int* entityIndices) - { - var typeLookupCache0 = 0; - var typeLookupCache1 = 0; - var typeLookupCache2 = 0; - var typeLookupCache3 = 0; - - for (var blockIndex = begin; blockIndex != end; ++blockIndex) - { - var chunk = chunks[blockIndex]; - int beginIndex = entityIndices[blockIndex]; - var count = chunk.Count; - #if ENABLE_UNITY_COLLECTIONS_CHECKS - JobsUtility.PatchBufferMinMaxRanges(bufferRangePatchData, UnsafeUtility.AddressOf(ref jobData), beginIndex, beginIndex + count); - #endif - ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex0, ref typeLookupCache0); - var ptr0 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly0 == 0, typeLookupCache0, jobData.Iterator.GlobalSystemVersion)); - ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex1, ref typeLookupCache1); - var ptr1 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly1 == 0, typeLookupCache1, jobData.Iterator.GlobalSystemVersion)); - ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex2, ref typeLookupCache2); - var ptr2 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly2 == 0, typeLookupCache2, jobData.Iterator.GlobalSystemVersion)); - ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex3, ref typeLookupCache3); - var ptr3 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly3 == 0, typeLookupCache3, jobData.Iterator.GlobalSystemVersion)); - - - for (var i = 0; i != count; i++) - { - jobData.Data.Execute(ref UnsafeUtilityEx.ArrayElementAsRef(ptr0, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr1, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr2, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr3, i)); - } - } - } - } - - internal static unsafe JobHandle ScheduleInternal_EDDDD(ref T jobData, ComponentSystemBase system, ComponentGroup componentGroup, int innerloopBatchCount, JobHandle dependsOn, ScheduleMode mode) - where T : struct - { - JobStruct_ProcessInfer_EDDDD fullData; - fullData.Data = jobData; - - var isParallelFor = innerloopBatchCount != -1; - Initialize(system, componentGroup, typeof(T), typeof(JobStruct_Process_EDDDD<,,,,>), isParallelFor, ref JobStruct_ProcessInfer_EDDDD.Cache, out fullData.Iterator); - - var unfilteredChunkCount = fullData.Iterator.m_Length; - var iterator = JobStruct_ProcessInfer_EDDDD.Cache.ComponentGroup.GetComponentChunkIterator(); - - var prefilterHandle = ComponentChunkIterator.PreparePrefilteredChunkLists(unfilteredChunkCount, iterator.m_MatchingArchetypeList, iterator.m_Filter, dependsOn, mode, out fullData.PrefilterData, out var deferredCountData); - - return Schedule(UnsafeUtility.AddressOf(ref fullData), fullData.PrefilterData, fullData.Iterator.m_Length, innerloopBatchCount, isParallelFor, iterator.RequiresFilter(), ref JobStruct_ProcessInfer_EDDDD.Cache, deferredCountData, prefilterHandle, mode); - } - - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public interface IBaseJobProcessComponentData_EDDDD : IBaseJobProcessComponentData { } - - [StructLayout(LayoutKind.Sequential)] - private struct JobStruct_ProcessInfer_EDDDD where T : struct - { - public static JobProcessComponentDataCache Cache; - - public ProcessIterationData Iterator; - public T Data; - - [DeallocateOnJobCompletion] - [NativeDisableContainerSafetyRestriction] - public NativeArray PrefilterData; - } - - [StructLayout(LayoutKind.Sequential)] - internal struct JobStruct_Process_EDDDD - where T : struct, IJobProcessComponentDataWithEntity - where U0 : struct, IComponentData - where U1 : struct, IComponentData - where U2 : struct, IComponentData - where U3 : struct, IComponentData - { - public ProcessIterationData Iterator; - public T Data; - - [DeallocateOnJobCompletion] - [NativeDisableContainerSafetyRestriction] - public NativeArray PrefilterData; - - [Preserve] - public static IntPtr Initialize(JobType jobType) - { - return JobsUtility.CreateJobReflectionData(typeof(JobStruct_Process_EDDDD), typeof(T), jobType, (ExecuteJobFunction) Execute); - } - - delegate void ExecuteJobFunction(ref JobStruct_Process_EDDDD data, IntPtr additionalPtr, IntPtr bufferRangePatchData, ref JobRanges ranges, int jobIndex); - - public static unsafe void Execute(ref JobStruct_Process_EDDDD jobData, IntPtr additionalPtr, IntPtr bufferRangePatchData, ref JobRanges ranges, int jobIndex) - { - var chunks = (ArchetypeChunk*)NativeArrayUnsafeUtility.GetUnsafePtr(jobData.PrefilterData); - var entityIndices = (int*) (chunks + jobData.Iterator.m_Length); - var chunkCount = *(entityIndices + jobData.Iterator.m_Length); - - if (jobData.Iterator.m_IsParallelFor) - { - int begin, end; - while (JobsUtility.GetWorkStealingRange(ref ranges, jobIndex, out begin, out end)) - ExecuteChunk(ref jobData, bufferRangePatchData, begin, end, chunks, entityIndices); - } - else - { - ExecuteChunk(ref jobData, bufferRangePatchData, 0, chunkCount, chunks, entityIndices); - } - } - - static unsafe void ExecuteChunk(ref JobStruct_Process_EDDDD jobData, IntPtr bufferRangePatchData, int begin, int end, ArchetypeChunk* chunks, int* entityIndices) - { - var typeLookupCache0 = 0; - var typeLookupCache1 = 0; - var typeLookupCache2 = 0; - var typeLookupCache3 = 0; - - for (var blockIndex = begin; blockIndex != end; ++blockIndex) - { - var chunk = chunks[blockIndex]; - int beginIndex = entityIndices[blockIndex]; - var count = chunk.Count; - #if ENABLE_UNITY_COLLECTIONS_CHECKS - JobsUtility.PatchBufferMinMaxRanges(bufferRangePatchData, UnsafeUtility.AddressOf(ref jobData), beginIndex, beginIndex + count); - #endif - var ptrE = (Entity*)UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, false, 0, jobData.Iterator.GlobalSystemVersion)); - ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex0, ref typeLookupCache0); - var ptr0 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly0 == 0, typeLookupCache0, jobData.Iterator.GlobalSystemVersion)); - ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex1, ref typeLookupCache1); - var ptr1 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly1 == 0, typeLookupCache1, jobData.Iterator.GlobalSystemVersion)); - ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex2, ref typeLookupCache2); - var ptr2 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly2 == 0, typeLookupCache2, jobData.Iterator.GlobalSystemVersion)); - ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex3, ref typeLookupCache3); - var ptr3 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly3 == 0, typeLookupCache3, jobData.Iterator.GlobalSystemVersion)); - - - for (var i = 0; i != count; i++) - { - jobData.Data.Execute(ptrE[i], i + beginIndex, ref UnsafeUtilityEx.ArrayElementAsRef(ptr0, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr1, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr2, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr3, i)); - } - } - } - } - - internal static unsafe JobHandle ScheduleInternal_DDDDD(ref T jobData, ComponentSystemBase system, ComponentGroup componentGroup, int innerloopBatchCount, JobHandle dependsOn, ScheduleMode mode) - where T : struct - { - JobStruct_ProcessInfer_DDDDD fullData; - fullData.Data = jobData; - - var isParallelFor = innerloopBatchCount != -1; - Initialize(system, componentGroup, typeof(T), typeof(JobStruct_Process_DDDDD<,,,,,>), isParallelFor, ref JobStruct_ProcessInfer_DDDDD.Cache, out fullData.Iterator); - - var unfilteredChunkCount = fullData.Iterator.m_Length; - var iterator = JobStruct_ProcessInfer_DDDDD.Cache.ComponentGroup.GetComponentChunkIterator(); - - var prefilterHandle = ComponentChunkIterator.PreparePrefilteredChunkLists(unfilteredChunkCount, iterator.m_MatchingArchetypeList, iterator.m_Filter, dependsOn, mode, out fullData.PrefilterData, out var deferredCountData); - - return Schedule(UnsafeUtility.AddressOf(ref fullData), fullData.PrefilterData, fullData.Iterator.m_Length, innerloopBatchCount, isParallelFor, iterator.RequiresFilter(), ref JobStruct_ProcessInfer_DDDDD.Cache, deferredCountData, prefilterHandle, mode); - } - - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public interface IBaseJobProcessComponentData_DDDDD : IBaseJobProcessComponentData { } - - [StructLayout(LayoutKind.Sequential)] - private struct JobStruct_ProcessInfer_DDDDD where T : struct - { - public static JobProcessComponentDataCache Cache; - - public ProcessIterationData Iterator; - public T Data; - - [DeallocateOnJobCompletion] - [NativeDisableContainerSafetyRestriction] - public NativeArray PrefilterData; - } - - [StructLayout(LayoutKind.Sequential)] - internal struct JobStruct_Process_DDDDD - where T : struct, IJobProcessComponentData - where U0 : struct, IComponentData - where U1 : struct, IComponentData - where U2 : struct, IComponentData - where U3 : struct, IComponentData - where U4 : struct, IComponentData - { - public ProcessIterationData Iterator; - public T Data; - - [DeallocateOnJobCompletion] - [NativeDisableContainerSafetyRestriction] - public NativeArray PrefilterData; - - [Preserve] - public static IntPtr Initialize(JobType jobType) - { - return JobsUtility.CreateJobReflectionData(typeof(JobStruct_Process_DDDDD), typeof(T), jobType, (ExecuteJobFunction) Execute); - } - - delegate void ExecuteJobFunction(ref JobStruct_Process_DDDDD data, IntPtr additionalPtr, IntPtr bufferRangePatchData, ref JobRanges ranges, int jobIndex); - - public static unsafe void Execute(ref JobStruct_Process_DDDDD jobData, IntPtr additionalPtr, IntPtr bufferRangePatchData, ref JobRanges ranges, int jobIndex) - { - var chunks = (ArchetypeChunk*)NativeArrayUnsafeUtility.GetUnsafePtr(jobData.PrefilterData); - var entityIndices = (int*) (chunks + jobData.Iterator.m_Length); - var chunkCount = *(entityIndices + jobData.Iterator.m_Length); - - if (jobData.Iterator.m_IsParallelFor) - { - int begin, end; - while (JobsUtility.GetWorkStealingRange(ref ranges, jobIndex, out begin, out end)) - ExecuteChunk(ref jobData, bufferRangePatchData, begin, end, chunks, entityIndices); - } - else - { - ExecuteChunk(ref jobData, bufferRangePatchData, 0, chunkCount, chunks, entityIndices); - } - } - - static unsafe void ExecuteChunk(ref JobStruct_Process_DDDDD jobData, IntPtr bufferRangePatchData, int begin, int end, ArchetypeChunk* chunks, int* entityIndices) - { - var typeLookupCache0 = 0; - var typeLookupCache1 = 0; - var typeLookupCache2 = 0; - var typeLookupCache3 = 0; - var typeLookupCache4 = 0; - - for (var blockIndex = begin; blockIndex != end; ++blockIndex) - { - var chunk = chunks[blockIndex]; - int beginIndex = entityIndices[blockIndex]; - var count = chunk.Count; - #if ENABLE_UNITY_COLLECTIONS_CHECKS - JobsUtility.PatchBufferMinMaxRanges(bufferRangePatchData, UnsafeUtility.AddressOf(ref jobData), beginIndex, beginIndex + count); - #endif - ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex0, ref typeLookupCache0); - var ptr0 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly0 == 0, typeLookupCache0, jobData.Iterator.GlobalSystemVersion)); - ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex1, ref typeLookupCache1); - var ptr1 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly1 == 0, typeLookupCache1, jobData.Iterator.GlobalSystemVersion)); - ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex2, ref typeLookupCache2); - var ptr2 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly2 == 0, typeLookupCache2, jobData.Iterator.GlobalSystemVersion)); - ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex3, ref typeLookupCache3); - var ptr3 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly3 == 0, typeLookupCache3, jobData.Iterator.GlobalSystemVersion)); - ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex4, ref typeLookupCache4); - var ptr4 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly4 == 0, typeLookupCache4, jobData.Iterator.GlobalSystemVersion)); - - - for (var i = 0; i != count; i++) - { - jobData.Data.Execute(ref UnsafeUtilityEx.ArrayElementAsRef(ptr0, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr1, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr2, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr3, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr4, i)); - } - } - } - } - - internal static unsafe JobHandle ScheduleInternal_EDDDDD(ref T jobData, ComponentSystemBase system, ComponentGroup componentGroup, int innerloopBatchCount, JobHandle dependsOn, ScheduleMode mode) - where T : struct - { - JobStruct_ProcessInfer_EDDDDD fullData; - fullData.Data = jobData; - - var isParallelFor = innerloopBatchCount != -1; - Initialize(system, componentGroup, typeof(T), typeof(JobStruct_Process_EDDDDD<,,,,,>), isParallelFor, ref JobStruct_ProcessInfer_EDDDDD.Cache, out fullData.Iterator); - - var unfilteredChunkCount = fullData.Iterator.m_Length; - var iterator = JobStruct_ProcessInfer_EDDDDD.Cache.ComponentGroup.GetComponentChunkIterator(); - - var prefilterHandle = ComponentChunkIterator.PreparePrefilteredChunkLists(unfilteredChunkCount, iterator.m_MatchingArchetypeList, iterator.m_Filter, dependsOn, mode, out fullData.PrefilterData, out var deferredCountData); - - return Schedule(UnsafeUtility.AddressOf(ref fullData), fullData.PrefilterData, fullData.Iterator.m_Length, innerloopBatchCount, isParallelFor, iterator.RequiresFilter(), ref JobStruct_ProcessInfer_EDDDDD.Cache, deferredCountData, prefilterHandle, mode); - } - - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public interface IBaseJobProcessComponentData_EDDDDD : IBaseJobProcessComponentData { } - - [StructLayout(LayoutKind.Sequential)] - private struct JobStruct_ProcessInfer_EDDDDD where T : struct - { - public static JobProcessComponentDataCache Cache; - - public ProcessIterationData Iterator; - public T Data; - - [DeallocateOnJobCompletion] - [NativeDisableContainerSafetyRestriction] - public NativeArray PrefilterData; - } - - [StructLayout(LayoutKind.Sequential)] - internal struct JobStruct_Process_EDDDDD - where T : struct, IJobProcessComponentDataWithEntity - where U0 : struct, IComponentData - where U1 : struct, IComponentData - where U2 : struct, IComponentData - where U3 : struct, IComponentData - where U4 : struct, IComponentData - { - public ProcessIterationData Iterator; - public T Data; - - [DeallocateOnJobCompletion] - [NativeDisableContainerSafetyRestriction] - public NativeArray PrefilterData; - - [Preserve] - public static IntPtr Initialize(JobType jobType) - { - return JobsUtility.CreateJobReflectionData(typeof(JobStruct_Process_EDDDDD), typeof(T), jobType, (ExecuteJobFunction) Execute); - } - - delegate void ExecuteJobFunction(ref JobStruct_Process_EDDDDD data, IntPtr additionalPtr, IntPtr bufferRangePatchData, ref JobRanges ranges, int jobIndex); - - public static unsafe void Execute(ref JobStruct_Process_EDDDDD jobData, IntPtr additionalPtr, IntPtr bufferRangePatchData, ref JobRanges ranges, int jobIndex) - { - var chunks = (ArchetypeChunk*)NativeArrayUnsafeUtility.GetUnsafePtr(jobData.PrefilterData); - var entityIndices = (int*) (chunks + jobData.Iterator.m_Length); - var chunkCount = *(entityIndices + jobData.Iterator.m_Length); - - if (jobData.Iterator.m_IsParallelFor) - { - int begin, end; - while (JobsUtility.GetWorkStealingRange(ref ranges, jobIndex, out begin, out end)) - ExecuteChunk(ref jobData, bufferRangePatchData, begin, end, chunks, entityIndices); - } - else - { - ExecuteChunk(ref jobData, bufferRangePatchData, 0, chunkCount, chunks, entityIndices); - } - } - - static unsafe void ExecuteChunk(ref JobStruct_Process_EDDDDD jobData, IntPtr bufferRangePatchData, int begin, int end, ArchetypeChunk* chunks, int* entityIndices) - { - var typeLookupCache0 = 0; - var typeLookupCache1 = 0; - var typeLookupCache2 = 0; - var typeLookupCache3 = 0; - var typeLookupCache4 = 0; - - for (var blockIndex = begin; blockIndex != end; ++blockIndex) - { - var chunk = chunks[blockIndex]; - int beginIndex = entityIndices[blockIndex]; - var count = chunk.Count; - #if ENABLE_UNITY_COLLECTIONS_CHECKS - JobsUtility.PatchBufferMinMaxRanges(bufferRangePatchData, UnsafeUtility.AddressOf(ref jobData), beginIndex, beginIndex + count); - #endif - var ptrE = (Entity*)UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, false, 0, jobData.Iterator.GlobalSystemVersion)); - ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex0, ref typeLookupCache0); - var ptr0 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly0 == 0, typeLookupCache0, jobData.Iterator.GlobalSystemVersion)); - ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex1, ref typeLookupCache1); - var ptr1 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly1 == 0, typeLookupCache1, jobData.Iterator.GlobalSystemVersion)); - ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex2, ref typeLookupCache2); - var ptr2 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly2 == 0, typeLookupCache2, jobData.Iterator.GlobalSystemVersion)); - ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex3, ref typeLookupCache3); - var ptr3 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly3 == 0, typeLookupCache3, jobData.Iterator.GlobalSystemVersion)); - ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex4, ref typeLookupCache4); - var ptr4 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly4 == 0, typeLookupCache4, jobData.Iterator.GlobalSystemVersion)); - - - for (var i = 0; i != count; i++) - { - jobData.Data.Execute(ptrE[i], i + beginIndex, ref UnsafeUtilityEx.ArrayElementAsRef(ptr0, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr1, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr2, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr3, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr4, i)); - } - } - } - } - - internal static unsafe JobHandle ScheduleInternal_DDDDDD(ref T jobData, ComponentSystemBase system, ComponentGroup componentGroup, int innerloopBatchCount, JobHandle dependsOn, ScheduleMode mode) - where T : struct - { - JobStruct_ProcessInfer_DDDDDD fullData; - fullData.Data = jobData; - - var isParallelFor = innerloopBatchCount != -1; - Initialize(system, componentGroup, typeof(T), typeof(JobStruct_Process_DDDDDD<,,,,,,>), isParallelFor, ref JobStruct_ProcessInfer_DDDDDD.Cache, out fullData.Iterator); - - var unfilteredChunkCount = fullData.Iterator.m_Length; - var iterator = JobStruct_ProcessInfer_DDDDDD.Cache.ComponentGroup.GetComponentChunkIterator(); - - var prefilterHandle = ComponentChunkIterator.PreparePrefilteredChunkLists(unfilteredChunkCount, iterator.m_MatchingArchetypeList, iterator.m_Filter, dependsOn, mode, out fullData.PrefilterData, out var deferredCountData); - - return Schedule(UnsafeUtility.AddressOf(ref fullData), fullData.PrefilterData, fullData.Iterator.m_Length, innerloopBatchCount, isParallelFor, iterator.RequiresFilter(), ref JobStruct_ProcessInfer_DDDDDD.Cache, deferredCountData, prefilterHandle, mode); - } - - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public interface IBaseJobProcessComponentData_DDDDDD : IBaseJobProcessComponentData { } - - [StructLayout(LayoutKind.Sequential)] - private struct JobStruct_ProcessInfer_DDDDDD where T : struct - { - public static JobProcessComponentDataCache Cache; - - public ProcessIterationData Iterator; - public T Data; - - [DeallocateOnJobCompletion] - [NativeDisableContainerSafetyRestriction] - public NativeArray PrefilterData; - } - - [StructLayout(LayoutKind.Sequential)] - internal struct JobStruct_Process_DDDDDD - where T : struct, IJobProcessComponentData - where U0 : struct, IComponentData - where U1 : struct, IComponentData - where U2 : struct, IComponentData - where U3 : struct, IComponentData - where U4 : struct, IComponentData - where U5 : struct, IComponentData - { - public ProcessIterationData Iterator; - public T Data; - - [DeallocateOnJobCompletion] - [NativeDisableContainerSafetyRestriction] - public NativeArray PrefilterData; - - [Preserve] - public static IntPtr Initialize(JobType jobType) - { - return JobsUtility.CreateJobReflectionData(typeof(JobStruct_Process_DDDDDD), typeof(T), jobType, (ExecuteJobFunction) Execute); - } - - delegate void ExecuteJobFunction(ref JobStruct_Process_DDDDDD data, IntPtr additionalPtr, IntPtr bufferRangePatchData, ref JobRanges ranges, int jobIndex); - - public static unsafe void Execute(ref JobStruct_Process_DDDDDD jobData, IntPtr additionalPtr, IntPtr bufferRangePatchData, ref JobRanges ranges, int jobIndex) - { - var chunks = (ArchetypeChunk*)NativeArrayUnsafeUtility.GetUnsafePtr(jobData.PrefilterData); - var entityIndices = (int*) (chunks + jobData.Iterator.m_Length); - var chunkCount = *(entityIndices + jobData.Iterator.m_Length); - - if (jobData.Iterator.m_IsParallelFor) - { - int begin, end; - while (JobsUtility.GetWorkStealingRange(ref ranges, jobIndex, out begin, out end)) - ExecuteChunk(ref jobData, bufferRangePatchData, begin, end, chunks, entityIndices); - } - else - { - ExecuteChunk(ref jobData, bufferRangePatchData, 0, chunkCount, chunks, entityIndices); - } - } - - static unsafe void ExecuteChunk(ref JobStruct_Process_DDDDDD jobData, IntPtr bufferRangePatchData, int begin, int end, ArchetypeChunk* chunks, int* entityIndices) - { - var typeLookupCache0 = 0; - var typeLookupCache1 = 0; - var typeLookupCache2 = 0; - var typeLookupCache3 = 0; - var typeLookupCache4 = 0; - var typeLookupCache5 = 0; - - for (var blockIndex = begin; blockIndex != end; ++blockIndex) - { - var chunk = chunks[blockIndex]; - int beginIndex = entityIndices[blockIndex]; - var count = chunk.Count; - #if ENABLE_UNITY_COLLECTIONS_CHECKS - JobsUtility.PatchBufferMinMaxRanges(bufferRangePatchData, UnsafeUtility.AddressOf(ref jobData), beginIndex, beginIndex + count); - #endif - ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex0, ref typeLookupCache0); - var ptr0 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly0 == 0, typeLookupCache0, jobData.Iterator.GlobalSystemVersion)); - ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex1, ref typeLookupCache1); - var ptr1 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly1 == 0, typeLookupCache1, jobData.Iterator.GlobalSystemVersion)); - ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex2, ref typeLookupCache2); - var ptr2 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly2 == 0, typeLookupCache2, jobData.Iterator.GlobalSystemVersion)); - ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex3, ref typeLookupCache3); - var ptr3 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly3 == 0, typeLookupCache3, jobData.Iterator.GlobalSystemVersion)); - ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex4, ref typeLookupCache4); - var ptr4 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly4 == 0, typeLookupCache4, jobData.Iterator.GlobalSystemVersion)); - ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex5, ref typeLookupCache5); - var ptr5 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly5 == 0, typeLookupCache5, jobData.Iterator.GlobalSystemVersion)); - - - for (var i = 0; i != count; i++) - { - jobData.Data.Execute(ref UnsafeUtilityEx.ArrayElementAsRef(ptr0, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr1, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr2, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr3, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr4, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr5, i)); - } - } - } - } - - internal static unsafe JobHandle ScheduleInternal_EDDDDDD(ref T jobData, ComponentSystemBase system, ComponentGroup componentGroup, int innerloopBatchCount, JobHandle dependsOn, ScheduleMode mode) - where T : struct - { - JobStruct_ProcessInfer_EDDDDDD fullData; - fullData.Data = jobData; - - var isParallelFor = innerloopBatchCount != -1; - Initialize(system, componentGroup, typeof(T), typeof(JobStruct_Process_EDDDDDD<,,,,,,>), isParallelFor, ref JobStruct_ProcessInfer_EDDDDDD.Cache, out fullData.Iterator); - - var unfilteredChunkCount = fullData.Iterator.m_Length; - var iterator = JobStruct_ProcessInfer_EDDDDDD.Cache.ComponentGroup.GetComponentChunkIterator(); - - var prefilterHandle = ComponentChunkIterator.PreparePrefilteredChunkLists(unfilteredChunkCount, iterator.m_MatchingArchetypeList, iterator.m_Filter, dependsOn, mode, out fullData.PrefilterData, out var deferredCountData); - - return Schedule(UnsafeUtility.AddressOf(ref fullData), fullData.PrefilterData, fullData.Iterator.m_Length, innerloopBatchCount, isParallelFor, iterator.RequiresFilter(), ref JobStruct_ProcessInfer_EDDDDDD.Cache, deferredCountData, prefilterHandle, mode); - } - - [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)] - public interface IBaseJobProcessComponentData_EDDDDDD : IBaseJobProcessComponentData { } - - [StructLayout(LayoutKind.Sequential)] - private struct JobStruct_ProcessInfer_EDDDDDD where T : struct - { - public static JobProcessComponentDataCache Cache; - - public ProcessIterationData Iterator; - public T Data; - - [DeallocateOnJobCompletion] - [NativeDisableContainerSafetyRestriction] - public NativeArray PrefilterData; - } - - [StructLayout(LayoutKind.Sequential)] - internal struct JobStruct_Process_EDDDDDD - where T : struct, IJobProcessComponentDataWithEntity - where U0 : struct, IComponentData - where U1 : struct, IComponentData - where U2 : struct, IComponentData - where U3 : struct, IComponentData - where U4 : struct, IComponentData - where U5 : struct, IComponentData - { - public ProcessIterationData Iterator; - public T Data; - - [DeallocateOnJobCompletion] - [NativeDisableContainerSafetyRestriction] - public NativeArray PrefilterData; - - [Preserve] - public static IntPtr Initialize(JobType jobType) - { - return JobsUtility.CreateJobReflectionData(typeof(JobStruct_Process_EDDDDDD), typeof(T), jobType, (ExecuteJobFunction) Execute); - } - - delegate void ExecuteJobFunction(ref JobStruct_Process_EDDDDDD data, IntPtr additionalPtr, IntPtr bufferRangePatchData, ref JobRanges ranges, int jobIndex); - - public static unsafe void Execute(ref JobStruct_Process_EDDDDDD jobData, IntPtr additionalPtr, IntPtr bufferRangePatchData, ref JobRanges ranges, int jobIndex) - { - var chunks = (ArchetypeChunk*)NativeArrayUnsafeUtility.GetUnsafePtr(jobData.PrefilterData); - var entityIndices = (int*) (chunks + jobData.Iterator.m_Length); - var chunkCount = *(entityIndices + jobData.Iterator.m_Length); - - if (jobData.Iterator.m_IsParallelFor) - { - int begin, end; - while (JobsUtility.GetWorkStealingRange(ref ranges, jobIndex, out begin, out end)) - ExecuteChunk(ref jobData, bufferRangePatchData, begin, end, chunks, entityIndices); - } - else - { - ExecuteChunk(ref jobData, bufferRangePatchData, 0, chunkCount, chunks, entityIndices); - } - } - - static unsafe void ExecuteChunk(ref JobStruct_Process_EDDDDDD jobData, IntPtr bufferRangePatchData, int begin, int end, ArchetypeChunk* chunks, int* entityIndices) - { - var typeLookupCache0 = 0; - var typeLookupCache1 = 0; - var typeLookupCache2 = 0; - var typeLookupCache3 = 0; - var typeLookupCache4 = 0; - var typeLookupCache5 = 0; - - for (var blockIndex = begin; blockIndex != end; ++blockIndex) - { - var chunk = chunks[blockIndex]; - int beginIndex = entityIndices[blockIndex]; - var count = chunk.Count; - #if ENABLE_UNITY_COLLECTIONS_CHECKS - JobsUtility.PatchBufferMinMaxRanges(bufferRangePatchData, UnsafeUtility.AddressOf(ref jobData), beginIndex, beginIndex + count); - #endif - var ptrE = (Entity*)UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, false, 0, jobData.Iterator.GlobalSystemVersion)); - ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex0, ref typeLookupCache0); - var ptr0 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly0 == 0, typeLookupCache0, jobData.Iterator.GlobalSystemVersion)); - ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex1, ref typeLookupCache1); - var ptr1 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly1 == 0, typeLookupCache1, jobData.Iterator.GlobalSystemVersion)); - ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex2, ref typeLookupCache2); - var ptr2 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly2 == 0, typeLookupCache2, jobData.Iterator.GlobalSystemVersion)); - ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex3, ref typeLookupCache3); - var ptr3 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly3 == 0, typeLookupCache3, jobData.Iterator.GlobalSystemVersion)); - ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex4, ref typeLookupCache4); - var ptr4 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly4 == 0, typeLookupCache4, jobData.Iterator.GlobalSystemVersion)); - ChunkDataUtility.GetIndexInTypeArray(chunk.m_Chunk->Archetype, jobData.Iterator.TypeIndex5, ref typeLookupCache5); - var ptr5 = UnsafeUtilityEx.RestrictNoAlias(ComponentChunkIterator.GetChunkComponentDataPtr(chunk.m_Chunk, jobData.Iterator.IsReadOnly5 == 0, typeLookupCache5, jobData.Iterator.GlobalSystemVersion)); - - - for (var i = 0; i != count; i++) - { - jobData.Data.Execute(ptrE[i], i + beginIndex, ref UnsafeUtilityEx.ArrayElementAsRef(ptr0, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr1, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr2, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr3, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr4, i), ref UnsafeUtilityEx.ArrayElementAsRef(ptr5, i)); - } - } - } - } - - } -} - -#endif diff --git a/Unity.Entities/IJobProcessComponentData.generated.cs.meta b/Unity.Entities/IJobProcessComponentData.generated.cs.meta deleted file mode 100644 index be44c35c..00000000 --- a/Unity.Entities/IJobProcessComponentData.generated.cs.meta +++ /dev/null @@ -1,11 +0,0 @@ -fileFormatVersion: 2 -guid: a0728d3be945c4afb91aa33feaef01a8 -MonoImporter: - externalObjects: {} - serializedVersion: 2 - defaultReferences: [] - executionOrder: 0 - icon: {instanceID: 0} - userData: - assetBundleName: - assetBundleVariant: diff --git a/Unity.Entities/Injection/ComponentSystemInjection.cs b/Unity.Entities/Injection/ComponentSystemInjection.cs deleted file mode 100644 index 5f5c60a1..00000000 --- a/Unity.Entities/Injection/ComponentSystemInjection.cs +++ /dev/null @@ -1,115 +0,0 @@ -#if !UNITY_ZEROPLAYER -using System; -using System.Collections.Generic; -using System.Reflection; - -#pragma warning disable 618 - -namespace Unity.Entities -{ - internal static class ComponentSystemInjection - { - public static string GetFieldString(FieldInfo info) - { - return $"{info.DeclaringType.Name}.{info.Name}"; - } - - public static void Inject(ComponentSystemBase componentSystem, World world, EntityManager entityManager, - out InjectComponentGroupData[] outInjectGroups, out InjectFromEntityData outInjectFromEntityData) - { - var componentSystemType = componentSystem.GetType(); - - ValidateNoStaticInjectDependencies(componentSystemType); - - InjectFields(componentSystem, world, entityManager, out outInjectGroups, out outInjectFromEntityData); - } - - static void InjectFields(ComponentSystemBase componentSystem, World world, EntityManager entityManager, - out InjectComponentGroupData[] outInjectGroups, out InjectFromEntityData outInjectFromEntityData) - { - var componentSystemType = componentSystem.GetType(); - var fields = componentSystemType.GetFields(BindingFlags.Instance | BindingFlags.NonPublic | BindingFlags.Public); - var injectGroups = new List(); - - var injectFromEntity = new List(); - var injectFromFixedArray = new List(); - - foreach (var field in fields) - { - var attr = field.GetCustomAttributes(typeof(InjectAttribute), true); - if (attr.Length == 0) - continue; - - if (field.FieldType.IsClass) - { - InjectConstructorDependencies(componentSystem, world, field); - } - else - { - if (InjectFromEntityData.SupportsInjections(field)) - InjectFromEntityData.CreateInjection(field, entityManager, injectFromEntity, - injectFromFixedArray); - else - injectGroups.Add( - InjectComponentGroupData.CreateInjection(field.FieldType, field, componentSystem)); - } - } - - - outInjectGroups = injectGroups.ToArray(); - - outInjectFromEntityData = new InjectFromEntityData(injectFromEntity.ToArray(), injectFromFixedArray.ToArray()); - } - - private static void ValidateNoStaticInjectDependencies(Type type) - { -#if UNITY_EDITOR - var fields = type.GetFields(BindingFlags.Static | BindingFlags.FlattenHierarchy | BindingFlags.NonPublic); - - foreach (var field in fields) - if (field.GetCustomAttributes(typeof(InjectAttribute), true).Length != 0) - throw new ArgumentException( - $"[Inject] may not be used on static variables: {GetFieldString(field)}"); -#endif - } - - private static void InjectConstructorDependencies(ScriptBehaviourManager manager, World world, FieldInfo field) - { - if (field.FieldType.IsSubclassOf(typeof(ScriptBehaviourManager))) - field.SetValue(manager, world.GetOrCreateManager(field.FieldType)); - else - ThrowUnsupportedInjectException(field); - } - - public static void ThrowUnsupportedInjectException(FieldInfo field) - { - throw new ArgumentException( - $"[Inject] is not supported for type '{field.FieldType}'. At: {GetFieldString(field)}"); - } - - internal static T[] GetAllInjectedManagers(ScriptBehaviourManager host, World world) - where T : ScriptBehaviourManager - { - var result = new List(); - var fields = host.GetType().GetFields(BindingFlags.Instance | BindingFlags.NonPublic | BindingFlags.Public); - - foreach (var field in fields) - { - var attr = field.GetCustomAttributes(typeof(InjectAttribute), true); - if (attr.Length == 0) - continue; - - if (!field.FieldType.IsClass) - continue; - - if (!field.FieldType.IsSubclassOf(typeof(T))) - continue; - - result.Add((T) world.GetOrCreateManager(field.FieldType)); - } - - return result.ToArray(); - } - } -} -#endif \ No newline at end of file diff --git a/Unity.Entities/Injection/ComponentSystemInjection.cs.meta b/Unity.Entities/Injection/ComponentSystemInjection.cs.meta deleted file mode 100644 index 365a26b7..00000000 --- a/Unity.Entities/Injection/ComponentSystemInjection.cs.meta +++ /dev/null @@ -1,3 +0,0 @@ -fileFormatVersion: 2 -guid: 15f0c9e8418343e09f1835b588049d8b -timeCreated: 1511049156 \ No newline at end of file diff --git a/Unity.Entities/Injection/InjectAttribute.cs b/Unity.Entities/Injection/InjectAttribute.cs deleted file mode 100644 index f92055ff..00000000 --- a/Unity.Entities/Injection/InjectAttribute.cs +++ /dev/null @@ -1,10 +0,0 @@ -using System; - -namespace Unity.Entities -{ - [AttributeUsage(AttributeTargets.Field)] - [Obsolete("Injection API is deprecated. Use ComponentGroup API instead.")] - public class InjectAttribute : Attribute - { - } -} diff --git a/Unity.Entities/Injection/InjectAttribute.cs.meta b/Unity.Entities/Injection/InjectAttribute.cs.meta deleted file mode 100644 index 35f0db95..00000000 --- a/Unity.Entities/Injection/InjectAttribute.cs.meta +++ /dev/null @@ -1,3 +0,0 @@ -fileFormatVersion: 2 -guid: 0cd570c938a441eb828ce5faca7c5418 -timeCreated: 1512064623 \ No newline at end of file diff --git a/Unity.Entities/Injection/InjectComponentFromEntity.cs b/Unity.Entities/Injection/InjectComponentFromEntity.cs deleted file mode 100644 index ad9b3d83..00000000 --- a/Unity.Entities/Injection/InjectComponentFromEntity.cs +++ /dev/null @@ -1,87 +0,0 @@ -#if !UNITY_ZEROPLAYER -using System.Collections.Generic; -using System.Reflection; -using Unity.Collections; -using Unity.Collections.LowLevel.Unsafe; - -namespace Unity.Entities -{ - internal struct InjectFromEntityData - { - private readonly InjectionData[] m_InjectComponentDataFromEntity; - private readonly InjectionData[] m_InjectBufferFromEntity; - - public InjectFromEntityData(InjectionData[] componentDataFromEntity, InjectionData[] bufferFromEntity) - { - m_InjectComponentDataFromEntity = componentDataFromEntity; - m_InjectBufferFromEntity = bufferFromEntity; - } - - public static bool SupportsInjections(FieldInfo field) - { - if (field.FieldType.IsGenericType && - field.FieldType.GetGenericTypeDefinition() == typeof(ComponentDataFromEntity<>)) - return true; - if (field.FieldType.IsGenericType && - field.FieldType.GetGenericTypeDefinition() == typeof(BufferFromEntity<>)) - return true; - return false; - } - - public static void CreateInjection(FieldInfo field, EntityManager entityManager, - List componentDataFromEntity, List bufferFromEntity) - { - var isReadOnly = field.GetCustomAttributes(typeof(ReadOnlyAttribute), true).Length != 0; - - if (field.FieldType.IsGenericType && - field.FieldType.GetGenericTypeDefinition() == typeof(ComponentDataFromEntity<>)) - { - var injection = new InjectionData(field, field.FieldType.GetGenericArguments()[0], isReadOnly); - componentDataFromEntity.Add(injection); - } - else if (field.FieldType.IsGenericType && - field.FieldType.GetGenericTypeDefinition() == typeof(BufferFromEntity<>)) - { - var injection = new InjectionData(field, field.FieldType.GetGenericArguments()[0], isReadOnly); - bufferFromEntity.Add(injection); - } - else - { - ComponentSystemInjection.ThrowUnsupportedInjectException(field); - } - } - - public unsafe void UpdateInjection(byte* pinnedSystemPtr, EntityManager entityManager) - { - for (var i = 0; i != m_InjectComponentDataFromEntity.Length; i++) - { - var array = entityManager.GetComponentDataFromEntity( - m_InjectComponentDataFromEntity[i].ComponentType.TypeIndex, - m_InjectComponentDataFromEntity[i].IsReadOnly); - UnsafeUtility.CopyStructureToPtr(ref array, - pinnedSystemPtr + m_InjectComponentDataFromEntity[i].FieldOffset); - } - - for (var i = 0; i != m_InjectBufferFromEntity.Length; i++) - { - var array = entityManager.GetBufferFromEntity( - m_InjectBufferFromEntity[i].ComponentType.TypeIndex, - m_InjectBufferFromEntity[i].IsReadOnly); - UnsafeUtility.CopyStructureToPtr(ref array, - pinnedSystemPtr + m_InjectBufferFromEntity[i].FieldOffset); - } - } - - public void ExtractJobDependencyTypes(ComponentSystemBase system) - { - if (m_InjectComponentDataFromEntity != null) - foreach (var injection in m_InjectComponentDataFromEntity) - system.AddReaderWriter(injection.ComponentType); - - if (m_InjectBufferFromEntity != null) - foreach (var injection in m_InjectBufferFromEntity) - system.AddReaderWriter(injection.ComponentType); - } - } -} -#endif \ No newline at end of file diff --git a/Unity.Entities/Injection/InjectComponentFromEntity.cs.meta b/Unity.Entities/Injection/InjectComponentFromEntity.cs.meta deleted file mode 100644 index 3676195f..00000000 --- a/Unity.Entities/Injection/InjectComponentFromEntity.cs.meta +++ /dev/null @@ -1,3 +0,0 @@ -fileFormatVersion: 2 -guid: 1be2212180ea46758457f3d63f4aabf5 -timeCreated: 1511048217 \ No newline at end of file diff --git a/Unity.Entities/Injection/InjectComponentGroup.cs b/Unity.Entities/Injection/InjectComponentGroup.cs deleted file mode 100644 index c753ef56..00000000 --- a/Unity.Entities/Injection/InjectComponentGroup.cs +++ /dev/null @@ -1,264 +0,0 @@ -#if !UNITY_ZEROPLAYER - -using System; -using System.Collections.Generic; -using System.Linq; -using System.Reflection; -using Unity.Collections; -using Unity.Collections.LowLevel.Unsafe; - -// Injection marked Obsolete -#pragma warning disable 618 - -namespace Unity.Entities -{ - internal struct ProxyComponentData : IComponentData - { - private byte m_Internal; - } - - internal struct ProxyBufferElementData : IBufferElementData - { - private byte m_Internal; - } - - internal struct ProxySharedComponentData : ISharedComponentData - { - private byte m_Internal; - } - - internal class InjectComponentGroupData - { - private readonly InjectionData[] m_ComponentDataInjections; - - private readonly int m_EntityArrayOffset; - private readonly InjectionData[] m_BufferArrayInjections; - private readonly int m_GroupFieldOffset; - - private readonly InjectionContext m_InjectionContext; - private readonly int m_LengthOffset; - private readonly InjectionData[] m_SharedComponentInjections; - private readonly ComponentGroup m_EntityGroup; - - private readonly int m_ComponentGroupIndex; - - private unsafe InjectComponentGroupData(ComponentSystemBase system, FieldInfo groupField, - InjectionData[] componentDataInjections, InjectionData[] bufferArrayInjections, - InjectionData[] sharedComponentInjections, - FieldInfo entityArrayInjection, FieldInfo indexFromEntityInjection, InjectionContext injectionContext, - FieldInfo lengthInjection, FieldInfo componentGroupIndexField, ComponentType[] componentRequirements) - { - m_EntityGroup = system.GetComponentGroupInternal(componentRequirements); - - m_ComponentGroupIndex = Array.IndexOf(system.ComponentGroups, m_EntityGroup); - - m_ComponentDataInjections = componentDataInjections; - m_BufferArrayInjections = bufferArrayInjections; - m_SharedComponentInjections = sharedComponentInjections; - m_InjectionContext = injectionContext; - - PatchGetIndexInComponentGroup(m_ComponentDataInjections); - PatchGetIndexInComponentGroup(m_BufferArrayInjections); - PatchGetIndexInComponentGroup(m_SharedComponentInjections); - - injectionContext.PrepareEntries(m_EntityGroup); - - if (entityArrayInjection != null) - m_EntityArrayOffset = UnsafeUtility.GetFieldOffset(entityArrayInjection); - else - m_EntityArrayOffset = -1; - - if (lengthInjection != null) - m_LengthOffset = UnsafeUtility.GetFieldOffset(lengthInjection); - else - m_LengthOffset = -1; - - m_GroupFieldOffset = UnsafeUtility.GetFieldOffset(groupField); - - if (componentGroupIndexField != null) - { - ulong gchandle; - var pinnedSystemPtr = (byte*)UnsafeUtility.PinGCObjectAndGetAddress(system, out gchandle); - var groupIndexPtr = pinnedSystemPtr + m_GroupFieldOffset + UnsafeUtility.GetFieldOffset(componentGroupIndexField); - - int groupIndex = m_ComponentGroupIndex; - UnsafeUtility.CopyStructureToPtr(ref groupIndex, groupIndexPtr); - - UnsafeUtility.ReleaseGCObject(gchandle); - } - } - - private void PatchGetIndexInComponentGroup(InjectionData[] componentInjections) - { - for (var i = 0; i != componentInjections.Length; i++) - componentInjections[i].IndexInComponentGroup = - m_EntityGroup.GetIndexInComponentGroup(componentInjections[i].ComponentType.TypeIndex); - } - - public unsafe void UpdateInjection(byte* systemPtr) - { - var groupStructPtr = systemPtr + m_GroupFieldOffset; - - int length = m_EntityGroup.CalculateLength(); - ComponentChunkIterator iterator = m_EntityGroup.GetComponentChunkIterator(); - - for (var i = 0; i != m_ComponentDataInjections.Length; i++) - { - ComponentDataArray data; - m_EntityGroup.GetComponentDataArray(ref iterator, m_ComponentDataInjections[i].IndexInComponentGroup, - length, out data); - UnsafeUtility.CopyStructureToPtr(ref data, groupStructPtr + m_ComponentDataInjections[i].FieldOffset); - } - - for (var i = 0; i != m_SharedComponentInjections.Length; i++) - { - SharedComponentDataArray data; - m_EntityGroup.GetSharedComponentDataArray(ref iterator, - m_SharedComponentInjections[i].IndexInComponentGroup, length, out data); - UnsafeUtility.CopyStructureToPtr(ref data, groupStructPtr + m_SharedComponentInjections[i].FieldOffset); - } - - for (var i = 0; i != m_BufferArrayInjections.Length; i++) - { - BufferArray data; - m_EntityGroup.GetBufferArray(ref iterator, m_BufferArrayInjections[i].IndexInComponentGroup, length, - out data); - UnsafeUtility.CopyStructureToPtr(ref data, groupStructPtr + m_BufferArrayInjections[i].FieldOffset); - } - - if (m_EntityArrayOffset != -1) - { - EntityArray entityArray = m_EntityGroup.GetEntityArray(); - UnsafeUtility.CopyStructureToPtr(ref entityArray, groupStructPtr + m_EntityArrayOffset); - } - - if (m_InjectionContext.HasEntries) - m_InjectionContext.UpdateEntries(m_EntityGroup, ref iterator, length, groupStructPtr); - - if (m_LengthOffset != -1) UnsafeUtility.CopyStructureToPtr(ref length, groupStructPtr + m_LengthOffset); - } - - public static InjectComponentGroupData CreateInjection(Type injectedGroupType, FieldInfo groupField, - ComponentSystemBase system) - { - FieldInfo entityArrayField; - FieldInfo indexFromEntityField; - FieldInfo lengthField; - FieldInfo componentGroupIndexField; - - var injectionContext = new InjectionContext(); - var componentDataInjections = new List(); - var bufferDataInjections = new List(); - var sharedComponentInjections = new List(); - - var componentRequirements = new HashSet(); - var error = CollectInjectedGroup(system, groupField, injectedGroupType, out entityArrayField, - out indexFromEntityField, injectionContext, out lengthField, out componentGroupIndexField, - componentRequirements, componentDataInjections, bufferDataInjections, sharedComponentInjections); - if (error != null) - throw new ArgumentException(error); - - return new InjectComponentGroupData(system, groupField, componentDataInjections.ToArray(), - bufferDataInjections.ToArray(), sharedComponentInjections.ToArray(), entityArrayField, - indexFromEntityField, injectionContext, lengthField, componentGroupIndexField, - componentRequirements.ToArray()); - } - - private static string CollectInjectedGroup(ComponentSystemBase system, FieldInfo groupField, - Type injectedGroupType, out FieldInfo entityArrayField, out FieldInfo indexFromEntityField, - InjectionContext injectionContext, out FieldInfo lengthField, out FieldInfo componentGroupIndexField, - ISet componentRequirements, ICollection componentDataInjections, - ICollection bufferDataInjections, ICollection sharedComponentInjections) - { - //@TODO: Improve error messages... - var fields = - injectedGroupType.GetFields(BindingFlags.Instance | BindingFlags.NonPublic | BindingFlags.Public); - entityArrayField = null; - indexFromEntityField = null; - lengthField = null; - componentGroupIndexField = null; - - foreach (var field in fields) - { - var isReadOnly = field.GetCustomAttributes(typeof(ReadOnlyAttribute), true).Length != 0; - //@TODO: Prevent using GameObjectEntity, it will never show up. Point to GameObjectArray instead... - - if (field.FieldType.IsGenericType && - field.FieldType.GetGenericTypeDefinition() == typeof(ComponentDataArray<>)) - { - var injection = new InjectionData(field, field.FieldType.GetGenericArguments()[0], isReadOnly); - componentDataInjections.Add(injection); - componentRequirements.Add(injection.ComponentType); - } - else if (field.FieldType.IsGenericType && - field.FieldType.GetGenericTypeDefinition() == typeof(ExcludeComponent<>)) - { - componentRequirements.Add(ComponentType.Exclude(field.FieldType.GetGenericArguments()[0])); - } - else if (field.FieldType.IsGenericType && - field.FieldType.GetGenericTypeDefinition() == typeof(BufferArray<>)) - { - var injection = new InjectionData(field, field.FieldType.GetGenericArguments()[0], isReadOnly); - - bufferDataInjections.Add(injection); - componentRequirements.Add(injection.ComponentType); - } - else if (field.FieldType.IsGenericType && - field.FieldType.GetGenericTypeDefinition() == typeof(SharedComponentDataArray<>)) - { - if (!isReadOnly) - return - $"{system.GetType().Name}:{groupField.Name} SharedComponentDataArray<> must always be injected as [ReadOnly]"; - var injection = new InjectionData(field, field.FieldType.GetGenericArguments()[0], true); - - sharedComponentInjections.Add(injection); - componentRequirements.Add(injection.ComponentType); - } - else if (field.FieldType == typeof(EntityArray)) - { - // Error on multiple EntityArray - if (entityArrayField != null) - return - $"{system.GetType().Name}:{groupField.Name} An [Inject] struct, may only contain a single EntityArray"; - - entityArrayField = field; - } - else if (field.FieldType == typeof(int)) - { - if (field.Name != "Length" && field.Name != "GroupIndex") - return - $"{system.GetType().Name}:{groupField.Name} Int in an [Inject] struct should be named \"Length\" (group length) or \"GroupIndex\" (index in ComponentGroup[])"; - - if (!field.IsInitOnly) - return - $"{system.GetType().Name}:{groupField.Name} {field.Name} must use the \"readonly\" keyword"; - - if(field.Name == "Length") - lengthField = field; - - if (field.Name == "GroupIndex") - componentGroupIndexField = field; - } - else - { - var hook = InjectionHookSupport.HookFor(field); - if (hook == null) - return - $"{system.GetType().Name}:{groupField.Name} [Inject] may only be used on ComponentDataArray<>, ComponentArray<>, TransformAccessArray, EntityArray, {string.Join(",", InjectionHookSupport.Hooks.Select(h => h.FieldTypeOfInterest.Name))} and int Length."; - - var error = hook.ValidateField(field, isReadOnly, injectionContext); - if (error != null) return error; - - injectionContext.AddEntry(hook.CreateInjectionInfoFor(field, isReadOnly)); - } - } - - if (injectionContext.HasComponentRequirements) - foreach (var requirement in injectionContext.ComponentRequirements) - componentRequirements.Add(requirement); - - return null; - } - } -} -#endif \ No newline at end of file diff --git a/Unity.Entities/Injection/InjectComponentGroup.cs.meta b/Unity.Entities/Injection/InjectComponentGroup.cs.meta deleted file mode 100644 index beb0080a..00000000 --- a/Unity.Entities/Injection/InjectComponentGroup.cs.meta +++ /dev/null @@ -1,12 +0,0 @@ -fileFormatVersion: 2 -guid: e808dc4ed0eb648f1a15659b3a73fc32 -timeCreated: 1491948448 -licenseType: Pro -MonoImporter: - serializedVersion: 2 - defaultReferences: [] - executionOrder: 0 - icon: {instanceID: 0} - userData: - assetBundleName: - assetBundleVariant: diff --git a/Unity.Entities/Injection/InjectionData.cs b/Unity.Entities/Injection/InjectionData.cs deleted file mode 100644 index cb2462b8..00000000 --- a/Unity.Entities/Injection/InjectionData.cs +++ /dev/null @@ -1,26 +0,0 @@ -#if !UNITY_ZEROPLAYER -using System; -using System.Reflection; -using Unity.Collections.LowLevel.Unsafe; - -namespace Unity.Entities -{ - internal struct InjectionData - { - public ComponentType ComponentType; - public int IndexInComponentGroup; - public readonly bool IsReadOnly; - public readonly int FieldOffset; - - public InjectionData(FieldInfo field, Type genericType, bool isReadOnly) - { - IndexInComponentGroup = -1; - FieldOffset = UnsafeUtility.GetFieldOffset(field); - - var accessMode = isReadOnly ? ComponentType.AccessMode.ReadOnly : ComponentType.AccessMode.ReadWrite; - ComponentType = new ComponentType(genericType, accessMode); - IsReadOnly = isReadOnly; - } - } -} -#endif \ No newline at end of file diff --git a/Unity.Entities/Injection/InjectionData.cs.meta b/Unity.Entities/Injection/InjectionData.cs.meta deleted file mode 100644 index d1d20a95..00000000 --- a/Unity.Entities/Injection/InjectionData.cs.meta +++ /dev/null @@ -1,3 +0,0 @@ -fileFormatVersion: 2 -guid: 8f94d591047f4a7cae2b25c92134da90 -timeCreated: 1511048294 \ No newline at end of file diff --git a/Unity.Entities/Injection/InjectionHook.cs b/Unity.Entities/Injection/InjectionHook.cs deleted file mode 100644 index 5d39ba2a..00000000 --- a/Unity.Entities/Injection/InjectionHook.cs +++ /dev/null @@ -1,134 +0,0 @@ -#if !UNITY_ZEROPLAYER -using System; -using System.Collections.Generic; -using System.Reflection; - -namespace Unity.Entities -{ - internal sealed class CustomInjectionHookAttribute : Attribute - { - } - - public sealed class InjectionContext - { - private readonly List m_Entries = new List(); - - public bool HasComponentRequirements { get; private set; } - - public bool HasEntries => m_Entries.Count != 0; - - public IReadOnlyCollection Entries => m_Entries; - - public IEnumerable ComponentRequirements - { - get - { - foreach (var info in m_Entries) - foreach (var requirement in info.ComponentRequirements) - yield return requirement; - } - } - - internal void AddEntry(Entry entry) - { - HasComponentRequirements = HasComponentRequirements || entry.ComponentRequirements.Length > 0; - m_Entries.Add(entry); - } - - public void PrepareEntries(ComponentGroup entityGroup) - { - if (!HasEntries) - return; - - for (var index = 0; index < m_Entries.Count; index++) - { - var entry = m_Entries[index]; - entry.Hook.PrepareEntry(ref entry, entityGroup); - m_Entries[index] = entry; - } - } - - internal unsafe void UpdateEntries(ComponentGroup entityGroup, ref ComponentChunkIterator iterator, int length, - byte* groupStructPtr) - { - if (!HasEntries) - return; - - foreach (var info in m_Entries) - info.Hook.InjectEntry(info, entityGroup, ref iterator, length, groupStructPtr); - } - - public struct Entry - { - public int FieldOffset; - public FieldInfo FieldInfo; - public Type[] ComponentRequirements; - public InjectionHook Hook; - public ComponentType.AccessMode AccessMode; - public int IndexInComponentGroup; - public bool IsReadOnly; - public ComponentType ComponentType; - } - } - - public abstract unsafe class InjectionHook - { - public abstract Type FieldTypeOfInterest { get; } - public abstract bool IsInterestedInField(FieldInfo fieldInfo); - public abstract InjectionContext.Entry CreateInjectionInfoFor(FieldInfo field, bool isReadOnly); - - internal abstract void InjectEntry(InjectionContext.Entry entry, ComponentGroup entityGroup, - ref ComponentChunkIterator iterator, int length, byte* groupStructPtr); - - public abstract string ValidateField(FieldInfo field, bool isReadOnly, InjectionContext injectionInfo); - - public virtual void PrepareEntry(ref InjectionContext.Entry entry, ComponentGroup entityGroup) - { - } - } - - public static class InjectionHookSupport - { - private static bool s_HasHooks; - private static readonly List k_Hooks = new List(); - - internal static IReadOnlyCollection Hooks => k_Hooks; - - public static void RegisterHook(InjectionHook hook) - { - s_HasHooks = true; - k_Hooks.Add(hook); - } - - public static void UnregisterHook(InjectionHook hook) - { - k_Hooks.Remove(hook); - s_HasHooks = k_Hooks.Count != 0; - } - - internal static InjectionHook HookFor(FieldInfo fieldInfo) - { - if (!s_HasHooks) - return null; - - // TODO: in case of multiple hooks interested in a single field type, we should drop an error (in Editor) - foreach (var hook in k_Hooks) - if (hook.IsInterestedInField(fieldInfo)) - return hook; - - return null; - } - - public static bool IsValidHook(Type type) - { - if (type.IsAbstract) - return false; - if (type.ContainsGenericParameters) - return false; - if (!typeof(InjectionHook).IsAssignableFrom(type)) - return false; - return type.GetCustomAttributes(typeof(CustomInjectionHookAttribute), true).Length != 0; - } - } -} -#endif diff --git a/Unity.Entities/Injection/InjectionHook.cs.meta b/Unity.Entities/Injection/InjectionHook.cs.meta deleted file mode 100644 index a0bbf3a1..00000000 --- a/Unity.Entities/Injection/InjectionHook.cs.meta +++ /dev/null @@ -1,11 +0,0 @@ -fileFormatVersion: 2 -guid: ebb4176a338ea4d3aaed1f846062468e -MonoImporter: - externalObjects: {} - serializedVersion: 2 - defaultReferences: [] - executionOrder: 0 - icon: {instanceID: 0} - userData: - assetBundleName: - assetBundleVariant: diff --git a/Unity.Entities/Injection/World.cs b/Unity.Entities/Injection/World.cs deleted file mode 100644 index af3cbfac..00000000 --- a/Unity.Entities/Injection/World.cs +++ /dev/null @@ -1,543 +0,0 @@ -using System; -using System.Collections.Generic; -using System.Collections.ObjectModel; -using System.Reflection; - -namespace Unity.Entities -{ -#if !UNITY_ZEROPLAYER - public class World : IDisposable - { - static readonly List allWorlds = new List(); -#if ENABLE_UNITY_COLLECTIONS_CHECKS - bool m_AllowGetManager = true; -#endif - - //@TODO: What about multiple managers of the same type... - Dictionary m_BehaviourManagerLookup = - new Dictionary(); - - List m_BehaviourManagers = new List(); - static int ms_SystemIDAllocator = 0; - - public World(string name) - { - // Debug.LogError("Create World "+ name + " - " + GetHashCode()); - Name = name; - allWorlds.Add(this); - } - - public IEnumerable BehaviourManagers => - new ReadOnlyCollection(m_BehaviourManagers); - - public string Name { get; } - - public override string ToString() - { - return Name; - } - - public int Version { get; private set; } - - public static World Active { get; set; } - - public EntityManager EntityManager - { - get { return GetOrCreateManager(); } - } - - public static ReadOnlyCollection AllWorlds => new ReadOnlyCollection(allWorlds); - - public bool IsCreated => m_BehaviourManagers != null; - - public void Dispose() - { - if (!IsCreated) - throw new ArgumentException("The World has already been Disposed."); - // Debug.LogError("Dispose World "+ Name + " - " + GetHashCode()); - - if (allWorlds.Contains(this)) - allWorlds.Remove(this); - -#if ENABLE_UNITY_COLLECTIONS_CHECKS - m_AllowGetManager = false; -#endif - // Destruction should happen in reverse order to construction - ScriptBehaviourManager em = null; - for (int i = m_BehaviourManagers.Count - 1; i >= 0; --i) - { - var mgr = m_BehaviourManagers[i]; - if (mgr is EntityManager) - { - em = mgr; - continue; - } - try - { - mgr.DestroyInstance(); - } - catch (Exception e) - { - Debug.LogException(e); - } - } - - // Destroy EntityManager last - if (em != null) - { - try - { - em.DestroyInstance(); - } - catch (Exception e) - { - Debug.LogException(e); - } - } - - if (Active == this) - Active = null; - - m_BehaviourManagers.Clear(); - m_BehaviourManagerLookup.Clear(); - - m_BehaviourManagers = null; - m_BehaviourManagerLookup = null; - } - - public static void DisposeAllWorlds() - { - while (allWorlds.Count != 0) - allWorlds[0].Dispose(); - } - - ScriptBehaviourManager CreateManagerInternal(Type type, object[] constructorArguments) - { - if (!typeof(ScriptBehaviourManager).IsAssignableFrom(type)) - { - throw new ArgumentException($"Type {type} must be derived from ScriptBehaviourManager."); - } - -#if ENABLE_UNITY_COLLECTIONS_CHECKS - - if (constructorArguments != null && constructorArguments.Length != 0) - { - var constructors = - type.GetConstructors(BindingFlags.Instance | BindingFlags.NonPublic | BindingFlags.Public); - if (constructors.Length == 1 && constructors[0].IsPrivate) - throw new MissingMethodException( - $"Constructing {type} failed because the constructor was private, it must be public."); - } - - m_AllowGetManager = false; -#endif - ScriptBehaviourManager manager; - try - { - manager = Activator.CreateInstance(type, constructorArguments) as ScriptBehaviourManager; - } - catch (MissingMethodException) - { - Debug.LogError($"System/Manager {type} must be mentioned in a link.xml file, or annotated " + - "with a [Preserve] attribute to prevent its constructor from being stripped. " + - "See https://docs.unity3d.com/Manual/ManagedCodeStripping.html for more information."); - throw; - } - finally - { -#if ENABLE_UNITY_COLLECTIONS_CHECKS - m_AllowGetManager = true; -#endif - } - - return AddManager(manager); - } - - ScriptBehaviourManager GetExistingManagerInternal(Type type) - { - ScriptBehaviourManager manager; - if (m_BehaviourManagerLookup.TryGetValue(type, out manager)) - return manager; - - return null; - } - - ScriptBehaviourManager GetOrCreateManagerInternal(Type type) - { - var manager = GetExistingManagerInternal(type); - - return manager ?? CreateManagerInternal(type, null); - } - - void AddTypeLookup(Type type, ScriptBehaviourManager manager) - { - while (type != typeof(ScriptBehaviourManager)) - { - if (!m_BehaviourManagerLookup.ContainsKey(type)) - m_BehaviourManagerLookup.Add(type, manager); - - type = type.BaseType; - } - } - - void RemoveManagerInternal(ScriptBehaviourManager manager) - { - if (!m_BehaviourManagers.Remove(manager)) - throw new ArgumentException($"manager does not exist in the world"); - ++Version; - - var type = manager.GetType(); - while (type != typeof(ScriptBehaviourManager)) - { - if (m_BehaviourManagerLookup[type] == manager) - { - m_BehaviourManagerLookup.Remove(type); - - foreach (var otherManager in m_BehaviourManagers) - if (otherManager.GetType().IsSubclassOf(type)) - AddTypeLookup(otherManager.GetType(), otherManager); - } - - type = type.BaseType; - } - } - - void CheckGetOrCreateManager() - { -#if ENABLE_UNITY_COLLECTIONS_CHECKS - if (!IsCreated) - throw new ArgumentException("The World has already been Disposed."); - if (!m_AllowGetManager) - throw new ArgumentException( - "During destruction and constructor of a system you are not allowed to get or create more systems."); -#endif - } - - void CheckCreated() - { -#if ENABLE_UNITY_COLLECTIONS_CHECKS - if (!IsCreated) - throw new ArgumentException("The World has already been Disposed."); -#endif - } - - public ScriptBehaviourManager CreateManager(Type type, params object[] constructorArguments) - { - CheckGetOrCreateManager(); - - return CreateManagerInternal(type, constructorArguments); - } - - public T CreateManager(params object[] constructorArguments) where T : ScriptBehaviourManager - { - CheckGetOrCreateManager(); - - return (T) CreateManagerInternal(typeof(T), constructorArguments); - } - - public T GetOrCreateManager() where T : ScriptBehaviourManager - { - CheckGetOrCreateManager(); - - return (T) GetOrCreateManagerInternal(typeof(T)); - } - - public ScriptBehaviourManager GetOrCreateManager(Type type) - { - CheckGetOrCreateManager(); - - return GetOrCreateManagerInternal(type); - } - - public T AddManager(T manager) where T : ScriptBehaviourManager - { - CheckGetOrCreateManager(); - - m_BehaviourManagers.Add(manager); - AddTypeLookup(manager.GetType(), manager); - - try - { - manager.CreateInstance(this); - } - catch - { - RemoveManagerInternal(manager); - throw; - } - ++Version; - return manager; - } - - public T GetExistingManager() where T : ScriptBehaviourManager - { - CheckGetOrCreateManager(); - - return (T) GetExistingManagerInternal(typeof(T)); - } - - public ScriptBehaviourManager GetExistingManager(Type type) - { - CheckGetOrCreateManager(); - - return GetExistingManagerInternal(type); - } - - public void DestroyManager(ScriptBehaviourManager manager) - { - CheckGetOrCreateManager(); - - RemoveManagerInternal(manager); - manager.DestroyInstance(); - } - - public bool QuitUpdate { get; set; } - - internal static int AllocateSystemID() - { - return ++ms_SystemIDAllocator; - } - } -#else - public class World : IDisposable - { - static readonly List allWorlds = new List(); -#if ENABLE_UNITY_COLLECTIONS_CHECKS - bool m_AllowGetManager = true; -#endif - - //@TODO: What about multiple managers of the same type... - List m_BehaviourManagers = new List(); - - int m_SystemIDAllocator = 0; - - public World(string name) - { - // Debug.LogError("Create World "+ name + " - " + GetHashCode()); - Name = name; - allWorlds.Add(this); - } - - // XXX fix me -- we need a readonly wrapper - public ScriptBehaviourManager[] BehaviourManagers => m_BehaviourManagers.ToArray(); - - public string Name { get; } - - public override string ToString() - { - return Name; - } - - public int Version { get; private set; } - - public static World Active { get; set; } - - public static World[] AllWorlds => allWorlds.ToArray(); - - public bool IsCreated => true; - - public void Dispose() - { - if (!IsCreated) - throw new ArgumentException("World is already disposed"); - // Debug.LogError("Dispose World "+ Name + " - " + GetHashCode()); - - if (allWorlds.Contains(this)) - allWorlds.Remove(this); - - // Destruction should happen in reverse order to construction - ScriptBehaviourManager em = null; - for (int i = m_BehaviourManagers.Count - 1; i >= 0; --i) - { - var mgr = m_BehaviourManagers[i]; - if (mgr is EntityManager) - { - em = mgr; - continue; - } - try - { - mgr.DestroyInstance(); - } - catch (Exception e) - { - Debug.LogException(e); - } - } - - // Destroy EntityManager last - if (em != null) - { - try - { - em.DestroyInstance(); - } - catch (Exception e) - { - Debug.LogException(e); - } - } - - if (Active == this) - Active = null; - - m_BehaviourManagers.Clear(); - m_BehaviourManagers = null; - } - - public static void DisposeAllWorlds() - { - while (allWorlds.Count != 0) - allWorlds[0].Dispose(); - } - - private ScriptBehaviourManager CreateManagerInternal() where T : new() - { -#if ENABLE_UNITY_COLLECTIONS_CHECKS - if (!m_AllowGetManager) - throw new ArgumentException( - "During destruction of a system you are not allowed to create more systems."); - - m_AllowGetManager = true; -#endif - ScriptBehaviourManager manager; - try - { -#if !UNITY_CSHARP_TINY - manager = new T() as ScriptBehaviourManager; -#else - manager = TypeManager.ConstructSystem(typeof(T)); -#endif - } - catch - { -#if ENABLE_UNITY_COLLECTIONS_CHECKS - m_AllowGetManager = false; -#endif - throw; - } - - return AddManager(manager); - } - - private ScriptBehaviourManager GetExistingManagerInternal() - { -#if ENABLE_UNITY_COLLECTIONS_CHECKS - if (!IsCreated) - throw new ArgumentException("During destruction "); - if (!m_AllowGetManager) - throw new ArgumentException( - "During destruction of a system you are not allowed to get or create more systems."); -#endif - - ScriptBehaviourManager manager; - for (int i = 0; i < m_BehaviourManagers.Count; ++i) { - var mgr = m_BehaviourManagers[i]; - if (mgr is T) - return mgr; - } - - return null; - } - - private ScriptBehaviourManager GetExistingManagerInternal(Type type) - { -#if ENABLE_UNITY_COLLECTIONS_CHECKS - if (!IsCreated) - throw new ArgumentException("During destruction "); - if (!m_AllowGetManager) - throw new ArgumentException( - "During destruction of a system you are not allowed to get or create more systems."); -#endif - - ScriptBehaviourManager manager; - for (int i = 0; i < m_BehaviourManagers.Count; ++i) { - var mgr = m_BehaviourManagers[i]; - if (type.IsAssignableFrom(mgr.GetType())) - return mgr; - } - - return null; - } - - private ScriptBehaviourManager GetOrCreateManagerInternal() where T : new() - { - var manager = GetExistingManagerInternal(); - return manager ?? CreateManagerInternal(); - } - - private void RemoveManagerInternal(ScriptBehaviourManager manager) - { - if (!m_BehaviourManagers.Remove(manager)) - throw new ArgumentException($"manager does not exist in the world"); - ++Version; - } - - public T CreateManager() where T : ScriptBehaviourManager, new() - { - return (T) CreateManagerInternal(); - } - - public T GetOrCreateManager() where T : ScriptBehaviourManager, new() - { - return (T) GetOrCreateManagerInternal(); - } - - public T AddManager(T manager) where T : ScriptBehaviourManager - { - m_BehaviourManagers.Add(manager); - try - { - manager.CreateInstance(this); - } - catch - { - RemoveManagerInternal(manager); - throw; - } - - ++Version; - return manager; - } - - public T GetExistingManager() where T : ScriptBehaviourManager - { - return (T) GetExistingManagerInternal(typeof(T)); - } - - public ScriptBehaviourManager GetExistingManager(Type type) - { - return GetExistingManagerInternal(type); - } - - public void DestroyManager(ScriptBehaviourManager manager) - { - RemoveManagerInternal(manager); - manager.DestroyInstance(); - } - - static int ms_SystemIDAllocator = 0; - internal static int AllocateSystemID() - { - return ++ms_SystemIDAllocator; - } - - public bool QuitUpdate { get; set; } - - public void Update() - { - InitializationSystemGroup initializationSystemGroup = - GetExistingManager(typeof(InitializationSystemGroup)) as InitializationSystemGroup; - SimulationSystemGroup simulationSystemGroup = - GetExistingManager(typeof(SimulationSystemGroup)) as SimulationSystemGroup; - PresentationSystemGroup presentationSystemGroup = - GetExistingManager(typeof(PresentationSystemGroup)) as PresentationSystemGroup; - - initializationSystemGroup?.Update(); - simulationSystemGroup?.Update(); - presentationSystemGroup?.Update(); - } - - } -#endif -} diff --git a/Unity.Entities/Injection/WorldDebuggingTools.cs b/Unity.Entities/Injection/WorldDebuggingTools.cs deleted file mode 100644 index cdbe1849..00000000 --- a/Unity.Entities/Injection/WorldDebuggingTools.cs +++ /dev/null @@ -1,54 +0,0 @@ -#if UNITY_EDITOR -using System; -using System.Collections.Generic; -using System.Linq; -using Unity.Collections; - -namespace Unity.Entities -{ - internal class WorldDebuggingTools - { - internal static void MatchEntityInComponentGroups(World world, Entity entity, - List>> matchList) - { - using (var entityComponentTypes = - world.GetExistingManager().GetComponentTypes(entity, Allocator.Temp)) - { - foreach (var manager in World.Active.BehaviourManagers) - { - var componentGroupList = new List(); - var system = manager as ComponentSystemBase; - if (system == null) continue; - foreach (var componentGroup in system.ComponentGroups) - if (Match(componentGroup, entityComponentTypes)) - componentGroupList.Add(componentGroup); - - if (componentGroupList.Count > 0) - matchList.Add( - new Tuple>(manager, componentGroupList)); - } - } - } - - private static bool Match(ComponentGroup group, NativeArray entityComponentTypes) - { - foreach (var groupType in group.GetQueryTypes().Skip(1)) - { - var found = false; - foreach (var type in entityComponentTypes) - { - if (type.TypeIndex != groupType.TypeIndex) - continue; - found = true; - break; - } - - if (found == (groupType.AccessModeType == ComponentType.AccessMode.Exclude)) - return false; - } - - return true; - } - } -} -#endif \ No newline at end of file diff --git a/Unity.Entities/Iterators/BufferArray.cs b/Unity.Entities/Iterators/BufferArray.cs deleted file mode 100644 index 693fe9aa..00000000 --- a/Unity.Entities/Iterators/BufferArray.cs +++ /dev/null @@ -1,92 +0,0 @@ -using System; -using Unity.Collections.LowLevel.Unsafe; - -namespace Unity.Entities -{ - [NativeContainer] - [NativeContainerSupportsMinMaxWriteRestriction] - [Obsolete("BufferArray is deprecated. Use ComponentGroup APIs instead.")] - public unsafe struct BufferArray where T : struct, IBufferElementData - { - private ComponentChunkCache m_Cache; - private ComponentChunkIterator m_Iterator; - private readonly bool m_IsReadOnly; - - - private readonly int m_Length; -#if ENABLE_UNITY_COLLECTIONS_CHECKS - private readonly int m_MinIndex; - private readonly int m_MaxIndex; - private readonly AtomicSafetyHandle m_Safety0; - private readonly AtomicSafetyHandle m_ArrayInvalidationSafety; -#pragma warning disable 0414 // assigned but its value is never used - private int m_SafetyReadOnlyCount; - private int m_SafetyReadWriteCount; -#pragma warning restore 0414 -#endif - public int Length => m_Length; - -#if ENABLE_UNITY_COLLECTIONS_CHECKS - internal BufferArray(ComponentChunkIterator iterator, int length, bool isReadOnly, - AtomicSafetyHandle safety, AtomicSafetyHandle arrayInvalidationSafety) -#else - internal BufferArray(ComponentChunkIterator iterator, int length, bool isReadOnly) -#endif - { - m_Length = length; - m_IsReadOnly = isReadOnly; - m_Iterator = iterator; - m_Cache = default(ComponentChunkCache); - -#if ENABLE_UNITY_COLLECTIONS_CHECKS - m_MinIndex = 0; - m_MaxIndex = length - 1; - m_Safety0 = safety; - m_ArrayInvalidationSafety = arrayInvalidationSafety; - m_SafetyReadOnlyCount = isReadOnly ? 2 : 0; - m_SafetyReadWriteCount = isReadOnly ? 0 : 2; -#endif - } - - public DynamicBuffer this[int index] - { - get - { -#if ENABLE_UNITY_COLLECTIONS_CHECKS - AtomicSafetyHandle.CheckReadAndThrow(m_Safety0); - if (index < m_MinIndex || index > m_MaxIndex) - FailOutOfRangeError(index); -#endif - - if (index < m_Cache.CachedBeginIndex || index >= m_Cache.CachedEndIndex) - { - m_Iterator.MoveToEntityIndexAndUpdateCache(index, out m_Cache, !m_IsReadOnly); -#if ENABLE_UNITY_COLLECTIONS_CHECKS - if (m_Cache.CachedSizeOf < sizeof(BufferHeader)) - throw new InvalidOperationException("size cache info is broken"); -#endif - } - - BufferHeader* header = (BufferHeader*) ((byte*)m_Cache.CachedPtr + index * m_Cache.CachedSizeOf); - -#if ENABLE_UNITY_COLLECTIONS_CHECKS - return new DynamicBuffer(header, m_Safety0, m_ArrayInvalidationSafety, m_IsReadOnly); -#else - return new DynamicBuffer(header); -#endif - } - } - -#if ENABLE_UNITY_COLLECTIONS_CHECKS - private void FailOutOfRangeError(int index) - { - //@TODO: Make error message utility and share with NativeArray... - if (index < Length && (m_MinIndex != 0 || m_MaxIndex != Length - 1)) - throw new IndexOutOfRangeException( - $"Index {index} is out of restricted IJobParallelFor range [{m_MinIndex}...{m_MaxIndex}] in ReadWriteBuffer.\nReadWriteBuffers are restricted to only read & write the element at the job index. You can use double buffering strategies to avoid race conditions due to reading & writing in parallel to the same elements from a job."); - - throw new IndexOutOfRangeException($"Index {index} is out of range of '{Length}' Length."); - } -#endif - } -} diff --git a/Unity.Entities/Iterators/BufferArray.cs.meta b/Unity.Entities/Iterators/BufferArray.cs.meta deleted file mode 100644 index 3e91ca74..00000000 --- a/Unity.Entities/Iterators/BufferArray.cs.meta +++ /dev/null @@ -1,13 +0,0 @@ -fileFormatVersion: 2 -guid: 24ee6c53bdd9546d0982af7d8b40e84d -timeCreated: 1504713176 -licenseType: Pro -MonoImporter: - externalObjects: {} - serializedVersion: 2 - defaultReferences: [] - executionOrder: 0 - icon: {instanceID: 0} - userData: - assetBundleName: - assetBundleVariant: diff --git a/Unity.Entities/Iterators/ChunkDataGatherJobs.cs b/Unity.Entities/Iterators/ChunkDataGatherJobs.cs index d696a347..8160541c 100644 --- a/Unity.Entities/Iterators/ChunkDataGatherJobs.cs +++ b/Unity.Entities/Iterators/ChunkDataGatherJobs.cs @@ -69,7 +69,7 @@ public void Execute() unsafe struct GatherChunksWithFiltering : IJobParallelFor { [NativeDisableUnsafePtrRestriction] public MatchingArchetype** MatchingArchetypes; - public ComponentGroupFilter Filter; + public EntityQueryFilter Filter; [ReadOnly] public NativeArray Offsets; public NativeArray FilteredCounts; @@ -88,9 +88,9 @@ public void Execute(int index) if (filter.Type == FilterType.SharedComponent) { - var indexInComponentGroup1 = filter.Shared.IndexInComponentGroup[0]; + var indexInEntityQuery1 = filter.Shared.IndexInEntityQuery[0]; var sharedComponentIndex1 = filter.Shared.SharedComponentIndex[0]; - var componentIndexInChunk1 = match->IndexInArchetype[indexInComponentGroup1] - archetype->FirstSharedComponent; + var componentIndexInChunk1 = match->IndexInArchetype[indexInEntityQuery1] - archetype->FirstSharedComponent; var sharedComponents1 = archetype->Chunks.GetSharedComponentValueArrayForType(componentIndexInChunk1); if (filter.Shared.Count == 1) @@ -103,9 +103,9 @@ public void Execute(int index) } else { - var indexInComponentGroup2 = filter.Shared.IndexInComponentGroup[1]; + var indexInEntityQuery2 = filter.Shared.IndexInEntityQuery[1]; var sharedComponentIndex2 = filter.Shared.SharedComponentIndex[1]; - var componentIndexInChunk2 = match->IndexInArchetype[indexInComponentGroup2] - archetype->FirstSharedComponent; + var componentIndexInChunk2 = match->IndexInArchetype[indexInEntityQuery2] - archetype->FirstSharedComponent; var sharedComponents2 = archetype->Chunks.GetSharedComponentValueArrayForType(componentIndexInChunk2); for (var i = 0; i < chunkCount; ++i) @@ -118,8 +118,8 @@ public void Execute(int index) } else { - var indexInComponentGroup1 = filter.Changed.IndexInComponentGroup[0]; - var componentIndexInChunk1 = match->IndexInArchetype[indexInComponentGroup1]; + var indexInEntityQuery1 = filter.Changed.IndexInEntityQuery[0]; + var componentIndexInChunk1 = match->IndexInArchetype[indexInEntityQuery1]; var changeVersions1 = archetype->Chunks.GetChangeVersionArrayForType(componentIndexInChunk1); var requiredVersion = filter.RequiredChangeVersion; @@ -133,8 +133,8 @@ public void Execute(int index) } else { - var indexInComponentGroup2 = filter.Changed.IndexInComponentGroup[1]; - var componentIndexInChunk2 = match->IndexInArchetype[indexInComponentGroup2]; + var indexInEntityQuery2 = filter.Changed.IndexInEntityQuery[1]; + var componentIndexInChunk2 = match->IndexInArchetype[indexInEntityQuery2]; var changeVersions2 = archetype->Chunks.GetChangeVersionArrayForType(componentIndexInChunk2); for (var i = 0; i < chunkCount; ++i) @@ -154,7 +154,7 @@ public void Execute(int index) internal unsafe struct GatherChunksAndOffsetsWithFilteringJob : IJob { public MatchingArchetypeList Archetypes; - public ComponentGroupFilter Filter; + public EntityQueryFilter Filter; [NativeDisableUnsafePtrRestriction] public void* PrefilterData; @@ -181,10 +181,10 @@ public void Execute() if (filter.Type == FilterType.SharedComponent) { - var indexInComponentGroup0 = filter.Shared.IndexInComponentGroup[0]; + var indexInEntityQuery0 = filter.Shared.IndexInEntityQuery[0]; var sharedComponentIndex0 = filter.Shared.SharedComponentIndex[0]; var componentIndexInChunk0 = - match->IndexInArchetype[indexInComponentGroup0] - archetype->FirstSharedComponent; + match->IndexInArchetype[indexInEntityQuery0] - archetype->FirstSharedComponent; var sharedComponents0 = archetype->Chunks.GetSharedComponentValueArrayForType(componentIndexInChunk0); @@ -203,10 +203,10 @@ public void Execute() } else { - var indexInComponentGroup1 = filter.Shared.IndexInComponentGroup[1]; + var indexInEntityQuery1 = filter.Shared.IndexInEntityQuery[1]; var sharedComponentIndex1 = filter.Shared.SharedComponentIndex[1]; var componentIndexInChunk1 = - match->IndexInArchetype[indexInComponentGroup1] - archetype->FirstSharedComponent; + match->IndexInArchetype[indexInEntityQuery1] - archetype->FirstSharedComponent; var sharedComponents1 = archetype->Chunks.GetSharedComponentValueArrayForType(componentIndexInChunk1); @@ -225,8 +225,8 @@ public void Execute() } else { - var indexInComponentGroup0 = filter.Changed.IndexInComponentGroup[0]; - var componentIndexInChunk0 = match->IndexInArchetype[indexInComponentGroup0]; + var indexInEntityQuery0 = filter.Changed.IndexInEntityQuery[0]; + var componentIndexInChunk0 = match->IndexInArchetype[indexInEntityQuery0]; var changeVersions0 = archetype->Chunks.GetChangeVersionArrayForType(componentIndexInChunk0); var requiredVersion = filter.RequiredChangeVersion; @@ -245,8 +245,8 @@ public void Execute() } else { - var indexInComponentGroup1 = filter.Changed.IndexInComponentGroup[1]; - var componentIndexInChunk1 = match->IndexInArchetype[indexInComponentGroup1]; + var indexInEntityQuery1 = filter.Changed.IndexInEntityQuery[1]; + var componentIndexInChunk1 = match->IndexInArchetype[indexInEntityQuery1]; var changeVersions1 = archetype->Chunks.GetChangeVersionArrayForType(componentIndexInChunk1); diff --git a/Unity.Entities/Iterators/ComponentChunkIterator.cs b/Unity.Entities/Iterators/ComponentChunkIterator.cs index d84658f2..1be96e21 100644 --- a/Unity.Entities/Iterators/ComponentChunkIterator.cs +++ b/Unity.Entities/Iterators/ComponentChunkIterator.cs @@ -18,12 +18,12 @@ internal enum FilterType } //@TODO: Use field offset / union here... There seems to be an issue in mono preventing it... - internal unsafe struct ComponentGroupFilter + internal unsafe struct EntityQueryFilter { public struct SharedComponentData { public int Count; - public fixed int IndexInComponentGroup[2]; + public fixed int IndexInEntityQuery[2]; public fixed int SharedComponentIndex[2]; } @@ -33,7 +33,7 @@ public struct ChangedFilter public const int Capacity = 2; public int Count; - public fixed int IndexInComponentGroup[2]; + public fixed int IndexInEntityQuery[2]; } public FilterType Type; @@ -91,11 +91,11 @@ internal unsafe struct ComponentChunkIterator private int m_CurrentArchetypeIndex; private int m_CurrentChunkIndex; - internal ComponentGroupFilter m_Filter; + internal EntityQueryFilter m_Filter; internal readonly uint m_GlobalSystemVersion; - public int IndexInComponentGroup; + public int IndexInEntityQuery; internal int GetSharedComponentFromCurrentChunk(int sharedComponentIndex) { @@ -106,11 +106,11 @@ internal int GetSharedComponentFromCurrentChunk(int sharedComponentIndex) } public ComponentChunkIterator(MatchingArchetypeList match, uint globalSystemVersion, - ref ComponentGroupFilter filter) + ref EntityQueryFilter filter) { m_MatchingArchetypeList = match; m_CurrentMatchingArchetypeIndex = match.Count - 1; - IndexInComponentGroup = -1; + IndexInEntityQuery = -1; m_CurrentChunk = null; m_CurrentArchetypeIndex = m_CurrentArchetypeEntityIndex = @@ -129,14 +129,14 @@ public object GetManagedObject(ArchetypeManager typeMan, int typeIndexInArchetyp public object GetManagedObject(ArchetypeManager typeMan, int cachedBeginIndex, int index) { return typeMan.GetManagedObject(*m_CurrentChunk, - m_CurrentMatchingArchetype->IndexInArchetype[IndexInComponentGroup], index - cachedBeginIndex); + m_CurrentMatchingArchetype->IndexInArchetype[IndexInEntityQuery], index - cachedBeginIndex); } public object[] GetManagedObjectRange(ArchetypeManager typeMan, int cachedBeginIndex, int index, out int rangeStart, out int rangeLength) { var objs = typeMan.GetManagedObjectRange(*m_CurrentChunk, - m_CurrentMatchingArchetype->IndexInArchetype[IndexInComponentGroup], out rangeStart, + m_CurrentMatchingArchetype->IndexInArchetype[IndexInEntityQuery], out rangeStart, out rangeLength); rangeStart += index - cachedBeginIndex; rangeLength -= index - cachedBeginIndex; @@ -169,7 +169,7 @@ internal static int CalculateNumberOfChunksWithoutFiltering(MatchingArchetypeLis /// Handle to the GatherChunks job used to fill the output array. /// NativeArray of all the chunks in the matchingArchetypes list. public static NativeArray CreateArchetypeChunkArray(MatchingArchetypeList matchingArchetypes, - Allocator allocator, out JobHandle jobHandle, ref ComponentGroupFilter filter, + Allocator allocator, out JobHandle jobHandle, ref EntityQueryFilter filter, JobHandle dependsOn = default(JobHandle)) { var archetypeCount = matchingArchetypes.Count; @@ -239,21 +239,21 @@ public static NativeArray CreateArchetypeChunkArray(MatchingArch } /// - /// Creates a NativeArray containing the entities in a given ComponentGroup. + /// Creates a NativeArray containing the entities in a given EntityQuery. /// /// List of matching archetypes. /// Allocator to use for the array. /// An atomic safety handle required by GatherEntitiesJob so it can call GetNativeArray() on chunks. - /// ComponentGroup to gather entities from. - /// ComponentGroupFilter for calculating the length of the output array. + /// EntityQuery to gather entities from. + /// EntityQueryFilter for calculating the length of the output array. /// Handle to the GatherEntitiesJob job used to fill the output array. /// Handle to a job this GatherEntitiesJob must wait on. - /// NativeArray of the entities in a given ComponentGroup. + /// NativeArray of the entities in a given EntityQuery. public static NativeArray CreateEntityArray(MatchingArchetypeList matchingArchetypes, Allocator allocator, ArchetypeChunkEntityType type, - ComponentGroup componentGroup, - ref ComponentGroupFilter filter, + EntityQuery entityQuery, + ref EntityQueryFilter filter, out JobHandle jobHandle, JobHandle dependsOn) @@ -265,7 +265,7 @@ public static NativeArray CreateEntityArray(MatchingArchetypeList matchi EntityType = type, Entities = new NativeArray(entityCount, allocator) }; - jobHandle = job.Schedule(componentGroup, dependsOn); + jobHandle = job.Schedule(entityQuery, dependsOn); return job.Entities; } @@ -273,8 +273,8 @@ public static NativeArray CreateEntityArray(MatchingArchetypeList matchi public static NativeArray CreateComponentDataArray(MatchingArchetypeList matchingArchetypes, Allocator allocator, ArchetypeChunkComponentType type, - ComponentGroup componentGroup, - ref ComponentGroupFilter filter, + EntityQuery entityQuery, + ref EntityQueryFilter filter, out JobHandle jobHandle, JobHandle dependsOn) where T :struct, IComponentData @@ -286,7 +286,7 @@ public static NativeArray CreateComponentDataArray(MatchingArchetypeList m ComponentData = new NativeArray(entityCount, allocator), ComponentType = type }; - jobHandle = job.Schedule(componentGroup, dependsOn); + jobHandle = job.Schedule(entityQuery, dependsOn); return job.ComponentData; } @@ -294,8 +294,8 @@ public static NativeArray CreateComponentDataArray(MatchingArchetypeList m public static void CopyFromComponentDataArray(MatchingArchetypeList matchingArchetypes, NativeArray componentDataArray, ArchetypeChunkComponentType type, - ComponentGroup componentGroup, - ref ComponentGroupFilter filter, + EntityQuery entityQuery, + ref EntityQueryFilter filter, out JobHandle jobHandle, JobHandle dependsOn) where T :struct, IComponentData @@ -305,16 +305,16 @@ public static void CopyFromComponentDataArray(MatchingArchetypeList matchingA ComponentData = componentDataArray, ComponentType = type }; - jobHandle = job.Schedule(componentGroup, dependsOn); + jobHandle = job.Schedule(entityQuery, dependsOn); } /// /// Total number of entities contained in a given MatchingArchetype list. /// /// List of matching archetypes. - /// ComponentGroupFilter to use when calculating total number of entities. + /// EntityQueryFilter to use when calculating total number of entities. /// Number of entities - public static int CalculateLength(MatchingArchetypeList matchingArchetypes, ref ComponentGroupFilter filter) + public static int CalculateLength(MatchingArchetypeList matchingArchetypes, ref EntityQueryFilter filter) { var filterCopy = filter; // Necessary to avoid a nasty compiler error cause by fixed buffer types @@ -342,10 +342,10 @@ public static int CalculateLength(MatchingArchetypeList matchingArchetypes, ref if (filter.Type == FilterType.SharedComponent) { - var indexInComponentGroup0 = filterCopy.Shared.IndexInComponentGroup[0]; + var indexInEntityQuery0 = filterCopy.Shared.IndexInEntityQuery[0]; var sharedComponentIndex0 = filterCopy.Shared.SharedComponentIndex[0]; var componentIndexInChunk0 = - match->IndexInArchetype[indexInComponentGroup0] - archetype->FirstSharedComponent; + match->IndexInArchetype[indexInEntityQuery0] - archetype->FirstSharedComponent; var sharedComponents0 = archetype->Chunks.GetSharedComponentValueArrayForType(componentIndexInChunk0); @@ -359,10 +359,10 @@ public static int CalculateLength(MatchingArchetypeList matchingArchetypes, ref } else { - var indexInComponentGroup1 = filterCopy.Shared.IndexInComponentGroup[1]; + var indexInEntityQuery1 = filterCopy.Shared.IndexInEntityQuery[1]; var sharedComponentIndex1 = filterCopy.Shared.SharedComponentIndex[1]; var componentIndexInChunk1 = - match->IndexInArchetype[indexInComponentGroup1] - archetype->FirstSharedComponent; + match->IndexInArchetype[indexInEntityQuery1] - archetype->FirstSharedComponent; var sharedComponents1 = archetype->Chunks.GetSharedComponentValueArrayForType(componentIndexInChunk1); @@ -376,8 +376,8 @@ public static int CalculateLength(MatchingArchetypeList matchingArchetypes, ref } else { - var indexInComponentGroup0 = filterCopy.Changed.IndexInComponentGroup[0]; - var componentIndexInChunk0 = match->IndexInArchetype[indexInComponentGroup0]; + var indexInEntityQuery0 = filterCopy.Changed.IndexInEntityQuery[0]; + var componentIndexInChunk0 = match->IndexInArchetype[indexInEntityQuery0]; var changeVersions0 = archetype->Chunks.GetChangeVersionArrayForType(componentIndexInChunk0); var requiredVersion = filter.RequiredChangeVersion; @@ -391,8 +391,8 @@ public static int CalculateLength(MatchingArchetypeList matchingArchetypes, ref } else { - var indexInComponentGroup1 = filterCopy.Changed.IndexInComponentGroup[1]; - var componentIndexInChunk1 = match->IndexInArchetype[indexInComponentGroup1]; + var indexInEntityQuery1 = filterCopy.Changed.IndexInEntityQuery[1]; + var componentIndexInChunk1 = match->IndexInArchetype[indexInEntityQuery1]; var changeVersions1 = archetype->Chunks.GetChangeVersionArrayForType(componentIndexInChunk1); @@ -550,16 +550,16 @@ public bool RequiresFilter() return m_Filter.RequiresMatchesFilter; } - public int GetIndexInArchetypeFromCurrentChunk(int indexInComponentGroup) + public int GetIndexInArchetypeFromCurrentChunk(int indexInEntityQuery) { - return m_CurrentMatchingArchetype->IndexInArchetype[indexInComponentGroup]; + return m_CurrentMatchingArchetype->IndexInArchetype[indexInEntityQuery]; } - public void UpdateCacheToCurrentChunk(out ComponentChunkCache cache, bool isWriting, int indexInComponentGroup) + public void UpdateCacheToCurrentChunk(out ComponentChunkCache cache, bool isWriting, int indexInEntityQuery) { var archetype = m_CurrentMatchingArchetype->Archetype; - int indexInArchetype = m_CurrentMatchingArchetype->IndexInArchetype[indexInComponentGroup]; + int indexInArchetype = m_CurrentMatchingArchetype->IndexInArchetype[indexInEntityQuery]; cache.CachedBeginIndex = m_CurrentChunkEntityIndex + m_CurrentArchetypeEntityIndex; cache.CachedEndIndex = cache.CachedBeginIndex + (*m_CurrentChunk)->Count; @@ -601,23 +601,23 @@ public void GetCurrentChunkRange(out int beginIndex, out int endIndex) return chunk->Buffer + archetype->Offsets[indexInArchetype]; } - public void* GetCurrentChunkComponentDataPtr(bool isWriting, int indexInComponentGroup) + public void* GetCurrentChunkComponentDataPtr(bool isWriting, int indexInEntityQuery) { - int indexInArchetype = m_CurrentMatchingArchetype->IndexInArchetype[indexInComponentGroup]; + int indexInArchetype = m_CurrentMatchingArchetype->IndexInArchetype[indexInEntityQuery]; return GetChunkComponentDataPtr(*m_CurrentChunk, isWriting, indexInArchetype, m_GlobalSystemVersion); } public void UpdateChangeVersion() { - int indexInArchetype = m_CurrentMatchingArchetype->IndexInArchetype[IndexInComponentGroup]; + int indexInArchetype = m_CurrentMatchingArchetype->IndexInArchetype[IndexInEntityQuery]; (*m_CurrentChunk)->SetChangeVersion(indexInArchetype, m_GlobalSystemVersion); } public void MoveToEntityIndexAndUpdateCache(int index, out ComponentChunkCache cache, bool isWriting) { - Assert.IsTrue(-1 != IndexInComponentGroup); + Assert.IsTrue(-1 != IndexInEntityQuery); MoveToEntityIndex(index); - UpdateCacheToCurrentChunk(out cache, isWriting, IndexInComponentGroup); + UpdateCacheToCurrentChunk(out cache, isWriting, IndexInEntityQuery); } internal ArchetypeChunk GetCurrentChunk() @@ -712,7 +712,7 @@ internal int GetIndexOfFirstEntityInCurrentChunk() return index; } - internal static JobHandle PreparePrefilteredChunkLists(int unfilteredChunkCount, MatchingArchetypeList archetypes, ComponentGroupFilter filter, JobHandle dependsOn, ScheduleMode mode, out NativeArray prefilterDataArray, out void* deferredCountData) + internal static JobHandle PreparePrefilteredChunkLists(int unfilteredChunkCount, MatchingArchetypeList archetypes, EntityQueryFilter filter, JobHandle dependsOn, ScheduleMode mode, out NativeArray prefilterDataArray, out void* deferredCountData) { // Allocate one buffer for all prefilter data and distribute it // We keep the full buffer as a "dummy array" so we can deallocate it later with [DeallocateOnJobCompletion] @@ -764,5 +764,13 @@ internal static JobHandle PreparePrefilteredChunkLists(int unfilteredChunkCount, return prefilterHandle; } + + internal static void UnpackPrefilterData(NativeArray prefilterData, out ArchetypeChunk* chunks, out int* entityOffsets, out int filteredChunkCount) + { + chunks = (ArchetypeChunk*) prefilterData.GetUnsafePtr(); + + filteredChunkCount = *(int*)((byte*) prefilterData.GetUnsafePtr() + prefilterData.Length - sizeof(int)); + entityOffsets = (int*) (chunks + filteredChunkCount); + } } } diff --git a/Unity.Entities/Iterators/ComponentDataArray.cs b/Unity.Entities/Iterators/ComponentDataArray.cs deleted file mode 100644 index ac3fbcfe..00000000 --- a/Unity.Entities/Iterators/ComponentDataArray.cs +++ /dev/null @@ -1,163 +0,0 @@ -using System; -using System.Diagnostics; -using Unity.Collections; -using Unity.Collections.LowLevel.Unsafe; - -namespace Unity.Entities -{ - [NativeContainer] - [NativeContainerSupportsMinMaxWriteRestriction] - [Obsolete("ComponentDataArray is deprecated. Use IJobProcessComponentData or ComponentGroup ToComponentDataArray/CopyFromComponentDataArray APIs instead.")] - public unsafe struct ComponentDataArray where T : struct, IComponentData - { - private ComponentChunkIterator m_Iterator; - private ComponentChunkCache m_Cache; - - private readonly int m_Length; -#if ENABLE_UNITY_COLLECTIONS_CHECKS - private readonly int m_MinIndex; - private readonly int m_MaxIndex; - private readonly AtomicSafetyHandle m_Safety; -#endif - public int Length => m_Length; - -#if ENABLE_UNITY_COLLECTIONS_CHECKS - internal ComponentDataArray(ComponentChunkIterator iterator, int length, AtomicSafetyHandle safety) -#else - internal ComponentDataArray(ComponentChunkIterator iterator, int length) -#endif - { - m_Iterator = iterator; - m_Cache = default(ComponentChunkCache); - - m_Length = length; -#if ENABLE_UNITY_COLLECTIONS_CHECKS - m_MinIndex = 0; - m_MaxIndex = length - 1; - m_Safety = safety; -#endif - } - - internal void* GetUnsafeChunkPtr(int startIndex, int maxCount, out int actualCount, bool isWriting) - { -#if ENABLE_UNITY_COLLECTIONS_CHECKS - GetUnsafeChunkPtrCheck(startIndex, maxCount); -#endif - - m_Iterator.MoveToEntityIndexAndUpdateCache(startIndex, out m_Cache, isWriting); - - void* ptr = (byte*) m_Cache.CachedPtr + startIndex * m_Cache.CachedSizeOf; - actualCount = Math.Min(maxCount, m_Cache.CachedEndIndex - startIndex); - - return ptr; - } - -#if ENABLE_UNITY_COLLECTIONS_CHECKS - [Conditional("ENABLE_UNITY_COLLECTIONS_CHECKS")] - private void GetUnsafeChunkPtrCheck(int startIndex, int maxCount) - { - AtomicSafetyHandle.CheckReadAndThrow(m_Safety); - - if (startIndex < m_MinIndex) - FailOutOfRangeError(startIndex); - else if (startIndex + maxCount > m_MaxIndex + 1) - FailOutOfRangeError(startIndex + maxCount); - } -#endif - - public NativeArray GetChunkArray(int startIndex, int maxCount) - { - int count; - //@TODO: How should we declare read / write here? - var ptr = GetUnsafeChunkPtr(startIndex, maxCount, out count, true); - - var arr = NativeArrayUnsafeUtility.ConvertExistingDataToNativeArray(ptr, count, Allocator.Invalid); - -#if ENABLE_UNITY_COLLECTIONS_CHECKS - NativeArrayUnsafeUtility.SetAtomicSafetyHandle(ref arr, m_Safety); -#endif - - return arr; - } - - public void CopyTo(NativeSlice dst, int startIndex = 0) - { - var copiedCount = 0; - while (copiedCount < dst.Length) - { - var chunkArray = GetChunkArray(startIndex + copiedCount, dst.Length - copiedCount); - dst.Slice(copiedCount, chunkArray.Length).CopyFrom(chunkArray); - - copiedCount += chunkArray.Length; - } - } - - public T this[int index] - { - get - { -#if ENABLE_UNITY_COLLECTIONS_CHECKS - GetValueCheck(index); -#endif - - if (index < m_Cache.CachedBeginIndex || index >= m_Cache.CachedEndIndex) - m_Iterator.MoveToEntityIndexAndUpdateCache(index, out m_Cache, false); - - return UnsafeUtility.ReadArrayElement(m_Cache.CachedPtr, index); - } - - set - { -#if ENABLE_UNITY_COLLECTIONS_CHECKS - SetValueCheck(index); -#endif - - if (index < m_Cache.CachedBeginIndex || index >= m_Cache.CachedEndIndex) - m_Iterator.MoveToEntityIndexAndUpdateCache(index, out m_Cache, true); - else if (!m_Cache.IsWriting) - { - m_Cache.IsWriting = true; - m_Iterator.UpdateChangeVersion(); - } - - UnsafeUtility.WriteArrayElement(m_Cache.CachedPtr, index, value); - } - } - -#if ENABLE_UNITY_COLLECTIONS_CHECKS - [Conditional("ENABLE_UNITY_COLLECTIONS_CHECKS")] - private void SetValueCheck(int index) - { - AtomicSafetyHandle.CheckWriteAndThrow(m_Safety); - if (index < m_MinIndex || index > m_MaxIndex) - FailOutOfRangeError(index); - } -#endif - -#if ENABLE_UNITY_COLLECTIONS_CHECKS - [Conditional("ENABLE_UNITY_COLLECTIONS_CHECKS")] - private void GetValueCheck(int index) - { - AtomicSafetyHandle.CheckReadAndThrow(m_Safety); - if (index < m_MinIndex || index > m_MaxIndex) - FailOutOfRangeError(index); - } -#endif - -#if ENABLE_UNITY_COLLECTIONS_CHECKS - [Conditional("ENABLE_UNITY_COLLECTIONS_CHECKS")] - private void FailOutOfRangeError(int index) - { - //@TODO: Make error message utility and share with NativeArray... - if (index < Length && (m_MinIndex != 0 || m_MaxIndex != Length - 1)) - throw new IndexOutOfRangeException( - $"Index {index} is out of restricted IJobParallelFor range [{m_MinIndex}...{m_MaxIndex}] in ReadWriteBuffer.\n" + - "ReadWriteBuffers are restricted to only read & write the element at the job index. " + - "You can use double buffering strategies to avoid race conditions due to " + - "reading & writing in parallel to the same elements from a job."); - - throw new IndexOutOfRangeException($"Index {index} is out of range of '{Length}' Length."); - } -#endif - } -} diff --git a/Unity.Entities/Iterators/ComponentDataArray.cs.meta b/Unity.Entities/Iterators/ComponentDataArray.cs.meta deleted file mode 100644 index 85a8eb73..00000000 --- a/Unity.Entities/Iterators/ComponentDataArray.cs.meta +++ /dev/null @@ -1,13 +0,0 @@ -fileFormatVersion: 2 -guid: e899c8ac6ed7b40afacf51d570844e48 -timeCreated: 1504713176 -licenseType: Pro -MonoImporter: - externalObjects: {} - serializedVersion: 2 - defaultReferences: [] - executionOrder: 0 - icon: {instanceID: 0} - userData: - assetBundleName: - assetBundleVariant: diff --git a/Unity.Entities/Iterators/ComponentGroup.cs.meta b/Unity.Entities/Iterators/ComponentGroup.cs.meta deleted file mode 100644 index 08a41540..00000000 --- a/Unity.Entities/Iterators/ComponentGroup.cs.meta +++ /dev/null @@ -1,13 +0,0 @@ -fileFormatVersion: 2 -guid: e5e6e5c15d4a044a699fce82fa67186b -timeCreated: 1506266985 -licenseType: Pro -MonoImporter: - externalObjects: {} - serializedVersion: 2 - defaultReferences: [] - executionOrder: 0 - icon: {instanceID: 0} - userData: - assetBundleName: - assetBundleVariant: diff --git a/Unity.Entities/Iterators/ComponentGroup.obsolete.cs b/Unity.Entities/Iterators/ComponentGroup.obsolete.cs new file mode 100644 index 00000000..0f717a23 --- /dev/null +++ b/Unity.Entities/Iterators/ComponentGroup.obsolete.cs @@ -0,0 +1,275 @@ +using System; +using System.Collections.Generic; +using System.ComponentModel; +using System.Diagnostics; +using Unity.Burst; +using Unity.Collections; +using Unity.Collections.LowLevel.Unsafe; +using Unity.Jobs; + +namespace Unity.Entities +{ + public sealed unsafe partial class EntityManager + { + // The script updater has some bugs that make it unable to handle these as automatic updates + [Obsolete("CreateComponentGroup has been renamed to CreateEntityQuery.", true)] + public ComponentGroup CreateComponentGroup(params ComponentType[] requiredComponents) + { + throw new NotImplementedException(); + } + + [Obsolete("CreateComponentGroup has been renamed to CreateEntityQuery.", true)] + public ComponentGroup CreateComponentGroup(params EntityArchetypeQuery[] queriesDesc) + { + throw new NotImplementedException(); + } + + [Obsolete("UniversalGroup has been renamed to UniversalQuery. (UnityUpgradable) -> UniversalQuery", true)] + public EntityQuery UniversalGroup => null; + } + + [Obsolete("EntityArchetypeQuery has been renamed to EntityQueryDesc. (UnityUpgradable) -> EntityQueryDesc", true)] + public class EntityArchetypeQuery + { + public ComponentType[] Any = null; + public ComponentType[] None = null; + public ComponentType[] All = null; + public EntityArchetypeQueryOptions Options = EntityArchetypeQueryOptions.Default; + } + + [Flags] + [Obsolete("EntityArchetypeQueryOptions has been renamed to EntityQueryOptions. (UnityUpgradable) -> EntityQueryOptions", true)] + public enum EntityArchetypeQueryOptions + { + Default = 0, + IncludePrefab = 1, + IncludeDisabled = 2, + FilterWriteGroup = 4, + } + + [Obsolete("ComponentGroupExtensionsForComponentArray has been renamed to EntityQueryExtensionsForComponentArray. (UnityUpgradable) -> EntityQueryExtensionsForComponentArray", true)] + public static class ComponentGroupExtensionsForComponentArray + { + } + + [Obsolete("ComponentGroupExtensionsForTransformAccessArray has been renamed to EntityQueryExtensionsForTransformAccessArray. (UnityUpgradable) -> EntityQueryExtensionsForTransformAccessArray", true)] + public static class ComponentGroupExtensionsForTransformAccessArray + { + } + + public unsafe abstract partial class ComponentSystemBase + { + //[Obsolete("GetComponentGroup has been renamed to GetEntityQuery. (UnityUpgradable) -> GetEntityQuery(*)", true)] + [Obsolete("GetComponentGroup has been renamed to GetEntityQuery.", true)] + protected internal ComponentGroup GetComponentGroup(params ComponentType[] componentTypes) + { + throw new NotImplementedException(); + } + + //[Obsolete("GetComponentGroup has been renamed to GetEntityQuery. (UnityUpgradable) -> GetEntityQuery(*)", true)] + [Obsolete("GetComponentGroup has been renamed to GetEntityQuery.", true)] + protected ComponentGroup GetComponentGroup(NativeArray componentTypes) + { + throw new NotImplementedException(); + } + + //[Obsolete("GetComponentGroup has been renamed to GetEntityQuery. (UnityUpgradable) -> GetEntityQuery(*)", true)] + [Obsolete("GetComponentGroup has been renamed to GetEntityQuery.", true)] + protected internal ComponentGroup GetComponentGroup(params EntityArchetypeQuery[] queryDesc) + { + throw new NotImplementedException(); + } + + [Obsolete("GetComponentGroup has been renamed to GetEntityQuery.", true)] + protected internal ComponentGroup GetComponentGroup(params EntityQueryDesc[] queryDesc) + { + throw new NotImplementedException(); + } + } + +#if !UNITY_CSHARP_TINY + public static partial class JobForEachExtensions + { + [Obsolete("GetComponentGroupForIJobForEach has been renamed to GetEntityQueryForIJobForEach. (UnityUpgradable) -> GetEntityQueryForIJobForEach(*)", true)] + public static ComponentGroup GetComponentGroupForIJobForEach(this ComponentSystemBase system, + Type jobType) + { + throw new NotImplementedException(); + } + + [Obsolete("PrepareComponentGroup has been renamed to PrepareEntityQuery. (UnityUpgradable) -> PrepareEntityQuery(*)", true)] + public static void PrepareComponentGroup(this T jobData, ComponentSystemBase system) + where T : struct, JobForEachExtensions.IBaseJobForEach + { + throw new NotImplementedException(); + } + + [Obsolete("ScheduleGroup has been renamed to Schedule. (UnityUpgradable) -> Schedule(*)", true)] + public static JobHandle ScheduleGroup(this T jobData, ComponentGroup cg, JobHandle dependsOn = default(JobHandle)) + where T : struct, IBaseJobForEach + { + throw new NotImplementedException(); + } + + [Obsolete("ScheduleGroupSingle has been renamed to ScheduleSingle. (UnityUpgradable) -> ScheduleSingle(*)", true)] + public static JobHandle ScheduleGroupSingle(this T jobData, ComponentGroup cg, JobHandle dependsOn = default(JobHandle)) + where T : struct, IBaseJobForEach + { + throw new NotImplementedException(); + } + + [Obsolete("RunGroup has been renamed to Run. (UnityUpgradable) -> Run(*)", true)] + public static JobHandle Run(this T jobData, ComponentGroup cg, JobHandle dependsOn = default(JobHandle)) + where T : struct, IBaseJobForEach + { + throw new NotImplementedException(); + } + } +#endif + + [EditorBrowsable(EditorBrowsableState.Never)] + [Obsolete("ComponentGroup has been renamed to EntityQuery. (UnityUpgradable) -> EntityQuery", true)] + public class ComponentGroup : IDisposable + { + public bool IsEmptyIgnoreFilter + { + get { throw new NotImplementedException(); } + } + + public int CalculateLength() + { + throw new NotImplementedException(); + } + + public NativeArray CreateArchetypeChunkArray(Allocator allocator, out JobHandle jobhandle) + { + throw new NotImplementedException(); + } + + public NativeArray CreateArchetypeChunkArray(Allocator allocator) + { + throw new NotImplementedException(); + } + + + public NativeArray ToEntityArray(Allocator allocator, out JobHandle jobhandle) + { + throw new NotImplementedException(); + } + + public NativeArray ToEntityArray(Allocator allocator) + { + throw new NotImplementedException(); + } + + public NativeArray ToComponentDataArray(Allocator allocator, out JobHandle jobhandle) + where T : struct, IComponentData + { + throw new NotImplementedException(); + } + + public NativeArray ToComponentDataArray(Allocator allocator) + where T : struct, IComponentData + { + throw new NotImplementedException(); + } + + public void CopyFromComponentDataArray(NativeArray componentDataArray) + where T : struct, IComponentData + { + throw new NotImplementedException(); + } + + public void CopyFromComponentDataArray(NativeArray componentDataArray, out JobHandle jobhandle) + where T : struct, IComponentData + { + throw new NotImplementedException(); + } + + public Entity GetSingletonEntity() + { + throw new NotImplementedException(); + } + + public T GetSingleton() + where T : struct, IComponentData + { + throw new NotImplementedException(); + } + + public void SetSingleton(T value) + where T : struct, IComponentData + { + throw new NotImplementedException(); + } + + public bool CompareComponents(ComponentType[] componentTypes) + { + throw new NotImplementedException(); + } + + public bool CompareComponents(NativeArray componentTypes) + { + throw new NotImplementedException(); + } + + public bool CompareQuery(EntityArchetypeQuery[] queryDesc) + { + throw new NotImplementedException(); + } + + public void ResetFilter() + { + throw new NotImplementedException(); + } + + public void SetFilter(SharedComponent1 sharedComponent1) + where SharedComponent1 : struct, ISharedComponentData + { + throw new NotImplementedException(); + } + + public void SetFilter(SharedComponent1 sharedComponent1, + SharedComponent2 sharedComponent2) + where SharedComponent1 : struct, ISharedComponentData + where SharedComponent2 : struct, ISharedComponentData + { + throw new NotImplementedException(); + } + + public void SetFilterChanged(ComponentType componentType) + { + throw new NotImplementedException(); + } + + public void SetFilterChanged(ComponentType[] componentType) + { + throw new NotImplementedException(); + } + + public void CompleteDependency() + { + throw new NotImplementedException(); + } + + public JobHandle GetDependency() + { + throw new NotImplementedException(); + } + + public void AddDependency(JobHandle job) + { + throw new NotImplementedException(); + } + + public int GetCombinedComponentOrderVersion() + { + throw new NotImplementedException(); + } + + public void Dispose() + { + throw new NotImplementedException(); + } + } +} diff --git a/Unity.Entities/Iterators/ComponentGroup.obsolete.cs.meta b/Unity.Entities/Iterators/ComponentGroup.obsolete.cs.meta new file mode 100644 index 00000000..bf03362b --- /dev/null +++ b/Unity.Entities/Iterators/ComponentGroup.obsolete.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: 4f44190df1d8f7846866e1f2cce00448 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/Unity.Entities/Iterators/ComponentGroupArray.cs b/Unity.Entities/Iterators/ComponentGroupArray.cs deleted file mode 100644 index 414d48b6..00000000 --- a/Unity.Entities/Iterators/ComponentGroupArray.cs +++ /dev/null @@ -1,417 +0,0 @@ -#if !UNITY_CSHARP_TINY -using System; -using System.Collections; -using System.Collections.Generic; -using System.Reflection; -using Unity.Assertions; -using Unity.Burst; -using Unity.Collections; -using Unity.Collections.LowLevel.Unsafe; - -namespace Unity.Entities -{ - internal class ComponentGroupArrayStaticCache - { - public readonly Type CachedType; - internal readonly int ComponentCount; - internal readonly int ComponentDataCount; - internal readonly int[] ComponentFieldOffsets; - internal readonly ComponentGroup ComponentGroup; - - internal readonly ComponentType[] ComponentTypes; - internal readonly ComponentJobSafetyManager SafetyManager; - - public ComponentGroupArrayStaticCache(Type type, EntityManager entityManager, ComponentSystemBase system) - { - var fields = type.GetFields(BindingFlags.Instance | BindingFlags.NonPublic | BindingFlags.Public); - - var componentFieldOffsetsBuilder = new List(); - var componentTypesBuilder = new List(); - - var componentDataFieldOffsetsBuilder = new List(); - var componentDataTypesBuilder = new List(); - - var subtractiveComponentTypesBuilder = new List(); - - foreach (var field in fields) - { - var fieldType = field.FieldType; - var offset = UnsafeUtility.GetFieldOffset(field); - - if (fieldType.IsPointer) - { - var isReadOnly = field.GetCustomAttributes(typeof(ReadOnlyAttribute), true).Length != 0; - var accessMode = - isReadOnly ? ComponentType.AccessMode.ReadOnly : ComponentType.AccessMode.ReadWrite; - - var elementType = fieldType.GetElementType(); - -#if ENABLE_UNITY_COLLECTIONS_CHECKS - if (!typeof(IComponentData).IsAssignableFrom(elementType) && elementType != typeof(Entity)) - throw new ArgumentException( - $"{type}.{field.Name} is a pointer type but not a IComponentData. Only IComponentData or Entity may be a pointer type for enumeration."); -#endif - componentDataFieldOffsetsBuilder.Add(offset); - componentDataTypesBuilder.Add(new ComponentType(elementType, accessMode)); - } - else if (TypeManager.UnityEngineComponentType?.IsAssignableFrom(fieldType) ?? false) - { - componentFieldOffsetsBuilder.Add(offset); - componentTypesBuilder.Add(fieldType); - } - else if (fieldType.IsGenericType && - fieldType.GetGenericTypeDefinition() == typeof(ExcludeComponent<>)) - { - subtractiveComponentTypesBuilder.Add(ComponentType.Exclude(fieldType.GetGenericArguments()[0])); - } -#if ENABLE_UNITY_COLLECTIONS_CHECKS - else if (typeof(IComponentData).IsAssignableFrom(fieldType)) - { - throw new ArgumentException( - $"{type}.{field.Name} must be an unsafe pointer to the {fieldType}. Like this: {fieldType}* {field.Name};"); - } - else - { - throw new ArgumentException($"{type}.{field.Name} can not be used in a component enumerator"); - } -#endif - } -#if ENABLE_UNITY_COLLECTIONS_CHECKS - if (componentTypesBuilder.Count + componentDataTypesBuilder.Count > ComponentGroupArrayData.kMaxStream) - throw new ArgumentException( - $"{type} has too many component references. A ComponentGroup Array can have up to {ComponentGroupArrayData.kMaxStream}."); -#endif - - ComponentDataCount = componentDataTypesBuilder.Count; - ComponentCount = componentTypesBuilder.Count; - - componentDataTypesBuilder.AddRange(componentTypesBuilder); - componentDataTypesBuilder.AddRange(subtractiveComponentTypesBuilder); - ComponentTypes = componentDataTypesBuilder.ToArray(); - - componentDataFieldOffsetsBuilder.AddRange(componentFieldOffsetsBuilder); - ComponentFieldOffsets = componentDataFieldOffsetsBuilder.ToArray(); - - ComponentGroup = system.GetComponentGroupInternal(ComponentTypes); - SafetyManager = entityManager.ComponentJobSafetyManager; - - CachedType = type; - } - } - - [NativeContainer] - [NativeContainerSupportsMinMaxWriteRestriction] - internal unsafe struct ComponentGroupArrayData - { - public const int kMaxStream = 6; - - private struct ComponentGroupStream - { - public byte* CachedPtr; - public int SizeOf; - public ushort FieldOffset; - public ushort TypeIndexInArchetype; - } - - private fixed byte m_Caches[16 * kMaxStream]; - - private readonly int m_ComponentDataCount; - private readonly int m_ComponentCount; - - // The following 3 fields must not be renamed, unless JobReflectionData.cpp is changed accordingly. - // TODO: make JobDebugger logic more solid, either by using codegen proxies or attributes. - public readonly int m_Length; - public readonly int m_MinIndex; - public readonly int m_MaxIndex; - - public int CacheBeginIndex; - public int CacheEndIndex; - - private ComponentChunkIterator m_ChunkIterator; - private fixed int m_IndexInComponentGroup[kMaxStream]; - private fixed bool m_IsWriting[kMaxStream]; - - // The following fields must not be renamed, unless JobReflectionData.cpp is changed accordingly. - // TODO: make JobDebugger logic more solid, either by using codegen proxies or attributes. -#if ENABLE_UNITY_COLLECTIONS_CHECKS - private readonly int m_SafetyReadOnlyCount; - private readonly int m_SafetyReadWriteCount; -#pragma warning disable 414 - private AtomicSafetyHandle m_Safety0; - private AtomicSafetyHandle m_Safety1; - private AtomicSafetyHandle m_Safety2; - private AtomicSafetyHandle m_Safety3; - private AtomicSafetyHandle m_Safety4; - private AtomicSafetyHandle m_Safety5; -#pragma warning restore -#endif - - [NativeSetClassTypeToNullOnSchedule] private readonly ArchetypeManager m_ArchetypeManager; - - public ComponentGroupArrayData(ComponentGroupArrayStaticCache staticCache) - { - var length = staticCache.ComponentGroup.CalculateLength(); - m_ChunkIterator = staticCache.ComponentGroup.GetComponentChunkIterator(); - m_ChunkIterator.IndexInComponentGroup = 0; - - m_Length = length; - m_MinIndex = 0; - m_MaxIndex = length - 1; - - CacheBeginIndex = 0; - CacheEndIndex = 0; - m_ArchetypeManager = staticCache.ComponentGroup.ArchetypeManager; - - m_ComponentDataCount = staticCache.ComponentDataCount; - m_ComponentCount = staticCache.ComponentCount; - - fixed (int* indexInComponentGroup = m_IndexInComponentGroup) - fixed (byte* cacheBytes = m_Caches) - fixed (bool* isWriting = m_IsWriting) - { - var streams = (ComponentGroupStream*) cacheBytes; - - for (var i = 0; i < staticCache.ComponentDataCount + staticCache.ComponentCount; i++) - { - indexInComponentGroup[i] = staticCache.ComponentGroup.GetIndexInComponentGroup(staticCache.ComponentTypes[i].TypeIndex); - streams[i].FieldOffset = (ushort) staticCache.ComponentFieldOffsets[i]; - isWriting[i] = staticCache.ComponentTypes[i].AccessModeType == ComponentType.AccessMode.ReadWrite; - } - } - -#if ENABLE_UNITY_COLLECTIONS_CHECKS - m_Safety0 = new AtomicSafetyHandle(); - m_Safety1 = new AtomicSafetyHandle(); - m_Safety2 = new AtomicSafetyHandle(); - m_Safety3 = new AtomicSafetyHandle(); - m_Safety4 = new AtomicSafetyHandle(); - m_Safety5 = new AtomicSafetyHandle(); - Assert.AreEqual(6, kMaxStream); - - m_SafetyReadWriteCount = 0; - m_SafetyReadOnlyCount = 0; - var safetyManager = staticCache.SafetyManager; - fixed (AtomicSafetyHandle* safety = &m_Safety0) - { - for (var i = 0; i != staticCache.ComponentTypes.Length; i++) - { - var type = staticCache.ComponentTypes[i]; - if (type.IsZeroSized || type.AccessModeType != ComponentType.AccessMode.ReadOnly) - continue; - - safety[m_SafetyReadOnlyCount] = safetyManager.GetSafetyHandle(type.TypeIndex, true); - m_SafetyReadOnlyCount++; - } - - for (var i = 0; i != staticCache.ComponentTypes.Length; i++) - { - var type = staticCache.ComponentTypes[i]; - if (type.IsZeroSized || type.AccessModeType != ComponentType.AccessMode.ReadWrite) - continue; - - safety[m_SafetyReadOnlyCount + m_SafetyReadWriteCount] = - safetyManager.GetSafetyHandle(type.TypeIndex, false); - m_SafetyReadWriteCount++; - } - } -#endif - } - - public void UpdateCache(int index) - { - ComponentChunkCache cache; - - m_ChunkIterator.MoveToEntityIndex(index); - - fixed (int* indexInComponentGroup = m_IndexInComponentGroup) - fixed (byte* cacheBytes = m_Caches) - fixed (bool* isWriting = m_IsWriting) - { - var streams = (ComponentGroupStream*) cacheBytes; - var totalCount = m_ComponentDataCount + m_ComponentCount; - for (var i = 0; i < totalCount; i++) - { - m_ChunkIterator.UpdateCacheToCurrentChunk(out cache, isWriting[i], indexInComponentGroup[i]); - CacheBeginIndex = cache.CachedBeginIndex; - CacheEndIndex = cache.CachedEndIndex; - - int indexInArcheType = m_ChunkIterator.GetIndexInArchetypeFromCurrentChunk(indexInComponentGroup[i]); - - streams[i].SizeOf = cache.CachedSizeOf; - streams[i].CachedPtr = (byte*) cache.CachedPtr; -#if ENABLE_UNITY_COLLECTIONS_CHECKS - if (indexInArcheType > ushort.MaxValue) - throw new ArgumentException( - $"There is a maximum of {ushort.MaxValue} components on one entity."); -#endif - streams[i].TypeIndexInArchetype = (ushort) indexInArcheType; - } - } - } - - public void CheckAccess() - { -#if ENABLE_UNITY_COLLECTIONS_CHECKS - fixed (AtomicSafetyHandle* safety = &m_Safety0) - { - for (var i = 0; i < m_SafetyReadOnlyCount; i++) - AtomicSafetyHandle.CheckReadAndThrow(safety[i]); - - for (var i = m_SafetyReadOnlyCount; i < m_SafetyReadOnlyCount + m_SafetyReadWriteCount; i++) - AtomicSafetyHandle.CheckWriteAndThrow(safety[i]); - } -#endif - } - - public void PatchPtrs(int index, byte* valuePtr) - { - fixed (byte* cacheBytes = m_Caches) - { - var streams = (ComponentGroupStream*) cacheBytes; - for (var i = 0; i != m_ComponentDataCount; i++) - { - var componentPtr = (void*) (streams[i].CachedPtr + streams[i].SizeOf * index); - var valuePtrOffsetted = (void**) (valuePtr + streams[i].FieldOffset); - - *valuePtrOffsetted = componentPtr; - } - } - } - - [BurstDiscard] - public void PatchManagedPtrs(int index, byte* valuePtr) - { - fixed (byte* cacheBytes = m_Caches) - { - var streams = (ComponentGroupStream*) cacheBytes; - for (var i = m_ComponentDataCount; i != m_ComponentDataCount + m_ComponentCount; i++) - { - var component = m_ChunkIterator.GetManagedObject(m_ArchetypeManager, - streams[i].TypeIndexInArchetype, CacheBeginIndex, index); - var valuePtrOffsetted = valuePtr + streams[i].FieldOffset; - UnsafeUtility.CopyObjectAddressToPtr(component, valuePtrOffsetted); - } - } - } - -#if ENABLE_UNITY_COLLECTIONS_CHECKS - [BurstDiscard] - public void FailOutOfRangeError(int index) - { - /* - //@TODO: Make error message utility and share with NativeArray... - if (index < Length && (m_MinIndex != 0 || m_MaxIndex != Length - 1)) - throw new IndexOutOfRangeException(string.Format( - "Index {0} is out of restricted component group array range [{1}...{2}] in ReadWriteBuffer.\n" + - "ReadWriteBuffers are restricted to only read & write the element at the job index. " + - "You can use double buffering strategies to avoid race conditions due to " + - "reading & writing in parallel to the same elements from a job.", - index, m_MinIndex, m_MaxIndex)); -*/ - throw new IndexOutOfRangeException($"Index {index} is out of range of '{m_Length}' Length."); - } -#endif - } - - [Obsolete("ComponentGroupArray has been deprecated. Use ComponentSystem.ForEach to access managed components.")] - public struct ComponentGroupArray : IDisposable where T : struct - { - internal ComponentGroupArrayData m_Data; - - internal ComponentGroupArray(ComponentGroupArrayStaticCache cache) - { - m_Data = new ComponentGroupArrayData(cache); - } - - public void Dispose() - { - } - - public int Length => m_Data.m_Length; - - public unsafe T this[int index] - { - get - { -#if ENABLE_UNITY_COLLECTIONS_CHECKS - m_Data.CheckAccess(); - if (index < m_Data.m_MinIndex || index > m_Data.m_MaxIndex) - m_Data.FailOutOfRangeError(index); -#endif - - if (index < m_Data.CacheBeginIndex || index >= m_Data.CacheEndIndex) - m_Data.UpdateCache(index); - - var value = default(T); - var valuePtr = (byte*) UnsafeUtility.AddressOf(ref value); - m_Data.PatchPtrs(index, valuePtr); - m_Data.PatchManagedPtrs(index, valuePtr); - return value; - } - } - - public ComponentGroupEnumerator GetEnumerator() - { - return new ComponentGroupEnumerator(m_Data); - } - - public unsafe struct ComponentGroupEnumerator : IEnumerator where U : struct - { - private ComponentGroupArrayData m_Data; - private int m_Index; - - internal ComponentGroupEnumerator(ComponentGroupArrayData arrayData) - { - m_Data = arrayData; - m_Index = -1; - } - - public void Dispose() - { - } - - public bool MoveNext() - { - m_Index++; - - if (m_Index >= m_Data.CacheBeginIndex && m_Index < m_Data.CacheEndIndex) - return true; - - if (m_Index >= m_Data.m_Length) - return false; - - m_Data.CheckAccess(); - m_Data.UpdateCache(m_Index); - - return true; - } - - public void Reset() - { - m_Index = -1; - } - - public U Current - { - get - { - m_Data.CheckAccess(); - -#if ENABLE_UNITY_COLLECTIONS_CHECKS - if (m_Index < m_Data.m_MinIndex || m_Index > m_Data.m_MaxIndex) - m_Data.FailOutOfRangeError(m_Index); -#endif - - var value = default(U); - var valuePtr = (byte*) UnsafeUtility.AddressOf(ref value); - m_Data.PatchPtrs(m_Index, valuePtr); - m_Data.PatchManagedPtrs(m_Index, valuePtr); - return value; - } - } - - object IEnumerator.Current => Current; - } - } -} -#endif diff --git a/Unity.Entities/Iterators/ComponentGroupArray.cs.meta b/Unity.Entities/Iterators/ComponentGroupArray.cs.meta deleted file mode 100644 index e3bcd5e2..00000000 --- a/Unity.Entities/Iterators/ComponentGroupArray.cs.meta +++ /dev/null @@ -1,11 +0,0 @@ -fileFormatVersion: 2 -guid: b2149666441034bd2be2b403b9bc5dbd -MonoImporter: - externalObjects: {} - serializedVersion: 2 - defaultReferences: [] - executionOrder: 0 - icon: {instanceID: 0} - userData: - assetBundleName: - assetBundleVariant: diff --git a/Unity.Entities/Iterators/DynamicBuffer.cs b/Unity.Entities/Iterators/DynamicBuffer.cs index 99b1081b..d99e5981 100644 --- a/Unity.Entities/Iterators/DynamicBuffer.cs +++ b/Unity.Entities/Iterators/DynamicBuffer.cs @@ -38,6 +38,9 @@ internal DynamicBuffer(BufferHeader* header) } #endif + /// + /// The number of elements the buffer holds. + /// public int Length { get @@ -47,6 +50,9 @@ public int Length } } + /// + /// The number of elements the buffer can hold. + /// public int Capacity { get @@ -116,11 +122,24 @@ public T this [int index] } } + /// + /// Increases the buffer capacity and length. + /// + /// The new length of the buffer. public void ResizeUninitialized(int length) + { + Reserve(length); + m_Buffer->Length = length; + } + + /// + /// Increases the buffer capacity without increasing its length. + /// + /// The new buffer capacity. + public void Reserve(int length) { CheckWriteAccessAndInvalidateArrayAliases(); BufferHeader.EnsureCapacity(m_Buffer, length, UnsafeUtility.SizeOf(), UnsafeUtility.AlignOf(), BufferHeader.TrashMode.RetainOldData); - m_Buffer->Length = length; } public void Clear() @@ -143,8 +162,8 @@ public void TrimExcess() int elemSize = UnsafeUtility.SizeOf(); int elemAlign = UnsafeUtility.AlignOf(); - byte* newPtr = (byte*) UnsafeUtility.Malloc(elemSize * length, elemAlign, Allocator.Persistent); - UnsafeUtility.MemCpy(newPtr, oldPtr, elemSize * length); + byte* newPtr = (byte*) UnsafeUtility.Malloc((long)elemSize * length, elemAlign, Allocator.Persistent); + UnsafeUtility.MemCpy(newPtr, oldPtr, (long)elemSize * length); m_Buffer->Capacity = length; m_Buffer->Pointer = newPtr; @@ -168,7 +187,7 @@ public void Insert(int index, T elem) CheckBounds(index); //CheckBounds after ResizeUninitialized since index == length is allowed int elemSize = UnsafeUtility.SizeOf(); byte* basePtr = BufferHeader.GetElementPointer(m_Buffer); - UnsafeUtility.MemMove(basePtr + (index + 1) * elemSize, basePtr + index * elemSize, elemSize * (length - index)); + UnsafeUtility.MemMove(basePtr + (index + 1) * elemSize, basePtr + index * elemSize, (long)elemSize * (length - index)); this[index] = elem; } @@ -180,7 +199,7 @@ public void AddRange(NativeArray newElems) ResizeUninitialized(oldLength + newElems.Length); byte* basePtr = BufferHeader.GetElementPointer(m_Buffer); - UnsafeUtility.MemCpy(basePtr + oldLength * elemSize, newElems.GetUnsafeReadOnlyPtr(), elemSize * newElems.Length); + UnsafeUtility.MemCpy(basePtr + (long)oldLength * elemSize, newElems.GetUnsafeReadOnlyPtr(), (long)elemSize * newElems.Length); } public void RemoveRange(int index, int count) @@ -191,7 +210,7 @@ public void RemoveRange(int index, int count) int elemSize = UnsafeUtility.SizeOf(); byte* basePtr = BufferHeader.GetElementPointer(m_Buffer); - UnsafeUtility.MemMove(basePtr + index * elemSize, basePtr + (index + count) * elemSize, elemSize * (Length - count - index)); + UnsafeUtility.MemMove(basePtr + index * elemSize, basePtr + (index + count) * elemSize, (long)elemSize * (Length - count - index)); m_Buffer->Length -= count; } @@ -243,7 +262,11 @@ public NativeArray ToNativeArray() return AsNativeArray(); } - + public NativeArray ToNativeArray(Allocator allocator) + { + return new NativeArray(AsNativeArray(), allocator); + } + public void CopyFrom(NativeArray v) { ResizeUninitialized(v.Length); @@ -252,7 +275,7 @@ public void CopyFrom(NativeArray v) public void CopyFrom(T[] v) { -#if UNITY_CSHARP_TINY +#if NET_DOTS Clear(); foreach (var d in v) { diff --git a/Unity.Entities/Iterators/EntityArray.cs b/Unity.Entities/Iterators/EntityArray.cs deleted file mode 100644 index 1423a473..00000000 --- a/Unity.Entities/Iterators/EntityArray.cs +++ /dev/null @@ -1,109 +0,0 @@ -using System; -using Unity.Collections; -using Unity.Collections.LowLevel.Unsafe; - -namespace Unity.Entities -{ - /// - /// Enables array-like iteration over entities in a set of chunks. - /// - [NativeContainer] - [NativeContainerIsReadOnly] - [Obsolete("EntityArray is deprecated. Use IJobProcessComponentDataWithEntity or ComponentGroup.ToEntityArray(...) instead.")] - public unsafe struct EntityArray - { - private ComponentChunkIterator m_Iterator; - private ComponentChunkCache m_Cache; - - private readonly int m_Length; -#if ENABLE_UNITY_COLLECTIONS_CHECKS - private readonly AtomicSafetyHandle m_Safety; -#endif - public int Length => m_Length; - -#if ENABLE_UNITY_COLLECTIONS_CHECKS - internal EntityArray(ComponentChunkIterator iterator, int length, AtomicSafetyHandle safety) -#else - internal unsafe EntityArray(ComponentChunkIterator iterator, int length) -#endif - { - m_Length = length; - m_Iterator = iterator; - m_Cache = default(ComponentChunkCache); - -#if ENABLE_UNITY_COLLECTIONS_CHECKS - m_Safety = safety; -#endif - } - - public Entity this[int index] - { - get - { -#if ENABLE_UNITY_COLLECTIONS_CHECKS - AtomicSafetyHandle.CheckReadAndThrow(m_Safety); - if ((uint)index >= (uint)m_Length) - FailOutOfRangeError(index); -#endif - - if (index < m_Cache.CachedBeginIndex || index >= m_Cache.CachedEndIndex) - m_Iterator.MoveToEntityIndexAndUpdateCache(index, out m_Cache, false); - - return UnsafeUtility.ReadArrayElement(m_Cache.CachedPtr, index); - } - } - -#if ENABLE_UNITY_COLLECTIONS_CHECKS - private void FailOutOfRangeError(int index) - { - throw new IndexOutOfRangeException($"Index {index} is out of range of '{Length}' Length."); - } -#endif - - public NativeArray GetChunkArray(int startIndex, int maxCount) - { -#if ENABLE_UNITY_COLLECTIONS_CHECKS - AtomicSafetyHandle.CheckReadAndThrow(m_Safety); - - if (startIndex < 0) - FailOutOfRangeError(startIndex); - else if (startIndex + maxCount > m_Length) - FailOutOfRangeError(startIndex + maxCount); -#endif - - - m_Iterator.MoveToEntityIndexAndUpdateCache(startIndex, out m_Cache, false); - - void* ptr = (byte*) m_Cache.CachedPtr + startIndex * m_Cache.CachedSizeOf; - var count = Math.Min(maxCount, m_Cache.CachedEndIndex - startIndex); - - var arr = NativeArrayUnsafeUtility.ConvertExistingDataToNativeArray(ptr, count, Allocator.Invalid); - -#if ENABLE_UNITY_COLLECTIONS_CHECKS - NativeArrayUnsafeUtility.SetAtomicSafetyHandle(ref arr, m_Safety); -#endif - - return arr; - } - - public void CopyTo(NativeSlice dst, int startIndex = 0) - { - var copiedCount = 0; - while (copiedCount < dst.Length) - { - var chunkArray = GetChunkArray(startIndex + copiedCount, dst.Length - copiedCount); - dst.Slice(copiedCount, chunkArray.Length).CopyFrom(chunkArray); - - copiedCount += chunkArray.Length; - } - } - - public Entity[] ToArray() - { - var array = new Entity[Length]; - for (var i = 0; i != array.Length; i++) - array[i] = this[i]; - return array; - } - } -} diff --git a/Unity.Entities/Iterators/EntityArray.cs.meta b/Unity.Entities/Iterators/EntityArray.cs.meta deleted file mode 100644 index 3c408a56..00000000 --- a/Unity.Entities/Iterators/EntityArray.cs.meta +++ /dev/null @@ -1,13 +0,0 @@ -fileFormatVersion: 2 -guid: 5e10ee0838de946aca61081698f37214 -timeCreated: 1504806398 -licenseType: Pro -MonoImporter: - externalObjects: {} - serializedVersion: 2 - defaultReferences: [] - executionOrder: 0 - icon: {instanceID: 0} - userData: - assetBundleName: - assetBundleVariant: diff --git a/Unity.Entities/Iterators/EntityGroupManager.cs b/Unity.Entities/Iterators/EntityGroupManager.cs index b11ea029..c0e9f505 100644 --- a/Unity.Entities/Iterators/EntityGroupManager.cs +++ b/Unity.Entities/Iterators/EntityGroupManager.cs @@ -11,7 +11,7 @@ namespace Unity.Entities { internal unsafe class EntityGroupManager : IDisposable { - private readonly ComponentJobSafetyManager m_JobSafetyManager; + private readonly ComponentJobSafetyManager* m_JobSafetyManager; private ChunkAllocator m_GroupDataChunkAllocator; private EntityGroupDataList m_EntityGroupDatas; private NativeMultiHashMap m_EntityGroupDataCache; @@ -20,12 +20,12 @@ ref UnsafePtrList m_EntityGroupDatasUnsafePtrList { get { return ref *(UnsafePtrList*)UnsafeUtility.AddressOf(ref m_EntityGroupDatas); } } - - public EntityGroupManager(ComponentJobSafetyManager safetyManager) + + public EntityGroupManager(ComponentJobSafetyManager* safetyManager) { m_JobSafetyManager = safetyManager; m_GroupDataChunkAllocator = new ChunkAllocator(); - m_EntityGroupDataCache = new NativeMultiHashMap(1024, Allocator.Persistent); + m_EntityGroupDataCache = new NativeMultiHashMap(1024, Allocator.Persistent); } public void Dispose() @@ -49,7 +49,7 @@ public void Dispose() else allList.Add(requiredTypes[i]); } - + // NativeList.ToArray requires GC Pinning, not supported in Tiny var allCount = allList.Length; var noneCount = noneList.Length; @@ -60,16 +60,16 @@ public void Dispose() for (int i = 0; i < noneCount; i++) noneTypes[i] = noneList[i]; - var query = new EntityArchetypeQuery + var query = new EntityQueryDesc { All = allTypes, None = noneTypes }; - + allList.Dispose(); noneList.Dispose(); - return CreateQuery(ref scratchAllocator, new EntityArchetypeQuery[] { query }); + return CreateQuery(ref scratchAllocator, new EntityQueryDesc[] { query }); } void ConstructTypeArray(ref ScratchAllocator scratchAllocator, ComponentType[] types, out int* outTypes, out byte* outAccessModes, out int outLength) @@ -85,7 +85,7 @@ void ConstructTypeArray(ref ScratchAllocator scratchAllocator, ComponentType[] t outLength = types.Length; outTypes = (int*)scratchAllocator.Allocate(types.Length); outAccessModes = (byte*)scratchAllocator.Allocate(types.Length); - + var sortedTypes = stackalloc ComponentType[types.Length]; for (var i = 0; i < types.Length; ++i) SortingUtilities.InsertSorted(sortedTypes, i, types[i]); @@ -102,14 +102,14 @@ void IncludeDependentWriteGroups(ComponentType type, NativeList e { if (type.AccessModeType != ComponentType.AccessMode.ReadOnly) return; - + var writeGroupTypes = TypeManager.GetWriteGroupTypes(type.TypeIndex); for (int i = 0; i < writeGroupTypes.Length; i++) { var excludedComponentType = GetWriteGroupReadOnlyComponentType(writeGroupTypes, i); if (explicitList.Contains(excludedComponentType)) continue; - + explicitList.Add(excludedComponentType); IncludeDependentWriteGroups(excludedComponentType, explicitList); } @@ -128,7 +128,7 @@ void ExcludeWriteGroups(ComponentType type, NativeList noneList, { if (type.AccessModeType == ComponentType.AccessMode.ReadOnly) return; - + var writeGroupTypes = TypeManager.GetWriteGroupTypes(type.TypeIndex); for (int i = 0; i < writeGroupTypes.Length; i++) { @@ -137,7 +137,7 @@ void ExcludeWriteGroups(ComponentType type, NativeList noneList, continue; if (explicitList.Contains(excludedComponentType)) continue; - + noneList.Add(excludedComponentType); } } @@ -154,30 +154,18 @@ NativeList CreateExplicitTypeList(ComponentType[] typesNone, Comp return explicitList; } - ArchetypeQuery* CreateQuery(ref ScratchAllocator scratchAllocator, EntityArchetypeQuery[] query) + ArchetypeQuery* CreateQuery(ref ScratchAllocator scratchAllocator, EntityQueryDesc[] queryDesc) { - var outQuery = (ArchetypeQuery*)scratchAllocator.Allocate(sizeof(ArchetypeQuery) * query.Length, UnsafeUtility.AlignOf()); - for (int q = 0; q != query.Length; q++) + var outQuery = (ArchetypeQuery*)scratchAllocator.Allocate(sizeof(ArchetypeQuery) * queryDesc.Length, UnsafeUtility.AlignOf()); + for (int q = 0; q != queryDesc.Length; q++) { - var typesNone = query[q].None; - var typesAll = query[q].All; - var typesAny = query[q].Any; - -#if ENABLE_UNITY_COLLECTIONS_CHECKS - // Check that query doesn't contain any ExcludeComponent... - { - for (int i=0;i CreateExplicitTypeList(ComponentType[] typesNone, Comp typesNone[i] = ComponentType.ReadOnly(typesNone[i].TypeIndex); } - var isFilterWriteGroup = (query[q].Options & EntityArchetypeQueryOptions.FilterWriteGroup) != 0; + var isFilterWriteGroup = (queryDesc[q].Options & EntityQueryOptions.FilterWriteGroup) != 0; if (isFilterWriteGroup) { // Each ReadOnly in any or all @@ -201,7 +189,7 @@ NativeList CreateExplicitTypeList(ComponentType[] typesNone, Comp // Each ReadWrite in any or all // if has WriteGroup types, - // Add to none (if not exist in any or all or none) + // Add to none (if not exist in any or all or none) var noneList = new NativeList(typesNone.Length, Allocator.Temp); for (int i = 0; i < typesNone.Length; i++) noneList.Add(typesNone[i]); @@ -219,7 +207,7 @@ NativeList CreateExplicitTypeList(ComponentType[] typesNone, Comp ConstructTypeArray(ref scratchAllocator, typesNone, out outQuery[q].None, out outQuery[q].NoneAccessMode, out outQuery[q].NoneCount); ConstructTypeArray(ref scratchAllocator, typesAll, out outQuery[q].All, out outQuery[q].AllAccessMode, out outQuery[q].AllCount); ConstructTypeArray(ref scratchAllocator, typesAny, out outQuery[q].Any, out outQuery[q].AnyAccessMode, out outQuery[q].AnyCount); - outQuery[q].Options = query[q].Options; + outQuery[q].Options = queryDesc[q].Options; } return outQuery; @@ -246,18 +234,18 @@ public static bool CompareQueryArray(ComponentType[] filter, int* typeArray, byt return true; } - public static bool CompareQuery(EntityArchetypeQuery[] query, EntityGroupData* groupData) + public static bool CompareQuery(EntityQueryDesc[] queryDesc, EntityGroupData* groupData) { - if (groupData->ArchetypeQueryCount != query.Length) + if (groupData->ArchetypeQueryCount != queryDesc.Length) return false; - for (int i = 0; i != query.Length; i++) + for (int i = 0; i != queryDesc.Length; i++) { - if (!CompareQueryArray(query[i].All, groupData->ArchetypeQuery[i].All, groupData->ArchetypeQuery[i].AllAccessMode, groupData->ArchetypeQuery[i].AllCount)) + if (!CompareQueryArray(queryDesc[i].All, groupData->ArchetypeQuery[i].All, groupData->ArchetypeQuery[i].AllAccessMode, groupData->ArchetypeQuery[i].AllCount)) return false; - if (!CompareQueryArray(query[i].None, groupData->ArchetypeQuery[i].None, groupData->ArchetypeQuery[i].NoneAccessMode, groupData->ArchetypeQuery[i].NoneCount)) + if (!CompareQueryArray(queryDesc[i].None, groupData->ArchetypeQuery[i].None, groupData->ArchetypeQuery[i].NoneAccessMode, groupData->ArchetypeQuery[i].NoneCount)) return false; - if (!CompareQueryArray(query[i].Any, groupData->ArchetypeQuery[i].Any, groupData->ArchetypeQuery[i].AnyAccessMode, groupData->ArchetypeQuery[i].AnyCount)) + if (!CompareQueryArray(queryDesc[i].Any, groupData->ArchetypeQuery[i].Any, groupData->ArchetypeQuery[i].AnyAccessMode, groupData->ArchetypeQuery[i].AnyCount)) return false; } @@ -270,7 +258,7 @@ public static bool CompareComponents(ComponentType* componentTypes, int componen for (var k = 0; k < componentTypesCount; ++k) if (componentTypes[k].TypeIndex == TypeManager.GetTypeIndex()) throw new ArgumentException( - "ComponentGroup.CompareComponents may not include typeof(Entity), it is implicit"); + "EntityQuery.CompareComponents may not include typeof(Entity), it is implicit"); #endif var sortedTypes = stackalloc ComponentType[componentTypesCount]; @@ -278,8 +266,8 @@ public static bool CompareComponents(ComponentType* componentTypes, int componen { SortingUtilities.InsertSorted(sortedTypes, i, componentTypes[i]); } - - // ComponentGroups are constructed including the Entity ID + + // EntityQueries are constructed including the Entity ID if (componentTypesCount + 1 != groupData->RequiredComponentsCount) return false; @@ -327,31 +315,31 @@ private int IntersectSortedComponentIndexArrays(int* arrayA, int arrayACount, in var intersectionCount = 0; var i = 0; - var j = 0; - while (i < arrayACount && j < arrayBCount) - { - if (arrayA[i] < arrayB[j]) - i++; - else if (arrayB[j] < arrayA[i]) - j++; + var j = 0; + while (i < arrayACount && j < arrayBCount) + { + if (arrayA[i] < arrayB[j]) + i++; + else if (arrayB[j] < arrayA[i]) + j++; else { outArray[intersectionCount++] = arrayB[j]; - i++; - j++; - } + i++; + j++; + } } return intersectionCount; } - // Calculates the intersection of "All" queries + // Calculates the intersection of "All" queriesDesc private ComponentType* CalculateRequiredComponentsFromQuery(ref ScratchAllocator allocator, ArchetypeQuery* queries, int queryCount, out int outRequiredComponentsCount) { var maxIntersectionCount = 0; for (int queryIndex = 0; queryIndex < queryCount; ++queryIndex) - maxIntersectionCount = math.max(maxIntersectionCount, queries[queryIndex].AllCount); - + maxIntersectionCount = math.max(maxIntersectionCount, queries[queryIndex].AllCount); + var intersection = (int*)allocator.Allocate(maxIntersectionCount); UnsafeUtility.MemCpy(intersection, queries[0].All, sizeof(int) * queries[0].AllCount); @@ -361,31 +349,31 @@ private int IntersectSortedComponentIndexArrays(int* arrayA, int arrayACount, in intersectionCount = IntersectSortedComponentIndexArrays(intersection, intersectionCount, queries[i].All, queries[i].AllCount, intersection); } - + var outRequiredComponents = (ComponentType*)allocator.Allocate(intersectionCount + 1); outRequiredComponents[0] = ComponentType.ReadWrite(); for (int i = 0; i < intersectionCount; ++i) { outRequiredComponents[i + 1] = ComponentType.FromTypeIndex(intersection[i]); } - + outRequiredComponentsCount = intersectionCount + 1; return outRequiredComponents; } - public ComponentGroup CreateEntityGroup(ArchetypeManager typeMan, EntityDataManager* entityDataManager, EntityArchetypeQuery[] query) + public EntityQuery CreateEntityGroup(ArchetypeManager typeMan, EntityDataManager* entityDataManager, EntityQueryDesc[] queryDesc) { - //@TODO: Support for CreateEntityGroup with query but using ComponentDataArray etc + //@TODO: Support for CreateEntityGroup with queryDesc but using ComponentDataArray etc var buffer = stackalloc byte[1024]; var scratchAllocator = new ScratchAllocator(buffer, 1024); - var archetypeQuery = CreateQuery(ref scratchAllocator, query); + var archetypeQuery = CreateQuery(ref scratchAllocator, queryDesc); - var outRequiredComponents = CalculateRequiredComponentsFromQuery(ref scratchAllocator, archetypeQuery, query.Length, out var outRequiredComponentsCount); - - return CreateEntityGroup(typeMan, entityDataManager, archetypeQuery, query.Length, outRequiredComponents, outRequiredComponentsCount); + var outRequiredComponents = CalculateRequiredComponentsFromQuery(ref scratchAllocator, archetypeQuery, queryDesc.Length, out var outRequiredComponentsCount); + + return CreateEntityGroup(typeMan, entityDataManager, archetypeQuery, queryDesc.Length, outRequiredComponents, outRequiredComponentsCount); } - public ComponentGroup CreateEntityGroup(ArchetypeManager typeMan, EntityDataManager* entityDataManager, ComponentType* inRequiredComponents, int inRequiredComponentsCount) + public EntityQuery CreateEntityGroup(ArchetypeManager typeMan, EntityDataManager* entityDataManager, ComponentType* inRequiredComponents, int inRequiredComponentsCount) { var buffer = stackalloc byte[1024]; var scratchAllocator = new ScratchAllocator(buffer, 1024); @@ -423,10 +411,10 @@ bool Matches(EntityGroupData* grp, ArchetypeQuery* archetypeQueries, int archety UnsafeUtility.MemCpy(pointer, source, bytes); return pointer; } - - public ComponentGroup CreateEntityGroup(ArchetypeManager typeMan, EntityDataManager* entityDataManager, + + public EntityQuery CreateEntityGroup(ArchetypeManager typeMan, EntityDataManager* entityDataManager, ArchetypeQuery* query, int queryCount, ComponentType* component, int componentCount) - { + { //@TODO: Validate that required types is subset of archetype filters all... int hash = (int)math.hash(component, componentCount * sizeof(ComponentType)); @@ -474,7 +462,7 @@ public ComponentGroup CreateEntityGroup(ArchetypeManager typeMan, EntityDataMana m_EntityGroupDatasUnsafePtrList.Add(cachedGroup); } - return new ComponentGroup(cachedGroup, m_JobSafetyManager, typeMan, entityDataManager); + return new EntityQuery(cachedGroup, m_JobSafetyManager, typeMan, entityDataManager); } void InitializeReaderWriter(EntityGroupData* grp, ComponentType* requiredTypes, int requiredCount) @@ -620,7 +608,7 @@ static bool TestMatchingArchetypeNone(Archetype* archetype, int* noneTypes, int return true; } - static bool TestMatchingArchetypeAll(Archetype* archetype, int* allTypes, int allCount, EntityArchetypeQueryOptions options) + static bool TestMatchingArchetypeAll(Archetype* archetype, int* allTypes, int allCount, EntityQueryOptions options) { var componentTypes = archetype->Types; var componentTypesCount = archetype->TypesCount; @@ -628,8 +616,8 @@ static bool TestMatchingArchetypeAll(Archetype* archetype, int* allTypes, int al var disabledTypeIndex = TypeManager.GetTypeIndex(); var prefabTypeIndex = TypeManager.GetTypeIndex(); var chunkHeaderTypeIndex = TypeManager.GetTypeIndex(); - var includeInactive = (options & EntityArchetypeQueryOptions.IncludeDisabled) != 0; - var includePrefab = (options & EntityArchetypeQueryOptions.IncludePrefab) != 0; + var includeInactive = (options & EntityQueryOptions.IncludeDisabled) != 0; + var includePrefab = (options & EntityQueryOptions.IncludePrefab) != 0; var includeChunkHeader = false; for (var i = 0; i < componentTypesCount; i++) @@ -644,7 +632,7 @@ static bool TestMatchingArchetypeAll(Archetype* archetype, int* allTypes, int al includePrefab = true; if (allTypeIndex == chunkHeaderTypeIndex) includeChunkHeader = true; - + if (componentTypeIndex == allTypeIndex) foundCount++; } } @@ -681,7 +669,7 @@ unsafe struct MatchingArchetypeList public int Count; public int Capacity; } - + unsafe struct ArchetypeQuery : IEquatable { public int* Any; @@ -695,9 +683,9 @@ unsafe struct ArchetypeQuery : IEquatable public int* None; public byte* NoneAccessMode; public int NoneCount; - - public EntityArchetypeQueryOptions Options; - + + public EntityQueryOptions Options; + public bool Equals(ArchetypeQuery other) { if (AnyCount != other.AnyCount) @@ -706,7 +694,7 @@ public bool Equals(ArchetypeQuery other) return false; if (NoneCount != other.NoneCount) return false; - if (AnyCount > 0 && UnsafeUtility.MemCmp(Any, other.Any, sizeof(int) * AnyCount) != 0 && + if (AnyCount > 0 && UnsafeUtility.MemCmp(Any, other.Any, sizeof(int) * AnyCount) != 0 && UnsafeUtility.MemCmp(AnyAccessMode, other.AnyAccessMode, sizeof(byte) * AnyCount) != 0) return false; if (AllCount > 0 && UnsafeUtility.MemCmp(All, other.All, sizeof(int) * AllCount) != 0 && @@ -717,9 +705,9 @@ public bool Equals(ArchetypeQuery other) return false; if (Options != other.Options) return false; - + return true; - } + } public override int GetHashCode() { unchecked @@ -745,7 +733,7 @@ unsafe struct EntityGroupDataList public int Count; public int Capacity; } - + unsafe struct EntityGroupData : IDisposable { //@TODO: better name or remove entirely... @@ -767,7 +755,7 @@ public ref UnsafePtrList MatchingArchetypesUnsafePtrList { get { return ref *(UnsafePtrList*)UnsafeUtility.AddressOf(ref MatchingArchetypes); } } - + public void Dispose() { MatchingArchetypesUnsafePtrList.Dispose(); diff --git a/Unity.Entities/Iterators/ComponentGroup.cs b/Unity.Entities/Iterators/EntityQuery.cs similarity index 65% rename from Unity.Entities/Iterators/ComponentGroup.cs rename to Unity.Entities/Iterators/EntityQuery.cs index b0c50b64..82735a59 100644 --- a/Unity.Entities/Iterators/ComponentGroup.cs +++ b/Unity.Entities/Iterators/EntityQuery.cs @@ -1,5 +1,7 @@ using System; using System.Collections.Generic; +using System.ComponentModel; +using System.Diagnostics; using Unity.Burst; using Unity.Collections; using Unity.Collections.LowLevel.Unsafe; @@ -7,11 +9,19 @@ namespace Unity.Entities { + public class EntityQueryDescValidationException : Exception + { + public EntityQueryDescValidationException(string message) : base(message) + { + + } + } + /// - /// Defines a query to find archetypes with specific components. + /// Defines a queryDesc to find archetypes with specific components. /// /// - /// A query combines components in the All, Any, and None sets according to the + /// A queryDesc combines components in the All, Any, and None sets according to the /// following rules: /// /// * All - Includes archetypes that have every component in this set @@ -24,106 +34,165 @@ namespace Unity.Entities /// * Enemy1 has components: Position, Rotation, Melee /// * Enemy2 has components: Position, Rotation, Ranger /// - /// The query below would give you all of the archetypes that: + /// The queryDesc below would give you all of the archetypes that: /// have any of [Melee or Ranger], AND have none of [Player], AND have all of [Position and Rotation] /// - /// new EntityArchetypeQuery { + /// new EntityQueryDesc { /// Any = new ComponentType[] {typeof(Melee), typeof(Ranger)}, /// None = new ComponentType[] {typeof(Player)}, /// All = new ComponentType[] {typeof(Position), typeof(Rotation)} /// } /// /// - /// In other words, the query selects the Enemy1 and Enemy2 entities, but not the Player entity. + /// In other words, the queryDesc selects the Enemy1 and Enemy2 entities, but not the Player entity. /// - public class EntityArchetypeQuery + public class EntityQueryDesc { /// - /// The query includes archetypes that contain at least one (but possibly more) of the + /// The queryDesc includes archetypes that contain at least one (but possibly more) of the /// components in the Any list. /// public ComponentType[] Any = Array.Empty(); /// - /// The query excludes archetypes that contain any of the + /// The queryDesc excludes archetypes that contain any of the /// components in the None list. /// public ComponentType[] None = Array.Empty(); /// - /// The query includes archetypes that contain all of the + /// The queryDesc includes archetypes that contain all of the /// components in the All list. /// public ComponentType[] All = Array.Empty(); /// - /// Specialized query options. + /// Specialized queryDesc options. /// /// - /// You should not need to set these options for most queries. + /// You should not need to set these options for most queriesDesc. /// /// Options is a bit mask; use the bitwise OR operator to combine multiple options. /// - public EntityArchetypeQueryOptions Options = EntityArchetypeQueryOptions.Default; + public EntityQueryOptions Options = EntityQueryOptions.Default; + + [Conditional("ENABLE_UNITY_COLLECTIONS_CHECKS")] + public void Validate() + { + // Determine the number of ComponentTypes contained in the filters + var itemCount = None.Length + All.Length + Any.Length; + + // Project all the ComponentType Ids of None, All, Any queryDesc filters into the same array to identify duplicated later on + // Also, check that queryDesc doesn't contain any ExcludeComponent... + + var allComponentTypeIds = new NativeArray(itemCount, Allocator.Temp); + var curComponentTypeIndex = 0; + + for (int i = 0; i < None.Length; i++) + { + var componentType = None[i]; + allComponentTypeIds[curComponentTypeIndex++] = componentType.TypeIndex; + if (componentType.AccessModeType == ComponentType.AccessMode.Exclude) + throw new ArgumentException("EntityQueryDesc cannot contain Exclude Component types"); + } + for (int i = 0; i < All.Length; i++) + { + var componentType = All[i]; + allComponentTypeIds[curComponentTypeIndex++] = componentType.TypeIndex; + if (componentType.AccessModeType == ComponentType.AccessMode.Exclude) + throw new ArgumentException("EntityQueryDesc cannot contain Exclude Component types"); + } + for (int i = 0; i < Any.Length; i++) + { + var componentType = Any[i]; + allComponentTypeIds[curComponentTypeIndex++] = componentType.TypeIndex; + if (componentType.AccessModeType == ComponentType.AccessMode.Exclude) + throw new ArgumentException("EntityQueryDesc cannot contain Exclude Component types"); + } + + // Check for duplicate, only if necessary + if (itemCount > 1) + { + // Sort the Ids to have identical value adjacent + allComponentTypeIds.Sort(); + + // Check for identical values + var refId = allComponentTypeIds[0]; + for (int i = 1; i < allComponentTypeIds.Length; i++) + { + var curId = allComponentTypeIds[i]; + if (curId == refId) + { +#if NET_DOTS + throw new EntityQueryDescValidationException($"Cannot create a queryDesc with more than one filter containing the same component type, type index {curId}"); +#else + var compType = TypeManager.GetType(curId); + throw new EntityQueryDescValidationException($"Cannot create a queryDesc with more than one filter containing the same component type, type name {compType.Name}"); +#endif + } + + refId = curId; + } + } + } } /// - /// The bit flags to use for the field. + /// The bit flags to use for the field. /// [Flags] - public enum EntityArchetypeQueryOptions + public enum EntityQueryOptions { /// /// No options specified. /// Default = 0, /// - /// The query includes the special component. + /// The queryDesc includes the special component. /// IncludePrefab = 1, /// - /// The query includes the special component. + /// The queryDesc includes the special component. /// IncludeDisabled = 2, /// - /// The query should filter selected entities based on the - /// settings of the components specified in the query. + /// The queryDesc should filter selected entities based on the + /// settings of the components specified in the queryDesc. /// FilterWriteGroup = 4, } - //@TODO: Rename to EntityView /// - /// A ComponentGroup provides a query-based view of your component data. + /// A EntityQuery provides a queryDesc-based view of your component data. /// /// - /// A ComponentGroup defines a view of your data based on a query for the set of + /// A EntityQuery defines a view of your data based on a queryDesc for the set of /// component types that an archetype must contain in order for its chunks and entities /// to be included in the view. You can also exclude archetypes that contain specific types - /// of components. For simple queries, you can create a ComponentGroup based on an array of - /// component types. The following example defines a ComponentGroup that finds all entities + /// of components. For simple queriesDesc, you can create a EntityQuery based on an array of + /// component types. The following example defines a EntityQuery that finds all entities /// with both RotationQuaternion and RotationSpeed components. /// /// - /// ComponentGroup m_Group = GetComponentGroup(typeof(RotationQuaternion), + /// EntityQuery m_Group = GetEntityQuery(typeof(RotationQuaternion), /// ComponentType.ReadOnly{RotationSpeed}()); /// /// - /// The query uses `ComponentType.ReadOnly` instead of the simpler `typeof` expression + /// The queryDesc uses `ComponentType.ReadOnly` instead of the simpler `typeof` expression /// to designate that the system does not write to RotationSpeed. Always specify read only /// when possible, since there are fewer constraints on read access to data, which can help /// the Job scheduler execute your Jobs more efficiently. /// - /// For more complex queries, you can use an instead of a + /// For more complex queriesDesc, you can use an instead of a /// simple list of component types. /// - /// Use the or - /// functions - /// to get a ComponentGroup instance. + /// Use the or + /// functions + /// to get a EntityQuery instance. /// - public unsafe class ComponentGroup : IDisposable + public unsafe class EntityQuery : IDisposable { - readonly ComponentJobSafetyManager m_SafetyManager; - readonly EntityGroupData* m_GroupData; - readonly EntityDataManager* m_EntityDataManager; - ComponentGroupFilter m_Filter; + readonly ComponentJobSafetyManager* m_SafetyManager; + readonly EntityGroupData* m_GroupData; + readonly EntityDataManager* m_EntityDataManager; + EntityQueryFilter m_Filter; #if ENABLE_UNITY_COLLECTIONS_CHECKS internal string DisallowDisposing = null; @@ -132,23 +201,23 @@ public unsafe class ComponentGroup : IDisposable // TODO: this is temporary, used to cache some state to avoid recomputing the TransformAccessArray. We need to improve this. internal IDisposable m_CachedState; - internal ComponentGroup(EntityGroupData* groupData, ComponentJobSafetyManager safetyManager, ArchetypeManager typeManager, EntityDataManager* entityDataManager) + internal EntityQuery(EntityGroupData* groupData, ComponentJobSafetyManager* safetyManager, ArchetypeManager typeManager, EntityDataManager* entityDataManager) { m_GroupData = groupData; m_EntityDataManager = entityDataManager; - m_Filter = default(ComponentGroupFilter); + m_Filter = default(EntityQueryFilter); m_SafetyManager = safetyManager; ArchetypeManager = typeManager; EntityDataManager = entityDataManager; } internal EntityDataManager* EntityDataManager { get; } - internal ComponentJobSafetyManager SafetyManager => m_SafetyManager; + internal ComponentJobSafetyManager* SafetyManager => m_SafetyManager; /// - /// Ignore this ComponentGroup if it has no entities in any of its archetypes. + /// Ignore this EntityQuery if it has no entities in any of its archetypes. /// - /// True if this ComponentGroup has no entities. False if it has 1 or more entities. + /// True if this EntityQuery has no entities. False if it has 1 or more entities. public bool IsEmptyIgnoreFilter { get @@ -163,7 +232,7 @@ public bool IsEmptyIgnoreFilter return true; } } -#if UNITY_CSHARP_TINY +#if NET_DOTS internal class SlowListSet { internal List items; @@ -188,12 +257,12 @@ internal T[] ToArray() #endif /// - /// Gets the array of objects included in this ComponentGroup. + /// Gets the array of objects included in this EntityQuery. /// /// Array of ComponentTypes internal ComponentType[] GetQueryTypes() { -#if !UNITY_CSHARP_TINY +#if !NET_DOTS var types = new HashSet(); #else var types = new SlowListSet(); @@ -215,7 +284,7 @@ internal ComponentType[] GetQueryTypes() } } -#if !UNITY_CSHARP_TINY +#if !NET_DOTS var array = new ComponentType[types.Count]; var t = 0; foreach (var type in types) @@ -227,7 +296,7 @@ internal ComponentType[] GetQueryTypes() } /// - /// Packed array of this ComponentGroup's ReadOnly and writable ComponentTypes. + /// Packed array of this EntityQuery's ReadOnly and writable ComponentTypes. /// ReadOnly ComponentTypes come before writable types in this array. /// /// Array of ComponentTypes @@ -264,43 +333,43 @@ public void Dispose() #if ENABLE_UNITY_COLLECTIONS_CHECKS /// - /// Gets safety handle to a ComponentType required by this ComponentGroup. + /// Gets safety handle to a ComponentType required by this EntityQuery. /// - /// Index of a ComponentType in this ComponentGroup's RequiredComponents list./param> + /// Index of a ComponentType in this EntityQuery's RequiredComponents list./param> /// AtomicSafetyHandle for a ComponentType - internal AtomicSafetyHandle GetSafetyHandle(int indexInComponentGroup) + internal AtomicSafetyHandle GetSafetyHandle(int indexInEntityQuery) { - var type = m_GroupData->RequiredComponents + indexInComponentGroup; + var type = m_GroupData->RequiredComponents + indexInEntityQuery; var isReadOnly = type->AccessModeType == ComponentType.AccessMode.ReadOnly; - return m_SafetyManager.GetSafetyHandle(type->TypeIndex, isReadOnly); + return m_SafetyManager->GetSafetyHandle(type->TypeIndex, isReadOnly); } /// - /// Gets buffer safety handle to a ComponentType required by this ComponentGroup. + /// Gets buffer safety handle to a ComponentType required by this EntityQuery. /// - /// Index of a ComponentType in this ComponentGroup's RequiredComponents list./param> + /// Index of a ComponentType in this EntityQuery's RequiredComponents list./param> /// AtomicSafetyHandle for a buffer - internal AtomicSafetyHandle GetBufferSafetyHandle(int indexInComponentGroup) + internal AtomicSafetyHandle GetBufferSafetyHandle(int indexInEntityQuery) { - var type = m_GroupData->RequiredComponents + indexInComponentGroup; - return m_SafetyManager.GetBufferSafetyHandle(type->TypeIndex); + var type = m_GroupData->RequiredComponents + indexInEntityQuery; + return m_SafetyManager->GetBufferSafetyHandle(type->TypeIndex); } #endif - bool GetIsReadOnly(int indexInComponentGroup) + bool GetIsReadOnly(int indexInEntityQuery) { - var type = m_GroupData->RequiredComponents + indexInComponentGroup; + var type = m_GroupData->RequiredComponents + indexInEntityQuery; var isReadOnly = type->AccessModeType == ComponentType.AccessMode.ReadOnly; return isReadOnly; } /// - /// Calculates the number of entities selected by this ComponentGroup. + /// Calculates the number of entities selected by this EntityQuery. /// /// - /// The ComponentGroup must run the query and apply any filters to calculate the entity count. + /// The EntityQuery must run the queryDesc and apply any filters to calculate the entity count. /// - /// The number of entities based on the current ComponentGroup properties. + /// The number of entities based on the current EntityQuery properties. public int CalculateLength() { SyncFilterTypes(); @@ -308,17 +377,17 @@ public int CalculateLength() } /// - /// Gets iterator to chunks associated with this ComponentGroup. + /// Gets iterator to chunks associated with this EntityQuery. /// - /// ComponentChunkIterator for this ComponentGroup + /// ComponentChunkIterator for this EntityQuery internal ComponentChunkIterator GetComponentChunkIterator() { return new ComponentChunkIterator(m_GroupData->MatchingArchetypes, m_EntityDataManager->GlobalSystemVersion, ref m_Filter); } /// - /// Index of a ComponentType in this ComponentGroup's RequiredComponents list. - /// For example, you have a ComponentGroup that requires these ComponentTypes: Position, Velocity, and Color. + /// Index of a ComponentType in this EntityQuery's RequiredComponents list. + /// For example, you have a EntityQuery that requires these ComponentTypes: Position, Velocity, and Color. /// /// These are their type indices (according to the TypeManager): /// Position.TypeIndex == 3 @@ -330,119 +399,22 @@ internal ComponentChunkIterator GetComponentChunkIterator() /// /// Index of a ComponentType in the TypeManager /// An index into RequiredComponents. - internal int GetIndexInComponentGroup(int componentType) + internal int GetIndexInEntityQuery(int componentType) { - // Go through all the required component types in this ComponentGroup until you find the matching component type index. + // Go through all the required component types in this EntityQuery until you find the matching component type index. var componentIndex = 0; while (componentIndex < m_GroupData->RequiredComponentsCount && m_GroupData->RequiredComponents[componentIndex].TypeIndex != componentType) ++componentIndex; #if ENABLE_UNITY_COLLECTIONS_CHECKS - if (componentIndex >= m_GroupData->RequiredComponentsCount) + if (componentIndex >= m_GroupData->RequiredComponentsCount || m_GroupData->RequiredComponents[componentIndex].AccessModeType == ComponentType.AccessMode.Exclude) throw new InvalidOperationException( $"Trying to get iterator for {TypeManager.GetType(componentType)} but the required component type was not declared in the EntityGroup."); #endif return componentIndex; } - [Obsolete("GetComponentDataArray is deprecated. Use IJobProcessComponentData or ToComponentDataArray/CopyFromComponentDataArray instead.")] - internal void GetComponentDataArray(ref ComponentChunkIterator iterator, int indexInComponentGroup, - int length, out ComponentDataArray output) where T : struct, IComponentData - { -#if ENABLE_UNITY_COLLECTIONS_CHECKS - var typeIndex = TypeManager.GetTypeIndex(); - var componentType = ComponentType.FromTypeIndex(typeIndex); - if (componentType.IsZeroSized) - throw new ArgumentException($"GetComponentDataArray<{typeof(T)}> cannot be called on zero-sized IComponentData"); -#endif - - iterator.IndexInComponentGroup = indexInComponentGroup; -#if ENABLE_UNITY_COLLECTIONS_CHECKS - output = new ComponentDataArray(iterator, length, GetSafetyHandle(indexInComponentGroup)); -#else - output = new ComponentDataArray(iterator, length); -#endif - } - - [Obsolete("GetComponentDataArray is deprecated. Use IJobProcessComponentData or ToComponentDataArray/CopyFromComponentDataArray instead.")] - public ComponentDataArray GetComponentDataArray() where T : struct, IComponentData - { - var typeIndex = TypeManager.GetTypeIndex(); - -#if ENABLE_UNITY_COLLECTIONS_CHECKS - var componentType = ComponentType.FromTypeIndex(typeIndex); - if (componentType.IsZeroSized) - throw new ArgumentException($"GetComponentDataArray<{typeof(T)}> cannot be called on zero-sized IComponentData"); -#endif - - int length = CalculateLength(); - ComponentChunkIterator iterator = GetComponentChunkIterator(); - var indexInComponentGroup = GetIndexInComponentGroup(typeIndex); - - ComponentDataArray res; - GetComponentDataArray(ref iterator, indexInComponentGroup, length, out res); - return res; - } - - [Obsolete("GetSharedComponentDataArray is deprecated.")] - internal void GetSharedComponentDataArray(ref ComponentChunkIterator iterator, int indexInComponentGroup, - int length, out SharedComponentDataArray output) where T : struct, ISharedComponentData - { - iterator.IndexInComponentGroup = indexInComponentGroup; -#if ENABLE_UNITY_COLLECTIONS_CHECKS - var componentTypeIndex = m_GroupData->RequiredComponents[indexInComponentGroup].TypeIndex; - output = new SharedComponentDataArray(ArchetypeManager.GetSharedComponentDataManager(), - indexInComponentGroup, iterator, length, m_SafetyManager.GetSafetyHandle(componentTypeIndex, true)); -#else - output = new SharedComponentDataArray(ArchetypeManager.GetSharedComponentDataManager(), - indexInComponentGroup, iterator, length); -#endif - } - - /// - /// Creates an array containing ISharedComponentData of a given type T. - /// - /// NativeArray of ISharedComponentData in this ComponentGroup. - [Obsolete("GetSharedComponentDataArray is deprecated.")] - public SharedComponentDataArray GetSharedComponentDataArray() where T : struct, ISharedComponentData - { - int length = CalculateLength(); - ComponentChunkIterator iterator = GetComponentChunkIterator(); - var indexInComponentGroup = GetIndexInComponentGroup(TypeManager.GetTypeIndex()); - - SharedComponentDataArray res; - GetSharedComponentDataArray(ref iterator, indexInComponentGroup, length, out res); - return res; - } - - [Obsolete("GetBufferArray is deprecated.")] - internal void GetBufferArray(ref ComponentChunkIterator iterator, int indexInComponentGroup, int length, - out BufferArray output) where T : struct, IBufferElementData - { - iterator.IndexInComponentGroup = indexInComponentGroup; - -#if ENABLE_UNITY_COLLECTIONS_CHECKS - output = new BufferArray(iterator, length, GetIsReadOnly(indexInComponentGroup), - GetSafetyHandle(indexInComponentGroup), - GetBufferSafetyHandle(indexInComponentGroup)); -#else - output = new BufferArray(iterator, length, GetIsReadOnly(indexInComponentGroup)); -#endif - } - - [Obsolete("GetBufferArray is deprecated.")] - public BufferArray GetBufferArray() where T : struct, IBufferElementData - { - int length = CalculateLength(); - ComponentChunkIterator iterator = GetComponentChunkIterator(); - var indexInComponentGroup = GetIndexInComponentGroup(TypeManager.GetTypeIndex()); - - BufferArray res; - GetBufferArray(ref iterator, indexInComponentGroup, length, out res); - return res; - } - /// - /// Creates an array with all the chunks in this ComponentGroup. + /// Creates an array with all the chunks in this EntityQuery. /// Gives the caller a job handle so it can wait for GatherChunks to finish. /// /// Allocator to use for the array. @@ -456,18 +428,18 @@ public NativeArray CreateArchetypeChunkArray(Allocator allocator { var filterCount = m_Filter.Changed.Count; var readerTypes = stackalloc int[filterCount]; - fixed (int* indexInComponentGroupPtr = m_Filter.Changed.IndexInComponentGroup) + fixed (int* indexInEntityQueryPtr = m_Filter.Changed.IndexInEntityQuery) for (int i = 0; i < filterCount; ++i) - readerTypes[i] = m_GroupData->RequiredComponents[indexInComponentGroupPtr[i]].TypeIndex; + readerTypes[i] = m_GroupData->RequiredComponents[indexInEntityQueryPtr[i]].TypeIndex; - dependency = m_SafetyManager.GetDependency(readerTypes, filterCount,null, 0); + dependency = m_SafetyManager->GetDependency(readerTypes, filterCount,null, 0); } return ComponentChunkIterator.CreateArchetypeChunkArray(m_GroupData->MatchingArchetypes, allocator, out jobhandle, ref m_Filter, dependency); } /// - /// Creates an array with all the chunks in this ComponentGroup. + /// Creates an array with all the chunks in this EntityQuery. /// Waits for the GatherChunks job to complete here. /// /// Allocator to use for the array. @@ -488,11 +460,11 @@ public NativeArray CreateArchetypeChunkArray(Allocator allocator /// The type of memory to allocate. /// A handle that you can use as a dependency for a Job /// that uses the NativeArray. - /// An array containing all the entities selected by the ComponentGroup. + /// An array containing all the entities selected by the EntityQuery. public NativeArray ToEntityArray(Allocator allocator, out JobHandle jobhandle) { #if ENABLE_UNITY_COLLECTIONS_CHECKS - var entityType = new ArchetypeChunkEntityType(m_SafetyManager.GetEntityManagerSafetyHandle()); + var entityType = new ArchetypeChunkEntityType(m_SafetyManager->GetEntityManagerSafetyHandle()); #else var entityType = new ArchetypeChunkEntityType(); #endif @@ -505,11 +477,11 @@ public NativeArray ToEntityArray(Allocator allocator, out JobHandle jobh /// /// This version of the function blocks until the Job used to fill the array is complete. /// The type of memory to allocate. - /// An array containing all the entities selected by the ComponentGroup. + /// An array containing all the entities selected by the EntityQuery. public NativeArray ToEntityArray(Allocator allocator) { #if ENABLE_UNITY_COLLECTIONS_CHECKS - var entityType = new ArchetypeChunkEntityType(m_SafetyManager.GetEntityManagerSafetyHandle()); + var entityType = new ArchetypeChunkEntityType(m_SafetyManager->GetEntityManagerSafetyHandle()); #else var entityType = new ArchetypeChunkEntityType(); #endif @@ -527,12 +499,12 @@ public NativeArray ToEntityArray(Allocator allocator) /// that uses the NativeArray. /// The component type. /// An array containing the specified component for all the entities selected - /// by the ComponentGroup. + /// by the EntityQuery. public NativeArray ToComponentDataArray(Allocator allocator, out JobHandle jobhandle) where T : struct,IComponentData { #if ENABLE_UNITY_COLLECTIONS_CHECKS - var componentType = new ArchetypeChunkComponentType(m_SafetyManager.GetSafetyHandle(TypeManager.GetTypeIndex(), true), true, EntityDataManager->GlobalSystemVersion); + var componentType = new ArchetypeChunkComponentType(m_SafetyManager->GetSafetyHandle(TypeManager.GetTypeIndex(), true), true, EntityDataManager->GlobalSystemVersion); #else var componentType = new ArchetypeChunkComponentType(true, EntityDataManager->GlobalSystemVersion); #endif @@ -545,22 +517,22 @@ public NativeArray ToComponentDataArray(Allocator allocator, out JobHandle /// The type of memory to allocate. /// The component type. /// An array containing the specified component for all the entities selected - /// by the ComponentGroup. + /// by the EntityQuery. /// Thrown if you ask for a component that is not part of /// the group. public NativeArray ToComponentDataArray(Allocator allocator) where T : struct, IComponentData { #if ENABLE_UNITY_COLLECTIONS_CHECKS - var componentType = new ArchetypeChunkComponentType(m_SafetyManager.GetSafetyHandle(TypeManager.GetTypeIndex(), true), true, EntityDataManager->GlobalSystemVersion); + var componentType = new ArchetypeChunkComponentType(m_SafetyManager->GetSafetyHandle(TypeManager.GetTypeIndex(), true), true, EntityDataManager->GlobalSystemVersion); #else var componentType = new ArchetypeChunkComponentType(true, EntityDataManager->GlobalSystemVersion); #endif #if ENABLE_UNITY_COLLECTIONS_CHECKS int typeIndex = TypeManager.GetTypeIndex(); - int indexInComponentGroup = GetIndexInComponentGroup(typeIndex); - if (indexInComponentGroup == -1) + int indexInEntityQuery = GetIndexInEntityQuery(typeIndex); + if (indexInEntityQuery == -1) throw new InvalidOperationException( $"Trying ToComponentDataArray of {TypeManager.GetType(typeIndex)} but the required component type was not declared in the EntityGroup."); #endif @@ -575,10 +547,10 @@ public void CopyFromComponentDataArray(NativeArray componentDataArray) { // throw if non equal size #if ENABLE_UNITY_COLLECTIONS_CHECKS - var groupLength = CalculateLength(); - if(groupLength != componentDataArray.Length) - throw new ArgumentException($"Length of input array ({componentDataArray.Length}) does not match length of ComponentGroup ({groupLength})"); - var componentType = new ArchetypeChunkComponentType(m_SafetyManager.GetSafetyHandle(TypeManager.GetTypeIndex(), false), false, EntityDataManager->GlobalSystemVersion); + var entityCount = CalculateLength(); + if (entityCount != componentDataArray.Length) + throw new ArgumentException($"Length of input array ({componentDataArray.Length}) does not match length of EntityQuery ({entityCount})"); + var componentType = new ArchetypeChunkComponentType(m_SafetyManager->GetSafetyHandle(TypeManager.GetTypeIndex(), false), false, EntityDataManager->GlobalSystemVersion); #else var componentType = new ArchetypeChunkComponentType(false, EntityDataManager->GlobalSystemVersion); #endif @@ -592,13 +564,13 @@ public void CopyFromComponentDataArray(NativeArray componentDataArray, out { // throw if non equal size #if ENABLE_UNITY_COLLECTIONS_CHECKS - var groupLength = CalculateLength(); - if(groupLength != componentDataArray.Length) - throw new ArgumentException($"Length of input array ({componentDataArray.Length}) does not match length of ComponentGroup ({groupLength})"); + var entityCount = CalculateLength(); + if(entityCount != componentDataArray.Length) + throw new ArgumentException($"Length of input array ({componentDataArray.Length}) does not match length of EntityQuery ({entityCount})"); #endif #if ENABLE_UNITY_COLLECTIONS_CHECKS - var componentType = new ArchetypeChunkComponentType(m_SafetyManager.GetSafetyHandle(TypeManager.GetTypeIndex(), false), false, EntityDataManager->GlobalSystemVersion); + var componentType = new ArchetypeChunkComponentType(m_SafetyManager->GetSafetyHandle(TypeManager.GetTypeIndex(), false), false, EntityDataManager->GlobalSystemVersion); #else var componentType = new ArchetypeChunkComponentType(false, EntityDataManager->GlobalSystemVersion); #endif @@ -606,36 +578,12 @@ public void CopyFromComponentDataArray(NativeArray componentDataArray, out ComponentChunkIterator.CopyFromComponentDataArray(m_GroupData->MatchingArchetypes, componentDataArray, componentType, this, ref m_Filter, out jobhandle, GetDependency()); } - /// - /// Creates an EntityArray that gives you access to the entities in this ComponentGroup. - /// - /// EntityArray of all the entities in this ComponentGroup. - [Obsolete("GetEntityArray is deprecated. Use IJobProcessComponentDataWithEntity or ToEntityArray instead.")] - public EntityArray GetEntityArray() - { - int length = CalculateLength(); - ComponentChunkIterator iterator = GetComponentChunkIterator(); - - EntityArray output; - iterator.IndexInComponentGroup = 0; -#if ENABLE_UNITY_COLLECTIONS_CHECKS - - if (m_GroupData->RequiredComponentsCount == 0) - throw new InvalidOperationException( $"GetEntityArray() is currently not supported from a ComponentGroup created with EntityArchetypeQuery."); - - output = new EntityArray(iterator, length, m_SafetyManager.GetEntityManagerSafetyHandle()); -#else - output = new EntityArray(iterator, length); -#endif - return output; - } - public Entity GetSingletonEntity() { #if ENABLE_UNITY_COLLECTIONS_CHECKS - var groupLength = CalculateLength(); - if (groupLength != 1) - throw new System.InvalidOperationException($"GetSingletonEntity() requires that exactly one exists but there are {groupLength}."); + var entityCount = CalculateLength(); + if (entityCount != 1) + throw new System.InvalidOperationException($"GetSingletonEntity() requires that exactly one exists but there are {entityCount}."); #endif @@ -660,12 +608,12 @@ public T GetSingleton() where T : struct, IComponentData { #if ENABLE_UNITY_COLLECTIONS_CHECKS - if(GetIndexInComponentGroup(TypeManager.GetTypeIndex()) != 1) + if(GetIndexInEntityQuery(TypeManager.GetTypeIndex()) != 1) throw new System.InvalidOperationException($"GetSingleton<{typeof(T)}>() requires that {typeof(T)} is the only component type in its archetype."); - var groupLength = CalculateLength(); - if (groupLength != 1) - throw new System.InvalidOperationException($"GetSingleton<{typeof(T)}>() requires that exactly one {typeof(T)} exists but there are {groupLength}."); + var entityCount = CalculateLength(); + if (entityCount != 1) + throw new System.InvalidOperationException($"GetSingleton<{typeof(T)}>() requires that exactly one {typeof(T)} exists but there are {entityCount}."); #endif CompleteDependency(); @@ -699,11 +647,11 @@ public T GetSingleton() /// /// var entityManager = World.Active.EntityManager; /// var singletonEntity = entityManager.CreateEntity(typeof(Singlet)); - /// var singletonGroup = entityManager.CreateComponentGroup(typeof(Singlet)); + /// var singletonGroup = entityManager.CreateEntityQuery(typeof(Singlet)); /// singletonGroup.SetSingleton<Singlet>(new Singlet {Value = 1}); /// /// - /// You can set and get the singleton value from a ComponentGroup or a ComponentSystem. + /// You can set and get the singleton value from a EntityQuery or a ComponentSystem. /// /// An instance of type T containing the values to set. /// The component type. @@ -713,12 +661,12 @@ public void SetSingleton(T value) where T : struct, IComponentData { #if ENABLE_UNITY_COLLECTIONS_CHECKS - if(GetIndexInComponentGroup(TypeManager.GetTypeIndex()) != 1) + if(GetIndexInEntityQuery(TypeManager.GetTypeIndex()) != 1) throw new System.InvalidOperationException($"GetSingleton<{typeof(T)}>() requires that {typeof(T)} is the only component type in its archetype."); - var groupLength = CalculateLength(); - if (groupLength != 1) - throw new System.InvalidOperationException($"SetSingleton<{typeof(T)}>() requires that exactly one {typeof(T)} exists but there are {groupLength}."); + var entityCount = CalculateLength(); + if (entityCount != 1) + throw new System.InvalidOperationException($"SetSingleton<{typeof(T)}>() requires that exactly one {typeof(T)} exists but there are {entityCount}."); #endif CompleteDependency(); @@ -762,15 +710,15 @@ public bool CompareComponents(NativeArray componentTypes) /// /// /// - /// + /// /// - public bool CompareQuery(EntityArchetypeQuery[] query) + public bool CompareQuery(EntityQueryDesc[] queryDesc) { - return EntityGroupManager.CompareQuery(query, m_GroupData); + return EntityGroupManager.CompareQuery(queryDesc, m_GroupData); } /// - /// Resets this ComponentGroup's filter. + /// Resets this EntityQuery's filter. /// /// /// Removes references to shared component data, if applicable, then resets the filter type to None. @@ -793,10 +741,10 @@ public void ResetFilter() } /// - /// Sets this ComponentGroup's filter while preserving its version number. + /// Sets this EntityQuery's filter while preserving its version number. /// - /// ComponentGroupFilter to use all data but RequiredChangeVersion from. - void SetFilter(ref ComponentGroupFilter filter) + /// EntityQueryFilter to use all data but RequiredChangeVersion from. + void SetFilter(ref EntityQueryFilter filter) { #if ENABLE_UNITY_COLLECTIONS_CHECKS filter.AssertValid(); @@ -808,28 +756,28 @@ void SetFilter(ref ComponentGroupFilter filter) } /// - /// Filters this ComponentGroup so that it only selects entities with shared component values + /// Filters this EntityQuery so that it only selects entities with shared component values /// matching the values specified by the `sharedComponent1` parameter. /// /// The shared component values on which to filter. /// The type of shared component. (The type must also be - /// one of the types used to create the ComponentGroup. + /// one of the types used to create the EntityQuery. public void SetFilter(SharedComponent1 sharedComponent1) where SharedComponent1 : struct, ISharedComponentData { var sm = ArchetypeManager.GetSharedComponentDataManager(); - var filter = new ComponentGroupFilter(); + var filter = new EntityQueryFilter(); filter.Type = FilterType.SharedComponent; filter.Shared.Count = 1; - filter.Shared.IndexInComponentGroup[0] = GetIndexInComponentGroup(TypeManager.GetTypeIndex()); + filter.Shared.IndexInEntityQuery[0] = GetIndexInEntityQuery(TypeManager.GetTypeIndex()); filter.Shared.SharedComponentIndex[0] = sm.InsertSharedComponent(sharedComponent1); SetFilter(ref filter); } /// - /// Filters this ComponentGroup based on the values of two separate shared components. + /// Filters this EntityQuery based on the values of two separate shared components. /// /// /// The filter only selects entities for which both shared component values @@ -838,9 +786,9 @@ public void SetFilter(SharedComponent1 sharedComponent1) /// Shared component values on which to filter. /// Shared component values on which to filter. /// The type of shared component. (The type must also be - /// one of the types used to create the ComponentGroup. + /// one of the types used to create the EntityQuery. /// The type of shared component. (The type must also be - /// one of the types used to create the ComponentGroup. + /// one of the types used to create the EntityQuery. public void SetFilter(SharedComponent1 sharedComponent1, SharedComponent2 sharedComponent2) where SharedComponent1 : struct, ISharedComponentData @@ -848,13 +796,13 @@ public void SetFilter(SharedComponent1 share { var sm = ArchetypeManager.GetSharedComponentDataManager(); - var filter = new ComponentGroupFilter(); + var filter = new EntityQueryFilter(); filter.Type = FilterType.SharedComponent; filter.Shared.Count = 2; - filter.Shared.IndexInComponentGroup[0] = GetIndexInComponentGroup(TypeManager.GetTypeIndex()); + filter.Shared.IndexInEntityQuery[0] = GetIndexInEntityQuery(TypeManager.GetTypeIndex()); filter.Shared.SharedComponentIndex[0] = sm .InsertSharedComponent(sharedComponent1); - filter.Shared.IndexInComponentGroup[1] = GetIndexInComponentGroup(TypeManager.GetTypeIndex()); + filter.Shared.IndexInEntityQuery[1] = GetIndexInEntityQuery(TypeManager.GetTypeIndex()); filter.Shared.SharedComponentIndex[1] = sm.InsertSharedComponent(sharedComponent2); SetFilter(ref filter); @@ -866,13 +814,13 @@ public void SetFilter(SharedComponent1 share /// /// Saves a given ComponentType's index in RequiredComponents in this group's Changed filter. /// - /// ComponentType to mark as changed on this ComponentGroup's filter. + /// ComponentType to mark as changed on this EntityQuery's filter. public void SetFilterChanged(ComponentType componentType) { - var filter = new ComponentGroupFilter(); + var filter = new EntityQueryFilter(); filter.Type = FilterType.Changed; filter.Changed.Count = 1; - filter.Changed.IndexInComponentGroup[0] = GetIndexInComponentGroup(componentType.TypeIndex); + filter.Changed.IndexInEntityQuery[0] = GetIndexInEntityQuery(componentType.TypeIndex); SetFilter(ref filter); } @@ -888,50 +836,50 @@ internal void SetFilterChangedRequiredVersion(uint requiredVersion) /// /// Saves given ComponentTypes' indices in RequiredComponents in this group's Changed filter. /// - /// Array of up to two ComponentTypes to mark as changed on this ComponentGroup's filter. + /// Array of up to two ComponentTypes to mark as changed on this EntityQuery's filter. public void SetFilterChanged(ComponentType[] componentType) { - if (componentType.Length > ComponentGroupFilter.ChangedFilter.Capacity) + if (componentType.Length > EntityQueryFilter.ChangedFilter.Capacity) throw new ArgumentException( - $"ComponentGroup.SetFilterChanged accepts a maximum of {ComponentGroupFilter.ChangedFilter.Capacity} component array length"); + $"EntityQuery.SetFilterChanged accepts a maximum of {EntityQueryFilter.ChangedFilter.Capacity} component array length"); if (componentType.Length <= 0) throw new ArgumentException( - $"ComponentGroup.SetFilterChanged component array length must be larger than 0"); + $"EntityQuery.SetFilterChanged component array length must be larger than 0"); - var filter = new ComponentGroupFilter(); + var filter = new EntityQueryFilter(); filter.Type = FilterType.Changed; filter.Changed.Count = componentType.Length; for (var i = 0; i != componentType.Length; i++) - filter.Changed.IndexInComponentGroup[i] = GetIndexInComponentGroup(componentType[i].TypeIndex); + filter.Changed.IndexInEntityQuery[i] = GetIndexInEntityQuery(componentType[i].TypeIndex); SetFilter(ref filter); } /// - /// Ensures all jobs running on this ComponentGroup complete. + /// Ensures all jobs running on this EntityQuery complete. /// public void CompleteDependency() { - m_SafetyManager.CompleteDependenciesNoChecks(m_GroupData->ReaderTypes, m_GroupData->ReaderTypesCount, + m_SafetyManager->CompleteDependenciesNoChecks(m_GroupData->ReaderTypes, m_GroupData->ReaderTypesCount, m_GroupData->WriterTypes, m_GroupData->WriterTypesCount); } /// - /// Combines all dependencies in this ComponentGroup into a single JobHandle. + /// Combines all dependencies in this EntityQuery into a single JobHandle. /// - /// JobHandle that represents the combined dependencies of this ComponentGroup + /// JobHandle that represents the combined dependencies of this EntityQuery public JobHandle GetDependency() { - return m_SafetyManager.GetDependency(m_GroupData->ReaderTypes, m_GroupData->ReaderTypesCount, + return m_SafetyManager->GetDependency(m_GroupData->ReaderTypes, m_GroupData->ReaderTypesCount, m_GroupData->WriterTypes, m_GroupData->WriterTypesCount); } /// - /// Adds another job handle to this ComponentGroup's dependencies. + /// Adds another job handle to this EntityQuery's dependencies. /// public void AddDependency(JobHandle job) { - m_SafetyManager.AddDependency(m_GroupData->ReaderTypes, m_GroupData->ReaderTypesCount, + m_SafetyManager->AddDependency(m_GroupData->ReaderTypes, m_GroupData->ReaderTypesCount, m_GroupData->WriterTypes, m_GroupData->WriterTypesCount, job); } @@ -950,10 +898,10 @@ public int GetCombinedComponentOrderVersion() } /// - /// Total number of chunks in this ComponentGroup's MatchingArchetypes list. + /// Total number of chunks in this EntityQuery's MatchingArchetypes list. /// /// First node of MatchingArchetypes linked list. - /// Number of chunks in this ComponentGroup. + /// Number of chunks in this EntityQuery. internal int CalculateNumberOfChunksWithoutFiltering() { return ComponentChunkIterator.CalculateNumberOfChunksWithoutFiltering(m_GroupData->MatchingArchetypes); @@ -964,7 +912,7 @@ internal bool AddReaderWritersToLists(ref UnsafeList reading, ref UnsafeList wri bool anyAdded = false; for (int i = 0; i < m_GroupData->ReaderTypesCount; ++i) anyAdded |= CalculateReaderWriterDependency.AddReaderTypeIndex(m_GroupData->ReaderTypes[i], ref reading, ref writing); - + for (int i = 0; i < m_GroupData->WriterTypesCount; ++i) anyAdded |=CalculateReaderWriterDependency.AddWriterTypeIndex(m_GroupData->WriterTypes[i], ref reading, ref writing); return anyAdded; @@ -979,13 +927,33 @@ internal void SyncFilterTypes() { if (m_Filter.Type == FilterType.Changed) { - fixed (int* indexInComponentGroupPtr = m_Filter.Changed.IndexInComponentGroup) + fixed (int* indexInEntityQueryPtr = m_Filter.Changed.IndexInEntityQuery) for (int i = 0; i < m_Filter.Changed.Count; ++i) { - var type = m_GroupData->RequiredComponents[indexInComponentGroupPtr[i]]; - SafetyManager.CompleteWriteDependency(type.TypeIndex); + var type = m_GroupData->RequiredComponents[indexInEntityQueryPtr[i]]; + SafetyManager->CompleteWriteDependency(type.TypeIndex); } } } + + [EditorBrowsable(EditorBrowsableState.Never)] + [Obsolete("GetComponentDataArray is deprecated. Use IJobForEach or ToComponentDataArray/CopyFromComponentDataArray instead. More information: https://forum.unity.com/threads/api-deprecation-faq-0-0-23.636994/", true)] + public void GetComponentDataArray() where T : struct, IComponentData + { } + + [EditorBrowsable(EditorBrowsableState.Never)] + [Obsolete("GetSharedComponentDataArray is deprecated. Use ArchetypeChunk.GetSharedComponentData. More information: https://forum.unity.com/threads/api-deprecation-faq-0-0-23.636994/", true)] + public void GetSharedComponentDataArray() where T : struct, ISharedComponentData + { } + + [EditorBrowsable(EditorBrowsableState.Never)] + [Obsolete("GetBufferArray is deprecated. Use ArchetypeChunk.GetBufferAccessor() instead. More information: https://forum.unity.com/threads/api-deprecation-faq-0-0-23.636994/", true)] + public void GetBufferArray() where T : struct, IBufferElementData + { } + + [EditorBrowsable(EditorBrowsableState.Never)] + [Obsolete("GetEntityArray is deprecated. Use IJobForEachWithEntity or ToEntityArray instead. More information: https://forum.unity.com/threads/api-deprecation-faq-0-0-23.636994/", true)] + public void GetEntityArray() + { } } } diff --git a/Unity.Entities/Iterators/EntityQuery.cs.meta b/Unity.Entities/Iterators/EntityQuery.cs.meta new file mode 100644 index 00000000..454d2f5d --- /dev/null +++ b/Unity.Entities/Iterators/EntityQuery.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: f749f9fbe6d422c4799fc5957dc3f912 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/Unity.Entities/Iterators/SharedComponentDataArray.cs b/Unity.Entities/Iterators/SharedComponentDataArray.cs deleted file mode 100644 index 93e55559..00000000 --- a/Unity.Entities/Iterators/SharedComponentDataArray.cs +++ /dev/null @@ -1,63 +0,0 @@ -using System; -using Unity.Collections.LowLevel.Unsafe; - -namespace Unity.Entities -{ - [Obsolete("SharedComponentDataArray is deprecated. Use chunk API instead.")] - public struct SharedComponentDataArray where T : struct, ISharedComponentData - { - private ComponentChunkIterator m_Iterator; - private ComponentChunkCache m_Cache; - private readonly SharedComponentDataManager m_sharedComponentDataManager; - private readonly int m_sharedComponentIndex; - -#if ENABLE_UNITY_COLLECTIONS_CHECKS - private readonly AtomicSafetyHandle m_Safety; -#endif - -#if ENABLE_UNITY_COLLECTIONS_CHECKS - internal SharedComponentDataArray(SharedComponentDataManager sharedComponentDataManager, - int sharedComponentIndex, ComponentChunkIterator iterator, int length, AtomicSafetyHandle safety) -#else - internal unsafe SharedComponentDataArray(SharedComponentDataManager sharedComponentDataManager, int sharedComponentIndex, ComponentChunkIterator iterator, int length) -#endif - { - m_sharedComponentDataManager = sharedComponentDataManager; - m_sharedComponentIndex = sharedComponentIndex; - m_Iterator = iterator; - m_Cache = default(ComponentChunkCache); - - Length = length; -#if ENABLE_UNITY_COLLECTIONS_CHECKS - m_Safety = safety; -#endif - } - - public T this[int index] - { - get - { -#if ENABLE_UNITY_COLLECTIONS_CHECKS - AtomicSafetyHandle.CheckReadAndThrow(m_Safety); - if ((uint) index >= (uint) Length) - FailOutOfRangeError(index); -#endif - - if (index < m_Cache.CachedBeginIndex || index >= m_Cache.CachedEndIndex) - m_Iterator.MoveToEntityIndexAndUpdateCache(index, out m_Cache, false); - - var sharedComponent = m_Iterator.GetSharedComponentFromCurrentChunk(m_sharedComponentIndex); - return m_sharedComponentDataManager.GetSharedComponentData(sharedComponent); - } - } - -#if ENABLE_UNITY_COLLECTIONS_CHECKS - private void FailOutOfRangeError(int index) - { - throw new IndexOutOfRangeException($"Index {index} is out of range of '{Length}' Length."); - } -#endif - - public int Length { get; } - } -} diff --git a/Unity.Entities/Iterators/SharedComponentDataArray.cs.meta b/Unity.Entities/Iterators/SharedComponentDataArray.cs.meta deleted file mode 100644 index 98380507..00000000 --- a/Unity.Entities/Iterators/SharedComponentDataArray.cs.meta +++ /dev/null @@ -1,11 +0,0 @@ -fileFormatVersion: 2 -guid: 36d1b790446ccf84e8be1845686c0c98 -MonoImporter: - externalObjects: {} - serializedVersion: 2 - defaultReferences: [] - executionOrder: 0 - icon: {instanceID: 0} - userData: - assetBundleName: - assetBundleVariant: diff --git a/Unity.Entities/ScriptBehaviourManager.cs b/Unity.Entities/ScriptBehaviourManager.cs index c89cf50f..bf665782 100644 --- a/Unity.Entities/ScriptBehaviourManager.cs +++ b/Unity.Entities/ScriptBehaviourManager.cs @@ -29,76 +29,4 @@ public WorldSystemFilterAttribute(WorldSystemFilterFlags flags) FilterFlags = flags; } } - - - public abstract class ScriptBehaviourManager - { -#if UNITY_EDITOR - private CustomSampler m_Sampler; -#endif - internal void CreateInstance(World world) - { - OnBeforeCreateManagerInternal(world); - try - { - OnCreateManager(); -#if UNITY_EDITOR - var type = GetType(); - m_Sampler = CustomSampler.Create($"{world.Name} {type.FullName}"); -#endif - } - catch - { - OnBeforeDestroyManagerInternal(); - OnAfterDestroyManagerInternal(); - throw; - } - } - - internal void DestroyInstance() - { - OnBeforeDestroyManagerInternal(); - OnDestroyManager(); - OnAfterDestroyManagerInternal(); - } - - protected abstract void OnBeforeCreateManagerInternal(World world); - - protected abstract void OnBeforeDestroyManagerInternal(); - protected abstract void OnAfterDestroyManagerInternal(); - - /// - /// Called when the ScriptBehaviourManager is created. - /// When a new domain is loaded, OnCreate on the necessary manager will be invoked - /// before the ScriptBehaviour will receive its first OnCreate() call. - /// - protected virtual void OnCreateManager() - { - } - - /// - /// Called when the ScriptBehaviourManager is destroyed. - /// Before Playmode exits or scripts are reloaded OnDestroy will be called on all created ScriptBehaviourManagers. - /// - protected virtual void OnDestroyManager() - { - } - - internal abstract void InternalUpdate(); - - /// - /// Execute the manager immediately. - /// - public void Update() - { -#if UNITY_EDITOR - m_Sampler?.Begin(); -#endif - InternalUpdate(); - -#if UNITY_EDITOR - m_Sampler?.End(); -#endif - } - } } diff --git a/Unity.Entities/ScriptBehaviourUpdateOrder.cs b/Unity.Entities/ScriptBehaviourUpdateOrder.cs index e7b20d7d..323f3ee7 100644 --- a/Unity.Entities/ScriptBehaviourUpdateOrder.cs +++ b/Unity.Entities/ScriptBehaviourUpdateOrder.cs @@ -53,7 +53,7 @@ public UpdateInGroupAttribute(Type groupType) public static class ScriptBehaviourUpdateOrder { private static void InsertManagerIntoSubsystemList(PlayerLoopSystem[] subsystemList, int insertIndex, T mgr) - where T : ScriptBehaviourManager + where T : ComponentSystemBase { var del = new DummyDelegateWrapper(mgr); subsystemList[insertIndex].type = typeof(T); @@ -75,7 +75,7 @@ public static void UpdatePlayerLoop(World world) for (var j = 0; j < subsystemListLength; ++j) newSubsystemList[j] = playerLoop.subSystemList[i].subSystemList[j]; InsertManagerIntoSubsystemList(newSubsystemList, - subsystemListLength + 0, world.GetOrCreateManager()); + subsystemListLength + 0, world.GetOrCreateSystem()); playerLoop.subSystemList[i].subSystemList = newSubsystemList; } else if (playerLoop.subSystemList[i].type == typeof(PreLateUpdate)) @@ -84,7 +84,7 @@ public static void UpdatePlayerLoop(World world) for (var j = 0; j < subsystemListLength; ++j) newSubsystemList[j] = playerLoop.subSystemList[i].subSystemList[j]; InsertManagerIntoSubsystemList(newSubsystemList, - subsystemListLength + 0, world.GetOrCreateManager()); + subsystemListLength + 0, world.GetOrCreateSystem()); playerLoop.subSystemList[i].subSystemList = newSubsystemList; } else if (playerLoop.subSystemList[i].type == typeof(Initialization)) @@ -93,7 +93,7 @@ public static void UpdatePlayerLoop(World world) for (var j = 0; j < subsystemListLength; ++j) newSubsystemList[j] = playerLoop.subSystemList[i].subSystemList[j]; InsertManagerIntoSubsystemList(newSubsystemList, - subsystemListLength + 0, world.GetOrCreateManager()); + subsystemListLength + 0, world.GetOrCreateSystem()); playerLoop.subSystemList[i].subSystemList = newSubsystemList; } } @@ -116,17 +116,17 @@ public static void SetPlayerLoop(PlayerLoopSystem playerLoop) internal class DummyDelegateWrapper { - internal ScriptBehaviourManager Manager => m_Manager; - private readonly ScriptBehaviourManager m_Manager; + internal ComponentSystemBase System => m_System; + private readonly ComponentSystemBase m_System; - public DummyDelegateWrapper(ScriptBehaviourManager man) + public DummyDelegateWrapper(ComponentSystemBase sys) { - m_Manager = man; + m_System = sys; } public void TriggerUpdate() { - m_Manager.Update(); + m_System.Update(); } } } diff --git a/Unity.Entities/SerializeUtility.cs b/Unity.Entities/SerializeUtility.cs index 9dc8b85d..b99b9c06 100644 --- a/Unity.Entities/SerializeUtility.cs +++ b/Unity.Entities/SerializeUtility.cs @@ -1,4 +1,4 @@ -#if !UNITY_CSHARP_TINY +#if !NET_DOTS using System; using System.Collections.Generic; using System.Linq; diff --git a/Unity.Entities/SharedComponentManager.cs b/Unity.Entities/SharedComponentManager.cs index eeb1e114..33dff445 100644 --- a/Unity.Entities/SharedComponentManager.cs +++ b/Unity.Entities/SharedComponentManager.cs @@ -112,12 +112,7 @@ private unsafe int FindNonDefaultSharedComponentIndex(int typeIndex, int hashCod var data = m_SharedComponentData[itemIndex]; if (data != null && m_SharedComponentType[itemIndex] == typeIndex) { - ulong handle; - var value = PinGCObjectAndGetAddress(data, out handle); - var res = TypeManager.Equals(newData, value, typeIndex); - UnsafeUtility.ReleaseGCObject(handle); - - if (res) + if (TypeManager.Equals(data, newData, typeIndex)) return itemIndex; } } while (m_HashLookup.TryGetNextValue(out itemIndex, ref iter)); @@ -125,14 +120,30 @@ private unsafe int FindNonDefaultSharedComponentIndex(int typeIndex, int hashCod return -1; } - internal unsafe int InsertSharedComponentAssumeNonDefault(int typeIndex, int hashCode, object newData) + private unsafe int FindNonDefaultSharedComponentIndex(int typeIndex, int hashCode, object newData) { - ulong handle; - var newDataPtr = PinGCObjectAndGetAddress(newData, out handle); + int itemIndex; + NativeMultiHashMapIterator iter; - var index = FindNonDefaultSharedComponentIndex(typeIndex, hashCode, newDataPtr); + if (!m_HashLookup.TryGetFirstValue(hashCode, out itemIndex, out iter)) + return -1; - UnsafeUtility.ReleaseGCObject(handle); + do + { + var data = m_SharedComponentData[itemIndex]; + if (data != null && m_SharedComponentType[itemIndex] == typeIndex) + { + if (TypeManager.Equals(data, newData, typeIndex)) + return itemIndex; + } + } while (m_HashLookup.TryGetNextValue(out itemIndex, ref iter)); + + return -1; + } + + internal unsafe int InsertSharedComponentAssumeNonDefault(int typeIndex, int hashCode, object newData) + { + var index = FindNonDefaultSharedComponentIndex(typeIndex, hashCode, newData); if (-1 == index) index = Add(typeIndex, hashCode, newData); @@ -194,13 +205,12 @@ public T GetSharedComponentData(int index) where T : struct public object GetSharedComponentDataBoxed(int index, int typeIndex) { -#if !UNITY_CSHARP_TINY +#if !NET_DOTS if (index == 0) return Activator.CreateInstance(TypeManager.GetType(typeIndex)); #else if (index == 0) throw new InvalidOperationException("Implement TypeManager.GetType(typeIndex).DefaultValue"); - throw new NotImplementedException("SharedComponents not supported (yet) in Tiny"); #endif return m_SharedComponentData[index]; } @@ -219,23 +229,6 @@ public void AddReference(int index, int numRefs = 1) m_SharedComponentRefCount[index] += numRefs; } - public static unsafe int GetHashCodeFast(object target, int typeIndex) - { - ulong handle; - var ptr = PinGCObjectAndGetAddress(target, out handle); - var hashCode = TypeManager.GetHashCode(ptr, typeIndex); - UnsafeUtility.ReleaseGCObject(handle); - - return hashCode; - } - - private static unsafe void* PinGCObjectAndGetAddress(object target, out ulong handle) - { - var ptr = UnsafeUtility.PinGCObjectAndGetAddress(target, out handle); - return (byte*) ptr + TypeManager.ObjectOffset; - } - - public void RemoveReference(int index, int numRefs = 1) { if (index == 0) @@ -248,7 +241,7 @@ public void RemoveReference(int index, int numRefs = 1) return; var typeIndex = m_SharedComponentType[index]; - var hashCode = GetHashCodeFast(m_SharedComponentData[index], typeIndex); + var hashCode = TypeManager.GetHashCode(m_SharedComponentData[index], typeIndex); object sharedComponent = m_SharedComponentData[index]; (sharedComponent as IDisposable)?.Dispose(); @@ -327,7 +320,7 @@ public unsafe void MoveSharedComponents(SharedComponentDataManager srcSharedComp var srcData = srcSharedComponents.m_SharedComponentData[srcIndex]; var typeIndex = srcSharedComponents.m_SharedComponentType[srcIndex]; - var hashCode = GetHashCodeFast(srcData, typeIndex); + var hashCode = TypeManager.GetHashCode(srcData, typeIndex); var dstIndex = InsertSharedComponentAssumeNonDefault(typeIndex, hashCode, srcData); srcSharedComponents.RemoveReference(srcIndex); @@ -346,7 +339,7 @@ public unsafe void CopySharedComponents(SharedComponentDataManager srcSharedComp var srcData = srcSharedComponents.m_SharedComponentData[srcIndex]; var typeIndex = srcSharedComponents.m_SharedComponentType[srcIndex]; - var hashCode = GetHashCodeFast(srcData, typeIndex); + var hashCode = TypeManager.GetHashCode(srcData, typeIndex); var dstIndex = InsertSharedComponentAssumeNonDefault(typeIndex, hashCode, srcData); sharedComponentIndices[i] = dstIndex; @@ -375,26 +368,6 @@ public unsafe bool AllSharedComponentReferencesAreFromChunks(ArchetypeManager ar return cmp == 0; } - public static unsafe bool FastEquality_CompareBoxed(object lhs, object rhs, int typeIndex) - { - ulong lhsHandle, rhsHandle; - var valueLHS = PinGCObjectAndGetAddress(lhs, out lhsHandle); - var valueRHS = PinGCObjectAndGetAddress(rhs, out rhsHandle); - - var res = TypeManager.Equals(valueLHS, valueRHS, typeIndex); - - UnsafeUtility.ReleaseGCObject(lhsHandle); - UnsafeUtility.ReleaseGCObject(rhsHandle); - - return res; - } - - public static unsafe bool FastEquality_ComparePtr(void* lhs, void* rhs, int typeIndex) - { - var res = TypeManager.Equals(lhs, rhs, typeIndex); - return res; - } - public static unsafe bool FastEquality_CompareElements(void* lhs, void* rhs, int count, int typeIndex) { var typeInfo = TypeManager.GetTypeInfo(typeIndex); @@ -420,7 +393,7 @@ public unsafe NativeArray MoveAllSharedComponents(SharedComponentDataManage var typeIndex = srcSharedComponents.m_SharedComponentType[srcIndex]; - var hashCode = GetHashCodeFast(srcData, typeIndex); + var hashCode = TypeManager.GetHashCode(srcData, typeIndex); var dstIndex = InsertSharedComponentAssumeNonDefault(typeIndex, hashCode, srcData); m_SharedComponentRefCount[dstIndex] += srcSharedComponents.m_SharedComponentRefCount[srcIndex] - 1; @@ -465,7 +438,7 @@ public unsafe NativeArray MoveSharedComponents(SharedComponentDataManager s var srcData = srcSharedComponents.m_SharedComponentData[srcIndex]; var typeIndex = srcSharedComponents.m_SharedComponentType[srcIndex]; - var hashCode = GetHashCodeFast(srcData, typeIndex); + var hashCode = TypeManager.GetHashCode(srcData, typeIndex); var dstIndex = InsertSharedComponentAssumeNonDefault(typeIndex, hashCode, srcData); m_SharedComponentRefCount[dstIndex] += remap[srcIndex] - 1; diff --git a/Unity.Entities/Types/BufferHeader.cs b/Unity.Entities/Types/BufferHeader.cs index 4e06cf39..6554da29 100644 --- a/Unity.Entities/Types/BufferHeader.cs +++ b/Unity.Entities/Types/BufferHeader.cs @@ -34,14 +34,14 @@ public static void EnsureCapacity(BufferHeader* header, int count, int typeSize, return; int newCapacity = Math.Max(Math.Max(2 * header->Capacity, count), kMinimumCapacity); - long newBlockSize = newCapacity * typeSize; + long newBlockSize = (long)newCapacity * typeSize; byte* oldData = GetElementPointer(header); byte* newData = (byte*) UnsafeUtility.Malloc(newBlockSize, alignment, Allocator.Persistent); if (trashMode == TrashMode.RetainOldData) { - long oldBlockSize = header->Capacity * typeSize; + long oldBlockSize = (long)header->Capacity * typeSize; UnsafeUtility.MemCpy(newData, oldData, oldBlockSize); } @@ -62,7 +62,7 @@ public static void Assign(BufferHeader* header, byte* source, int count, int typ // Select between internal capacity buffer and heap buffer. byte* elementPtr = GetElementPointer(header); - UnsafeUtility.MemCpy(elementPtr, source, typeSize * count); + UnsafeUtility.MemCpy(elementPtr, source, (long)typeSize * count); header->Length = count; } @@ -103,9 +103,9 @@ public static void PatchAfterCloningChunk(Chunk* chunk) if (header->Pointer != null) // hoo boy, it's a malloc { BufferHeader newHeader = *header; - var bytesToAllocate = header->Capacity * ti.ElementSize; - var bytesToCopy = header->Length * ti.ElementSize; - newHeader.Pointer = (byte*)UnsafeUtility.Malloc(bytesToAllocate, 16, Allocator.Persistent); + long bytesToAllocate = (long)header->Capacity * ti.ElementSize; + long bytesToCopy = (long)header->Length * ti.ElementSize; + newHeader.Pointer = (byte*)UnsafeUtility.Malloc(bytesToAllocate, TypeManager.MaximumSupportedAlignment, Allocator.Persistent); UnsafeUtility.MemCpy(newHeader.Pointer, header->Pointer, bytesToCopy); *header = newHeader; } diff --git a/Unity.Entities/Types/ComponentType.cs b/Unity.Entities/Types/ComponentType.cs index 3e7f0382..d1128fca 100644 --- a/Unity.Entities/Types/ComponentType.cs +++ b/Unity.Entities/Types/ComponentType.cs @@ -30,6 +30,8 @@ public enum AccessMode public bool IsChunkComponent => TypeManager.IsChunkComponent(TypeIndex); public bool HasEntityReferences => TypeManager.HasEntityReferences(TypeIndex); + public bool IgnoreDuplicateAdd => TypeManager.IgnoreDuplicateAdd(TypeIndex); + [Obsolete("Create has been renamed. Use ReadWrite instead.", false)] [System.ComponentModel.EditorBrowsable(EditorBrowsableState.Never)] public static ComponentType Create() @@ -160,7 +162,7 @@ internal static unsafe bool CompareArray(ComponentType* type1, int typeCount1, C #if ENABLE_UNITY_COLLECTIONS_CHECKS public override string ToString() { -#if UNITY_CSHARP_TINY +#if NET_DOTS var name = TypeManager.GetTypeInfo(TypeIndex).StableTypeHash.ToString(); #else var name = GetManagedType().Name; diff --git a/Unity.Entities/Types/FastEquality.cs b/Unity.Entities/Types/FastEquality.cs index ead82cd2..bf70a1e6 100644 --- a/Unity.Entities/Types/FastEquality.cs +++ b/Unity.Entities/Types/FastEquality.cs @@ -11,7 +11,7 @@ namespace Unity.Entities { public static class FastEquality { -#if !UNITY_CSHARP_TINY +#if !NET_DOTS internal static TypeInfo CreateTypeInfo() where T : struct { return CreateTypeInfo(typeof(T)); @@ -67,7 +67,7 @@ private unsafe struct PointerSize { private void* pter; } -#if !UNITY_CSHARP_TINY +#if !NET_DOTS struct FieldData { public int Offset; diff --git a/Unity.Entities/Types/Hash128.cs b/Unity.Entities/Types/Hash128.cs index 85e46784..f8f47129 100644 --- a/Unity.Entities/Types/Hash128.cs +++ b/Unity.Entities/Types/Hash128.cs @@ -12,7 +12,7 @@ public struct Hash128 : IEquatable public unsafe override string ToString() { -#if !UNITY_CSHARP_TINY +#if !NET_DOTS var str = new string('0', 32); fixed (char* buf = str) { diff --git a/Unity.Entities/Types/TypeManager.cs b/Unity.Entities/Types/TypeManager.cs index 093c2951..51419a3d 100644 --- a/Unity.Entities/Types/TypeManager.cs +++ b/Unity.Entities/Types/TypeManager.cs @@ -3,6 +3,8 @@ using System.Diagnostics; using System.Linq; using System.Reflection; +using System.Runtime.CompilerServices; +using System.Runtime.InteropServices; using System.Threading; using Unity.Collections; using Unity.Collections.LowLevel.Unsafe; @@ -71,26 +73,38 @@ public enum TypeCategory public const int BufferComponentTypeFlag = 1 << 27; public const int SharedComponentTypeFlag = 1 << 28; public const int ChunkComponentTypeFlag = 1<<29; - public const int ZeroSizeTypeFlag = 1<<30; + public const int ZeroSizeInChunkTypeFlag = 1<<30; public const int ClearFlagsMask = 0x00FFFFFF; public const int SystemStateSharedComponentTypeFlag = SystemStateTypeFlag | SharedComponentTypeFlag; - + public const int MaximumChunkCapacity = int.MaxValue; + public const int MaximumSupportedAlignment = 16; public const int MaximumTypesCount = 1024 * 10; + private static volatile int s_Count; -#if !UNITY_CSHARP_TINY + private static int s_InitCount = 0; +#if !NET_DOTS private static SpinLock s_CreateTypeLock; #endif public static int ObjectOffset; -#if !UNITY_CSHARP_TINY - public static IEnumerable AllTypes { get { return Enumerable.Take(s_Types, s_Count); } } - private static Dictionary s_StableTypeHashToTypeIndex; +#if !NET_DOTS + public static IEnumerable AllTypes { get { return Enumerable.Take(s_TypeInfos, s_Count); } } private static Dictionary s_ManagedTypeToIndex; #endif - static TypeInfo[] s_Types; - static Type[] s_Systems; + private static TypeInfo[] s_TypeInfos; + private static Type[] s_Systems; + private static NativeHashMap s_StableTypeHashToTypeIndex; + +#if NET_DOTS + private static List s_FastEqualityTypeInfoList; + private static List s_DynamicTypeList; + private static NativeList s_WriteGroupList; + private static NativeList s_EntityOffsetList; + private static NativeList s_BlobAssetRefOffsetList; +#endif + #if !UNITY_ZEROPLAYER internal static Type UnityEngineComponentType; @@ -101,26 +115,26 @@ public static void RegisterUnityEngineComponentType(Type type) UnityEngineComponentType = type; } #endif - private struct StaticTypeLookup + public struct EntityOffsetInfo { - public static int typeIndex; + public int Offset; } - public struct EntityOffsetInfo + public struct StaticTypeLookup { - public int Offset; + public static int typeIndex; } public struct EqualityHelper { - public delegate bool EqualsFn(T left, T right); - public delegate int HashFn(T value); + public delegate bool EqualsFn(ref T left, ref T right); + public delegate int HashFn(ref T value); public static new EqualsFn Equals; public static HashFn Hash; } -#if !UNITY_CSHARP_TINY +#if !NET_DOTS // https://stackoverflow.com/a/27851610 static bool IsZeroSizeStruct(Type t) { @@ -129,10 +143,10 @@ static bool IsZeroSizeStruct(Type t) } #endif - // NOTE: This type will be moved into Unity.Entities.StaticTypeRegistry once Static Type Registry generation is hooked into #!UNITY_CSHARP_TINY builds + // NOTE: This type will be moved into Unity.Entities.StaticTypeRegistry once Static Type Registry generation is hooked into #!NET_DOTS builds public readonly struct TypeInfo { -#if !UNITY_CSHARP_TINY +#if !NET_DOTS public TypeInfo(Type type, int typeIndex, int size, TypeCategory category, FastEquality.TypeInfo typeInfo, EntityOffsetInfo[] entityOffsets, EntityOffsetInfo[] blobAssetRefOffsets, ulong memoryOrdering, int bufferCapacity, int elementSize, int alignmentInBytes, ulong stableTypeHash, int* writeGroups, int writeGroupCount, int maximumChunkCapacity) { Type = type; @@ -160,7 +174,7 @@ public TypeInfo(Type type, int typeIndex, int size, TypeCategory category, FastE if (typeIndex != 0) { if (SizeInChunk == 0) - TypeIndex |= ZeroSizeTypeFlag; + TypeIndex |= ZeroSizeInChunkTypeFlag; if(Category == TypeCategory.ISharedComponentData) TypeIndex |= SharedComponentTypeFlag; @@ -185,7 +199,8 @@ public TypeInfo(Type type, int typeIndex, int size, TypeCategory category, FastE public readonly int SizeInChunk; // Normally the same as SizeInChunk (for components), but for buffers means size of an individual element. public readonly int ElementSize; - // Sometimes we need to know not only the size, but the alignment. + // Sometimes we need to know not only the size, but the alignment. For buffers this is the alignment + // of an individual element. public readonly int AlignmentInBytes; public readonly int BufferCapacity; public readonly FastEquality.TypeInfo FastEqualityTypeInfo; @@ -201,53 +216,65 @@ public TypeInfo(Type type, int typeIndex, int size, TypeCategory category, FastE public readonly int WriteGroupCount; public readonly int MaximumChunkCapacity; + // Alignment of this type in a chunk. Normally the same + // as AlignmentInBytes, but that might be less than this + // for buffer elements, whereas the buffer itself must + // be aligned to the maximum. + public int AlignmentInChunkInBytes { + get { + if (Category == TypeCategory.BufferData) + return MaximumSupportedAlignment; + return AlignmentInBytes; + } + } + public bool IsZeroSized => SizeInChunk == 0; public bool HasWriteGroups => WriteGroupCount > 0; #else - public TypeInfo(int typeIndex, TypeCategory category, int entityOffsetCount, int entityOffsetStartIndex, ulong memoryOrdering, ulong stableTypeHash, int bufferCapacity, int typeSize, int elementSize, int alignmentInBytes, bool isSystemStateComponent, bool isSystemStateSharedComponent) + // NOTE: Any change to this constructor prototype requires a change in the TypeRegGen to match + public TypeInfo(int typeIndex, TypeCategory category, int entityOffsetCount, int entityOffsetStartIndex, + ulong memoryOrdering, ulong stableTypeHash, int bufferCapacity, int typeSize, int elementSize, + int alignmentInBytes, int maxChunkCapacity, int writeGroupCount, int writeGroupStartIndex, + int blobAssetRefOffsetCount, int blobAssetRefOffsetStartIndex, int fastEqualityIndex, bool usesDynamicInfo) { TypeIndex = typeIndex; Category = category; EntityOffsetCount = entityOffsetCount; EntityOffsetStartIndex = entityOffsetStartIndex; - //TODO: add BlobAssetRefOffset support to the static type registry - BlobAssetRefOffsetCount = 0; - BlobAssetRefOffsetStartIndex = 0; MemoryOrdering = memoryOrdering; StableTypeHash = stableTypeHash; BufferCapacity = bufferCapacity; SizeInChunk = typeSize; - AlignmentInBytes = alignmentInBytes; ElementSize = elementSize; - - if (typeIndex != 0) - { - if (SizeInChunk == 0) - TypeIndex |= ZeroSizeTypeFlag; - - if(Category == TypeCategory.ISharedComponentData) - TypeIndex |= SharedComponentTypeFlag; - - //System state shared components are also considered system state components - if (isSystemStateComponent || isSystemStateSharedComponent) - TypeIndex |= SystemStateTypeFlag; - - if (isSystemStateSharedComponent) - TypeIndex |= SystemStateSharedComponentTypeFlag; - - if (Category == TypeCategory.BufferData) - TypeIndex |= BufferComponentTypeFlag; - - if (EntityOffsetCount == 0) - TypeIndex |= HasNoEntityReferencesFlag; - } + AlignmentInBytes = alignmentInBytes; + MaximumChunkCapacity = maxChunkCapacity; + WriteGroupCount = writeGroupCount; + WriteGroupStartIndex = writeGroupStartIndex; + BlobAssetRefOffsetCount = blobAssetRefOffsetCount; + BlobAssetRefOffsetStartIndex = blobAssetRefOffsetStartIndex; + FastEqualityIndex = fastEqualityIndex; // Only used for Hybrid types (should be removed once we code gen all equality cases) + UsesDynamicInfo = usesDynamicInfo; } public readonly int TypeIndex; // Note that this includes internal capacity and header overhead for buffers. public readonly int SizeInChunk; - // Sometimes we need to know not only the size, but the alignment. + // Sometimes we need to know not only the size, but the alignment. For buffers this is the alignment + // of an individual element. public readonly int AlignmentInBytes; + // Alignment of this type in a chunk. Normally the same + // as AlignmentInBytes, but that might be less than this + // for buffer elements, whereas the buffer itself must + // be aligned to the maximum. + public int AlignmentInChunkInBytes + { + get + { + if (Category == TypeCategory.BufferData) + return MaximumSupportedAlignment; + return AlignmentInBytes; + } + } // Normally the same as SizeInChunk (for components), but for buffers means size of an individual element. public readonly int ElementSize; public readonly int BufferCapacity; @@ -255,33 +282,74 @@ public TypeInfo(int typeIndex, TypeCategory category, int entityOffsetCount, int public readonly ulong MemoryOrdering; public readonly ulong StableTypeHash; public readonly int EntityOffsetCount; - public readonly int EntityOffsetStartIndex; + internal readonly int EntityOffsetStartIndex; public readonly int BlobAssetRefOffsetCount; - public readonly int BlobAssetRefOffsetStartIndex; + internal readonly int BlobAssetRefOffsetStartIndex; + public readonly int WriteGroupCount; + internal readonly int WriteGroupStartIndex; + public readonly int MaximumChunkCapacity; + internal readonly int FastEqualityIndex; + internal readonly bool UsesDynamicInfo; + + public bool IsZeroSized => !UsesDynamicInfo && SizeInChunk == 0; + public bool HasWriteGroups => WriteGroupCount > 0; + + // NOTE: We explictly exclude Type as a member of TypeInfo so the type can remain a ValueType + public Type Type => StaticTypeRegistry.StaticTypeRegistry.Types[TypeIndex & ClearFlagsMask]; + + // To consider: pinning the static array ptr and storing the correct offset pinned ptr for TypeInfo and the nremove these getters + public EntityOffsetInfo* EntityOffsets + { + get + { + if (EntityOffsetCount > 0) + { + return ((EntityOffsetInfo*)UnsafeUtility.AddressOf(ref StaticTypeRegistry.StaticTypeRegistry.EntityOffsets[0])) + EntityOffsetStartIndex; + } - public bool IsZeroSized => SizeInChunk == 0; - public EntityOffsetInfo* EntityOffsets => EntityOffsetCount > 0 ? ((EntityOffsetInfo*) UnsafeUtility.AddressOf(ref StaticTypeRegistry.StaticTypeRegistry.EntityOffsets[0])) + EntityOffsetStartIndex : null; - public EntityOffsetInfo* BlobAssetRefOffsets => BlobAssetRefOffsetCount > 0 ? ((EntityOffsetInfo*) UnsafeUtility.AddressOf(ref StaticTypeRegistry.StaticTypeRegistry.EntityOffsets[0])) + BlobAssetRefOffsetStartIndex : null; + return null; + } + } + public EntityOffsetInfo* BlobAssetRefOffsets + { + get + { + if (BlobAssetRefOffsetCount > 0) + { + return ((EntityOffsetInfo*)UnsafeUtility.AddressOf(ref StaticTypeRegistry.StaticTypeRegistry.BlobAssetReferenceOffsets[0])) + BlobAssetRefOffsetStartIndex; + } + + return null; + } + } + public int* WriteGroups + { + get + { + if (WriteGroupCount> 0) + { + return ((int*)UnsafeUtility.AddressOf(ref StaticTypeRegistry.StaticTypeRegistry.WriteGroups[0])) + WriteGroupStartIndex; + } + + return null; + } + } #endif } public static unsafe TypeInfo GetTypeInfo(int typeIndex) { - return s_Types[typeIndex & ClearFlagsMask]; + return s_TypeInfos[typeIndex & ClearFlagsMask]; } public static TypeInfo GetTypeInfo() where T : struct { - return s_Types[GetTypeIndex() & ClearFlagsMask]; + return s_TypeInfos[GetTypeIndex() & ClearFlagsMask]; } public static Type GetType(int typeIndex) { - #if !UNITY_CSHARP_TINY - return s_Types[typeIndex & ClearFlagsMask].Type; - #else - return StaticTypeRegistry.StaticTypeRegistry.Types[typeIndex & ClearFlagsMask]; - #endif + return s_TypeInfos[typeIndex & ClearFlagsMask].Type; } public static int GetTypeCount() @@ -293,12 +361,14 @@ public static int GetTypeCount() public static bool IsSystemStateComponent(int typeIndex) => (typeIndex & SystemStateTypeFlag) != 0; public static bool IsSystemStateSharedComponent(int typeIndex) => (typeIndex & SystemStateSharedComponentTypeFlag) == SystemStateSharedComponentTypeFlag; public static bool IsSharedComponent(int typeIndex) => (typeIndex & SharedComponentTypeFlag) != 0; - public static bool IsZeroSized(int typeIndex) => (typeIndex & ZeroSizeTypeFlag) != 0; + public static bool IsZeroSized(int typeIndex) => (typeIndex & ZeroSizeInChunkTypeFlag) != 0; public static bool IsChunkComponent(int typeIndex) => (typeIndex & ChunkComponentTypeFlag) != 0; public static bool HasEntityReferences(int typeIndex) => (typeIndex & HasNoEntityReferencesFlag) == 0; - public static int MakeChunkComponentTypeIndex(int typeIndex) => (typeIndex | ChunkComponentTypeFlag | ZeroSizeTypeFlag); - public static int ChunkComponentToNormalTypeIndex(int typeIndex) => s_Types[typeIndex & ClearFlagsMask].TypeIndex; + public static bool IgnoreDuplicateAdd(int typeIndex) => (typeIndex & ZeroSizeInChunkTypeFlag) != 0 && (typeIndex & SharedComponentTypeFlag) == 0; + + public static int MakeChunkComponentTypeIndex(int typeIndex) => (typeIndex | ChunkComponentTypeFlag | ZeroSizeInChunkTypeFlag); + public static int ChunkComponentToNormalTypeIndex(int typeIndex) => s_TypeInfos[typeIndex & ClearFlagsMask].TypeIndex; // TODO: this creates a dependency on UnityEngine, but makes splitting code in separate assemblies easier. We need to remove it during the biggere refactor. private struct ObjectOffsetType @@ -307,100 +377,185 @@ private struct ObjectOffsetType private void* v1; } -#if !UNITY_CSHARP_TINY +#if !NET_DOTS private static void AddTypeInfoToTables(TypeInfo typeInfo) { - s_Types[typeInfo.TypeIndex & ClearFlagsMask] = typeInfo; - s_StableTypeHashToTypeIndex.Add(typeInfo.StableTypeHash, typeInfo.TypeIndex); + s_TypeInfos[typeInfo.TypeIndex & ClearFlagsMask] = typeInfo; + s_StableTypeHashToTypeIndex.TryAdd(typeInfo.StableTypeHash, typeInfo.TypeIndex); s_ManagedTypeToIndex.Add(typeInfo.Type, typeInfo.TypeIndex); ++s_Count; } #endif - public static void Initialize() + /// + /// Initializes the TypeManager with all ECS type information. Additional calls without a matching Shutdown() simply increment the TypeManager ref-count and return + /// + /// Returns the TypeManager ref-count. It is expected for each Initialize() call to be matched by a Shutdown() call. + public static int Initialize() { - if (s_Types != null) - return; + int initVal = Interlocked.Increment(ref s_InitCount); + if (initVal != 1) + return initVal; ObjectOffset = UnsafeUtility.SizeOf(); -#if !UNITY_CSHARP_TINY - s_CreateTypeLock = new SpinLock(); - s_ManagedTypeToIndex = new Dictionary(1000); -#endif - s_Types = new TypeInfo[MaximumTypesCount]; - - #if !UNITY_CSHARP_TINY - s_StableTypeHashToTypeIndex = new Dictionary(); + #if !NET_DOTS + s_CreateTypeLock = new SpinLock(); + s_ManagedTypeToIndex = new Dictionary(1000); #endif - s_Count = 0; - - #if !UNITY_CSHARP_TINY - s_Types[s_Count++] = new TypeInfo(null, 0, 0, TypeCategory.ComponentData, FastEquality.TypeInfo.Null, null, null, 0, -1, 0, 1, 0, null, 0, int.MaxValue); + s_TypeInfos = new TypeInfo[MaximumTypesCount]; - // This must always be first so that Entity is always index 0 in the archetype - AddTypeInfoToTables(new TypeInfo(typeof(Entity), 1, sizeof(Entity), TypeCategory.EntityData, - FastEquality.CreateTypeInfo(), EntityRemapUtility.CalculateEntityOffsets(), null, 0, -1, sizeof(Entity), UnsafeUtility.AlignOf(), CalculateStableTypeHash(typeof(Entity)), null, 0, int.MaxValue)); + s_Count = 0; + #if !NET_DOTS + s_TypeInfos[s_Count++] = new TypeInfo(null, 0, 0, TypeCategory.ComponentData, FastEquality.TypeInfo.Null, null, null, 0, -1, 0, 1, 0, null, 0, int.MaxValue); InitializeAllComponentTypes(); #else + s_FastEqualityTypeInfoList = new List(); + s_DynamicTypeList = new List(); + s_EntityOffsetList = new NativeList(Allocator.Persistent); + s_BlobAssetRefOffsetList = new NativeList(Allocator.Persistent); + s_WriteGroupList = new NativeList(Allocator.Persistent); + + // Registers all types and their static info from the static type rgistry + // Note: this will call AddStaticTypesToRegistry which will initialize s_StableTypeHashToTypeIndex StaticTypeRegistry.StaticTypeRegistry.RegisterStaticTypes(); #endif + + return initVal; + } + + /// + /// Removes all ECS type information and any allocated memory when the TypeManager ref-count == 1. That is, + /// the TypeManager will shutdown on the last call to Shutdown() matching the number of invocations of Initialize(). + /// + /// Returns the TypeManager ref-count. It is expected for each Shutdown() call to match a previous Initialize() call. + public static int Shutdown() + { + int initVal = Interlocked.Decrement(ref s_InitCount); + + if (initVal < 0) + throw new Exception("Calling TypeManager.Shutdown() before TypeManager.Initialize() has ever been called"); + + if (initVal == 0) + { + #if !NET_DOTS + ClearStaticTypeLookup(); + #endif + + s_TypeInfos = null; + s_Count = 0; + s_StableTypeHashToTypeIndex.Dispose(); + #if NET_DOTS + s_Systems = null; + s_EntityOffsetList.Dispose(); + s_BlobAssetRefOffsetList.Dispose(); + s_WriteGroupList.Dispose(); + #else + s_ManagedTypeToIndex.Clear(); + #endif + } + + return initVal; } -#if UNITY_CSHARP_TINY +#if !NET_DOTS + static void ClearStaticTypeLookup() + { + var staticLookupGenericType = Type.GetType("Unity.Entities.TypeManager+StaticTypeLookup`1"); + for (int i = 1; i < s_Count; ++i) + { + var type = s_TypeInfos[i].Type; + var staticLookupType = staticLookupGenericType.MakeGenericType(type); + var typeIndexField = staticLookupType.GetField("typeIndex", BindingFlags.Static | BindingFlags.Public); + typeIndexField.SetValue(null, 0); + } + } +#endif + +#if NET_DOTS // Called by the StaticTypeRegistry - internal static void AddStaticTypesFromRegistry(ref TypeInfo[] typeArray, int count) + internal static void AddStaticTypesFromRegistry(in TypeInfo[] typeInfoArray/*, int count*/) { - if (count >= MaximumTypesCount) + if (typeInfoArray.Length >= MaximumTypesCount) throw new Exception("More types detected than MaximumTypesCount. Increase the static buffer size."); s_Count = 0; - for (int i = 0; i < count; ++i) + + if (s_StableTypeHashToTypeIndex.IsCreated) + s_StableTypeHashToTypeIndex.Dispose(); + + s_StableTypeHashToTypeIndex = new NativeHashMap(typeInfoArray.Length * 2, Allocator.Persistent); // Extra room added for dynamically added types + + for (int i = 0; i < typeInfoArray.Length; ++i) { - s_Types[s_Count++] = typeArray[i]; + TypeInfo typeInfo = typeInfoArray[i]; + s_TypeInfos[s_Count++] = typeInfo; + + if (!s_StableTypeHashToTypeIndex.TryAdd(typeInfo.StableTypeHash, typeInfo.TypeIndex)) + throw new Exception("Failed to add hash to StableTypeHash -> typeIndex dictionary."); } } // Called by the StaticTypeRegistry - internal static void AddStaticSystemsFromRegistry(ref Type[] systemArray) + internal static void AddStaticSystemsFromRegistry(in Type[] systemArray) { s_Systems = systemArray; } #endif -#if !UNITY_CSHARP_TINY +#if !NET_DOTS static void InitializeAllComponentTypes() { - var componentTypeSet = new HashSet(); - - foreach (var assembly in AppDomain.CurrentDomain.GetAssemblies()) + var lockTaken = false; + try { - if (!IsAssemblyReferencingEntities(assembly)) - continue; + s_CreateTypeLock.Enter(ref lockTaken); - foreach (var type in assembly.GetTypes()) + var componentTypeSet = new HashSet(); + + foreach (var assembly in AppDomain.CurrentDomain.GetAssemblies()) { - if (type.IsAbstract || !type.IsValueType) - continue; - if (!UnsafeUtility.IsUnmanaged(type)) + if (!IsAssemblyReferencingEntities(assembly)) continue; - if (typeof(IComponentData).IsAssignableFrom(type) || - typeof(ISharedComponentData).IsAssignableFrom(type) || - typeof(IBufferElementData).IsAssignableFrom(type)) + foreach (var type in assembly.GetTypes()) { - componentTypeSet.Add(type); + if (type.IsAbstract || !type.IsValueType) + continue; + + // XXX There's a bug in the Unity Mono scripting backend where if the + // Mono type hasn't been initialized, the IsUnmanaged result is wrong. + // We force it to be fully initialized by creating an instance until + // that bug is fixed. + try + { + var inst = Activator.CreateInstance(type); + } catch (Exception) + { + // ignored + } + + if (!UnsafeUtility.IsUnmanaged(type)) + continue; + + if (typeof(IComponentData).IsAssignableFrom(type) || + typeof(ISharedComponentData).IsAssignableFrom(type) || + typeof(IBufferElementData).IsAssignableFrom(type)) + { + componentTypeSet.Add(type); + } } } - } - var lockTaken = false; - try - { - s_CreateTypeLock.Enter(ref lockTaken); + s_StableTypeHashToTypeIndex = new NativeHashMap(componentTypeSet.Count * 2, Allocator.Persistent); // Extra room added for dynamically added types + + // This must always be first so that Entity is always first in the archetype + AddTypeInfoToTables(new TypeInfo(typeof(Entity), 1, sizeof(Entity), TypeCategory.EntityData, + FastEquality.CreateTypeInfo(), EntityRemapUtility.CalculateEntityOffsets(), null, 0, -1, + sizeof(Entity), UnsafeUtility.AlignOf(), CalculateStableTypeHash(typeof(Entity)), null, 0, int.MaxValue)); var componentTypeCount = componentTypeSet.Count; var componentTypes = new Type[componentTypeCount]; @@ -411,7 +566,9 @@ static void InitializeAllComponentTypes() var startTypeIndex = s_Count; for (int i = 0; i < componentTypes.Length; i++) + { typeIndexByType[componentTypes[i]] = startTypeIndex + i; + } GatherWriteGroups(componentTypes, startTypeIndex, typeIndexByType, writeGroupByType); AddAllComponentTypes(componentTypes, startTypeIndex, writeGroupByType); @@ -472,6 +629,11 @@ private static void GatherWriteGroups(Type[] componentTypes, int startTypeIndex, foreach (var attribute in type.GetCustomAttributes(typeof(WriteGroupAttribute))) { var attr = (WriteGroupAttribute) attribute; + if (!typeIndexByType.ContainsKey(attr.TargetType)) + { + Debug.LogError($"GatherWriteGroups: looking for {attr.TargetType} but it hasn't been set up yet"); + } + int targetTypeIndex = typeIndexByType[attr.TargetType]; if (!writeGroupByType.ContainsKey(targetTypeIndex)) @@ -501,8 +663,8 @@ private static int FindTypeIndex(Type type) { for (var i = 0; i != s_Count; i++) { - var c = s_Types[i]; - if (StaticTypeRegistry.StaticTypeRegistry.Types[c.TypeIndex & ClearFlagsMask] == type) + var c = s_TypeInfos[i]; + if (c.Type == type) return c.TypeIndex; } @@ -513,6 +675,8 @@ private static int FindTypeIndex(Type type) public static int GetTypeIndex() { var typeIndex = StaticTypeLookup.typeIndex; + // with NET_DOTS, this could be a straight return without a 0 check, + // if the static typereg code were to set the generic field during init if (typeIndex != 0) return typeIndex; @@ -530,17 +694,17 @@ public static int GetTypeIndex(Type type) public static bool Equals(ref T left, ref T right) where T : struct { - #if !UNITY_CSHARP_TINY + #if !NET_DOTS var typeInfo = TypeManager.GetTypeInfo().FastEqualityTypeInfo; return FastEquality.Equals(ref left, ref right, typeInfo); #else - return EqualityHelper.Equals(left, right); + return EqualityHelper.Equals(ref left, ref right); #endif } public static bool Equals(void* left, void* right, int typeIndex) { - #if !UNITY_CSHARP_TINY + #if !NET_DOTS var typeInfo = TypeManager.GetTypeInfo(typeIndex).FastEqualityTypeInfo; return FastEquality.Equals(left, right, typeInfo); #else @@ -548,19 +712,51 @@ public static bool Equals(void* left, void* right, int typeIndex) #endif } + public static bool Equals(object left, object right, int typeIndex) + { + #if !NET_DOTS + var leftptr = (byte*) UnsafeUtility.PinGCObjectAndGetAddress(left, out var lhandle) + ObjectOffset; + var rightptr = (byte*) UnsafeUtility.PinGCObjectAndGetAddress(right, out var rhandle) + ObjectOffset; + + var typeInfo = GetTypeInfo(typeIndex).FastEqualityTypeInfo; + var result = FastEquality.Equals(leftptr, rightptr, typeInfo); + + UnsafeUtility.ReleaseGCObject(lhandle); + UnsafeUtility.ReleaseGCObject(rhandle); + return result; + #else + return StaticTypeRegistry.StaticTypeRegistry.Equals(left, right, typeIndex & ClearFlagsMask); + #endif + } + + public static bool Equals(object left, void* right, int typeIndex) + { + #if !NET_DOTS + var leftptr = (byte*) UnsafeUtility.PinGCObjectAndGetAddress(left, out var lhandle) + ObjectOffset; + + var typeInfo = GetTypeInfo(typeIndex).FastEqualityTypeInfo; + var result = FastEquality.Equals(leftptr, right, typeInfo); + + UnsafeUtility.ReleaseGCObject(lhandle); + return result; + #else + return StaticTypeRegistry.StaticTypeRegistry.Equals(left, right, typeIndex & ClearFlagsMask); + #endif + } + public static int GetHashCode(ref T val) where T : struct { - #if !UNITY_CSHARP_TINY + #if !NET_DOTS var typeInfo = TypeManager.GetTypeInfo().FastEqualityTypeInfo; return FastEquality.GetHashCode(ref val, typeInfo); #else - return EqualityHelper.Hash(val); + return EqualityHelper.Hash(ref val); #endif } public static int GetHashCode(void* val, int typeIndex) { - #if !UNITY_CSHARP_TINY + #if !NET_DOTS var typeInfo = TypeManager.GetTypeInfo(typeIndex).FastEqualityTypeInfo; return FastEquality.GetHashCode(val, typeInfo); #else @@ -568,18 +764,33 @@ public static int GetHashCode(void* val, int typeIndex) #endif } + public static int GetHashCode(object val, int typeIndex) + { + #if !NET_DOTS + var ptr = (byte*) UnsafeUtility.PinGCObjectAndGetAddress(val, out var handle) + ObjectOffset; + + var typeInfo = GetTypeInfo(typeIndex).FastEqualityTypeInfo; + var result = FastEquality.GetHashCode(ptr, typeInfo); + + UnsafeUtility.ReleaseGCObject(handle); + return result; + #else + return StaticTypeRegistry.StaticTypeRegistry.BoxedGetHashCode(val, typeIndex & ClearFlagsMask); + #endif + } + public static int GetTypeIndexFromStableTypeHash(ulong stableTypeHash) { -#if !UNITY_CSHARP_TINY - if(s_StableTypeHashToTypeIndex.TryGetValue(stableTypeHash, out var typeIndex)) - return typeIndex; - return -1; -#else - throw new InvalidOperationException("Not allowed in Project Tiny"); -#endif + #if !NET_DOTS + if(s_StableTypeHashToTypeIndex.TryGetValue(stableTypeHash, out var typeIndex)) + return typeIndex; + return -1; + #else + throw new InvalidOperationException("Not allowed in Project Tiny"); + #endif } -#if !UNITY_CSHARP_TINY +#if !NET_DOTS public static bool IsAssemblyReferencingEntities(Assembly assembly) { const string entitiesAssemblyName = "Unity.Entities"; @@ -610,13 +821,13 @@ public static string[] SystemNames public static string SystemName(Type t) { -#if UNITY_CSHARP_TINY - int index = GetSystemTypeIndex(t); - if (index < 0 || index >= SystemNames.Length) return "null"; - return SystemNames[index]; -#else - return t.FullName; -#endif + #if NET_DOTS + int index = GetSystemTypeIndex(t); + if (index < 0 || index >= SystemNames.Length) return "null"; + return SystemNames[index]; + #else + return t.FullName; + #endif } public static int GetSystemTypeIndex(Type t) @@ -631,13 +842,13 @@ public static int GetSystemTypeIndex(Type t) public static bool IsSystemAGroup(Type t) { -#if !UNITY_CSHARP_TINY - return t.IsSubclassOf(typeof(ComponentSystemGroup)); -#else - int index = GetSystemTypeIndex(t); - var isGroup = StaticTypeRegistry.StaticTypeRegistry.SystemIsGroup[index]; - return isGroup; -#endif + #if !NET_DOTS + return t.IsSubclassOf(typeof(ComponentSystemGroup)); + #else + int index = GetSystemTypeIndex(t); + var isGroup = StaticTypeRegistry.StaticTypeRegistry.SystemIsGroup[index]; + return isGroup; + #endif } /// @@ -665,34 +876,34 @@ public static Attribute[] GetSystemAttributes(Type systemType) /// public static Attribute[] GetSystemAttributes(Type systemType, Type attributeType) { -#if !UNITY_CSHARP_TINY - var objArr = systemType.GetCustomAttributes(attributeType, true); - var attr = new Attribute[objArr.Length]; - for (int i = 0; i < objArr.Length; i++) { - attr[i] = objArr[i] as Attribute; - } - return attr; -#else - Attribute[] attr = StaticTypeRegistry.StaticTypeRegistry.GetSystemAttributes(systemType); - int count = 0; - for (int i = 0; i < attr.Length; ++i) - { - if (attr[i].GetType() == attributeType) + #if !NET_DOTS + var objArr = systemType.GetCustomAttributes(attributeType, true); + var attr = new Attribute[objArr.Length]; + for (int i = 0; i < objArr.Length; i++) { + attr[i] = objArr[i] as Attribute; + } + return attr; + #else + Attribute[] attr = StaticTypeRegistry.StaticTypeRegistry.GetSystemAttributes(systemType); + int count = 0; + for (int i = 0; i < attr.Length; ++i) { - ++count; + if (attr[i].GetType() == attributeType) + { + ++count; + } } - } - Attribute[] result = new Attribute[count]; - count = 0; - for (int i = 0; i < attr.Length; ++i) - { - if (attr[i].GetType() == attributeType) + Attribute[] result = new Attribute[count]; + count = 0; + for (int i = 0; i < attr.Length; ++i) { - result[count++] = attr[i]; + if (attr[i].GetType() == attributeType) + { + result[count++] = attr[i]; + } } - } - return result; -#endif + return result; + #endif } #if ENABLE_UNITY_COLLECTIONS_CHECKS @@ -719,21 +930,43 @@ internal static void CheckComponentType(Type type) public static NativeArray GetWriteGroupTypes(int typeIndex) { -#if UNITY_CSHARP_TINY - var arr = NativeArrayUnsafeUtility.ConvertExistingDataToNativeArray(null, 0, Allocator.None); -#else var type = GetTypeInfo(typeIndex); var writeGroups = type.WriteGroups; var writeGroupCount = type.WriteGroupCount; var arr = NativeArrayUnsafeUtility.ConvertExistingDataToNativeArray(writeGroups, writeGroupCount, Allocator.None); -#endif #if ENABLE_UNITY_COLLECTIONS_CHECKS NativeArrayUnsafeUtility.SetAtomicSafetyHandle(ref arr, AtomicSafetyHandle.Create()); #endif return arr; } -#if !UNITY_CSHARP_TINY + internal static int NextPowerOfTwo(int n) + { + if (n == 0) + return 1; + n--; + n |= n >> 1; + n |= n >> 2; + n |= n >> 4; + n |= n >> 8; + n |= n >> 16; + return n + 1; + } + + internal static int AlignUp(int value, int alignment) + { + int mask = alignment - 1; + if ((value & ~mask) == 0) + return value; + return (value + mask) & ~mask; + } + + internal static bool IsPowerOfTwo(int value) + { + return (value & (value - 1)) == 0; + } + +#if !NET_DOTS // // The reflection-based type registration path that we can't support with tiny csharp profile. // A generics compile-time path is temporarily used (see later in the file) until we have @@ -766,6 +999,7 @@ static void CalculatBlobAssetRefOffsetsRecurse(ref List offset } } + // Todo: Replace with code in Entities.BuildUtils static ulong CalculateMemoryOrdering(Type type) { if (type == typeof(Entity)) @@ -823,6 +1057,7 @@ private static ulong HashStringWithFNV1A64(string text) return result; } + // Todo: Replace with code in Entities.BuildUtils static ulong CalculateStableTypeHash(Type type) { return HashStringWithFNV1A64(type.AssemblyQualifiedName); @@ -881,7 +1116,7 @@ internal static TypeInfo BuildComponentType(Type type, int* writeGroups, int wri int bufferCapacity = -1; var memoryOrdering = CalculateMemoryOrdering(type); var stableTypeHash = CalculateStableTypeHash(type); - var maxChunkCapacity = int.MaxValue; + var maxChunkCapacity = MaximumChunkCapacity; var maxCapacityAttribute = type.GetCustomAttribute(); if (maxCapacityAttribute != null) @@ -898,27 +1133,37 @@ internal static TypeInfo BuildComponentType(Type type, int* writeGroups, int wri CheckIsAllowedAsComponentData(type, nameof(IComponentData)); category = TypeCategory.ComponentData; + + int sizeInBytes = UnsafeUtility.SizeOf(type); + // TODO: Implement UnsafeUtility.AlignOf(type) + // Assume an alignment of 16, unless the type itself is smaller and a power of two + alignmentInBytes = MaximumSupportedAlignment; + if (sizeInBytes < alignmentInBytes && IsPowerOfTwo(sizeInBytes)) + alignmentInBytes = sizeInBytes; + if (TypeManager.IsZeroSizeStruct(type)) componentSize = 0; else - componentSize = UnsafeUtility.SizeOf(type); + componentSize = sizeInBytes; typeInfo = FastEquality.CreateTypeInfo(type); entityOffsets = EntityRemapUtility.CalculateEntityOffsets(type); blobAssetRefOffsets = CalculatBlobAssetRefOffsets(type); - - int sizeInBytes = UnsafeUtility.SizeOf(type); - // TODO: Implement UnsafeUtility.AlignOf(type) - alignmentInBytes = 16; - if(sizeInBytes < 16 && (sizeInBytes & (sizeInBytes-1))==0) - alignmentInBytes = sizeInBytes; } else if (typeof(IBufferElementData).IsAssignableFrom(type)) { CheckIsAllowedAsComponentData(type, nameof(IBufferElementData)); category = TypeCategory.BufferData; - elementSize = UnsafeUtility.SizeOf(type); + + int sizeInBytes = UnsafeUtility.SizeOf(type); + // TODO: Implement UnsafeUtility.AlignOf(type) + // Assume an alignment of 16, unless the type itself is smaller + alignmentInBytes = MaximumSupportedAlignment; + if (sizeInBytes < alignmentInBytes && IsPowerOfTwo(sizeInBytes)) + alignmentInBytes = sizeInBytes; + + elementSize = sizeInBytes; var capacityAttribute = (InternalBufferCapacityAttribute) type.GetCustomAttribute(typeof(InternalBufferCapacityAttribute)); if (capacityAttribute != null) @@ -930,12 +1175,6 @@ internal static TypeInfo BuildComponentType(Type type, int* writeGroups, int wri typeInfo = FastEquality.CreateTypeInfo(type); entityOffsets = EntityRemapUtility.CalculateEntityOffsets(type); blobAssetRefOffsets = CalculatBlobAssetRefOffsets(type); - - int sizeInBytes = UnsafeUtility.SizeOf(type); - // TODO: Implement UnsafeUtility.AlignOf(type) - alignmentInBytes = 16; - if(sizeInBytes < 16 && (sizeInBytes & (sizeInBytes-1))==0) - alignmentInBytes = sizeInBytes; } else if (typeof(ISharedComponentData).IsAssignableFrom(type)) { diff --git a/Unity.Entities/World.cs b/Unity.Entities/World.cs new file mode 100644 index 00000000..045a5870 --- /dev/null +++ b/Unity.Entities/World.cs @@ -0,0 +1,562 @@ +using System; +using System.Collections.Generic; +using System.Collections.ObjectModel; +using System.Reflection; + +namespace Unity.Entities +{ + // Obsolete + public partial class World + { + [Obsolete("BehaviourManagers have been renamed to Systems. (UnityUpgradable) -> Systems", true)] + public IEnumerable BehaviourManagers => null; + + [Obsolete("CreateManager has been renamed to CreateSystem. (UnityUpgradable) -> CreateSystem(*)", true)] + public ComponentSystemBase CreateManager(Type type, params object[] constructorArgumnents) + { + throw new NotImplementedException(); + } + + [Obsolete("GetOrCreateManager has been renamed to GetOrCreateSystem. (UnityUpgradable) -> GetOrCreateSystem(*)", true)] + public ComponentSystemBase GetOrCreateManager(Type type) + { + throw new NotImplementedException(); + } + + [Obsolete("AddManager has been renamed to AddSystem. (UnityUpgradable) -> AddSystem(*)", true)] + public T AddManager(T manager) where T : ComponentSystemBase + { + throw new NotImplementedException(); + } + + [Obsolete("GetExistingManager has been renamed to GetExistingSystem. (UnityUpgradable) -> GetExistingSystem(*)", true)] + public ComponentSystemBase GetExistingManager(Type type) + { + throw new NotImplementedException(); + } + + [Obsolete("DestroyManager has been renamed to DestroySystem. (UnityUpgradable) -> DestroySystem(*)", true)] + public void DestroyManager(ComponentSystemBase manager) + { + throw new NotImplementedException(); + } + + // These updates can not be configured automatically inline, due to the generic type param. + [Obsolete("CreateManager has been renamed to CreateSystem. (UnityUpgradable) -> CreateSystem(*)", true)] + public T CreateManager(params object[] constructorArgumnents) where T : ComponentSystemBase + { + throw new NotImplementedException(); + } + + [Obsolete("GetOrCreateManager has been renamed to GetOrCreateSystem. (UnityUpgradable) -> GetOrCreateSystem()", true)] + public T GetOrCreateManager() where T : ComponentSystemBase + { + throw new NotImplementedException(); + } + + [Obsolete("GetExistingManager has been renamed to GetExistingSystem. (UnityUpgradable) -> GetExistingSystem()", true)] + public T GetExistingManager() where T : ComponentSystemBase + { + throw new NotImplementedException(); + } + } + +#if !UNITY_ZEROPLAYER + public partial class World : IDisposable + { + public static World Active { get; set; } + + static readonly List allWorlds = new List(); + + public static ReadOnlyCollection AllWorlds => new ReadOnlyCollection(allWorlds); + +#if ENABLE_UNITY_COLLECTIONS_CHECKS + bool m_AllowGetSystem = true; +#endif + + //@TODO: What about multiple managers of the same type... + Dictionary m_SystemLookup = + new Dictionary(); + + List m_Systems = new List(); + private EntityManager m_EntityManager; + + static int ms_SystemIDAllocator = 0; + + public IEnumerable Systems => + new ReadOnlyCollection(m_Systems); + + public string Name { get; } + + public override string ToString() + { + return Name; + } + + public int Version { get; private set; } + + public EntityManager EntityManager => m_EntityManager; + + public bool IsCreated => m_Systems != null; + + public World(string name) + { + // Debug.LogError("Create World "+ name + " - " + GetHashCode()); + Name = name; + allWorlds.Add(this); + + m_EntityManager = new EntityManager(this); + } + + public void Dispose() + { + if (!IsCreated) + throw new ArgumentException("The World has already been Disposed."); + // Debug.LogError("Dispose World "+ Name + " - " + GetHashCode()); + + if (allWorlds.Contains(this)) + allWorlds.Remove(this); + +#if ENABLE_UNITY_COLLECTIONS_CHECKS + m_AllowGetSystem = false; +#endif + // Destruction should happen in reverse order to construction + for (int i = m_Systems.Count - 1; i >= 0; --i) + { + try + { + m_Systems[i].DestroyInstance(); + } + catch (Exception e) + { + Debug.LogException(e); + } + } + + // Destroy EntityManager last + m_EntityManager.DestroyInstance(); + m_EntityManager = null; + + m_Systems.Clear(); + m_SystemLookup.Clear(); + + m_Systems = null; + m_SystemLookup = null; + + if (Active == this) + Active = null; + } + + public static void DisposeAllWorlds() + { + while (allWorlds.Count != 0) + allWorlds[0].Dispose(); + } + + ComponentSystemBase CreateSystemInternal(Type type, object[] constructorArguments) + { + if (!typeof(ComponentSystemBase).IsAssignableFrom(type)) + { + throw new ArgumentException($"Type {type} must be derived from ComponentSystem or JobComponentSystem."); + } + +#if ENABLE_UNITY_COLLECTIONS_CHECKS + + if (constructorArguments != null && constructorArguments.Length != 0) + { + var constructors = + type.GetConstructors(BindingFlags.Instance | BindingFlags.NonPublic | BindingFlags.Public); + if (constructors.Length == 1 && constructors[0].IsPrivate) + throw new MissingMethodException( + $"Constructing {type} failed because the constructor was private, it must be public."); + } + + m_AllowGetSystem = false; +#endif + ComponentSystemBase system; + try + { + system = Activator.CreateInstance(type, constructorArguments) as ComponentSystemBase; + } + catch (MissingMethodException) + { + Debug.LogError($"[Job]ComponentSystem {type} must be mentioned in a link.xml file, or annotated " + + "with a [Preserve] attribute to prevent its constructor from being stripped. " + + "See https://docs.unity3d.com/Manual/ManagedCodeStripping.html for more information."); + throw; + } + finally + { +#if ENABLE_UNITY_COLLECTIONS_CHECKS + m_AllowGetSystem = true; +#endif + } + + return AddSystem(system); + } + + ComponentSystemBase GetExistingSystemInternal(Type type) + { + ComponentSystemBase manager; + if (m_SystemLookup.TryGetValue(type, out manager)) + return manager; + + return null; + } + + ComponentSystemBase GetOrCreateSystemInternal(Type type) + { + var manager = GetExistingSystemInternal(type); + + return manager ?? CreateSystemInternal(type, null); + } + + void AddTypeLookup(Type type, ComponentSystemBase manager) + { + while (type != typeof(ComponentSystemBase)) + { + if (!m_SystemLookup.ContainsKey(type)) + m_SystemLookup.Add(type, manager); + + type = type.BaseType; + } + } + + void RemoveSystemInternal(ComponentSystemBase manager) + { + if (!m_Systems.Remove(manager)) + throw new ArgumentException($"manager does not exist in the world"); + ++Version; + + var type = manager.GetType(); + while (type != typeof(ComponentSystemBase)) + { + if (m_SystemLookup[type] == manager) + { + m_SystemLookup.Remove(type); + + foreach (var otherManager in m_Systems) + if (otherManager.GetType().IsSubclassOf(type)) + AddTypeLookup(otherManager.GetType(), otherManager); + } + + type = type.BaseType; + } + } + + void CheckGetOrCreateSystem() + { +#if ENABLE_UNITY_COLLECTIONS_CHECKS + if (!IsCreated) + throw new ArgumentException("The World has already been Disposed."); + if (!m_AllowGetSystem) + throw new ArgumentException( + "You are not allowed to get or create more systems during destruction and constructor of a system."); +#endif + } + + void CheckCreated() + { +#if ENABLE_UNITY_COLLECTIONS_CHECKS + if (!IsCreated) + throw new ArgumentException("The World has already been Disposed."); +#endif + } + + public ComponentSystemBase CreateSystem(Type type, params object[] constructorArgumnents) + { + CheckGetOrCreateSystem(); + + return CreateSystemInternal(type, constructorArgumnents); + } + + public T CreateSystem(params object[] constructorArgumnents) where T : ComponentSystemBase + { + CheckGetOrCreateSystem(); + + return (T) CreateSystemInternal(typeof(T), constructorArgumnents); + } + + public T GetOrCreateSystem() where T : ComponentSystemBase + { + CheckGetOrCreateSystem(); + + return (T) GetOrCreateSystemInternal(typeof(T)); + } + + public ComponentSystemBase GetOrCreateSystem(Type type) + { + CheckGetOrCreateSystem(); + + return GetOrCreateSystemInternal(type); + } + + public T AddSystem(T system) where T : ComponentSystemBase + { + CheckGetOrCreateSystem(); + + m_Systems.Add(system); + AddTypeLookup(system.GetType(), system); + + try + { + system.CreateInstance(this); + } + catch + { + RemoveSystemInternal(system); + throw; + } + ++Version; + return system; + } + + public T GetExistingSystem() where T : ComponentSystemBase + { + CheckGetOrCreateSystem(); + + return (T) GetExistingSystemInternal(typeof(T)); + } + + public ComponentSystemBase GetExistingSystem(Type type) + { + CheckGetOrCreateSystem(); + + return GetExistingSystemInternal(type); + } + + public void DestroySystem(ComponentSystemBase system) + { + CheckGetOrCreateSystem(); + + RemoveSystemInternal(system); + system.DestroyInstance(); + } + + public bool QuitUpdate { get; set; } + + internal static int AllocateSystemID() + { + return ++ms_SystemIDAllocator; + } + } +#else + public partial class World : IDisposable + { + public static World Active { get; set; } + + static readonly List allWorlds = new List(); + + public static World[] AllWorlds => allWorlds.ToArray(); + +#if ENABLE_UNITY_COLLECTIONS_CHECKS + bool m_AllowGetSystem = true; +#endif + + //@TODO: What about multiple managers of the same type... + List m_Systems = new List(); + private EntityManager m_EntityManager; + + int m_SystemIDAllocator = 0; + + public ComponentSystemBase[] Systems => m_Systems.ToArray(); + + public string Name { get; } + + public override string ToString() + { + return Name; + } + + public int Version { get; private set; } + + public EntityManager EntityManager => m_EntityManager; + + public bool IsCreated => true; + + public World(string name) + { + // Debug.LogError("Create World "+ name + " - " + GetHashCode()); + Name = name; + allWorlds.Add(this); + + m_EntityManager = new EntityManager(this); + Version++; + } + + public void Dispose() + { + if (!IsCreated) + throw new ArgumentException("World is already disposed"); + // Debug.LogError("Dispose World "+ Name + " - " + GetHashCode()); + + if (allWorlds.Contains(this)) + allWorlds.Remove(this); + +#if ENABLE_UNITY_COLLECTIONS_CHECKS + m_AllowGetSystem = false; +#endif + + // Destruction should happen in reverse order to construction + for (int i = m_Systems.Count - 1; i >= 0; --i) + { + try + { + m_Systems[i].DestroyInstance(); + } + catch (Exception e) + { + Debug.LogException(e); + } + } + + // Destroy EntityManager last + m_EntityManager.DestroyInstance(); + m_EntityManager = null; + + m_Systems.Clear(); + m_Systems = null; + + if (Active == this) + Active = null; + } + + public static void DisposeAllWorlds() + { + while (allWorlds.Count != 0) + allWorlds[0].Dispose(); + } + + private ComponentSystemBase CreateSystemInternal() where T : new() + { +#if ENABLE_UNITY_COLLECTIONS_CHECKS + if (!m_AllowGetSystem) + throw new ArgumentException( + "During destruction of a system you are not allowed to create more systems."); + + m_AllowGetSystem = true; +#endif + ComponentSystemBase system; + try + { +#if !NET_DOTS + system = new T() as ComponentSystemBase; +#else + system = TypeManager.ConstructSystem(typeof(T)); +#endif + } + catch + { +#if ENABLE_UNITY_COLLECTIONS_CHECKS + m_AllowGetSystem = false; +#endif + throw; + } + + return AddSystem(system); + } + + private ComponentSystemBase GetExistingSystemInternal() + { + return GetExistingSystem(typeof(T)); + } + + private ComponentSystemBase GetExistingSystemInternal(Type type) + { +#if ENABLE_UNITY_COLLECTIONS_CHECKS + if (!IsCreated) + throw new ArgumentException("During destruction "); + if (!m_AllowGetSystem) + throw new ArgumentException( + "During destruction of a system you are not allowed to get or create more systems."); +#endif + + for (int i = 0; i < m_Systems.Count; ++i) { + var mgr = m_Systems[i]; + if (type.IsAssignableFrom(mgr.GetType())) + return mgr; + } + + return null; + } + + private ComponentSystemBase GetOrCreateSystemInternal() where T : new() + { + var manager = GetExistingSystemInternal(); + return manager ?? CreateSystemInternal(); + } + + private void RemoveSystemInternal(ComponentSystemBase system) + { + if (!m_Systems.Remove(system)) + throw new ArgumentException($"manager does not exist in the world"); + ++Version; + } + + public T CreateSystem() where T : ComponentSystemBase, new() + { + return (T) CreateSystemInternal(); + } + + public T GetOrCreateSystem() where T : ComponentSystemBase, new() + { + return (T) GetOrCreateSystemInternal(); + } + + public T AddSystem(T system) where T : ComponentSystemBase + { + m_Systems.Add(system); + try + { + system.CreateInstance(this); + } + catch + { + RemoveSystemInternal(system); + throw; + } + + ++Version; + return system; + } + + public T GetExistingSystem() where T : ComponentSystemBase + { + return (T) GetExistingSystemInternal(typeof(T)); + } + + public ComponentSystemBase GetExistingSystem(Type type) + { + return GetExistingSystemInternal(type); + } + + public void DestroySystem(ComponentSystemBase system) + { + RemoveSystemInternal(system); + system.DestroyInstance(); + } + + static int ms_SystemIDAllocator = 0; + internal static int AllocateSystemID() + { + return ++ms_SystemIDAllocator; + } + + public bool QuitUpdate { get; set; } + + public void Update() + { + InitializationSystemGroup initializationSystemGroup = + GetExistingSystem(typeof(InitializationSystemGroup)) as InitializationSystemGroup; + SimulationSystemGroup simulationSystemGroup = + GetExistingSystem(typeof(SimulationSystemGroup)) as SimulationSystemGroup; + PresentationSystemGroup presentationSystemGroup = + GetExistingSystem(typeof(PresentationSystemGroup)) as PresentationSystemGroup; + + initializationSystemGroup?.Update(); + simulationSystemGroup?.Update(); + presentationSystemGroup?.Update(); + } + } +#endif +} diff --git a/Unity.Entities/Injection/World.cs.meta b/Unity.Entities/World.cs.meta similarity index 100% rename from Unity.Entities/Injection/World.cs.meta rename to Unity.Entities/World.cs.meta diff --git a/Unity.Entities/WorldDebuggingTools.cs b/Unity.Entities/WorldDebuggingTools.cs new file mode 100644 index 00000000..5e7c1000 --- /dev/null +++ b/Unity.Entities/WorldDebuggingTools.cs @@ -0,0 +1,53 @@ +#if UNITY_EDITOR +using System; +using System.Collections.Generic; +using System.Linq; +using Unity.Collections; + +namespace Unity.Entities +{ + internal class WorldDebuggingTools + { + internal static void MatchEntityInEntityQueries(World world, Entity entity, + List>> matchList) + { + using (var entityComponentTypes = + world.EntityManager.GetComponentTypes(entity, Allocator.Temp)) + { + foreach (var system in World.Active.Systems) + { + var queryList = new List(); + if (system == null) continue; + foreach (var query in system.EntityQueries) + if (Match(query, entityComponentTypes)) + queryList.Add(query); + + if (queryList.Count > 0) + matchList.Add( + new Tuple>(system, queryList)); + } + } + } + + private static bool Match(EntityQuery query, NativeArray entityComponentTypes) + { + foreach (var groupType in query.GetQueryTypes().Skip(1)) + { + var found = false; + foreach (var type in entityComponentTypes) + { + if (type.TypeIndex != groupType.TypeIndex) + continue; + found = true; + break; + } + + if (found == (groupType.AccessModeType == ComponentType.AccessMode.Exclude)) + return false; + } + + return true; + } + } +} +#endif diff --git a/Unity.Entities/Injection/WorldDebuggingTools.cs.meta b/Unity.Entities/WorldDebuggingTools.cs.meta similarity index 100% rename from Unity.Entities/Injection/WorldDebuggingTools.cs.meta rename to Unity.Entities/WorldDebuggingTools.cs.meta diff --git a/Unity.Entities/WorldDiff.cs b/Unity.Entities/WorldDiff.cs index d62ec2a5..4f963d64 100644 --- a/Unity.Entities/WorldDiff.cs +++ b/Unity.Entities/WorldDiff.cs @@ -11,7 +11,7 @@ using Unity.Profiling; namespace Unity.Entities -{ +{ public static class HashMapUtility { public static bool TryRemove(this NativeMultiHashMap h, K k, V v) where K : struct, IEquatable where V : struct, IEquatable @@ -130,38 +130,38 @@ public struct SetSharedComponentDiff } public static class DiffUtil - { - public static ComponentGroup CreateAllChunksGroup(EntityManager manager) + { + public static EntityQuery CreateAllChunksQuery(EntityManager manager) { var guidQuery = new[] { - new EntityArchetypeQuery + new EntityQueryDesc { All = new ComponentType[] {typeof(EntityGuid)}, }, - new EntityArchetypeQuery + new EntityQueryDesc { All = new ComponentType[] {typeof(EntityGuid), typeof(Disabled)}, }, - new EntityArchetypeQuery + new EntityQueryDesc { All = new ComponentType[] {typeof(EntityGuid), typeof(Prefab)}, }, - new EntityArchetypeQuery + new EntityQueryDesc { All = new ComponentType[] {typeof(EntityGuid), typeof(Prefab), typeof(Disabled)}, } }; - return manager.CreateComponentGroup(guidQuery); + return manager.CreateEntityQuery(guidQuery); } public static NativeArray GetAllChunks(EntityManager manager) { - var group = CreateAllChunksGroup(manager); + var query = CreateAllChunksQuery(manager); - var chunks = group.CreateArchetypeChunkArray(Allocator.TempJob); + var chunks = query.CreateArchetypeChunkArray(Allocator.TempJob); - group.Dispose(); + query.Dispose(); return chunks; } @@ -246,13 +246,13 @@ public static void DiffAndApply(World newState, World previousStateShadowWorld, using (var diff = WorldDiffer.UpdateDiff(newState, previousStateShadowWorld, Allocator.TempJob)) { m_Diff.End(); - + m_ApplyDiff.Begin(); ApplyDiff(dstEntityWorld, diff); m_ApplyDiff.End(); } } - + public static WorldDiff UpdateDiff(World Source, World ShadowWorld, Allocator resultAllocator) { if (Source == null) @@ -264,8 +264,8 @@ public static WorldDiff UpdateDiff(World Source, World ShadowWorld, Allocator re if (resultAllocator == Allocator.Temp) throw new ArgumentException("Allocator can not be Allocator.Temp. Use Allocator.TempJob instead."); - var smgr = Source.GetOrCreateManager(); - var dmgr = ShadowWorld.GetOrCreateManager(); + var smgr = Source.EntityManager; + var dmgr = ShadowWorld.EntityManager; var schunks = DiffUtil.GetAllChunks(smgr); var dchunks = DiffUtil.GetAllChunks(dmgr); @@ -553,7 +553,7 @@ static void BuildComponentDiff(NativeList modifiedEntities byte* beforeAddress = mod.BeforeChunk->Buffer + beforeArch->Offsets[beforeType] + beforeArch->SizeOfs[beforeType] * mod.BeforeIndex; byte* afterAddress = mod.AfterChunk->Buffer + afterArch->Offsets[afterType] + afterArch->SizeOfs[afterType] * mod.AfterIndex; - if (!SharedComponentDataManager.FastEquality_ComparePtr(beforeAddress, afterAddress, targetTypeIndex)) + if (!TypeManager.Equals(beforeAddress, afterAddress, targetTypeIndex)) { // For now do full component replacement, because we need to dig into field information to make this work. dataDiffs.Add(new DataDiff @@ -577,7 +577,7 @@ static void BuildComponentDiff(NativeList modifiedEntities object beforeObject = GetSharedComponentObject(dsharedComponentDataManager, mod.BeforeChunk, beforeType, targetTypeIndex); object afterObject = GetSharedComponentObject(ssharedComponentDataManager, mod.AfterChunk, afterType, targetTypeIndex); - if (!SharedComponentDataManager.FastEquality_CompareBoxed(beforeObject, afterObject, targetTypeIndex)) + if (!TypeManager.Equals(beforeObject, afterObject, targetTypeIndex)) { sharedComponentDiffs.Add(new SetSharedComponentDiff { @@ -694,7 +694,7 @@ private static void ExtractGuidPatches(NativeList entityPatches int elementOffset = 0; for (var element = 0; element < elementCount; ++element) { - #if !UNITY_CSHARP_TINY + #if !NET_DOTS foreach (var eo in ft.EntityOffsets) { #else @@ -723,7 +723,7 @@ private static void ExtractGuidPatches(NativeList entityPatches } -#if !UNITY_CSHARP_TINY +#if !NET_DOTS static object GetSharedComponentObject(SharedComponentDataManager sharedComponentDataManager, Chunk* chunk, int typeIndexInArchetype, int targetTypeIndex) { int off = typeIndexInArchetype - chunk->Archetype->FirstSharedComponent; @@ -1071,7 +1071,7 @@ public void Execute(ArchetypeChunk chunk, int entityIndex, int chunkIndex) var entities = chunk.GetNativeArray(Entity); for (int i = 0; i != entities.Length; i++) GuidToDestWorldPrefabEntity.TryAdd(guids[i], entities[i]); - } + } } struct BuildEntityToRootEntityLookup : IJobChunk @@ -1097,8 +1097,8 @@ public void Execute(ArchetypeChunk chunk, int entityIndex, int chunkIndex) internal struct DiffApplier : IDisposable { static ProfilerMarker s_AllocateLookups = new ProfilerMarker("DiffApplier.AllocateLookups"); - static ProfilerMarker s_BuildDestWorldLookups = new ProfilerMarker("DiffApplier.BuildDestWorldLookups"); - static ProfilerMarker s_BuildDiffToDestWorldLookups = new ProfilerMarker("DiffApplier.BuildDiffToDestWorldLookups"); + static ProfilerMarker s_BuildDestWorldLookups = new ProfilerMarker("DiffApplier.BuildDestWorldLookups"); + static ProfilerMarker s_BuildDiffToDestWorldLookups = new ProfilerMarker("DiffApplier.BuildDiffToDestWorldLookups"); static ProfilerMarker s_CreateEntities = new ProfilerMarker("DiffApplier.CreateEntities"); static ProfilerMarker s_DestroyEntities = new ProfilerMarker("DiffApplier.DestroyEntities"); static ProfilerMarker s_AddComponents = new ProfilerMarker("DiffApplier.AddComponents"); @@ -1108,7 +1108,7 @@ internal struct DiffApplier : IDisposable static ProfilerMarker s_ApplyLinkedEntityGroupAdds = new ProfilerMarker("DiffApplier.ApplyLinkedEntityGroupAdds"); static ProfilerMarker s_ApplyLinkedEntityGroupRemoves = new ProfilerMarker("DiffApplier.ApplyLinkedEntityGroupRemoves"); static ProfilerMarker s_PatchEntities = new ProfilerMarker("DiffApplier.PatchEntities"); - + // World to which we apply the diff internal World destWorld; @@ -1153,7 +1153,7 @@ public DiffApplier(World dest_, WorldDiff diff_) { destWorld = dest_; diff = diff_; - DestWorldManager = destWorld.GetOrCreateManager(); + DestWorldManager = destWorld.EntityManager; DestWorldManager.CompleteAllJobs(); DiffIndexToDestWorldEntities = default; DiffIndexToDestWorldTypes = default; @@ -1209,19 +1209,19 @@ void BuildDestWorldLookups() // we need to ask for guids, guids and disabled, guids and prefab, and guids/prefab/disabled. var guidQuery = new[] { - new EntityArchetypeQuery + new EntityQueryDesc { All = new ComponentType[] {typeof(EntityGuid)} }, - new EntityArchetypeQuery + new EntityQueryDesc { All = new ComponentType[] {typeof(EntityGuid), typeof(Disabled)} }, - new EntityArchetypeQuery + new EntityQueryDesc { All = new ComponentType[] {typeof(EntityGuid), typeof(Prefab)} }, - new EntityArchetypeQuery + new EntityQueryDesc { All = new ComponentType[] {typeof(EntityGuid), typeof(Prefab), typeof(Disabled)} } @@ -1230,19 +1230,19 @@ void BuildDestWorldLookups() // we need to ask for linkedentitygroups, linkedentitygroups and disabled, linkedentitygroups and prefab, and linkedentitygroups/prefab/disabled. var linkedEntityGroupQuery = new[] { - new EntityArchetypeQuery + new EntityQueryDesc { All = new ComponentType[] {typeof(LinkedEntityGroup)} }, - new EntityArchetypeQuery + new EntityQueryDesc { All = new ComponentType[] {typeof(LinkedEntityGroup), typeof(Disabled)} }, - new EntityArchetypeQuery + new EntityQueryDesc { All = new ComponentType[] {typeof(LinkedEntityGroup), typeof(Prefab)} }, - new EntityArchetypeQuery + new EntityQueryDesc { All = new ComponentType[] {typeof(LinkedEntityGroup), typeof(Prefab), typeof(Disabled)} } @@ -1250,18 +1250,18 @@ void BuildDestWorldLookups() // this is for finding entities that have GUIDs, and which are Prefabs. var guidPrefabQuery = new[] { - new EntityArchetypeQuery + new EntityQueryDesc { All = new ComponentType[] {typeof(EntityGuid), typeof(Prefab)} }, - new EntityArchetypeQuery + new EntityQueryDesc { All = new ComponentType[] {typeof(EntityGuid), typeof(Prefab), typeof(Disabled)} } }; - using(ComponentGroup DestChunksWithGuids = DestWorldManager.CreateComponentGroup(guidQuery)) - using (ComponentGroup DestChunksWithLinkedEntityGroups = DestWorldManager.CreateComponentGroup(linkedEntityGroupQuery)) - using(ComponentGroup DestchunksWithGuidAndPrefab = DestWorldManager.CreateComponentGroup(guidPrefabQuery)) + using (EntityQuery DestChunksWithGuids = DestWorldManager.CreateEntityQuery(guidQuery)) + using (EntityQuery DestChunksWithLinkedEntityGroups = DestWorldManager.CreateEntityQuery(linkedEntityGroupQuery)) + using (EntityQuery DestchunksWithGuidAndPrefab = DestWorldManager.CreateEntityQuery(guidPrefabQuery)) { DestWorldEntitiesWithGuids = DestChunksWithGuids.CalculateLength(); DestWorldEntitiesWithLinkedEntityGroups = DestChunksWithLinkedEntityGroups.CalculateLength(); @@ -1293,7 +1293,7 @@ void BuildDestWorldLookups() { GuidToDestWorldPrefabEntity = GuidToDestWorldPrefabEntity.ToConcurrent(), Entity = DestWorldManager.GetArchetypeChunkEntityType(), - EntityGUID = DestWorldManager.GetArchetypeChunkComponentType(true) + EntityGUID = DestWorldManager.GetArchetypeChunkComponentType(true) }; handle = buildDestWorldGuidPrefabLookups.Schedule(DestchunksWithGuidAndPrefab); handle.Complete(); @@ -1362,14 +1362,14 @@ void SetDebugNames() { do { -#if UNITY_EDITOR +#if UNITY_EDITOR DestWorldManager.SetName(entity, diff.EntityNames[i].ToString()); #endif } while (DiffIndexToDestWorldEntities.TryGetNextValue(out entity, ref it)); } } } - + void DestroyEntities() { s_DestroyEntities.Begin(); @@ -1399,7 +1399,7 @@ void DestroyEntities() void AddComponents() { s_AddComponents.Begin(); - var linkedEntityGroupTypeIndex = TypeManager.GetTypeIndex(); + var linkedEntityGroupTypeIndex = TypeManager.GetTypeIndex(); foreach (var addition in diff.AddComponents) { @@ -1414,7 +1414,7 @@ void AddComponents() // magic is required to force the first entity in the LinkedEntityGroup to be the entity // that owns the component. this magic doesn't seem to exist at a lower level, so let's // shim it in here. we'll probably need to move the magic lower someday. - if (componentType.TypeIndex == linkedEntityGroupTypeIndex) + if (componentType.TypeIndex == linkedEntityGroupTypeIndex) { var buffer = DestWorldManager.GetBuffer(entity); buffer.Add(entity); @@ -1680,12 +1680,12 @@ void ApplyLinkedEntityGroupAdds() } else { - Debug.LogWarning($"Tried to add a child to a linked entity group, but root entity didn't exist in destination world."); + Debug.LogWarning($"Tried to add a child to a linked entity group, but root entity didn't exist in destination world."); } } else { - Debug.LogWarning($"Tried to add a child to a linked entity group, but no such prefab exists in destination world."); + Debug.LogWarning($"Tried to add a child to a linked entity group, but no such prefab exists in destination world."); } } for (var a = 0; a < instanceChildren.Length; ++a) @@ -1731,7 +1731,7 @@ void ApplyLinkedEntityGroupRemoves() for (var a = 0; a < pending.Length; ++a) RemoveChildFromTables(pending[a]); } - s_ApplyLinkedEntityGroupRemoves.End(); + s_ApplyLinkedEntityGroupRemoves.End(); } } diff --git a/Unity.Scenes.Editor/ConversionWarningsEditor.cs b/Unity.Scenes.Editor/ConversionWarningsEditor.cs index 184212b3..05adff75 100644 --- a/Unity.Scenes.Editor/ConversionWarningsEditor.cs +++ b/Unity.Scenes.Editor/ConversionWarningsEditor.cs @@ -49,7 +49,7 @@ public static string GetWarnings(GameObject gameobject) Type convertType = null; foreach (var behaviour in gameobject.GetComponents()) { - if (behaviour.GetType().GetCustomAttribute(true) != null) + if (behaviour != null && behaviour.GetType().GetCustomAttribute(true) != null) { convertType = behaviour.GetType(); break; diff --git a/Unity.Scenes.Editor/EditorEntityScenes.cs b/Unity.Scenes.Editor/EditorEntityScenes.cs index fee1efc0..08896f75 100644 --- a/Unity.Scenes.Editor/EditorEntityScenes.cs +++ b/Unity.Scenes.Editor/EditorEntityScenes.cs @@ -8,6 +8,7 @@ using Unity.Profiling; using UnityEditor; using UnityEngine; +using UnityEngine.Assertions; using UnityEngine.SceneManagement; using static Unity.Entities.GameObjectConversionUtility; using Hash128 = Unity.Entities.Hash128; @@ -44,7 +45,7 @@ public static bool HasEntitySceneCache(Hash128 sceneGUID) public static SceneData[] WriteEntityScene(Scene scene, Hash128 sceneGUID, ConversionFlags conversionFlags) { var world = new World("ConversionWorld"); - var entityManager = world.GetOrCreateManager(); + var entityManager = world.EntityManager; var boundsEntity = entityManager.CreateEntity(typeof(SceneBoundingVolume)); entityManager.SetComponentData(boundsEntity, new SceneBoundingVolume { Value = MinMaxAABB.Empty } ); @@ -63,11 +64,11 @@ public static SceneData[] WriteEntityScene(Scene scene, Hash128 sceneGUID, Conve NativeArray entitiesInMainSection; - var sectionGrp = entityManager.CreateComponentGroup( - new EntityArchetypeQuery + var sectionGrp = entityManager.CreateEntityQuery( + new EntityQueryDesc { All = new[] {ComponentType.ReadWrite()}, - Options = EntityArchetypeQueryOptions.IncludePrefab | EntityArchetypeQueryOptions.IncludeDisabled + Options = EntityQueryOptions.IncludePrefab | EntityQueryOptions.IncludeDisabled } ); @@ -82,6 +83,10 @@ public static SceneData[] WriteEntityScene(Scene scene, Hash128 sceneGUID, Conve // otherwise they would reuse entities that have been moved and mess up the remapping tables. for(int sectionIndex = 1; sectionIndex < subSectionList.Count; ++sectionIndex) { + if (subSectionList[sectionIndex].Section == 0) + // Main section, the only one that doesn't need an external ref array + continue; + var extRefInfoEntity = entityManager.CreateEntity(); entityManager.AddSharedComponentData(extRefInfoEntity, subSectionList[sectionIndex]); extRefInfoEntities[sectionIndex] = extRefInfoEntity; @@ -106,7 +111,7 @@ public static SceneData[] WriteEntityScene(Scene scene, Hash128 sceneGUID, Conve // Save main section var sectionWorld = new World("SectionWorld"); - var sectionManager = sectionWorld.GetOrCreateManager(); + var sectionManager = sectionWorld.EntityManager; var entityRemapping = entityManager.CreateEntityRemapArray(Allocator.TempJob); sectionManager.MoveEntitiesFrom(entityManager, sectionGrp, entityRemapping); @@ -114,8 +119,8 @@ public static SceneData[] WriteEntityScene(Scene scene, Hash128 sceneGUID, Conve // The section component is only there to break the conversion world into different sections // We don't want to store that on the disk //@TODO: Component should be removed but currently leads to corrupt data file. Figure out why. - //sectionManager.RemoveComponent(sectionManager.UniversalGroup, typeof(SceneSection)); - + //sectionManager.RemoveComponent(sectionManager.UniversalQuery, typeof(SceneSection)); + var sectionFileSize = WriteEntityScene(sectionManager, sceneGUID, "0"); sceneSections.Add(new SceneData { @@ -137,7 +142,7 @@ public static SceneData[] WriteEntityScene(Scene scene, Hash128 sceneGUID, Conve var subSection = subSectionList[subSectionIndex]; if (subSection.Section == 0) continue; - + sectionGrp.SetFilter(subSection); var entitiesInSection = sectionGrp.ToEntityArray(Allocator.TempJob); @@ -155,10 +160,13 @@ public static SceneData[] WriteEntityScene(Scene scene, Hash128 sceneGUID, Conve ExternalEntityRef.Add(ref externRefs, new ExternalEntityRef{entityIndex = i}); } - // Entities will be remapped to a contiguous range in the section world, - // so any range after that is fine for the external references - //@TODO why are we not mapping anything to entity 0? we use the range [1;count], hence +1 - var externEntityIndexStart = entitiesInSection.Length + 1; + var entityRemapping = entityManager.CreateEntityRemapArray(Allocator.TempJob); + + // Entities will be remapped to a contiguous range in the section world, but they will + // also come with an unpredictable amount of meta entities. We have the guarantee that + // the entities in the main section won't be moved over, so there's a free range of that + // size at the end of the remapping table. So we use that range for external references. + var externEntityIndexStart = entityRemapping.Length - entitiesInMainSection.Length; entityManager.AddComponentData(refInfoEntity, new ExternalEntityRefInfo @@ -168,9 +176,7 @@ public static SceneData[] WriteEntityScene(Scene scene, Hash128 sceneGUID, Conve }); var sectionWorld = new World("SectionWorld"); - var sectionManager = sectionWorld.GetOrCreateManager(); - - var entityRemapping = entityManager.CreateEntityRemapArray(Allocator.TempJob); + var sectionManager = sectionWorld.EntityManager; // Insert mapping for external references, conversion world entity to virtual index in section for (int i = 0; i < entitiesInMainSection.Length; ++i) @@ -180,19 +186,47 @@ public static SceneData[] WriteEntityScene(Scene scene, Hash128 sceneGUID, Conve } sectionManager.MoveEntitiesFrom(entityManager, sectionGrp, entityRemapping); - + + // Now that all the required entities have been moved over, we can get rid of the gap between + // real entities and external references. This allows remapping during load to deal with a + // smaller remap table, containing only useful entries. + + int highestEntityIndexInUse = 0; + for (int i = 0; i < externEntityIndexStart; ++i) + { + var targetIndex = entityRemapping[i].Target.Index; + if (targetIndex < externEntityIndexStart && targetIndex > highestEntityIndexInUse) + highestEntityIndexInUse = targetIndex; + } + + var oldExternEntityIndexStart = externEntityIndexStart; + externEntityIndexStart = highestEntityIndexInUse + 1; + + sectionManager.SetComponentData + ( + EntityRemapUtility.RemapEntity(ref entityRemapping, refInfoEntity), + new ExternalEntityRefInfo + { + SceneGUID = sceneGUID, + EntityIndexStart = externEntityIndexStart + } + ); + // When writing the scene, references to missing entities are set to Entity.Null by default + // (but only if they have been used, otherwise they remain untouched) // We obviously don't want that to happen to our external references, so we add explicit mapping + // And at the same time, we put them back at the end of the effective range of real entities. for (int i = 0; i < entitiesInMainSection.Length; ++i) { - var entity = new Entity {Index = i + externEntityIndexStart, Version = 1}; - EntityRemapUtility.AddEntityRemapping(ref entityRemapping, entity, entity); + var src = new Entity {Index = i + oldExternEntityIndexStart, Version = 1}; + var dst = new Entity {Index = i + externEntityIndexStart, Version = 1}; + EntityRemapUtility.AddEntityRemapping(ref entityRemapping, src, dst); } // The section component is only there to break the conversion world into different sections // We don't want to store that on the disk //@TODO: Component should be removed but currently leads to corrupt data file. Figure out why. - //sectionManager.RemoveComponent(sectionManager.UniversalGroup, typeof(SceneSection)); + //sectionManager.RemoveComponent(sectionManager.UniversalQuery, typeof(SceneSection)); var fileSize = WriteEntityScene(sectionManager, sceneGUID, subSection.Section.ToString(), entityRemapping); sceneSections.Add(new SceneData @@ -213,11 +247,11 @@ public static SceneData[] WriteEntityScene(Scene scene, Hash128 sceneGUID, Conve } { - var noSectionGrp = entityManager.CreateComponentGroup( - new EntityArchetypeQuery + var noSectionGrp = entityManager.CreateEntityQuery( + new EntityQueryDesc { None = new[] {ComponentType.ReadWrite()}, - Options = EntityArchetypeQueryOptions.IncludePrefab | EntityArchetypeQueryOptions.IncludeDisabled + Options = EntityQueryOptions.IncludePrefab | EntityQueryOptions.IncludeDisabled } ); if (noSectionGrp.CalculateLength() != 0) diff --git a/Unity.Scenes.Editor/SubSceneInspector.cs b/Unity.Scenes.Editor/SubSceneInspector.cs index 0d0d37f5..b8c158ac 100644 --- a/Unity.Scenes.Editor/SubSceneInspector.cs +++ b/Unity.Scenes.Editor/SubSceneInspector.cs @@ -88,7 +88,7 @@ public override void OnInspectorGUI() if (World.Active != null) { - var entityManager = World.Active.GetOrCreateManager(); + var entityManager = World.Active.EntityManager; foreach (var scene in scenes) { diff --git a/Unity.Scenes.Editor/SubSceneLiveLinkSystem.cs b/Unity.Scenes.Editor/SubSceneLiveLinkSystem.cs index 0710e81c..a28d5328 100644 --- a/Unity.Scenes.Editor/SubSceneLiveLinkSystem.cs +++ b/Unity.Scenes.Editor/SubSceneLiveLinkSystem.cs @@ -27,14 +27,14 @@ static void OnPostprocessAllAssets(string[] importedAssets, string[] deletedAsse } } } - + static int GlobalDirtyID = 0; static int PreviousGlobalDirtyID = 0; MethodInfo m_GetDirtyIDMethod = typeof(Scene).GetProperty("dirtyID", BindingFlags.Instance | BindingFlags.NonPublic | BindingFlags.Public).GetMethod; - + UInt64 m_LiveLinkEditSceneViewMask = 1UL << 60; UInt64 m_LiveLinkEditGameViewMask = 1UL << 58; - HashSet m_EditingSceneAssets = new HashSet(); + HashSet m_EditingSceneAssets = new HashSet(); static void AddUnique(ref List list, SubScene scene) { @@ -51,9 +51,9 @@ protected override void OnUpdate() List markSceneLoadedFromLiveLink = null; List removeSceneLoadedFromLiveLink = null; m_EditingSceneAssets.Clear(); - + var liveLinkEnabled = SubSceneInspectorUtility.LiveLinkEnabled; - + // By default all scenes need to have m_GameObjectSceneCullingMask, otherwise they won't show up in game view for (int i = 0; i != EditorSceneManager.sceneCount; i++) { @@ -67,10 +67,10 @@ protected override void OnUpdate() var sceneAsset = AssetDatabase.LoadAssetAtPath(scene.path); if (scene.isLoaded && sceneAsset != null) - m_EditingSceneAssets.Add(sceneAsset); + m_EditingSceneAssets.Add(sceneAsset); } - } - + } + if (PreviousGlobalDirtyID != GlobalDirtyID) { Entities.ForEach((SubScene subScene) => @@ -79,7 +79,7 @@ protected override void OnUpdate() }); PreviousGlobalDirtyID = GlobalDirtyID; } - + Entities.ForEach((SubScene subScene) => { var isLoaded = m_EditingSceneAssets.Contains(subScene.SceneAsset); @@ -171,9 +171,9 @@ void CleanupScene(SubScene scene) { // Debug.Log("CleanupScene: " + scene.SceneName); scene.CleanupLiveLink(); - - var streamingSystem = World.GetExistingManager(); - + + var streamingSystem = World.GetExistingSystem(); + foreach (var sceneEntity in scene._SceneEntities) { streamingSystem.UnloadSceneImmediate(sceneEntity); @@ -187,17 +187,17 @@ void CleanupScene(SubScene scene) void ApplyLiveLink(SubScene scene) { //Debug.Log("ApplyLiveLink: " + scene.SceneName); - - var streamingSystem = World.GetExistingManager(); + + var streamingSystem = World.GetExistingSystem(); var isFirstTime = scene.LiveLinkShadowWorld == null; if (scene.LiveLinkShadowWorld == null) scene.LiveLinkShadowWorld = new World("LiveLink"); - - + + using (var cleanConvertedEntityWorld = new World("Clean Entity Conversion World")) { - // Unload scene + // Unload scene //@TODO: We optimally shouldn't be unloading the scene here. We should simply prime the shadow world with the scene that we originally loaded into the player (Including Entity GUIDs) // This way we can continue the live link, compared to exactly what we loaded into the player. if (isFirstTime) @@ -207,46 +207,46 @@ void ApplyLiveLink(SubScene scene) streamingSystem.UnloadSceneImmediate(s); EntityManager.DestroyEntity(s); } - + var sceneEntity = EntityManager.CreateEntity(); EntityManager.SetName(sceneEntity, "Scene (LiveLink): " + scene.SceneName); EntityManager.AddComponentObject(sceneEntity, scene); EntityManager.AddComponentData(sceneEntity, new SubSceneStreamingSystem.StreamingState { Status = SubSceneStreamingSystem.StreamingStatus.Loaded}); EntityManager.AddComponentData(sceneEntity, new SubSceneStreamingSystem.IgnoreTag( )); - + scene._SceneEntities = new List(); scene._SceneEntities.Add(sceneEntity); } - + // Convert scene GameObjectConversionUtility.ConvertScene(scene.LoadedScene, scene.SceneGUID, cleanConvertedEntityWorld, GameObjectConversionUtility.ConversionFlags.AddEntityGUID | GameObjectConversionUtility.ConversionFlags.AssignName); - var convertedEntityManager = cleanConvertedEntityWorld.GetOrCreateManager(); + var convertedEntityManager = cleanConvertedEntityWorld.EntityManager; var liveLinkSceneEntity = scene._SceneEntities[0]; - - /// We want to let the live linked scene be able to reference the already existing Scene Entity (Specifically SceneTag should point to the scene Entity after live link completes) + + /// We want to let the live linked scene be able to reference the already existing Scene Entity (Specifically SceneTag should point to the scene Entity after live link completes) // Add Scene tag to all entities using the convertedSceneEntity that will map to the already existing scene entity. - convertedEntityManager.AddSharedComponentData(convertedEntityManager.UniversalGroup, new SceneTag { SceneEntity = liveLinkSceneEntity }); + convertedEntityManager.AddSharedComponentData(convertedEntityManager.UniversalQuery, new SceneTag { SceneEntity = liveLinkSceneEntity }); WorldDiffer.DiffAndApply(cleanConvertedEntityWorld, scene.LiveLinkShadowWorld, World); convertedEntityManager.Debug.CheckInternalConsistency(); - scene.LiveLinkShadowWorld.GetOrCreateManager().Debug.CheckInternalConsistency(); + scene.LiveLinkShadowWorld.EntityManager.Debug.CheckInternalConsistency(); - var group = EntityManager.CreateComponentGroup(typeof(SceneTag), ComponentType.Exclude()); + var group = EntityManager.CreateEntityQuery(typeof(SceneTag), ComponentType.Exclude()); group.SetFilter(new SceneTag {SceneEntity = liveLinkSceneEntity}); - + EntityManager.AddSharedComponentData(group, new EditorRenderData() { SceneCullingMask = m_LiveLinkEditGameViewMask, PickableObject = scene.gameObject }); group.Dispose(); - + scene.LiveLinkDirtyID = GetSceneDirtyID(scene.LoadedScene); EditorUpdateUtility.EditModeQueuePlayerLoopUpdate(); } } - - + + static GameObject GetGameObjectFromAny(Object target) { Component component = target as Component; @@ -254,13 +254,13 @@ static GameObject GetGameObjectFromAny(Object target) return component.gameObject; return target as GameObject; } - + internal static bool IsHotControlActive() { return GUIUtility.hotControl != 0; } - - + + UndoPropertyModification[] PostprocessModifications(UndoPropertyModification[] modifications) { foreach (var mod in modifications) @@ -273,15 +273,15 @@ UndoPropertyModification[] PostprocessModifications(UndoPropertyModification[] m { if (scene.IsLoaded && scene.LoadedScene == targetScene) { - scene.LiveLinkDirtyID = -1; + scene.LiveLinkDirtyID = -1; } }); } } - + return modifications; } - + int GetSceneDirtyID(Scene scene) { if (scene.IsValid()) @@ -291,17 +291,17 @@ int GetSceneDirtyID(Scene scene) else return -1; } - + static void GlobalDirtyLiveLink() { GlobalDirtyID++; } - - protected override void OnCreateManager() - { + + protected override void OnCreate() + { Undo.postprocessModifications += PostprocessModifications; Undo.undoRedoPerformed += GlobalDirtyLiveLink; - + Camera.onPreCull += OnPreCull; RenderPipeline.beginCameraRendering += OnPreCull; } @@ -326,14 +326,14 @@ void OnPreCull(Camera camera) camera.overrideSceneCullingMask = EditorSceneManager.DefaultSceneCullingMask | m_LiveLinkEditSceneViewMask; } } - + } - protected override void OnDestroyManager() - { + protected override void OnDestroy() + { Undo.postprocessModifications -= PostprocessModifications; Undo.undoRedoPerformed -= GlobalDirtyLiveLink; - + Camera.onPreCull -= OnPreCull; RenderPipeline.beginCameraRendering -= OnPreCull; } diff --git a/Unity.Scenes.Hybrid.Tests/SaveAndLoadEndToEnd.cs b/Unity.Scenes.Hybrid.Tests/SaveAndLoadEndToEnd.cs index d3c9f0a8..e4d42630 100644 --- a/Unity.Scenes.Hybrid.Tests/SaveAndLoadEndToEnd.cs +++ b/Unity.Scenes.Hybrid.Tests/SaveAndLoadEndToEnd.cs @@ -38,17 +38,20 @@ public IEnumerator EndToEnd() for (int w = 0; w != 1000; w++) { - World.GetOrCreateManager().Update(); + World.GetOrCreateSystem().Update(); if (1 != m_Manager.Debug.EntityCount) break; yield return null; } - Assert.AreEqual(4, m_Manager.Debug.EntityCount); + // 1. Scene entity + // 2. Public ref array + // 3. Mesh Renderer + Assert.AreEqual(3, m_Manager.Debug.EntityCount); m_Manager.RemoveComponent(sceneEntity); - World.GetOrCreateManager().Update(); + World.GetOrCreateSystem().Update(); Assert.AreEqual(1, m_Manager.Debug.EntityCount); } @@ -56,4 +59,3 @@ public IEnumerator EndToEnd() } } } - diff --git a/Unity.Scenes.Hybrid/EntitySceneOptimization.cs b/Unity.Scenes.Hybrid/EntitySceneOptimization.cs index 95fc6980..38c1bdca 100644 --- a/Unity.Scenes.Hybrid/EntitySceneOptimization.cs +++ b/Unity.Scenes.Hybrid/EntitySceneOptimization.cs @@ -14,7 +14,7 @@ public static class EntitySceneOptimization { static void MarkStaticFrozen(EntityManager entityManager) { - var staticGroup = entityManager.CreateComponentGroup(typeof(Static)); + var staticGroup = entityManager.CreateEntityQuery(typeof(Static)); entityManager.AddComponent(staticGroup, ComponentType.ReadWrite()); staticGroup.Dispose(); } @@ -26,7 +26,7 @@ static void RemoveSystemState(EntityManager entityManager) if (TypeManager.IsSystemStateComponent(s.TypeIndex)) { //@TODO: Make query instead of this crazy slow shit - entityManager.RemoveComponent(entityManager.UniversalGroup, ComponentType.FromTypeIndex(s.TypeIndex)); + entityManager.RemoveComponent(entityManager.UniversalQuery, ComponentType.FromTypeIndex(s.TypeIndex)); } } } @@ -35,7 +35,7 @@ public static void Optimize(World world) { var entityManager = world.EntityManager; - var group = world.GetOrCreateManager(); + var group = world.GetOrCreateSystem(); var systemTypes = DefaultWorldInitialization.GetAllSystems(WorldSystemFilterFlags.EntitySceneOptimizations); foreach (var systemType in systemTypes) @@ -50,7 +50,7 @@ public static void Optimize(World world) // Freeze static objects. // After this no more reordering is allowed - var staticGroup = entityManager.CreateComponentGroup(typeof(Static)); + var staticGroup = entityManager.CreateEntityQuery(typeof(Static)); entityManager.LockChunkOrder(staticGroup); staticGroup.Dispose(); @@ -66,7 +66,7 @@ static void AddSystemAndLogException(World world, ComponentSystemGroup group, Ty { try { - group.AddSystemToUpdateList(world.GetOrCreateManager(type) as ComponentSystemBase); + group.AddSystemToUpdateList(world.GetOrCreateSystem(type) as ComponentSystemBase); } catch (Exception e) { @@ -74,4 +74,4 @@ static void AddSystemAndLogException(World world, ComponentSystemGroup group, Ty } } } -} \ No newline at end of file +} diff --git a/Unity.Scenes.Hybrid/SubScene.cs b/Unity.Scenes.Hybrid/SubScene.cs index 8bc2a2b5..e26431dd 100644 --- a/Unity.Scenes.Hybrid/SubScene.cs +++ b/Unity.Scenes.Hybrid/SubScene.cs @@ -168,7 +168,7 @@ public void UpdateSceneEntities() DefaultWorldInitialization.DefaultLazyEditModeInitialize(); if (World.Active != null && m_SubSceneHeader != null) { - _SceneEntityManager = World.Active.GetOrCreateManager(); + _SceneEntityManager = World.Active.EntityManager; for (int i = 0; i < m_SubSceneHeader.Sections.Length; ++i) { diff --git a/Unity.Scenes.Hybrid/SubSceneStreamingSystem.cs b/Unity.Scenes.Hybrid/SubSceneStreamingSystem.cs index 06013af8..a3a4d60f 100644 --- a/Unity.Scenes.Hybrid/SubSceneStreamingSystem.cs +++ b/Unity.Scenes.Hybrid/SubSceneStreamingSystem.cs @@ -45,54 +45,54 @@ struct Stream int MaximumMoveEntitiesFromPerFrame = 1; Stream[] m_Streams = new Stream[LoadScenesPerFrame]; - ComponentGroup m_PendingStreamRequests; - ComponentGroup m_UnloadStreamRequests; - ComponentGroup m_SceneFilter; - ComponentGroup m_PublicRefFilter; - ComponentGroup m_SectionData; + EntityQuery m_PendingStreamRequests; + EntityQuery m_UnloadStreamRequests; + EntityQuery m_SceneFilter; + EntityQuery m_PublicRefFilter; + EntityQuery m_SectionData; ProfilerMarker m_MoveEntitiesFrom = new ProfilerMarker("SceneStreaming.MoveEntitiesFrom"); ProfilerMarker m_ExtractEntityRemapRefs = new ProfilerMarker("SceneStreaming.ExtractEntityRemapRefs"); ProfilerMarker m_AddSceneSharedComponents = new ProfilerMarker("SceneStreaming.AddSceneSharedComponents"); - protected override void OnCreateManager() + protected override void OnCreate() { for (int i = 0; i < LoadScenesPerFrame; ++i) CreateStreamWorld(i); - m_PendingStreamRequests = GetComponentGroup(new EntityArchetypeQuery() + m_PendingStreamRequests = GetEntityQuery(new EntityQueryDesc() { All = new[] {ComponentType.ReadWrite(), ComponentType.ReadWrite()}, None = new[] {ComponentType.ReadWrite(), ComponentType.ReadWrite() } }); - m_UnloadStreamRequests = GetComponentGroup(new EntityArchetypeQuery() + m_UnloadStreamRequests = GetEntityQuery(new EntityQueryDesc() { All = new[] {ComponentType.ReadWrite()}, None = new[] {ComponentType.ReadWrite(), ComponentType.ReadWrite()} }); - m_PublicRefFilter = GetComponentGroup + m_PublicRefFilter = GetEntityQuery ( ComponentType.ReadWrite(), ComponentType.ReadWrite() ); - m_SectionData = GetComponentGroup + m_SectionData = GetEntityQuery ( ComponentType.ReadWrite() ); - m_SceneFilter = GetComponentGroup( - new EntityArchetypeQuery + m_SceneFilter = GetEntityQuery( + new EntityQueryDesc { All = new [] {ComponentType.ReadWrite() }, - Options = EntityArchetypeQueryOptions.IncludeDisabled | EntityArchetypeQueryOptions.IncludePrefab + Options = EntityQueryOptions.IncludeDisabled | EntityQueryOptions.IncludePrefab } ); } - protected override void OnDestroyManager() + protected override void OnDestroy() { for (int i = 0; i != m_Streams.Length; i++) { @@ -100,7 +100,7 @@ protected override void OnDestroyManager() DestroyStreamWorld(i); } } - + void DestroyStreamWorld(int index) { m_Streams[index].World.Dispose(); @@ -111,12 +111,11 @@ void DestroyStreamWorld(int index) void CreateStreamWorld(int index) { m_Streams[index].World = new World("LoadingWorld" + index); - m_Streams[index].World.CreateManager(); } static NativeArray GetExternalRefEntities(EntityManager manager, Allocator allocator) { - using (var group = manager.CreateComponentGroup(typeof(ExternalEntityRefInfo))) + using (var group = manager.CreateEntityQuery(typeof(ExternalEntityRefInfo))) { return group.ToEntityArray(allocator); } @@ -125,7 +124,7 @@ static NativeArray GetExternalRefEntities(EntityManager manager, Allocat //@TODO There must be a more efficient way static Entity GetPublicRefEntity(EntityManager manager, Entities.Hash128 guid) { - using (var sceneDataGrp = manager.CreateComponentGroup(typeof(SceneData))) + using (var sceneDataGrp = manager.CreateEntityQuery(typeof(SceneData))) { using (var sceneEntities = sceneDataGrp.ToEntityArray(Allocator.TempJob)) { @@ -134,7 +133,7 @@ static Entity GetPublicRefEntity(EntityManager manager, Entities.Hash128 guid) var scenedata = manager.GetComponentData(sceneEntity); if (scenedata.SceneGUID == guid && scenedata.SubSectionIndex == 0) { - using(var group = manager.CreateComponentGroup(typeof(SceneTag), typeof(PublicEntityRef))) + using(var group = manager.CreateEntityQuery(typeof(SceneTag), typeof(PublicEntityRef))) { group.SetFilter(new SceneTag {SceneEntity = sceneEntity}); using (var entities = group.ToEntityArray(Allocator.TempJob)) @@ -196,10 +195,10 @@ bool MoveEntities(EntityManager srcManager, Entity sceneEntity) SceneCullingMask = UnityEditor.SceneManagement.EditorSceneManager.DefaultSceneCullingMask | (1UL << 59), PickableObject = EntityManager.HasComponent(sceneEntity) ? EntityManager.GetComponentObject(sceneEntity).gameObject : null }; - srcManager.AddSharedComponentData(srcManager.UniversalGroup, data); + srcManager.AddSharedComponentData(srcManager.UniversalQuery, data); #endif - srcManager.AddSharedComponentData(srcManager.UniversalGroup, new SceneTag { SceneEntity = sceneEntity}); + srcManager.AddSharedComponentData(srcManager.UniversalQuery, new SceneTag { SceneEntity = sceneEntity}); } @@ -289,8 +288,8 @@ bool ProcessActiveStreams() { var operation = m_Streams[i].Operation; var streamingWorld = m_Streams[i].World; - var streamingManager = streamingWorld.GetOrCreateManager(); - + var streamingManager = streamingWorld.EntityManager; + if (operation != null) { operation.Update(); @@ -328,7 +327,7 @@ bool ProcessActiveStreams() EntityManager.RemoveComponent(m_Streams[i].SceneEntity); - streamingManager.DestroyEntity(streamingManager.UniversalGroup); + streamingManager.DestroyEntity(streamingManager.UniversalQuery); streamingManager.PrepareForDeserialize(); moveEntitiesFromProcessed++; } @@ -408,17 +407,17 @@ protected override void OnUpdate() public void UnloadSceneImmediate(Entity scene) { if (EntityManager.HasComponent(scene)) - { + { m_SceneFilter.SetFilter(new SceneTag {SceneEntity = scene }); - + EntityManager.DestroyEntity(m_SceneFilter); - + m_SceneFilter.ResetFilter(); - + EntityManager.RemoveComponent(scene); } } - + int CreateAsyncLoadScene(Entity entity) { for (int i = 0; i != m_Streams.Length; i++) @@ -427,10 +426,10 @@ int CreateAsyncLoadScene(Entity entity) continue; var sceneData = EntityManager.GetComponentData(entity); - + var entitiesBinaryPath = EntityScenesPaths.GetLoadPath(sceneData.SceneGUID, EntityScenesPaths.PathType.EntitiesBinary, sceneData.SubSectionIndex); var resourcesPath = EntityScenesPaths.GetLoadPath(sceneData.SceneGUID, EntityScenesPaths.PathType.EntitiesSharedComponents, sceneData.SubSectionIndex); - var entityManager = m_Streams[i].World.GetOrCreateManager(); + var entityManager = m_Streams[i].World.EntityManager; m_Streams[i].Operation = new AsyncLoadSceneOperation(entitiesBinaryPath, sceneData.FileSize, sceneData.SharedComponentCount, resourcesPath, entityManager); m_Streams[i].SceneEntity = entity; diff --git a/Unity.Transforms.Hybrid/CopyInitialTransformFromGameObjectSystem.cs b/Unity.Transforms.Hybrid/CopyInitialTransformFromGameObjectSystem.cs index b30e532f..bfae21a9 100644 --- a/Unity.Transforms.Hybrid/CopyInitialTransformFromGameObjectSystem.cs +++ b/Unity.Transforms.Hybrid/CopyInitialTransformFromGameObjectSystem.cs @@ -33,7 +33,7 @@ public void Execute(int index, TransformAccess transform) } [BurstCompile] - struct CopyTransforms : IJobProcessComponentDataWithEntity + struct CopyTransforms : IJobForEachWithEntity { [DeallocateOnJobCompletion] public NativeArray transformStashes; @@ -69,12 +69,12 @@ public void Execute() EndInitializationEntityCommandBufferSystem m_EntityCommandBufferSystem; - ComponentGroup m_InitialTransformGroup; + EntityQuery m_InitialTransformGroup; - protected override void OnCreateManager() + protected override void OnCreate() { - m_EntityCommandBufferSystem = World.GetOrCreateManager(); - m_InitialTransformGroup = GetComponentGroup( + m_EntityCommandBufferSystem = World.GetOrCreateSystem(); + m_InitialTransformGroup = GetEntityQuery( ComponentType.ReadOnly(typeof(CopyInitialTransformFromGameObject)), typeof(UnityEngine.Transform), ComponentType.ReadWrite()); @@ -98,7 +98,7 @@ protected override JobHandle OnUpdate(JobHandle inputDeps) transformStashes = transformStashes, }; - var copyTransformsJobHandle = copyTransformsJob.ScheduleGroup(m_InitialTransformGroup, stashTransformsJobHandle); + var copyTransformsJobHandle = copyTransformsJob.Schedule(m_InitialTransformGroup, stashTransformsJobHandle); var removeComponentsJob = new RemoveCopyInitialTransformFromGameObjectComponent { diff --git a/Unity.Transforms.Hybrid/CopyTransformFromGameObjectSystem.cs b/Unity.Transforms.Hybrid/CopyTransformFromGameObjectSystem.cs index 5ad748b4..30c9326a 100644 --- a/Unity.Transforms.Hybrid/CopyTransformFromGameObjectSystem.cs +++ b/Unity.Transforms.Hybrid/CopyTransformFromGameObjectSystem.cs @@ -34,7 +34,7 @@ public void Execute(int index, TransformAccess transform) } [BurstCompile] - struct CopyTransforms : IJobProcessComponentDataWithEntity + struct CopyTransforms : IJobForEachWithEntity { [DeallocateOnJobCompletion] public NativeArray transformStashes; @@ -52,11 +52,11 @@ public void Execute(Entity entity, int index, ref LocalToWorld localToWorld) } } - ComponentGroup m_TransformGroup; + EntityQuery m_TransformGroup; - protected override void OnCreateManager() + protected override void OnCreate() { - m_TransformGroup = GetComponentGroup( + m_TransformGroup = GetEntityQuery( ComponentType.ReadOnly(typeof(CopyTransformFromGameObject)), typeof(UnityEngine.Transform), ComponentType.ReadWrite()); @@ -81,7 +81,7 @@ protected override JobHandle OnUpdate(JobHandle inputDeps) transformStashes = transformStashes, }; - return copyTransformsJob.ScheduleGroup(m_TransformGroup, stashTransformsJobHandle); + return copyTransformsJob.Schedule(m_TransformGroup, stashTransformsJobHandle); } } } diff --git a/Unity.Transforms.Hybrid/CopyTransformToGameObjectSystem.cs b/Unity.Transforms.Hybrid/CopyTransformToGameObjectSystem.cs index 2fe32d73..f64ae105 100644 --- a/Unity.Transforms.Hybrid/CopyTransformToGameObjectSystem.cs +++ b/Unity.Transforms.Hybrid/CopyTransformToGameObjectSystem.cs @@ -16,41 +16,33 @@ public class CopyTransformToGameObjectSystem : JobComponentSystem [BurstCompile] struct CopyTransforms : IJobParallelForTransform { - [ReadOnly] public ComponentDataFromEntity LocalToWorlds; - - [ReadOnly] [DeallocateOnJobCompletion] - public NativeArray entities; + [ReadOnly] public NativeArray LocalToWorlds; public void Execute(int index, TransformAccess transform) { - var entity = entities[index]; - - var value = LocalToWorlds[entity]; - transform.position = math.transform(value.Value, float3.zero); + var value = LocalToWorlds[index]; + transform.position = value.Position; transform.rotation = new quaternion(value.Value); } } - ComponentGroup m_TransformGroup; + EntityQuery m_TransformGroup; - protected override void OnCreateManager() + protected override void OnCreate() { - m_TransformGroup = GetComponentGroup(ComponentType.ReadOnly(typeof(CopyTransformToGameObject)), ComponentType.ReadOnly(), typeof(UnityEngine.Transform)); + m_TransformGroup = GetEntityQuery(ComponentType.ReadOnly(typeof(CopyTransformToGameObject)), ComponentType.ReadOnly(), typeof(UnityEngine.Transform)); } protected override JobHandle OnUpdate(JobHandle inputDeps) { var transforms = m_TransformGroup.GetTransformAccessArray(); - var entities = m_TransformGroup.ToEntityArray(Allocator.TempJob); - var copyTransformsJob = new CopyTransforms { - LocalToWorlds = GetComponentDataFromEntity(true), - entities = entities + LocalToWorlds = m_TransformGroup.ToComponentDataArray(Allocator.TempJob, out inputDeps), }; - return copyTransformsJob.Schedule(transforms,inputDeps); + return copyTransformsJob.Schedule(transforms, inputDeps); } } } \ No newline at end of file diff --git a/Unity.Transforms.Tests/TransformTests.cs b/Unity.Transforms.Tests/TransformTests.cs index 6da88e78..65945c81 100644 --- a/Unity.Transforms.Tests/TransformTests.cs +++ b/Unity.Transforms.Tests/TransformTests.cs @@ -25,8 +25,8 @@ public void TRS_ChildPosition() Value = math.mul( float4x4.RotateY((float)math.PI), float4x4.Translate( new float3(0.0f, 0.0f, 1.0f))) }); - World.GetOrCreateManager().Update(); - World.GetOrCreateManager().Update(); + World.GetOrCreateSystem().Update(); + World.GetOrCreateSystem().Update(); m_Manager.CompleteAllJobs(); var childWorldPosition = m_Manager.GetComponentData(child).Position; @@ -49,8 +49,8 @@ public void TRS_RemovedParentDoesNotAffectChildPosition() Value = math.mul( float4x4.RotateY((float)math.PI), float4x4.Translate( new float3(0.0f, 0.0f, 1.0f))) }); - World.GetOrCreateManager().Update(); - World.GetOrCreateManager().Update(); + World.GetOrCreateSystem().Update(); + World.GetOrCreateSystem().Update(); m_Manager.CompleteAllJobs(); var expectedChildWorldPosition = m_Manager.GetComponentData(child).Position; @@ -62,8 +62,8 @@ public void TRS_RemovedParentDoesNotAffectChildPosition() Value = math.mul( float4x4.RotateY((float)math.PI), float4x4.Translate( new float3(0.0f, 0.0f, 1.0f))) }); - World.GetOrCreateManager().Update(); - World.GetOrCreateManager().Update(); + World.GetOrCreateSystem().Update(); + World.GetOrCreateSystem().Update(); var childWorldPosition = m_Manager.GetComponentData(child).Position; @@ -251,14 +251,14 @@ public void CreateWithCompositeScaleParentScaleInverse() public void Update() { - World.GetOrCreateManager().Update(); - World.GetOrCreateManager().Update(); - World.GetOrCreateManager().Update(); - World.GetOrCreateManager().Update(); - World.GetOrCreateManager().Update(); - World.GetOrCreateManager().Update(); - World.GetOrCreateManager().Update(); - World.GetOrCreateManager().Update(); + World.GetOrCreateSystem().Update(); + World.GetOrCreateSystem().Update(); + World.GetOrCreateSystem().Update(); + World.GetOrCreateSystem().Update(); + World.GetOrCreateSystem().Update(); + World.GetOrCreateSystem().Update(); + World.GetOrCreateSystem().Update(); + World.GetOrCreateSystem().Update(); // Force complete so that main thread (tests) can have access to direct editing. m_Manager.CompleteAllJobs(); diff --git a/Unity.Transforms/CompositeRotation.cs b/Unity.Transforms/CompositeRotation.cs index 803b67d3..ddfd7aa2 100644 --- a/Unity.Transforms/CompositeRotation.cs +++ b/Unity.Transforms/CompositeRotation.cs @@ -39,7 +39,7 @@ public struct RotationPivotTranslation : IComponentData // CompositeRotation = RotationPivotTranslation * RotationPivot * Rotation * PostRotation * RotationPivot^-1 public abstract class CompositeRotationSystem : JobComponentSystem { - private ComponentGroup m_Group; + private EntityQuery m_Group; [BurstCompile] struct ToCompositeRotation : IJobChunk @@ -357,9 +357,9 @@ public void Execute(ArchetypeChunk chunk, int index, int entityOffset) } } - protected override void OnCreateManager() + protected override void OnCreate() { - m_Group = GetComponentGroup(new EntityArchetypeQuery + m_Group = GetEntityQuery(new EntityQueryDesc { All = new ComponentType[] { @@ -372,7 +372,7 @@ protected override void OnCreateManager() ComponentType.ReadOnly(), ComponentType.ReadOnly() }, - Options = EntityArchetypeQueryOptions.FilterWriteGroup + Options = EntityQueryOptions.FilterWriteGroup }); } diff --git a/Unity.Transforms/CompositeScale.cs b/Unity.Transforms/CompositeScale.cs index c1c17438..a6346f11 100644 --- a/Unity.Transforms/CompositeScale.cs +++ b/Unity.Transforms/CompositeScale.cs @@ -35,7 +35,7 @@ public struct ScalePivotTranslation : IComponentData // (or) CompositeScale = ScalePivotTranslation * ScalePivot * NonUniformScale * ScalePivot^-1 public abstract class CompositeScaleSystem : JobComponentSystem { - private ComponentGroup m_Group; + private EntityQuery m_Group; [BurstCompile] struct ToCompositeScale : IJobChunk @@ -272,9 +272,9 @@ public void Execute(ArchetypeChunk chunk, int index, int entityOffset) } } - protected override void OnCreateManager() + protected override void OnCreate() { - m_Group = GetComponentGroup(new EntityArchetypeQuery + m_Group = GetEntityQuery(new EntityQueryDesc { All = new ComponentType[] { @@ -287,7 +287,7 @@ protected override void OnCreateManager() ComponentType.ReadOnly(), ComponentType.ReadOnly() }, - Options = EntityArchetypeQueryOptions.FilterWriteGroup + Options = EntityQueryOptions.FilterWriteGroup }); } diff --git a/Unity.Transforms/LocalToParentSystem.cs b/Unity.Transforms/LocalToParentSystem.cs index 86406ede..8f8e4715 100644 --- a/Unity.Transforms/LocalToParentSystem.cs +++ b/Unity.Transforms/LocalToParentSystem.cs @@ -10,7 +10,7 @@ namespace Unity.Transforms { public abstract class LocalToParentSystem : JobComponentSystem { - private ComponentGroup m_RootsGroup; + private EntityQuery m_RootsGroup; // LocalToWorld = Parent.LocalToWorld * LocalToParent [BurstCompile] @@ -56,9 +56,9 @@ public void Execute(ArchetypeChunk chunk, int index, int entityOffset) } } - protected override void OnCreateManager() + protected override void OnCreate() { - m_RootsGroup = GetComponentGroup(new EntityArchetypeQuery + m_RootsGroup = GetEntityQuery(new EntityQueryDesc { All = new ComponentType[] { @@ -69,7 +69,7 @@ protected override void OnCreateManager() { typeof(Parent) }, - Options = EntityArchetypeQueryOptions.FilterWriteGroup + Options = EntityQueryOptions.FilterWriteGroup }); } diff --git a/Unity.Transforms/ParentScaleInverse.cs b/Unity.Transforms/ParentScaleInverse.cs index 332d4369..cafce220 100644 --- a/Unity.Transforms/ParentScaleInverse.cs +++ b/Unity.Transforms/ParentScaleInverse.cs @@ -25,7 +25,7 @@ public struct ParentScaleInverse : IComponentData // (or) ParentScaleInverse = Parent.NonUniformScale^-1 public abstract class ParentScaleInverseSystem : JobComponentSystem { - private ComponentGroup m_Group; + private EntityQuery m_Group; [BurstCompile] struct ToChildParentScaleInverse : IJobChunk @@ -115,9 +115,9 @@ public void Execute(ArchetypeChunk chunk, int chunkIndex, int firstEntityIndex) } } - protected override void OnCreateManager() + protected override void OnCreate() { - m_Group = GetComponentGroup(new EntityArchetypeQuery + m_Group = GetEntityQuery(new EntityQueryDesc { All = new ComponentType[] { @@ -129,7 +129,7 @@ protected override void OnCreateManager() ComponentType.ReadOnly(), ComponentType.ReadOnly(), }, - Options = EntityArchetypeQueryOptions.FilterWriteGroup + Options = EntityQueryOptions.FilterWriteGroup }); } diff --git a/Unity.Transforms/ParentSystem.cs b/Unity.Transforms/ParentSystem.cs index bae1cf79..9b17ba26 100644 --- a/Unity.Transforms/ParentSystem.cs +++ b/Unity.Transforms/ParentSystem.cs @@ -8,10 +8,10 @@ namespace Unity.Transforms { public abstract class ParentSystem : ComponentSystem { - private ComponentGroup m_NewParentsGroup; - private ComponentGroup m_RemovedParentsGroup; - private ComponentGroup m_ExistingParentsGroup; - private ComponentGroup m_DeletedParentsGroup; + private EntityQuery m_NewParentsGroup; + private EntityQuery m_RemovedParentsGroup; + private EntityQuery m_ExistingParentsGroup; + private EntityQuery m_DeletedParentsGroup; void AddChildToParent(Entity childEntity, Entity parentEntity) { @@ -96,9 +96,9 @@ public void Execute() } } - protected override void OnCreateManager() + protected override void OnCreate() { - m_NewParentsGroup = GetComponentGroup(new EntityArchetypeQuery + m_NewParentsGroup = GetEntityQuery(new EntityQueryDesc { All = new ComponentType[] { @@ -110,9 +110,9 @@ protected override void OnCreateManager() { typeof(PreviousParent) }, - Options = EntityArchetypeQueryOptions.FilterWriteGroup + Options = EntityQueryOptions.FilterWriteGroup }); - m_RemovedParentsGroup = GetComponentGroup(new EntityArchetypeQuery + m_RemovedParentsGroup = GetEntityQuery(new EntityQueryDesc { All = new ComponentType[] { @@ -122,9 +122,9 @@ protected override void OnCreateManager() { typeof(Parent) }, - Options = EntityArchetypeQueryOptions.FilterWriteGroup + Options = EntityQueryOptions.FilterWriteGroup }); - m_ExistingParentsGroup = GetComponentGroup(new EntityArchetypeQuery + m_ExistingParentsGroup = GetEntityQuery(new EntityQueryDesc { All = new ComponentType[] { @@ -133,9 +133,9 @@ protected override void OnCreateManager() ComponentType.ReadOnly(), typeof(PreviousParent) }, - Options = EntityArchetypeQueryOptions.FilterWriteGroup + Options = EntityQueryOptions.FilterWriteGroup }); - m_DeletedParentsGroup = GetComponentGroup(new EntityArchetypeQuery + m_DeletedParentsGroup = GetEntityQuery(new EntityQueryDesc { All = new ComponentType[] { @@ -145,7 +145,7 @@ protected override void OnCreateManager() { typeof(LocalToWorld) }, - Options = EntityArchetypeQueryOptions.FilterWriteGroup + Options = EntityQueryOptions.FilterWriteGroup }); } diff --git a/Unity.Transforms/PostRotationEuler.cs b/Unity.Transforms/PostRotationEuler.cs index 2fd36304..ecdddad4 100644 --- a/Unity.Transforms/PostRotationEuler.cs +++ b/Unity.Transforms/PostRotationEuler.cs @@ -66,11 +66,11 @@ public struct PostRotationEulerZYX : IComponentData // (or) PostRotation = PostRotationEulerZYX public abstract class PostRotationEulerSystem : JobComponentSystem { - private ComponentGroup m_Group; + private EntityQuery m_Group; - protected override void OnCreateManager() + protected override void OnCreate() { - m_Group = GetComponentGroup(new EntityArchetypeQuery + m_Group = GetEntityQuery(new EntityQueryDesc { All = new ComponentType[] { @@ -85,7 +85,7 @@ protected override void OnCreateManager() ComponentType.ReadOnly(), ComponentType.ReadOnly() }, - Options = EntityArchetypeQueryOptions.FilterWriteGroup + Options = EntityQueryOptions.FilterWriteGroup }); } diff --git a/Unity.Transforms/RotationEuler.cs b/Unity.Transforms/RotationEuler.cs index f9f85047..716b3ac7 100644 --- a/Unity.Transforms/RotationEuler.cs +++ b/Unity.Transforms/RotationEuler.cs @@ -66,11 +66,11 @@ public struct RotationEulerZYX : IComponentData // (or) Rotation = RotationEulerZYX public abstract class RotationEulerSystem : JobComponentSystem { - private ComponentGroup m_Group; + private EntityQuery m_Group; - protected override void OnCreateManager() + protected override void OnCreate() { - m_Group = GetComponentGroup(new EntityArchetypeQuery + m_Group = GetEntityQuery(new EntityQueryDesc { All = new ComponentType[] { @@ -85,7 +85,7 @@ protected override void OnCreateManager() ComponentType.ReadOnly(), ComponentType.ReadOnly() }, - Options = EntityArchetypeQueryOptions.FilterWriteGroup + Options = EntityQueryOptions.FilterWriteGroup }); } diff --git a/Unity.Transforms/TRSToLocalToParentSystem.cs b/Unity.Transforms/TRSToLocalToParentSystem.cs index 5036df92..9fe32113 100644 --- a/Unity.Transforms/TRSToLocalToParentSystem.cs +++ b/Unity.Transforms/TRSToLocalToParentSystem.cs @@ -31,7 +31,7 @@ namespace Unity.Transforms public abstract class TRSToLocalToParentSystem : JobComponentSystem { - private ComponentGroup m_Group; + private EntityQuery m_Group; [BurstCompile] struct TRSToLocalToParent : IJobChunk @@ -468,9 +468,9 @@ public void Execute(ArchetypeChunk chunk, int chunkIndex, int entityOffset) } } - protected override void OnCreateManager() + protected override void OnCreate() { - m_Group = GetComponentGroup(new EntityArchetypeQuery() + m_Group = GetEntityQuery(new EntityQueryDesc() { All = new ComponentType[] { @@ -486,7 +486,7 @@ protected override void OnCreateManager() ComponentType.ReadOnly(), ComponentType.ReadOnly() }, - Options = EntityArchetypeQueryOptions.FilterWriteGroup + Options = EntityQueryOptions.FilterWriteGroup }); } diff --git a/Unity.Transforms/TRSToLocalToWorldSystem.cs b/Unity.Transforms/TRSToLocalToWorldSystem.cs index 6c5b4029..292f9fe3 100644 --- a/Unity.Transforms/TRSToLocalToWorldSystem.cs +++ b/Unity.Transforms/TRSToLocalToWorldSystem.cs @@ -24,7 +24,7 @@ namespace Unity.Transforms // (or) LocalToWorld = Translation * CompositeRotation * CompositeScale public abstract class TRSToLocalToWorldSystem : JobComponentSystem { - private ComponentGroup m_Group; + private EntityQuery m_Group; [BurstCompile] struct TRSToLocalToWorld : IJobChunk @@ -223,9 +223,9 @@ public void Execute(ArchetypeChunk chunk, int chunkIndex, int entityOffset) } } - protected override void OnCreateManager() + protected override void OnCreate() { - m_Group = GetComponentGroup(new EntityArchetypeQuery() + m_Group = GetEntityQuery(new EntityQueryDesc() { All = new ComponentType[] { @@ -239,7 +239,7 @@ protected override void OnCreateManager() ComponentType.ReadOnly(), ComponentType.ReadOnly() }, - Options = EntityArchetypeQueryOptions.FilterWriteGroup + Options = EntityQueryOptions.FilterWriteGroup }); } diff --git a/Unity.Transforms/WorldToLocal.cs b/Unity.Transforms/WorldToLocal.cs index 75947577..9c5f3d75 100644 --- a/Unity.Transforms/WorldToLocal.cs +++ b/Unity.Transforms/WorldToLocal.cs @@ -19,7 +19,7 @@ public struct WorldToLocal : IComponentData public abstract class WorldToLocalSystem : JobComponentSystem { - private ComponentGroup m_Group; + private EntityQuery m_Group; struct ToWorldToLocal : IJobChunk { @@ -43,16 +43,16 @@ public void Execute(ArchetypeChunk chunk, int chunkIndex, int firstEntityIndex) } } - protected override void OnCreateManager() + protected override void OnCreate() { - m_Group = GetComponentGroup(new EntityArchetypeQuery + m_Group = GetEntityQuery(new EntityQueryDesc { All = new ComponentType[] { typeof(WorldToLocal), ComponentType.ReadOnly(), }, - Options = EntityArchetypeQueryOptions.FilterWriteGroup + Options = EntityQueryOptions.FilterWriteGroup }); } diff --git a/package.json b/package.json index 72973ec2..6f5bb73e 100644 --- a/package.json +++ b/package.json @@ -1,15 +1,16 @@ { + "displayName": "Entities", "name": "com.unity.entities", "unity": "2019.1", - "version": "0.0.12-preview.29", + "version": "0.0.12-preview.30", "dependencies": { "nuget.mono-cecil": "0.1.6-preview", "com.unity.test-framework.performance": "1.0.6-preview", "com.unity.properties": "0.4.0-preview", "com.unity.mathematics": "1.0.0-preview.1", - "com.unity.collections": "0.0.9-preview.16", - "com.unity.burst": "1.0.0-preview.4", - "com.unity.jobs": "0.0.7-preview.9" + "com.unity.collections": "0.0.9-preview.17", + "com.unity.burst": "1.0.0-preview.8", + "com.unity.jobs": "0.0.7-preview.10" }, "keywords": [ "entities",