using Ryujinx.Cpu.Tracking;
using Ryujinx.Graphics.GAL;
using Ryujinx.Graphics.Texture;
using Ryujinx.Memory.Range;
using System;
using System.Collections.Generic;
namespace Ryujinx.Graphics.Gpu.Image
{
    /// 
    /// A texture group represents a group of textures that belong to the same storage.
    /// When views are created, this class will track memory accesses for them separately.
    /// The group iteratively adds more granular tracking as views of different kinds are added.
    /// Note that a texture group can be absorbed into another when it becomes a view parent.
    /// 
    class TextureGroup : IDisposable
    {
        private delegate void HandlesCallbackDelegate(int baseHandle, int regionCount, bool split = false);
        /// 
        /// The storage texture associated with this group.
        /// 
        public Texture Storage { get; }
        /// 
        /// Indicates if the texture has copy dependencies. If true, then all modifications
        /// must be signalled to the group, rather than skipping ones still to be flushed.
        /// 
        public bool HasCopyDependencies { get; set; }
        private GpuContext _context;
        private int[] _allOffsets;
        private int[] _sliceSizes;
        private bool _is3D;
        private bool _hasMipViews;
        private bool _hasLayerViews;
        private int _layers;
        private int _levels;
        private MultiRange TextureRange => Storage.Range;
        /// 
        /// The views list from the storage texture.
        /// 
        private List _views;
        private TextureGroupHandle[] _handles;
        private bool[] _loadNeeded;
        /// 
        /// Create a new texture group.
        /// 
        /// GPU context that the texture group belongs to
        /// The storage texture for this group
        public TextureGroup(GpuContext context, Texture storage)
        {
            Storage = storage;
            _context = context;
            _is3D = storage.Info.Target == Target.Texture3D;
            _layers = storage.Info.GetSlices();
            _levels = storage.Info.Levels;
        }
        /// 
        /// Initialize a new texture group's dirty regions and offsets.
        /// 
        /// Size info for the storage texture
        /// True if the storage will have layer views
        /// True if the storage will have mip views
        public void Initialize(ref SizeInfo size, bool hasLayerViews, bool hasMipViews)
        {
            _allOffsets = size.AllOffsets;
            _sliceSizes = size.SliceSizes;
            (_hasLayerViews, _hasMipViews) = PropagateGranularity(hasLayerViews, hasMipViews);
            RecalculateHandleRegions();
        }
        /// 
        /// Consume the dirty flags for a given texture. The state is shared between views of the same layers and levels.
        /// 
        /// The texture being used
        /// True if a flag was dirty, false otherwise
        public bool ConsumeDirty(Texture texture)
        {
            bool dirty = false;
            EvaluateRelevantHandles(texture, (baseHandle, regionCount, split) =>
            {
                for (int i = 0; i < regionCount; i++)
                {
                    TextureGroupHandle group = _handles[baseHandle + i];
                    foreach (CpuRegionHandle handle in group.Handles)
                    {
                        if (handle.Dirty)
                        {
                            handle.Reprotect();
                            dirty = true;
                        }
                    }
                }
            });
            return dirty;
        }
        /// 
        /// Synchronize memory for a given texture. 
        /// If overlapping tracking handles are dirty, fully or partially synchronize the texture data.
        /// 
        /// The texture being used
        public void SynchronizeMemory(Texture texture)
        {
            EvaluateRelevantHandles(texture, (baseHandle, regionCount, split) =>
            {
                bool dirty = false;
                bool anyModified = false;
                bool anyUnmapped = false;
                for (int i = 0; i < regionCount; i++)
                {
                    TextureGroupHandle group = _handles[baseHandle + i];
                    bool modified = group.Modified;
                    bool handleDirty = false;
                    bool handleModified = false;
                    bool handleUnmapped = false;
                    foreach (CpuRegionHandle handle in group.Handles)
                    {
                        if (handle.Dirty)
                        {
                            handle.Reprotect();
                            handleDirty = true;
                        }
                        else
                        {
                            handleUnmapped |= handle.Unmapped;
                            handleModified |= modified;
                        }
                    }
                    // Evaluate if any copy dependencies need to be fulfilled. A few rules:
                    // If the copy handle needs to be synchronized, prefer our own state.
                    // If we need to be synchronized and there is a copy present, prefer the copy. 
                    if (group.NeedsCopy && group.Copy())
                    {
                        anyModified |= true; // The copy target has been modified.
                        handleDirty = false;
                    }
                    else
                    {
                        anyModified |= handleModified;
                        dirty |= handleDirty;
                    }
                    anyUnmapped |= handleUnmapped;
                    if (group.NeedsCopy)
                    {
                        // The texture we copied from is still being written to. Copy from it again the next time this texture is used.
                        texture.SignalGroupDirty();
                    }
                    _loadNeeded[baseHandle + i] = handleDirty && !handleUnmapped;
                }
                if (dirty)
                {
                    if (anyUnmapped || (_handles.Length > 1 && (anyModified || split)))
                    {
                        // Partial texture invalidation. Only update the layers/levels with dirty flags of the storage.
                        SynchronizePartial(baseHandle, regionCount);
                    }
                    else
                    {
                        // Full texture invalidation.
                        texture.SynchronizeFull();
                    }
                }
            });
        }
        /// 
        /// Synchronize part of the storage texture, represented by a given range of handles.
        /// Only handles marked by the _loadNeeded array will be synchronized.
        /// 
        /// The base index of the range of handles
        /// The number of handles to synchronize
        private void SynchronizePartial(int baseHandle, int regionCount)
        {
            for (int i = 0; i < regionCount; i++)
            {
                if (_loadNeeded[baseHandle + i])
                {
                    var info = GetHandleInformation(baseHandle + i);
                    int offsetIndex = info.Index;
                    // Only one of these will be greater than 1, as partial sync is only called when there are sub-image views.
                    for (int layer = 0; layer < info.Layers; layer++)
                    {
                        for (int level = 0; level < info.Levels; level++)
                        {
                            int offset = _allOffsets[offsetIndex];
                            int endOffset = (offsetIndex + 1 == _allOffsets.Length) ? (int)Storage.Size : _allOffsets[offsetIndex + 1];
                            int size = endOffset - offset;
                            ReadOnlySpan data = _context.PhysicalMemory.GetSpan(Storage.Range.GetSlice((ulong)offset, (ulong)size));
                            data = Storage.ConvertToHostCompatibleFormat(data, info.BaseLevel, true);
                            Storage.SetData(data, info.BaseLayer, info.BaseLevel);
                            offsetIndex++;
                        }
                    }
                }
            }
        }
        /// 
        /// Signal that a texture in the group has been modified by the GPU.
        /// 
        /// The texture that has been modified
        /// True if the flushing read action should be registered, false otherwise
        public void SignalModified(Texture texture, bool registerAction)
        {
            EvaluateRelevantHandles(texture, (baseHandle, regionCount, split) =>
            {
                for (int i = 0; i < regionCount; i++)
                {
                    TextureGroupHandle group = _handles[baseHandle + i];
                    group.SignalModified();
                    if (registerAction)
                    {
                        RegisterAction(group);
                    }
                }
            });
        }
        /// 
        /// Signal that a texture in the group is actively bound, or has been unbound by the GPU.
        /// 
        /// The texture that has been modified
        /// True if this texture is being bound, false if unbound
        /// True if the flushing read action should be registered, false otherwise
        public void SignalModifying(Texture texture, bool bound, bool registerAction)
        {
            EvaluateRelevantHandles(texture, (baseHandle, regionCount, split) =>
            {
                for (int i = 0; i < regionCount; i++)
                {
                    TextureGroupHandle group = _handles[baseHandle + i];
                    group.SignalModifying(bound);
                    if (registerAction)
                    {
                        RegisterAction(group);
                    }
                }
            });
        }
        /// 
        /// Register a read/write action to flush for a texture group.
        /// 
        /// The group to register an action for
        public void RegisterAction(TextureGroupHandle group)
        {
            foreach (CpuRegionHandle handle in group.Handles)
            {
                handle.RegisterAction((address, size) => FlushAction(group, address, size));
            }
        }
        /// 
        /// Propagates the mip/layer view flags depending on the texture type.
        /// When the most granular type of subresource has views, the other type of subresource must be segmented granularly too.
        /// 
        /// True if the storage has layer views
        /// True if the storage has mip views
        /// The input values after propagation
        private (bool HasLayerViews, bool HasMipViews) PropagateGranularity(bool hasLayerViews, bool hasMipViews)
        {
            if (_is3D)
            {
                hasMipViews |= hasLayerViews;
            }
            else
            {
                hasLayerViews |= hasMipViews;
            }
            return (hasLayerViews, hasMipViews);
        }
        /// 
        /// Evaluate the range of tracking handles which a view texture overlaps with.
        /// 
        /// The texture to get handles for
        /// 
        /// A function to be called with the base index of the range of handles for the given texture, and the number of handles it covers.
        /// This can be called for multiple disjoint ranges, if required.
        /// 
        private void EvaluateRelevantHandles(Texture texture, HandlesCallbackDelegate callback)
        {
            if (texture == Storage || !(_hasMipViews || _hasLayerViews))
            {
                callback(0, _handles.Length);
                return;
            }
            EvaluateRelevantHandles(texture.FirstLayer, texture.FirstLevel, texture.Info.GetSlices(), texture.Info.Levels, callback);
        }
        /// 
        /// Evaluate the range of tracking handles which a view texture overlaps with, 
        /// using the view's position and slice/level counts.
        /// 
        /// The first layer of the texture
        /// The first level of the texture
        /// The slice count of the texture
        /// The level count of the texture
        /// 
        /// A function to be called with the base index of the range of handles for the given texture, and the number of handles it covers.
        /// This can be called for multiple disjoint ranges, if required.
        /// 
        private void EvaluateRelevantHandles(int firstLayer, int firstLevel, int slices, int levels, HandlesCallbackDelegate callback)
        {
            int targetLayerHandles = _hasLayerViews ? slices : 1;
            int targetLevelHandles = _hasMipViews ? levels : 1;
            if (_is3D)
            {
                // Future mip levels come after all layers of the last mip level. Each mipmap has less layers (depth) than the last.
                
                if (!_hasLayerViews)
                {
                    // When there are no layer views, the mips are at a consistent offset.
                    callback(firstLevel, targetLevelHandles);
                }
                else
                {
                    (int levelIndex, int layerCount) = Get3DLevelRange(firstLevel);
                    if (levels > 1 && slices < _layers)
                    {
                        // The given texture only covers some of the depth of multiple mips. (a "depth slice")
                        // Callback with each mip's range separately.
                        // Can assume that the group is fully subdivided (both slices and levels > 1 for storage)
                        while (levels-- > 1)
                        {
                            callback(firstLayer + levelIndex, slices);
                            levelIndex += layerCount;
                            layerCount = Math.Max(layerCount >> 1, 1);
                            slices = Math.Max(layerCount >> 1, 1);
                        }
                    }
                    else
                    {
                        int totalSize = Math.Min(layerCount, slices);
                        while (levels-- > 1)
                        {
                            layerCount = Math.Max(layerCount >> 1, 1);
                            totalSize += layerCount;
                        }
                        callback(firstLayer + levelIndex, totalSize);
                    }
                }
            }
            else
            {
                // Future layers come after all mipmaps of the last.
                int levelHandles = _hasMipViews ? _levels : 1;
                if (slices > 1 && levels < _levels)
                {
                    // The given texture only covers some of the mipmaps of multiple slices. (a "mip slice")
                    // Callback with each layer's range separately.
                    // Can assume that the group is fully subdivided (both slices and levels > 1 for storage)
                    for (int i = 0; i < slices; i++)
                    {
                        callback(firstLevel + (firstLayer + i) * levelHandles, targetLevelHandles, true);
                    }
                }
                else
                {
                    callback(firstLevel + firstLayer * levelHandles, targetLevelHandles + (targetLayerHandles - 1) * levelHandles);
                }
            }
        }
        /// 
        /// Get the range of offsets for a given mip level of a 3D texture.
        /// 
        /// The level to return
        /// Start index and count of offsets for the given level
        private (int Index, int Count) Get3DLevelRange(int level)
        {
            int index = 0;
            int count = _layers; // Depth. Halves with each mip level.
            while (level-- > 0)
            {
                index += count;
                count = Math.Max(count >> 1, 1);
            }
            return (index, count);
        }
        /// 
        /// Get view information for a single tracking handle.
        /// 
        /// The index of the handle
        /// The layers and levels that the handle covers, and its index in the offsets array
        private (int BaseLayer, int BaseLevel, int Levels, int Layers, int Index) GetHandleInformation(int handleIndex)
        {
            int baseLayer;
            int baseLevel;
            int levels = _hasMipViews ? 1 : _levels;
            int layers = _hasLayerViews ? 1 : _layers;
            int index;
            if (_is3D)
            {
                if (_hasLayerViews)
                {
                    // NOTE: Will also have mip views, or only one level in storage.
                    index = handleIndex;
                    baseLevel = 0;
                    int layerLevels = _levels;
                    while (handleIndex >= layerLevels)
                    {
                        handleIndex -= layerLevels;
                        baseLevel++;
                        layerLevels = Math.Max(layerLevels >> 1, 1);
                    }
                    baseLayer = handleIndex;
                } 
                else
                {
                    baseLayer = 0;
                    baseLevel = handleIndex;
                    (index, _) = Get3DLevelRange(baseLevel);
                }
            }
            else
            {
                baseLevel = _hasMipViews ? handleIndex % _levels : 0;
                baseLayer = _hasMipViews ? handleIndex / _levels : handleIndex;
                index = baseLevel + baseLayer * _levels;
            }
            return (baseLayer, baseLevel, levels, layers, index);
        }
        /// 
        /// Gets the layer and level for a given view.
        /// 
        /// The index of the view
        /// The layer and level of the specified view
        private (int BaseLayer, int BaseLevel) GetLayerLevelForView(int index)
        {
            if (_is3D)
            {
                int baseLevel = 0;
                int layerLevels = _layers;
                while (index >= layerLevels)
                {
                    index -= layerLevels;
                    baseLevel++;
                    layerLevels = Math.Max(layerLevels >> 1, 1);
                }
                return (index, baseLevel);
            }
            else
            {
                return (index / _levels, index % _levels);
            }
        }
        /// 
        /// Find the byte offset of a given texture relative to the storage.
        /// 
        /// The texture to locate
        /// The offset of the texture in bytes
        public int FindOffset(Texture texture)
        {
            return _allOffsets[GetOffsetIndex(texture.FirstLayer, texture.FirstLevel)];
        }
        /// 
        /// Find the offset index of a given layer and level.
        /// 
        /// The view layer
        /// The view level
        /// The offset index of the given layer and level
        public int GetOffsetIndex(int layer, int level)
        {
            if (_is3D)
            {
                return layer + Get3DLevelRange(level).Index;
            }
            else
            {
                return level + layer * _levels;
            }
        }
        /// 
        /// The action to perform when a memory tracking handle is flipped to dirty.
        /// This notifies overlapping textures that the memory needs to be synchronized.
        /// 
        /// The handle that a dirty flag was set on
        private void DirtyAction(TextureGroupHandle groupHandle)
        {
            // Notify all textures that belong to this handle.
            Storage.SignalGroupDirty();
            lock (groupHandle.Overlaps)
            {
                foreach (Texture overlap in groupHandle.Overlaps)
                {
                    overlap.SignalGroupDirty();
                }
            }
        }
        /// 
        /// Generate a CpuRegionHandle for a given address and size range in CPU VA.
        /// 
        /// The start address of the tracked region
        /// The size of the tracked region
        /// A CpuRegionHandle covering the given range
        private CpuRegionHandle GenerateHandle(ulong address, ulong size)
        {
            return _context.PhysicalMemory.BeginTracking(address, size);
        }
        /// 
        /// Generate a TextureGroupHandle covering a specified range of views.
        /// 
        /// The start view of the handle
        /// The number of views to cover
        /// A TextureGroupHandle covering the given views
        private TextureGroupHandle GenerateHandles(int viewStart, int views)
        {
            int offset = _allOffsets[viewStart];
            int endOffset = (viewStart + views == _allOffsets.Length) ? (int)Storage.Size : _allOffsets[viewStart + views];
            int size = endOffset - offset;
            var result = new List();
            for (int i = 0; i < TextureRange.Count; i++)
            {
                MemoryRange item = TextureRange.GetSubRange(i);
                int subRangeSize = (int)item.Size;
                int sliceStart = Math.Clamp(offset, 0, subRangeSize);
                int sliceEnd = Math.Clamp(endOffset, 0, subRangeSize);
                if (sliceStart != sliceEnd)
                {
                    result.Add(GenerateHandle(item.Address + (ulong)sliceStart, (ulong)(sliceEnd - sliceStart)));
                }
                offset -= subRangeSize;
                endOffset -= subRangeSize;
                if (endOffset <= 0)
                {
                    break;
                }
            }
            (int firstLayer, int firstLevel) = GetLayerLevelForView(viewStart);
            if (_hasLayerViews && _hasMipViews)
            {
                size = _sliceSizes[firstLevel];
            }
            var groupHandle = new TextureGroupHandle(this, _allOffsets[viewStart], (ulong)size, _views, firstLayer, firstLevel, result.ToArray());
            foreach (CpuRegionHandle handle in result)
            {
                handle.RegisterDirtyEvent(() => DirtyAction(groupHandle));
            }
            return groupHandle;
        }
        /// 
        /// Update the views in this texture group, rebuilding the memory tracking if required.
        /// 
        /// The views list of the storage texture
        public void UpdateViews(List views)
        {
            // This is saved to calculate overlapping views for each handle.
            _views = views;
            bool layerViews = _hasLayerViews;
            bool mipViews = _hasMipViews;
            bool regionsRebuilt = false;
            if (!(layerViews && mipViews))
            {
                foreach (Texture view in views)
                {
                    if (view.Info.GetSlices() < _layers)
                    {
                        layerViews = true;
                    }
                    if (view.Info.Levels < _levels)
                    {
                        mipViews = true;
                    }
                }
                (layerViews, mipViews) = PropagateGranularity(layerViews, mipViews);
                if (layerViews != _hasLayerViews || mipViews != _hasMipViews)
                {
                    _hasLayerViews = layerViews;
                    _hasMipViews = mipViews;
                    RecalculateHandleRegions();
                    regionsRebuilt = true;
                }
            }
            if (!regionsRebuilt)
            {
                // Must update the overlapping views on all handles, but only if they were not just recreated.
                foreach (TextureGroupHandle handle in _handles)
                {
                    handle.RecalculateOverlaps(this, views);
                }
            }
            Storage.SignalGroupDirty();
            foreach (Texture texture in views)
            {
                texture.SignalGroupDirty();
            }
        }
        /// 
        /// Inherit handle state from an old set of handles, such as modified and dirty flags.
        /// 
        /// The set of handles to inherit state from
        /// The set of handles inheriting the state
        private void InheritHandles(TextureGroupHandle[] oldHandles, TextureGroupHandle[] handles)
        {
            foreach (var group in handles)
            {
                foreach (var handle in group.Handles)
                {
                    bool dirty = false;
                    foreach (var oldGroup in oldHandles)
                    {
                        if (group.OverlapsWith(oldGroup.Offset, oldGroup.Size))
                        {
                            foreach (var oldHandle in oldGroup.Handles)
                            {
                                if (handle.OverlapsWith(oldHandle.Address, oldHandle.Size))
                                {
                                    dirty |= oldHandle.Dirty;
                                }
                            }
                            
                            group.Inherit(oldGroup);
                        }
                    }
                    if (dirty && !handle.Dirty)
                    { 
                        handle.Reprotect(true);
                    }
                    if (group.Modified)
                    {
                        handle.RegisterAction((address, size) => FlushAction(group, address, size));
                    }
                }
            }
        }
        /// 
        /// Inherit state from another texture group.
        /// 
        /// The texture group to inherit from
        public void Inherit(TextureGroup other)
        {
            bool layerViews = _hasLayerViews || other._hasLayerViews;
            bool mipViews = _hasMipViews || other._hasMipViews;
            if (layerViews != _hasLayerViews || mipViews != _hasMipViews)
            {
                _hasLayerViews = layerViews;
                _hasMipViews = mipViews;
                RecalculateHandleRegions();
            }
            InheritHandles(other._handles, _handles);
        }
        /// 
        /// Replace the current handles with the new handles. It is assumed that the new handles start dirty.
        /// The dirty flags from the previous handles will be kept.
        /// 
        /// The handles to replace the current handles with
        private void ReplaceHandles(TextureGroupHandle[] handles)
        {
            if (_handles != null)
            {
                // When replacing handles, they should start as non-dirty.
                foreach (TextureGroupHandle groupHandle in handles)
                {
                    foreach (CpuRegionHandle handle in groupHandle.Handles)
                    {
                        handle.Reprotect();
                    }
                }
                InheritHandles(_handles, handles);
                foreach (var oldGroup in _handles)
                {
                    foreach (var oldHandle in oldGroup.Handles)
                    {
                        oldHandle.Dispose();
                    }
                }
            }
            _handles = handles;
            _loadNeeded = new bool[_handles.Length];
        }
        /// 
        /// Recalculate handle regions for this texture group, and inherit existing state into the new handles.
        /// 
        private void RecalculateHandleRegions()
        {
            TextureGroupHandle[] handles;
            if (!(_hasMipViews || _hasLayerViews))
            {
                // Single dirty region.
                var cpuRegionHandles = new CpuRegionHandle[TextureRange.Count];
                for (int i = 0; i < TextureRange.Count; i++)
                {
                    var currentRange = TextureRange.GetSubRange(i);
                    cpuRegionHandles[i] = GenerateHandle(currentRange.Address, currentRange.Size);
                }
                var groupHandle = new TextureGroupHandle(this, 0, Storage.Size, _views, 0, 0, cpuRegionHandles);
                foreach (CpuRegionHandle handle in cpuRegionHandles)
                {
                    handle.RegisterDirtyEvent(() => DirtyAction(groupHandle));
                }
                handles = new TextureGroupHandle[] { groupHandle };
            }
            else
            {
                // Get views for the host texture.
                // It's worth noting that either the texture has layer views or mip views when getting to this point, which simplifies the logic a little.
                // Depending on if the texture is 3d, either the mip views imply that layer views are present (2d) or the other way around (3d).
                // This is enforced by the way the texture matched as a view, so we don't need to check.
                int layerHandles = _hasLayerViews ? _layers : 1;
                int levelHandles = _hasMipViews ? _levels : 1;
                int handleIndex = 0;
                if (_is3D)
                {
                    var handlesList = new List();
                    for (int i = 0; i < levelHandles; i++)
                    {
                        for (int j = 0; j < layerHandles; j++)
                        {
                            (int viewStart, int views) = Get3DLevelRange(i);
                            viewStart += j;
                            views = _hasLayerViews ? 1 : views; // A layer view is also a mip view.
                            handlesList.Add(GenerateHandles(viewStart, views));
                        }
                        layerHandles = Math.Max(1, layerHandles >> 1);
                    }
                    handles = handlesList.ToArray();
                } 
                else
                {
                    handles = new TextureGroupHandle[layerHandles * levelHandles];
                    for (int i = 0; i < layerHandles; i++)
                    {
                        for (int j = 0; j < levelHandles; j++)
                        {
                            int viewStart = j + i * _levels;
                            int views = _hasMipViews ? 1 : _levels; // A mip view is also a layer view.
                            handles[handleIndex++] = GenerateHandles(viewStart, views);
                        }
                    }
                }
            }
            ReplaceHandles(handles);
        }
        /// 
        /// Ensure that there is a handle for each potential texture view. Required for copy dependencies to work.
        /// 
        private void EnsureFullSubdivision()
        {
            if (!(_hasLayerViews && _hasMipViews))
            {
                _hasLayerViews = true;
                _hasMipViews = true;
                RecalculateHandleRegions();
            }
        }
        /// 
        /// Create a copy dependency between this texture group, and a texture at a given layer/level offset.
        /// 
        /// The view compatible texture to create a dependency to
        /// The base layer of the given texture relative to the storage
        /// The base level of the given texture relative to the storage
        /// True if this texture is first copied to the given one, false for the opposite direction
        public void CreateCopyDependency(Texture other, int firstLayer, int firstLevel, bool copyTo)
        {
            TextureGroup otherGroup = other.Group;
            EnsureFullSubdivision();
            otherGroup.EnsureFullSubdivision();
            // Get the location of each texture within its storage, so we can find the handles to apply the dependency to.
            // This can consist of multiple disjoint regions, for example if this is a mip slice of an array texture.
            var targetRange = new List<(int BaseHandle, int RegionCount)>();
            var otherRange = new List<(int BaseHandle, int RegionCount)>();
            EvaluateRelevantHandles(firstLayer, firstLevel, other.Info.GetSlices(), other.Info.Levels, (baseHandle, regionCount, split) => targetRange.Add((baseHandle, regionCount)));
            otherGroup.EvaluateRelevantHandles(other, (baseHandle, regionCount, split) => otherRange.Add((baseHandle, regionCount)));
            int targetIndex = 0;
            int otherIndex = 0;
            (int Handle, int RegionCount) targetRegion = (0, 0);
            (int Handle, int RegionCount) otherRegion = (0, 0);
            while (true)
            {
                if (targetRegion.RegionCount == 0)
                {
                    if (targetIndex >= targetRange.Count)
                    {
                        break;
                    }
                    targetRegion = targetRange[targetIndex++];
                }
                if (otherRegion.RegionCount == 0)
                {
                    if (otherIndex >= otherRange.Count)
                    {
                        break;
                    }
                    otherRegion = otherRange[otherIndex++];
                }
                TextureGroupHandle handle = _handles[targetRegion.Handle++];
                TextureGroupHandle otherHandle = other.Group._handles[otherRegion.Handle++];
                targetRegion.RegionCount--;
                otherRegion.RegionCount--;
                handle.CreateCopyDependency(otherHandle, copyTo);
                // If "copyTo" is true, this texture must copy to the other.
                // Otherwise, it must copy to this texture.
                if (copyTo)
                {
                    otherHandle.Copy(handle);
                }
                else
                {
                    handle.Copy(otherHandle);
                }
            }
        }
        /// 
        /// A flush has been requested on a tracked region. Find an appropriate view to flush.
        /// 
        /// The handle this flush action is for
        /// The address of the flushing memory access
        /// The size of the flushing memory access
        public void FlushAction(TextureGroupHandle handle, ulong address, ulong size)
        {
            Storage.ExternalFlush(address, size);
            lock (handle.Overlaps)
            {
                foreach (Texture overlap in handle.Overlaps)
                {
                    overlap.ExternalFlush(address, size);
                }
            }
            handle.Modified = false;
        }
        /// 
        /// Dispose this texture group, disposing all related memory tracking handles.
        /// 
        public void Dispose()
        {
            foreach (TextureGroupHandle group in _handles)
            {
                group.Dispose();
            }
        }
    }
}