632 lines
		
	
	
		
			No EOL
		
	
	
		
			24 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			632 lines
		
	
	
		
			No EOL
		
	
	
		
			24 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| using Ryujinx.Audio.Renderer.Common;
 | |
| using Ryujinx.Audio.Renderer.Parameter;
 | |
| using Ryujinx.Audio.Renderer.Parameter.Performance;
 | |
| using Ryujinx.Audio.Renderer.Server.Effect;
 | |
| using Ryujinx.Audio.Renderer.Server.MemoryPool;
 | |
| using Ryujinx.Audio.Renderer.Server.Mix;
 | |
| using Ryujinx.Audio.Renderer.Server.Performance;
 | |
| using Ryujinx.Audio.Renderer.Server.Sink;
 | |
| using Ryujinx.Audio.Renderer.Server.Splitter;
 | |
| using Ryujinx.Audio.Renderer.Server.Voice;
 | |
| using Ryujinx.Audio.Renderer.Utils;
 | |
| using Ryujinx.Common.Logging;
 | |
| using System;
 | |
| using System.Diagnostics;
 | |
| using System.Runtime.CompilerServices;
 | |
| using System.Runtime.InteropServices;
 | |
| using static Ryujinx.Audio.Renderer.Common.BehaviourParameter;
 | |
| 
 | |
| namespace Ryujinx.Audio.Renderer.Server
 | |
| {
 | |
|     public class StateUpdater
 | |
|     {
 | |
|         private readonly ReadOnlyMemory<byte> _inputOrigin;
 | |
|         private ReadOnlyMemory<byte> _outputOrigin;
 | |
|         private ReadOnlyMemory<byte> _input;
 | |
| 
 | |
|         private Memory<byte> _output;
 | |
|         private uint _processHandle;
 | |
|         private BehaviourContext _behaviourContext;
 | |
| 
 | |
|         private UpdateDataHeader _inputHeader;
 | |
|         private Memory<UpdateDataHeader> _outputHeader;
 | |
| 
 | |
|         private ref UpdateDataHeader OutputHeader => ref _outputHeader.Span[0];
 | |
| 
 | |
|         public StateUpdater(ReadOnlyMemory<byte> input, Memory<byte> output, uint processHandle, BehaviourContext behaviourContext)
 | |
|         {
 | |
|             _input = input;
 | |
|             _inputOrigin = _input;
 | |
|             _output = output;
 | |
|             _outputOrigin = _output;
 | |
|             _processHandle = processHandle;
 | |
|             _behaviourContext = behaviourContext;
 | |
| 
 | |
|             _inputHeader = SpanIOHelper.Read<UpdateDataHeader>(ref _input);
 | |
| 
 | |
|             _outputHeader = SpanMemoryManager<UpdateDataHeader>.Cast(_output.Slice(0, Unsafe.SizeOf<UpdateDataHeader>()));
 | |
|             OutputHeader.Initialize(_behaviourContext.UserRevision);
 | |
|             _output = _output.Slice(Unsafe.SizeOf<UpdateDataHeader>());
 | |
|         }
 | |
| 
 | |
|         public ResultCode UpdateBehaviourContext()
 | |
|         {
 | |
|             BehaviourParameter parameter = SpanIOHelper.Read<BehaviourParameter>(ref _input);
 | |
| 
 | |
|             if (!BehaviourContext.CheckValidRevision(parameter.UserRevision) || parameter.UserRevision != _behaviourContext.UserRevision)
 | |
|             {
 | |
|                 return ResultCode.InvalidUpdateInfo;
 | |
|             }
 | |
| 
 | |
|             _behaviourContext.ClearError();
 | |
|             _behaviourContext.UpdateFlags(parameter.Flags);
 | |
| 
 | |
|             if (_inputHeader.BehaviourSize != Unsafe.SizeOf<BehaviourParameter>())
 | |
|             {
 | |
|                 return ResultCode.InvalidUpdateInfo;
 | |
|             }
 | |
| 
 | |
|             return ResultCode.Success;
 | |
|         }
 | |
| 
 | |
|         public ResultCode UpdateMemoryPools(Span<MemoryPoolState> memoryPools)
 | |
|         {
 | |
|             PoolMapper mapper = new PoolMapper(_processHandle, _behaviourContext.IsMemoryPoolForceMappingEnabled());
 | |
| 
 | |
|             if (memoryPools.Length * Unsafe.SizeOf<MemoryPoolInParameter>() != _inputHeader.MemoryPoolsSize)
 | |
|             {
 | |
|                 return ResultCode.InvalidUpdateInfo;
 | |
|             }
 | |
| 
 | |
|             foreach (ref MemoryPoolState memoryPool in memoryPools)
 | |
|             {
 | |
|                 MemoryPoolInParameter parameter = SpanIOHelper.Read<MemoryPoolInParameter>(ref _input);
 | |
| 
 | |
|                 ref MemoryPoolOutStatus outStatus = ref SpanIOHelper.GetWriteRef<MemoryPoolOutStatus>(ref _output)[0];
 | |
| 
 | |
|                 PoolMapper.UpdateResult updateResult = mapper.Update(ref memoryPool, ref parameter, ref outStatus);
 | |
| 
 | |
|                 if (updateResult != PoolMapper.UpdateResult.Success &&
 | |
|                     updateResult != PoolMapper.UpdateResult.MapError &&
 | |
|                     updateResult != PoolMapper.UpdateResult.UnmapError)
 | |
|                 {
 | |
|                     if (updateResult != PoolMapper.UpdateResult.InvalidParameter)
 | |
|                     {
 | |
|                         throw new InvalidOperationException($"{updateResult}");
 | |
|                     }
 | |
| 
 | |
|                     return ResultCode.InvalidUpdateInfo;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             OutputHeader.MemoryPoolsSize = (uint)(Unsafe.SizeOf<MemoryPoolOutStatus>() * memoryPools.Length);
 | |
|             OutputHeader.TotalSize += OutputHeader.MemoryPoolsSize;
 | |
| 
 | |
|             return ResultCode.Success;
 | |
|         }
 | |
| 
 | |
|         public ResultCode UpdateVoiceChannelResources(VoiceContext context)
 | |
|         {
 | |
|             if (context.GetCount() * Unsafe.SizeOf<VoiceChannelResourceInParameter>() != _inputHeader.VoiceResourcesSize)
 | |
|             {
 | |
|                 return ResultCode.InvalidUpdateInfo;
 | |
|             }
 | |
| 
 | |
|             for (int i = 0; i < context.GetCount(); i++)
 | |
|             {
 | |
|                 VoiceChannelResourceInParameter parameter = SpanIOHelper.Read<VoiceChannelResourceInParameter>(ref _input);
 | |
| 
 | |
|                 ref VoiceChannelResource resource = ref context.GetChannelResource(i);
 | |
| 
 | |
|                 resource.Id = parameter.Id;
 | |
|                 parameter.Mix.AsSpan().CopyTo(resource.Mix.AsSpan());
 | |
|                 resource.IsUsed = parameter.IsUsed;
 | |
|             }
 | |
| 
 | |
|             return ResultCode.Success;
 | |
|         }
 | |
| 
 | |
|         public ResultCode UpdateVoices(VoiceContext context, Memory<MemoryPoolState> memoryPools)
 | |
|         {
 | |
|             if (context.GetCount() * Unsafe.SizeOf<VoiceInParameter>() != _inputHeader.VoicesSize)
 | |
|             {
 | |
|                 return ResultCode.InvalidUpdateInfo;
 | |
|             }
 | |
| 
 | |
|             int initialOutputSize = _output.Length;
 | |
| 
 | |
|             ReadOnlySpan<VoiceInParameter> parameters = MemoryMarshal.Cast<byte, VoiceInParameter>(_input.Slice(0, (int)_inputHeader.VoicesSize).Span);
 | |
| 
 | |
|             _input = _input.Slice((int)_inputHeader.VoicesSize);
 | |
| 
 | |
|             PoolMapper mapper = new PoolMapper(_processHandle, memoryPools, _behaviourContext.IsMemoryPoolForceMappingEnabled());
 | |
| 
 | |
|             // First make everything not in use.
 | |
|             for (int i = 0; i < context.GetCount(); i++)
 | |
|             {
 | |
|                 ref VoiceState state = ref context.GetState(i);
 | |
| 
 | |
|                 state.InUse = false;
 | |
|             }
 | |
| 
 | |
|             // Start processing
 | |
|             for (int i = 0; i < context.GetCount(); i++)
 | |
|             {
 | |
|                 VoiceInParameter parameter = parameters[i];
 | |
| 
 | |
|                 Memory<VoiceUpdateState>[] voiceUpdateStates = new Memory<VoiceUpdateState>[Constants.VoiceChannelCountMax];
 | |
| 
 | |
|                 ref VoiceOutStatus outStatus = ref SpanIOHelper.GetWriteRef<VoiceOutStatus>(ref _output)[0];
 | |
| 
 | |
|                 if (parameter.InUse)
 | |
|                 {
 | |
|                     ref VoiceState currentVoiceState = ref context.GetState(i);
 | |
| 
 | |
|                     for (int channelResourceIndex = 0; channelResourceIndex < parameter.ChannelCount; channelResourceIndex++)
 | |
|                     {
 | |
|                         int channelId = parameter.ChannelResourceIds[channelResourceIndex];
 | |
| 
 | |
|                         Debug.Assert(channelId >= 0 && channelId < context.GetCount());
 | |
| 
 | |
|                         voiceUpdateStates[channelResourceIndex] = context.GetUpdateStateForCpu(channelId);
 | |
|                     }
 | |
| 
 | |
|                     if (parameter.IsNew)
 | |
|                     {
 | |
|                         currentVoiceState.Initialize();
 | |
|                     }
 | |
| 
 | |
|                     currentVoiceState.UpdateParameters(out ErrorInfo updateParameterError, ref parameter, ref mapper, ref _behaviourContext);
 | |
| 
 | |
|                     if (updateParameterError.ErrorCode != ResultCode.Success)
 | |
|                     {
 | |
|                         _behaviourContext.AppendError(ref updateParameterError);
 | |
|                     }
 | |
| 
 | |
|                     currentVoiceState.UpdateWaveBuffers(out ErrorInfo[] waveBufferUpdateErrorInfos, ref parameter, voiceUpdateStates, ref mapper, ref _behaviourContext);
 | |
| 
 | |
|                     foreach (ref ErrorInfo errorInfo in waveBufferUpdateErrorInfos.AsSpan())
 | |
|                     {
 | |
|                         if (errorInfo.ErrorCode != ResultCode.Success)
 | |
|                         {
 | |
|                             _behaviourContext.AppendError(ref errorInfo);
 | |
|                         }
 | |
|                     }
 | |
| 
 | |
|                     currentVoiceState.WriteOutStatus(ref outStatus, ref parameter, voiceUpdateStates);
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             int currentOutputSize = _output.Length;
 | |
| 
 | |
|             OutputHeader.VoicesSize = (uint)(Unsafe.SizeOf<VoiceOutStatus>() * context.GetCount());
 | |
|             OutputHeader.TotalSize += OutputHeader.VoicesSize;
 | |
| 
 | |
|             Debug.Assert((initialOutputSize - currentOutputSize) == OutputHeader.VoicesSize);
 | |
| 
 | |
|             return ResultCode.Success;
 | |
|         }
 | |
| 
 | |
|         private static void ResetEffect<T>(ref BaseEffect effect, ref T parameter, PoolMapper mapper) where T : unmanaged, IEffectInParameter
 | |
|         {
 | |
|             effect.ForceUnmapBuffers(mapper);
 | |
| 
 | |
|             switch (parameter.Type)
 | |
|             {
 | |
|                 case EffectType.Invalid:
 | |
|                     effect = new BaseEffect();
 | |
|                     break;
 | |
|                 case EffectType.BufferMix:
 | |
|                     effect = new BufferMixEffect();
 | |
|                     break;
 | |
|                 case EffectType.AuxiliaryBuffer:
 | |
|                     effect = new AuxiliaryBufferEffect();
 | |
|                     break;
 | |
|                 case EffectType.Delay:
 | |
|                     effect = new DelayEffect();
 | |
|                     break;
 | |
|                 case EffectType.Reverb:
 | |
|                     effect = new ReverbEffect();
 | |
|                     break;
 | |
|                 case EffectType.Reverb3d:
 | |
|                     effect = new Reverb3dEffect();
 | |
|                     break;
 | |
|                 case EffectType.BiquadFilter:
 | |
|                     effect = new BiquadFilterEffect();
 | |
|                     break;
 | |
|                 case EffectType.Limiter:
 | |
|                     effect = new LimiterEffect();
 | |
|                     break;
 | |
|                 case EffectType.CaptureBuffer:
 | |
|                     effect = new CaptureBufferEffect();
 | |
|                     break;
 | |
|                 default:
 | |
|                     throw new NotImplementedException($"EffectType {parameter.Type} not implemented!");
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public ResultCode UpdateEffects(EffectContext context, bool isAudioRendererActive, Memory<MemoryPoolState> memoryPools)
 | |
|         {
 | |
|             if (_behaviourContext.IsEffectInfoVersion2Supported())
 | |
|             {
 | |
|                 return UpdateEffectsVersion2(context, isAudioRendererActive, memoryPools);
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 return UpdateEffectsVersion1(context, isAudioRendererActive, memoryPools);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public ResultCode UpdateEffectsVersion2(EffectContext context, bool isAudioRendererActive, Memory<MemoryPoolState> memoryPools)
 | |
|         {
 | |
|             if (context.GetCount() * Unsafe.SizeOf<EffectInParameterVersion2>() != _inputHeader.EffectsSize)
 | |
|             {
 | |
|                 return ResultCode.InvalidUpdateInfo;
 | |
|             }
 | |
| 
 | |
|             int initialOutputSize = _output.Length;
 | |
| 
 | |
|             ReadOnlySpan<EffectInParameterVersion2> parameters = MemoryMarshal.Cast<byte, EffectInParameterVersion2>(_input.Slice(0, (int)_inputHeader.EffectsSize).Span);
 | |
| 
 | |
|             _input = _input.Slice((int)_inputHeader.EffectsSize);
 | |
| 
 | |
|             PoolMapper mapper = new PoolMapper(_processHandle, memoryPools, _behaviourContext.IsMemoryPoolForceMappingEnabled());
 | |
| 
 | |
|             for (int i = 0; i < context.GetCount(); i++)
 | |
|             {
 | |
|                 EffectInParameterVersion2 parameter = parameters[i];
 | |
| 
 | |
|                 ref EffectOutStatusVersion2 outStatus = ref SpanIOHelper.GetWriteRef<EffectOutStatusVersion2>(ref _output)[0];
 | |
| 
 | |
|                 ref BaseEffect effect = ref context.GetEffect(i);
 | |
| 
 | |
|                 if (!effect.IsTypeValid(ref parameter))
 | |
|                 {
 | |
|                     ResetEffect(ref effect, ref parameter, mapper);
 | |
|                 }
 | |
| 
 | |
|                 effect.Update(out ErrorInfo updateErrorInfo, ref parameter, mapper);
 | |
| 
 | |
|                 if (updateErrorInfo.ErrorCode != ResultCode.Success)
 | |
|                 {
 | |
|                     _behaviourContext.AppendError(ref updateErrorInfo);
 | |
|                 }
 | |
| 
 | |
|                 effect.StoreStatus(ref outStatus, isAudioRendererActive);
 | |
| 
 | |
|                 if (parameter.IsNew)
 | |
|                 {
 | |
|                     effect.InitializeResultState(ref context.GetDspState(i));
 | |
|                     effect.InitializeResultState(ref context.GetState(i));
 | |
|                 }
 | |
| 
 | |
|                 effect.UpdateResultState(ref outStatus.ResultState, ref context.GetState(i));
 | |
|             }
 | |
| 
 | |
|             int currentOutputSize = _output.Length;
 | |
| 
 | |
|             OutputHeader.EffectsSize = (uint)(Unsafe.SizeOf<EffectOutStatusVersion2>() * context.GetCount());
 | |
|             OutputHeader.TotalSize += OutputHeader.EffectsSize;
 | |
| 
 | |
|             Debug.Assert((initialOutputSize - currentOutputSize) == OutputHeader.EffectsSize);
 | |
| 
 | |
|             return ResultCode.Success;
 | |
|         }
 | |
| 
 | |
|         public ResultCode UpdateEffectsVersion1(EffectContext context, bool isAudioRendererActive, Memory<MemoryPoolState> memoryPools)
 | |
|         {
 | |
|             if (context.GetCount() * Unsafe.SizeOf<EffectInParameterVersion1>() != _inputHeader.EffectsSize)
 | |
|             {
 | |
|                 return ResultCode.InvalidUpdateInfo;
 | |
|             }
 | |
| 
 | |
|             int initialOutputSize = _output.Length;
 | |
| 
 | |
|             ReadOnlySpan<EffectInParameterVersion1> parameters = MemoryMarshal.Cast<byte, EffectInParameterVersion1>(_input.Slice(0, (int)_inputHeader.EffectsSize).Span);
 | |
| 
 | |
|             _input = _input.Slice((int)_inputHeader.EffectsSize);
 | |
| 
 | |
|             PoolMapper mapper = new PoolMapper(_processHandle, memoryPools, _behaviourContext.IsMemoryPoolForceMappingEnabled());
 | |
| 
 | |
|             for (int i = 0; i < context.GetCount(); i++)
 | |
|             {
 | |
|                 EffectInParameterVersion1 parameter = parameters[i];
 | |
| 
 | |
|                 ref EffectOutStatusVersion1 outStatus = ref SpanIOHelper.GetWriteRef<EffectOutStatusVersion1>(ref _output)[0];
 | |
| 
 | |
|                 ref BaseEffect effect = ref context.GetEffect(i);
 | |
| 
 | |
|                 if (!effect.IsTypeValid(ref parameter))
 | |
|                 {
 | |
|                     ResetEffect(ref effect, ref parameter, mapper);
 | |
|                 }
 | |
| 
 | |
|                 effect.Update(out ErrorInfo updateErrorInfo, ref parameter, mapper);
 | |
| 
 | |
|                 if (updateErrorInfo.ErrorCode != ResultCode.Success)
 | |
|                 {
 | |
|                     _behaviourContext.AppendError(ref updateErrorInfo);
 | |
|                 }
 | |
| 
 | |
|                 effect.StoreStatus(ref outStatus, isAudioRendererActive);
 | |
|             }
 | |
| 
 | |
|             int currentOutputSize = _output.Length;
 | |
| 
 | |
|             OutputHeader.EffectsSize = (uint)(Unsafe.SizeOf<EffectOutStatusVersion1>() * context.GetCount());
 | |
|             OutputHeader.TotalSize += OutputHeader.EffectsSize;
 | |
| 
 | |
|             Debug.Assert((initialOutputSize - currentOutputSize) == OutputHeader.EffectsSize);
 | |
| 
 | |
|             return ResultCode.Success;
 | |
|         }
 | |
| 
 | |
|         public ResultCode UpdateSplitter(SplitterContext context)
 | |
|         {
 | |
|             if (context.Update(_input.Span, out int consumedSize))
 | |
|             {
 | |
|                 _input = _input.Slice(consumedSize);
 | |
| 
 | |
|                 return ResultCode.Success;
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 return ResultCode.InvalidUpdateInfo;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         private bool CheckMixParametersValidity(MixContext mixContext, uint mixBufferCount, uint inputMixCount, ReadOnlySpan<MixParameter> parameters)
 | |
|         {
 | |
|             uint maxMixStateCount = mixContext.GetCount();
 | |
|             uint totalRequiredMixBufferCount = 0;
 | |
| 
 | |
|             for (int i = 0; i < inputMixCount; i++)
 | |
|             {
 | |
|                 if (parameters[i].IsUsed)
 | |
|                 {
 | |
|                     if (parameters[i].DestinationMixId != Constants.UnusedMixId &&
 | |
|                         parameters[i].DestinationMixId > maxMixStateCount &&
 | |
|                         parameters[i].MixId != Constants.FinalMixId)
 | |
|                     {
 | |
|                         return true;
 | |
|                     }
 | |
| 
 | |
|                     totalRequiredMixBufferCount += parameters[i].BufferCount;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             return totalRequiredMixBufferCount > mixBufferCount;
 | |
|         }
 | |
| 
 | |
|         public ResultCode UpdateMixes(MixContext mixContext, uint mixBufferCount, EffectContext effectContext, SplitterContext splitterContext)
 | |
|         {
 | |
|             uint mixCount;
 | |
|             uint inputMixSize;
 | |
|             uint inputSize = 0;
 | |
| 
 | |
|             if (_behaviourContext.IsMixInParameterDirtyOnlyUpdateSupported())
 | |
|             {
 | |
|                 MixInParameterDirtyOnlyUpdate parameter = MemoryMarshal.Cast<byte, MixInParameterDirtyOnlyUpdate>(_input.Span)[0];
 | |
| 
 | |
|                 mixCount = parameter.MixCount;
 | |
| 
 | |
|                 inputSize += (uint)Unsafe.SizeOf<MixInParameterDirtyOnlyUpdate>();
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 mixCount = mixContext.GetCount();
 | |
|             }
 | |
| 
 | |
|             inputMixSize = mixCount * (uint)Unsafe.SizeOf<MixParameter>();
 | |
| 
 | |
|             inputSize += inputMixSize;
 | |
| 
 | |
|             if (inputSize != _inputHeader.MixesSize)
 | |
|             {
 | |
|                 return ResultCode.InvalidUpdateInfo;
 | |
|             }
 | |
| 
 | |
|             if (_behaviourContext.IsMixInParameterDirtyOnlyUpdateSupported())
 | |
|             {
 | |
|                 _input = _input.Slice(Unsafe.SizeOf<MixInParameterDirtyOnlyUpdate>());
 | |
|             }
 | |
| 
 | |
|             ReadOnlySpan<MixParameter> parameters = MemoryMarshal.Cast<byte, MixParameter>(_input.Span.Slice(0, (int)inputMixSize));
 | |
| 
 | |
|             _input = _input.Slice((int)inputMixSize);
 | |
| 
 | |
|             if (CheckMixParametersValidity(mixContext, mixBufferCount, mixCount, parameters))
 | |
|             {
 | |
|                 return ResultCode.InvalidUpdateInfo;
 | |
|             }
 | |
| 
 | |
|             bool isMixContextDirty = false;
 | |
| 
 | |
|             for (int i = 0; i < parameters.Length; i++)
 | |
|             {
 | |
|                 MixParameter parameter = parameters[i];
 | |
| 
 | |
|                 int mixId = i;
 | |
| 
 | |
|                 if (_behaviourContext.IsMixInParameterDirtyOnlyUpdateSupported())
 | |
|                 {
 | |
|                     mixId = parameter.MixId;
 | |
|                 }
 | |
| 
 | |
|                 ref MixState mix = ref mixContext.GetState(mixId);
 | |
| 
 | |
|                 if (parameter.IsUsed != mix.IsUsed)
 | |
|                 {
 | |
|                     mix.IsUsed = parameter.IsUsed;
 | |
| 
 | |
|                     if (parameter.IsUsed)
 | |
|                     {
 | |
|                         mix.ClearEffectProcessingOrder();
 | |
|                     }
 | |
| 
 | |
|                     isMixContextDirty = true;
 | |
|                 }
 | |
| 
 | |
|                 if (mix.IsUsed)
 | |
|                 {
 | |
|                     isMixContextDirty |= mix.Update(mixContext.EdgeMatrix, ref parameter, effectContext, splitterContext, _behaviourContext);
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             if (isMixContextDirty)
 | |
|             {
 | |
|                 if (_behaviourContext.IsSplitterSupported() && splitterContext.UsingSplitter())
 | |
|                 {
 | |
|                     if (!mixContext.Sort(splitterContext))
 | |
|                     {
 | |
|                         return ResultCode.InvalidMixSorting;
 | |
|                     }
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     mixContext.Sort();
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             return ResultCode.Success;
 | |
|         }
 | |
| 
 | |
|         private static void ResetSink(ref BaseSink sink, ref SinkInParameter parameter)
 | |
|         {
 | |
|             sink.CleanUp();
 | |
| 
 | |
|             switch (parameter.Type)
 | |
|             {
 | |
|                 case SinkType.Invalid:
 | |
|                     sink = new BaseSink();
 | |
|                     break;
 | |
|                 case SinkType.CircularBuffer:
 | |
|                     sink = new CircularBufferSink();
 | |
|                     break;
 | |
|                 case SinkType.Device:
 | |
|                     sink = new DeviceSink();
 | |
|                     break;
 | |
|                 default:
 | |
|                     throw new NotImplementedException($"SinkType {parameter.Type} not implemented!");
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public ResultCode UpdateSinks(SinkContext context, Memory<MemoryPoolState> memoryPools)
 | |
|         {
 | |
|             PoolMapper mapper = new PoolMapper(_processHandle, memoryPools, _behaviourContext.IsMemoryPoolForceMappingEnabled());
 | |
| 
 | |
|             if (context.GetCount() * Unsafe.SizeOf<SinkInParameter>() != _inputHeader.SinksSize)
 | |
|             {
 | |
|                 return ResultCode.InvalidUpdateInfo;
 | |
|             }
 | |
| 
 | |
|             int initialOutputSize = _output.Length;
 | |
| 
 | |
|             ReadOnlySpan<SinkInParameter> parameters = MemoryMarshal.Cast<byte, SinkInParameter>(_input.Slice(0, (int)_inputHeader.SinksSize).Span);
 | |
| 
 | |
|             _input = _input.Slice((int)_inputHeader.SinksSize);
 | |
| 
 | |
|             for (int i = 0; i < context.GetCount(); i++)
 | |
|             {
 | |
|                 SinkInParameter parameter = parameters[i];
 | |
|                 ref SinkOutStatus outStatus = ref SpanIOHelper.GetWriteRef<SinkOutStatus>(ref _output)[0];
 | |
|                 ref BaseSink sink = ref context.GetSink(i);
 | |
| 
 | |
|                 if (!sink.IsTypeValid(ref parameter))
 | |
|                 {
 | |
|                     ResetSink(ref sink, ref parameter);
 | |
|                 }
 | |
| 
 | |
|                 sink.Update(out ErrorInfo updateErrorInfo, ref parameter, ref outStatus, mapper);
 | |
| 
 | |
|                 if (updateErrorInfo.ErrorCode != ResultCode.Success)
 | |
|                 {
 | |
|                     _behaviourContext.AppendError(ref updateErrorInfo);
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             int currentOutputSize = _output.Length;
 | |
| 
 | |
|             OutputHeader.SinksSize = (uint)(Unsafe.SizeOf<SinkOutStatus>() * context.GetCount());
 | |
|             OutputHeader.TotalSize += OutputHeader.SinksSize;
 | |
| 
 | |
|             Debug.Assert((initialOutputSize - currentOutputSize) == OutputHeader.SinksSize);
 | |
| 
 | |
|             return ResultCode.Success;
 | |
|         }
 | |
| 
 | |
|         public ResultCode UpdatePerformanceBuffer(PerformanceManager manager, Span<byte> performanceOutput)
 | |
|         {
 | |
|             if (Unsafe.SizeOf<PerformanceInParameter>() != _inputHeader.PerformanceBufferSize)
 | |
|             {
 | |
|                 return ResultCode.InvalidUpdateInfo;
 | |
|             }
 | |
| 
 | |
|             PerformanceInParameter parameter = SpanIOHelper.Read<PerformanceInParameter>(ref _input);
 | |
| 
 | |
|             ref PerformanceOutStatus outStatus = ref SpanIOHelper.GetWriteRef<PerformanceOutStatus>(ref _output)[0];
 | |
| 
 | |
|             if (manager != null)
 | |
|             {
 | |
|                 outStatus.HistorySize = manager.CopyHistories(performanceOutput);
 | |
| 
 | |
|                 manager.SetTargetNodeId(parameter.TargetNodeId);
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 outStatus.HistorySize = 0;
 | |
|             }
 | |
| 
 | |
|             OutputHeader.PerformanceBufferSize = (uint)Unsafe.SizeOf<PerformanceOutStatus>();
 | |
|             OutputHeader.TotalSize += OutputHeader.PerformanceBufferSize;
 | |
| 
 | |
|             return ResultCode.Success;
 | |
|         }
 | |
| 
 | |
|         public ResultCode UpdateErrorInfo()
 | |
|         {
 | |
|             ref BehaviourErrorInfoOutStatus outStatus = ref SpanIOHelper.GetWriteRef<BehaviourErrorInfoOutStatus>(ref _output)[0];
 | |
| 
 | |
|             _behaviourContext.CopyErrorInfo(outStatus.ErrorInfos.AsSpan(), out outStatus.ErrorInfosCount);
 | |
| 
 | |
|             OutputHeader.BehaviourSize = (uint)Unsafe.SizeOf<BehaviourErrorInfoOutStatus>();
 | |
|             OutputHeader.TotalSize += OutputHeader.BehaviourSize;
 | |
| 
 | |
|             return ResultCode.Success;
 | |
|         }
 | |
| 
 | |
|         public ResultCode UpdateRendererInfo(ulong elapsedFrameCount)
 | |
|         {
 | |
|             ref RendererInfoOutStatus outStatus = ref SpanIOHelper.GetWriteRef<RendererInfoOutStatus>(ref _output)[0];
 | |
| 
 | |
|             outStatus.ElapsedFrameCount = elapsedFrameCount;
 | |
| 
 | |
|             OutputHeader.RenderInfoSize = (uint)Unsafe.SizeOf<RendererInfoOutStatus>();
 | |
|             OutputHeader.TotalSize += OutputHeader.RenderInfoSize;
 | |
| 
 | |
|             return ResultCode.Success;
 | |
|         }
 | |
| 
 | |
|         public ResultCode CheckConsumedSize()
 | |
|         {
 | |
|             int consumedInputSize = _inputOrigin.Length - _input.Length;
 | |
|             int consumedOutputSize = _outputOrigin.Length - _output.Length;
 | |
| 
 | |
|             if (consumedInputSize != _inputHeader.TotalSize)
 | |
|             {
 | |
|                 Logger.Error?.Print(LogClass.AudioRenderer, $"Consumed input size mismatch (got {consumedInputSize} expected {_inputHeader.TotalSize})");
 | |
| 
 | |
|                 return ResultCode.InvalidUpdateInfo;
 | |
|             }
 | |
| 
 | |
|             if (consumedOutputSize != OutputHeader.TotalSize)
 | |
|             {
 | |
|                 Logger.Error?.Print(LogClass.AudioRenderer, $"Consumed output size mismatch (got {consumedOutputSize} expected {OutputHeader.TotalSize})");
 | |
| 
 | |
|                 return ResultCode.InvalidUpdateInfo;
 | |
|             }
 | |
| 
 | |
|             return ResultCode.Success;
 | |
|         }
 | |
|     }
 | |
| } | 
