2018-10-17 18:15:50 +01:00
|
|
|
using Ryujinx.Common.Logging;
|
2018-08-17 00:47:36 +01:00
|
|
|
using Ryujinx.HLE.HOS.Ipc;
|
|
|
|
using Ryujinx.HLE.HOS.Kernel;
|
|
|
|
using Ryujinx.HLE.HOS.SystemState;
|
2018-09-23 19:11:46 +01:00
|
|
|
using System;
|
2018-03-12 19:29:06 +00:00
|
|
|
using System.Collections.Generic;
|
|
|
|
using System.Text;
|
|
|
|
|
2018-08-17 00:47:36 +01:00
|
|
|
namespace Ryujinx.HLE.HOS.Services.Aud
|
2018-03-12 19:29:06 +00:00
|
|
|
{
|
2018-04-22 05:21:49 +01:00
|
|
|
class IAudioDevice : IpcService
|
2018-03-12 19:29:06 +00:00
|
|
|
{
|
|
|
|
private Dictionary<int, ServiceProcessRequest> m_Commands;
|
|
|
|
|
2018-03-19 18:58:46 +00:00
|
|
|
public override IReadOnlyDictionary<int, ServiceProcessRequest> Commands => m_Commands;
|
2018-03-12 19:29:06 +00:00
|
|
|
|
2018-04-18 02:52:20 +01:00
|
|
|
private KEvent SystemEvent;
|
|
|
|
|
2018-09-19 00:36:43 +01:00
|
|
|
public IAudioDevice(Horizon System)
|
2018-03-12 19:29:06 +00:00
|
|
|
{
|
|
|
|
m_Commands = new Dictionary<int, ServiceProcessRequest>()
|
|
|
|
{
|
2018-06-02 23:46:09 +01:00
|
|
|
{ 0, ListAudioDeviceName },
|
|
|
|
{ 1, SetAudioDeviceOutputVolume },
|
|
|
|
{ 3, GetActiveAudioDeviceName },
|
|
|
|
{ 4, QueryAudioDeviceSystemEvent },
|
|
|
|
{ 5, GetActiveChannelCount },
|
|
|
|
{ 6, ListAudioDeviceNameAuto },
|
|
|
|
{ 7, SetAudioDeviceOutputVolumeAuto },
|
|
|
|
{ 8, GetAudioDeviceOutputVolumeAuto },
|
|
|
|
{ 10, GetActiveAudioDeviceNameAuto },
|
|
|
|
{ 11, QueryAudioDeviceInputEvent },
|
|
|
|
{ 12, QueryAudioDeviceOutputEvent }
|
2018-03-12 19:29:06 +00:00
|
|
|
};
|
2018-04-18 02:52:20 +01:00
|
|
|
|
2018-09-19 00:36:43 +01:00
|
|
|
SystemEvent = new KEvent(System);
|
2018-04-19 04:19:22 +01:00
|
|
|
|
2018-04-18 02:52:20 +01:00
|
|
|
//TODO: We shouldn't be signaling this here.
|
2018-09-23 19:11:46 +01:00
|
|
|
SystemEvent.ReadableEvent.Signal();
|
2018-03-12 19:29:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public long ListAudioDeviceName(ServiceCtx Context)
|
|
|
|
{
|
2018-04-24 21:14:26 +01:00
|
|
|
string[] DeviceNames = SystemStateMgr.AudioOutputs;
|
2018-03-12 19:29:06 +00:00
|
|
|
|
2018-04-24 21:14:26 +01:00
|
|
|
Context.ResponseData.Write(DeviceNames.Length);
|
2018-03-12 19:29:06 +00:00
|
|
|
|
|
|
|
long Position = Context.Request.ReceiveBuff[0].Position;
|
|
|
|
long Size = Context.Request.ReceiveBuff[0].Size;
|
|
|
|
|
|
|
|
long BasePosition = Position;
|
|
|
|
|
2018-04-24 21:14:26 +01:00
|
|
|
foreach (string Name in DeviceNames)
|
2018-03-12 19:29:06 +00:00
|
|
|
{
|
2018-04-26 15:34:40 +01:00
|
|
|
byte[] Buffer = Encoding.ASCII.GetBytes(Name + "\0");
|
2018-03-12 19:29:06 +00:00
|
|
|
|
|
|
|
if ((Position - BasePosition) + Buffer.Length > Size)
|
|
|
|
{
|
2018-10-17 18:15:50 +01:00
|
|
|
Logger.PrintError(LogClass.ServiceAudio, $"Output buffer size {Size} too small!");
|
2018-04-24 21:14:26 +01:00
|
|
|
|
2018-03-12 19:29:06 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2018-06-09 17:05:41 +01:00
|
|
|
Context.Memory.WriteBytes(Position, Buffer);
|
2018-03-12 19:29:06 +00:00
|
|
|
|
|
|
|
Position += Buffer.Length;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
public long SetAudioDeviceOutputVolume(ServiceCtx Context)
|
|
|
|
{
|
|
|
|
float Volume = Context.RequestData.ReadSingle();
|
|
|
|
|
|
|
|
long Position = Context.Request.SendBuff[0].Position;
|
|
|
|
long Size = Context.Request.SendBuff[0].Size;
|
|
|
|
|
2018-06-09 01:15:02 +01:00
|
|
|
byte[] DeviceNameBuffer = Context.Memory.ReadBytes(Position, Size);
|
2018-04-24 21:14:26 +01:00
|
|
|
|
2018-04-26 15:34:40 +01:00
|
|
|
string DeviceName = Encoding.ASCII.GetString(DeviceNameBuffer);
|
2018-03-12 19:29:06 +00:00
|
|
|
|
2018-10-17 18:15:50 +01:00
|
|
|
Logger.PrintStub(LogClass.ServiceAudio, "Stubbed.");
|
2018-04-17 01:24:42 +01:00
|
|
|
|
2018-03-12 19:29:06 +00:00
|
|
|
return 0;
|
|
|
|
}
|
2018-04-18 02:52:20 +01:00
|
|
|
|
2018-04-22 05:21:49 +01:00
|
|
|
public long GetActiveAudioDeviceName(ServiceCtx Context)
|
|
|
|
{
|
2018-08-17 00:47:36 +01:00
|
|
|
string Name = Context.Device.System.State.ActiveAudioOutput;
|
2018-04-22 05:21:49 +01:00
|
|
|
|
|
|
|
long Position = Context.Request.ReceiveBuff[0].Position;
|
|
|
|
long Size = Context.Request.ReceiveBuff[0].Size;
|
|
|
|
|
2018-04-26 15:34:40 +01:00
|
|
|
byte[] DeviceNameBuffer = Encoding.ASCII.GetBytes(Name + "\0");
|
2018-04-22 05:21:49 +01:00
|
|
|
|
2018-04-24 21:14:26 +01:00
|
|
|
if ((ulong)DeviceNameBuffer.Length <= (ulong)Size)
|
|
|
|
{
|
2018-06-09 17:05:41 +01:00
|
|
|
Context.Memory.WriteBytes(Position, DeviceNameBuffer);
|
2018-04-24 21:14:26 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2018-10-17 18:15:50 +01:00
|
|
|
Logger.PrintError(LogClass.ServiceAudio, $"Output buffer size {Size} too small!");
|
2018-04-24 21:14:26 +01:00
|
|
|
}
|
2018-04-22 05:21:49 +01:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-04-18 02:52:20 +01:00
|
|
|
public long QueryAudioDeviceSystemEvent(ServiceCtx Context)
|
|
|
|
{
|
2018-09-23 19:11:46 +01:00
|
|
|
if (Context.Process.HandleTable.GenerateHandle(SystemEvent.ReadableEvent, out int Handle) != KernelResult.Success)
|
|
|
|
{
|
|
|
|
throw new InvalidOperationException("Out of handles!");
|
|
|
|
}
|
2018-04-18 02:52:20 +01:00
|
|
|
|
|
|
|
Context.Response.HandleDesc = IpcHandleDesc.MakeCopy(Handle);
|
|
|
|
|
2018-10-17 18:15:50 +01:00
|
|
|
Logger.PrintStub(LogClass.ServiceAudio, "Stubbed.");
|
2018-04-18 02:52:20 +01:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
public long GetActiveChannelCount(ServiceCtx Context)
|
|
|
|
{
|
|
|
|
Context.ResponseData.Write(2);
|
|
|
|
|
2018-10-17 18:15:50 +01:00
|
|
|
Logger.PrintStub(LogClass.ServiceAudio, "Stubbed.");
|
2018-04-18 02:52:20 +01:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2018-06-02 23:46:09 +01:00
|
|
|
|
|
|
|
public long ListAudioDeviceNameAuto(ServiceCtx Context)
|
|
|
|
{
|
|
|
|
string[] DeviceNames = SystemStateMgr.AudioOutputs;
|
|
|
|
|
|
|
|
Context.ResponseData.Write(DeviceNames.Length);
|
|
|
|
|
|
|
|
(long Position, long Size) = Context.Request.GetBufferType0x22();
|
|
|
|
|
|
|
|
long BasePosition = Position;
|
|
|
|
|
|
|
|
foreach (string Name in DeviceNames)
|
|
|
|
{
|
|
|
|
byte[] Buffer = Encoding.UTF8.GetBytes(Name + '\0');
|
|
|
|
|
|
|
|
if ((Position - BasePosition) + Buffer.Length > Size)
|
|
|
|
{
|
2018-10-17 18:15:50 +01:00
|
|
|
Logger.PrintError(LogClass.ServiceAudio, $"Output buffer size {Size} too small!");
|
2018-06-02 23:46:09 +01:00
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2018-06-09 17:05:41 +01:00
|
|
|
Context.Memory.WriteBytes(Position, Buffer);
|
2018-06-02 23:46:09 +01:00
|
|
|
|
|
|
|
Position += Buffer.Length;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
public long SetAudioDeviceOutputVolumeAuto(ServiceCtx Context)
|
|
|
|
{
|
|
|
|
float Volume = Context.RequestData.ReadSingle();
|
|
|
|
|
2018-06-04 06:09:41 +01:00
|
|
|
(long Position, long Size) = Context.Request.GetBufferType0x21();
|
2018-06-02 23:46:09 +01:00
|
|
|
|
2018-06-09 01:15:02 +01:00
|
|
|
byte[] DeviceNameBuffer = Context.Memory.ReadBytes(Position, Size);
|
2018-06-02 23:46:09 +01:00
|
|
|
|
|
|
|
string DeviceName = Encoding.UTF8.GetString(DeviceNameBuffer);
|
|
|
|
|
2018-10-17 18:15:50 +01:00
|
|
|
Logger.PrintStub(LogClass.ServiceAudio, "Stubbed.");
|
2018-06-02 23:46:09 +01:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
public long GetAudioDeviceOutputVolumeAuto(ServiceCtx Context)
|
|
|
|
{
|
|
|
|
Context.ResponseData.Write(1f);
|
|
|
|
|
2018-10-17 18:15:50 +01:00
|
|
|
Logger.PrintStub(LogClass.ServiceAudio, "Stubbed.");
|
2018-06-02 23:46:09 +01:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
public long GetActiveAudioDeviceNameAuto(ServiceCtx Context)
|
|
|
|
{
|
2018-08-17 00:47:36 +01:00
|
|
|
string Name = Context.Device.System.State.ActiveAudioOutput;
|
2018-06-02 23:46:09 +01:00
|
|
|
|
2018-06-04 06:09:41 +01:00
|
|
|
(long Position, long Size) = Context.Request.GetBufferType0x22();
|
2018-06-02 23:46:09 +01:00
|
|
|
|
|
|
|
byte[] DeviceNameBuffer = Encoding.UTF8.GetBytes(Name + '\0');
|
|
|
|
|
|
|
|
if ((ulong)DeviceNameBuffer.Length <= (ulong)Size)
|
|
|
|
{
|
2018-06-09 17:05:41 +01:00
|
|
|
Context.Memory.WriteBytes(Position, DeviceNameBuffer);
|
2018-06-02 23:46:09 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2018-10-17 18:15:50 +01:00
|
|
|
Logger.PrintError(LogClass.ServiceAudio, $"Output buffer size {Size} too small!");
|
2018-06-02 23:46:09 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
public long QueryAudioDeviceInputEvent(ServiceCtx Context)
|
|
|
|
{
|
2018-09-23 19:11:46 +01:00
|
|
|
if (Context.Process.HandleTable.GenerateHandle(SystemEvent.ReadableEvent, out int Handle) != KernelResult.Success)
|
|
|
|
{
|
|
|
|
throw new InvalidOperationException("Out of handles!");
|
|
|
|
}
|
2018-06-02 23:46:09 +01:00
|
|
|
|
|
|
|
Context.Response.HandleDesc = IpcHandleDesc.MakeCopy(Handle);
|
|
|
|
|
2018-10-17 18:15:50 +01:00
|
|
|
Logger.PrintStub(LogClass.ServiceAudio, "Stubbed.");
|
2018-06-02 23:46:09 +01:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
public long QueryAudioDeviceOutputEvent(ServiceCtx Context)
|
|
|
|
{
|
2018-09-23 19:11:46 +01:00
|
|
|
if (Context.Process.HandleTable.GenerateHandle(SystemEvent.ReadableEvent, out int Handle) != KernelResult.Success)
|
|
|
|
{
|
|
|
|
throw new InvalidOperationException("Out of handles!");
|
|
|
|
}
|
2018-06-02 23:46:09 +01:00
|
|
|
|
|
|
|
Context.Response.HandleDesc = IpcHandleDesc.MakeCopy(Handle);
|
|
|
|
|
2018-10-17 18:15:50 +01:00
|
|
|
Logger.PrintStub(LogClass.ServiceAudio, "Stubbed.");
|
2018-06-02 23:46:09 +01:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2018-03-12 19:29:06 +00:00
|
|
|
}
|
2018-04-18 02:52:20 +01:00
|
|
|
}
|