forked from Mirror/Ryujinx
9444b4a647
* Implement HwOpus multistream functions * Avoid one copy
116 lines
No EOL
4.9 KiB
C#
116 lines
No EOL
4.9 KiB
C#
using Ryujinx.HLE.HOS.Services.Audio.Types;
|
|
using System;
|
|
using System.Runtime.InteropServices;
|
|
|
|
namespace Ryujinx.HLE.HOS.Services.Audio.HardwareOpusDecoderManager
|
|
{
|
|
class IHardwareOpusDecoder : IpcService
|
|
{
|
|
private readonly IDecoder _decoder;
|
|
private readonly OpusDecoderFlags _flags;
|
|
|
|
public IHardwareOpusDecoder(int sampleRate, int channelsCount, OpusDecoderFlags flags)
|
|
{
|
|
_decoder = new Decoder(sampleRate, channelsCount);
|
|
_flags = flags;
|
|
}
|
|
|
|
public IHardwareOpusDecoder(int sampleRate, int channelsCount, int streams, int coupledStreams, OpusDecoderFlags flags, byte[] mapping)
|
|
{
|
|
_decoder = new MultiSampleDecoder(sampleRate, channelsCount, streams, coupledStreams, mapping);
|
|
_flags = flags;
|
|
}
|
|
|
|
[CommandHipc(0)]
|
|
// DecodeInterleavedOld(buffer<unknown, 5>) -> (u32, u32, buffer<unknown, 6>)
|
|
public ResultCode DecodeInterleavedOld(ServiceCtx context)
|
|
{
|
|
return DecodeInterleavedInternal(context, OpusDecoderFlags.None, reset: false, withPerf: false);
|
|
}
|
|
|
|
[CommandHipc(2)]
|
|
// DecodeInterleavedForMultiStreamOld(buffer<unknown, 5>) -> (u32, u32, buffer<unknown, 6>)
|
|
public ResultCode DecodeInterleavedForMultiStreamOld(ServiceCtx context)
|
|
{
|
|
return DecodeInterleavedInternal(context, OpusDecoderFlags.None, reset: false, withPerf: false);
|
|
}
|
|
|
|
[CommandHipc(4)] // 6.0.0+
|
|
// DecodeInterleavedWithPerfOld(buffer<unknown, 5>) -> (u32, u32, u64, buffer<unknown, 0x46>)
|
|
public ResultCode DecodeInterleavedWithPerfOld(ServiceCtx context)
|
|
{
|
|
return DecodeInterleavedInternal(context, OpusDecoderFlags.None, reset: false, withPerf: true);
|
|
}
|
|
|
|
[CommandHipc(5)] // 6.0.0+
|
|
// DecodeInterleavedForMultiStreamWithPerfOld(buffer<unknown, 5>) -> (u32, u32, u64, buffer<unknown, 0x46>)
|
|
public ResultCode DecodeInterleavedForMultiStreamWithPerfOld(ServiceCtx context)
|
|
{
|
|
return DecodeInterleavedInternal(context, OpusDecoderFlags.None, reset: false, withPerf: true);
|
|
}
|
|
|
|
[CommandHipc(6)] // 6.0.0+
|
|
// DecodeInterleavedWithPerfAndResetOld(bool reset, buffer<unknown, 5>) -> (u32, u32, u64, buffer<unknown, 0x46>)
|
|
public ResultCode DecodeInterleavedWithPerfAndResetOld(ServiceCtx context)
|
|
{
|
|
bool reset = context.RequestData.ReadBoolean();
|
|
|
|
return DecodeInterleavedInternal(context, OpusDecoderFlags.None, reset, withPerf: true);
|
|
}
|
|
|
|
[CommandHipc(7)] // 6.0.0+
|
|
// DecodeInterleavedForMultiStreamWithPerfAndResetOld(bool reset, buffer<unknown, 5>) -> (u32, u32, u64, buffer<unknown, 0x46>)
|
|
public ResultCode DecodeInterleavedForMultiStreamWithPerfAndResetOld(ServiceCtx context)
|
|
{
|
|
bool reset = context.RequestData.ReadBoolean();
|
|
|
|
return DecodeInterleavedInternal(context, OpusDecoderFlags.None, reset, withPerf: true);
|
|
}
|
|
|
|
[CommandHipc(8)] // 7.0.0+
|
|
// DecodeInterleaved(bool reset, buffer<unknown, 0x45>) -> (u32, u32, u64, buffer<unknown, 0x46>)
|
|
public ResultCode DecodeInterleaved(ServiceCtx context)
|
|
{
|
|
bool reset = context.RequestData.ReadBoolean();
|
|
|
|
return DecodeInterleavedInternal(context, _flags, reset, withPerf: true);
|
|
}
|
|
|
|
[CommandHipc(9)] // 7.0.0+
|
|
// DecodeInterleavedForMultiStream(bool reset, buffer<unknown, 0x45>) -> (u32, u32, u64, buffer<unknown, 0x46>)
|
|
public ResultCode DecodeInterleavedForMultiStream(ServiceCtx context)
|
|
{
|
|
bool reset = context.RequestData.ReadBoolean();
|
|
|
|
return DecodeInterleavedInternal(context, _flags, reset, withPerf: true);
|
|
}
|
|
|
|
private ResultCode DecodeInterleavedInternal(ServiceCtx context, OpusDecoderFlags flags, bool reset, bool withPerf)
|
|
{
|
|
ulong inPosition = context.Request.SendBuff[0].Position;
|
|
ulong inSize = context.Request.SendBuff[0].Size;
|
|
ulong outputPosition = context.Request.ReceiveBuff[0].Position;
|
|
ulong outputSize = context.Request.ReceiveBuff[0].Size;
|
|
|
|
ReadOnlySpan<byte> input = context.Memory.GetSpan(inPosition, (int)inSize);
|
|
|
|
ResultCode result = _decoder.DecodeInterleaved(reset, input, out short[] outPcmData, outputSize, out uint outConsumed, out int outSamples);
|
|
|
|
if (result == ResultCode.Success)
|
|
{
|
|
context.Memory.Write(outputPosition, MemoryMarshal.Cast<short, byte>(outPcmData.AsSpan()));
|
|
|
|
context.ResponseData.Write(outConsumed);
|
|
context.ResponseData.Write(outSamples);
|
|
|
|
if (withPerf)
|
|
{
|
|
// This is the time the DSP took to process the request, TODO: fill this.
|
|
context.ResponseData.Write(0UL);
|
|
}
|
|
}
|
|
|
|
return result;
|
|
}
|
|
}
|
|
} |