using System.Collections.Generic; using System.Linq; using Unity.Collections; using UnityEngine; using Unity.Jobs; using Unity.Burst; using Unity.Mathematics; public static class LocalToWorldJob { private static Dictionary _data = new Dictionary(); [BurstCompile] struct LocalToWorldConvertJob : IJob { [WriteOnly] public NativeArray positionsWorld; [ReadOnly] public Matrix4x4 matrix; [ReadOnly] public NativeArray positionsLocal; // The code actually running on the job public void Execute() { for (var i = 0; i < positionsLocal.Length; i++) { float4 pos = float4.zero; pos.xyz = positionsLocal[i]; pos.w = 1f; pos = matrix * pos; positionsWorld[i] = pos.xyz; } } } public static void ScheduleJob(int guid, Vector3[] positions, Matrix4x4 localToWorld) { if (!_data.ContainsKey(guid)) { TransformLocalToWorld jobData = new TransformLocalToWorld(); jobData.positionsWorld = new NativeArray(positions.Length, Allocator.Persistent); jobData.positionsLocal = new NativeArray(positions.Length, Allocator.Persistent); for (var i = 0; i < positions.Length; i++) jobData.positionsLocal[i] = positions[i]; _data.Add(guid, jobData); } if (_data[guid].processing) return; _data[guid].job = new LocalToWorldConvertJob() { positionsWorld = _data[guid].positionsWorld, positionsLocal = _data[guid].positionsLocal, matrix = localToWorld }; _data[guid].handle = _data[guid].job.Schedule(); _data[guid].processing = true; JobHandle.ScheduleBatchedJobs(); } public static float3[] CompleteJob(int guid) { if (_data.ContainsKey(guid)) { _data[guid].handle.Complete(); _data[guid].processing = false; return _data[guid].job.positionsWorld.ToArray(); } return null; } public static void Cleanup(int guid) { if (_data.ContainsKey(guid)) { _data[guid].handle.Complete(); _data[guid].positionsWorld.Dispose(); _data[guid].positionsLocal.Dispose(); _data.Remove(guid); } } class TransformLocalToWorld { public NativeArray positionsLocal; public NativeArray positionsWorld; public JobHandle handle; public LocalToWorldConvertJob job; public bool processing; } }