#if UNITY_2022_2_OR_NEWER using System; using System.Collections.Generic; using System.Linq; using UnityEditor.Build.Content; using UnityEditor.Build.Pipeline.Injector; using UnityEditor.Build.Pipeline.Interfaces; using UnityEditor.Build.Pipeline.Utilities; using UnityEditor.Build.Pipeline.WriteTypes; using UnityEditor.Build.Utilities; using UnityEngine; namespace UnityEditor.Build.Pipeline.Tasks { public interface IClusterOutput : IContextObject { Dictionary ObjectToCluster { get; } Dictionary ObjectToLocalID { get; } } public class ClusterOutput : IClusterOutput { private Dictionary m_ObjectToCluster = new Dictionary(); private Dictionary m_ObjectToLocalID = new Dictionary(); public Dictionary ObjectToCluster { get { return m_ObjectToCluster; } } public Dictionary ObjectToLocalID { get { return m_ObjectToLocalID; } } } /// /// Build task for creating content archives based asset co-location. /// public class ClusterBuildLayout : IBuildTask { private static void GetOrAdd(IDictionary dictionary, TKey key, out TValue value) where TValue : new() { if (dictionary.TryGetValue(key, out value)) return; value = new TValue(); dictionary.Add(key, value); } /// public int Version { get { return 1; } } #pragma warning disable 649 [InjectContext(ContextUsage.In)] IDependencyData m_DependencyData; [InjectContext] IBundleWriteData m_WriteData; [InjectContext(ContextUsage.In)] IDeterministicIdentifiers m_PackingMethod; [InjectContext] IBuildResults m_Results; [InjectContext] IClusterOutput m_ClusterResult; #pragma warning restore 649 /// public ReturnCode Run() { // Create usage maps Dictionary> objectToAssets = new Dictionary>(); foreach (var pair in m_DependencyData.AssetInfo) { ExtractAssets(objectToAssets, pair.Key, pair.Value.includedObjects); ExtractAssets(objectToAssets, pair.Key, pair.Value.referencedObjects); } foreach (var pair in m_DependencyData.SceneInfo) { ExtractAssets(objectToAssets, pair.Key, pair.Value.referencedObjects); } // Using usage maps, create clusters Dictionary> clusterToObjects = new Dictionary>(); foreach (var pair in objectToAssets) { HashSet assets = pair.Value; Hash128 cluster = HashingMethods.Calculate(assets.OrderBy(x => x)).ToHash128(); GetOrAdd(clusterToObjects, cluster, out var objectIds); objectIds.Add(pair.Key); m_ClusterResult.ObjectToCluster.Add(pair.Key, cluster); } // From clusters, create the final write data BuildUsageTagSet usageSet = new BuildUsageTagSet(); foreach (var pair in m_DependencyData.AssetUsage) usageSet.UnionWith(pair.Value); foreach (var pair in m_DependencyData.SceneUsage) usageSet.UnionWith(pair.Value); // Generates Content Archive Files from Clusters BuildReferenceMap referenceMap = new BuildReferenceMap(); foreach (var pair in clusterToObjects) { var cluster = pair.Key.ToString(); m_WriteData.FileToObjects.Add(cluster, pair.Value.ToList()); #pragma warning disable CS0618 // Type or member is obsolete var op = new RawWriteOperation(); #pragma warning restore CS0618 // Type or member is obsolete m_WriteData.WriteOperations.Add(op); op.Command = new WriteCommand(); op.Command.fileName = cluster; op.Command.internalName = cluster; op.Command.serializeObjects = new List(); foreach (var objectId in pair.Value) { var lfid = m_PackingMethod.SerializationIndexFromObjectIdentifier(objectId); op.Command.serializeObjects.Add(new SerializationInfo { serializationObject = objectId, serializationIndex = lfid }); referenceMap.AddMapping(cluster, lfid, objectId); m_ClusterResult.ObjectToLocalID.Add(objectId, lfid); } op.ReferenceMap = referenceMap; op.UsageSet = usageSet; m_WriteData.FileToBundle.Add(cluster, cluster); } // Generates Content Scene Archive Files from Scene Input foreach (var pair in m_DependencyData.SceneInfo) { #pragma warning disable CS0618 // Type or member is obsolete var op = new SceneRawWriteOperation(); #pragma warning restore CS0618 // Type or member is obsolete m_WriteData.WriteOperations.Add(op); op.Command = new WriteCommand(); op.Command.fileName = pair.Key.ToString(); op.Command.internalName = pair.Key.ToString(); op.Command.serializeObjects = new List(); op.ReferenceMap = referenceMap; op.UsageSet = usageSet; op.Scene = pair.Value.scene; m_WriteData.FileToBundle.Add(pair.Key.ToString(), pair.Key.ToString()); } return ReturnCode.Success; } private static void ExtractAssets(Dictionary> objectToAssets, GUID asset, IEnumerable objectIds) { foreach (var objectId in objectIds) { if (objectId.filePath.Equals(CommonStrings.UnityDefaultResourcePath, StringComparison.OrdinalIgnoreCase)) continue; GetOrAdd(objectToAssets, objectId, out var assets); assets.Add(asset); } } } } #endif