9 changed files with 630 additions and 562 deletions
-
12src/core/CMakeLists.txt
-
5src/core/hle/service/audio/audio.cpp
-
183src/core/hle/service/audio/audio_device.cpp
-
35src/core/hle/service/audio/audio_device.h
-
210src/core/hle/service/audio/audio_renderer.cpp
-
45src/core/hle/service/audio/audio_renderer.h
-
143src/core/hle/service/audio/audio_renderer_manager.cpp
-
7src/core/hle/service/audio/audio_renderer_manager.h
-
552src/core/hle/service/audio/audren_u.cpp
@ -0,0 +1,183 @@ |
|||||
|
// SPDX-FileCopyrightText: Copyright 2024 yuzu Emulator Project
|
||||
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||
|
|
||||
|
#include "audio_core/audio_core.h"
|
||||
|
#include "common/string_util.h"
|
||||
|
#include "core/hle/service/audio/audio_device.h"
|
||||
|
#include "core/hle/service/ipc_helpers.h"
|
||||
|
|
||||
|
namespace Service::Audio { |
||||
|
using namespace AudioCore::Renderer; |
||||
|
|
||||
|
IAudioDevice::IAudioDevice(Core::System& system_, u64 applet_resource_user_id, u32 revision, |
||||
|
u32 device_num) |
||||
|
: ServiceFramework{system_, "IAudioDevice"}, service_context{system_, "IAudioDevice"}, |
||||
|
impl{std::make_unique<AudioDevice>(system_, applet_resource_user_id, revision)}, |
||||
|
event{service_context.CreateEvent(fmt::format("IAudioDeviceEvent-{}", device_num))} { |
||||
|
static const FunctionInfo functions[] = { |
||||
|
{0, &IAudioDevice::ListAudioDeviceName, "ListAudioDeviceName"}, |
||||
|
{1, &IAudioDevice::SetAudioDeviceOutputVolume, "SetAudioDeviceOutputVolume"}, |
||||
|
{2, &IAudioDevice::GetAudioDeviceOutputVolume, "GetAudioDeviceOutputVolume"}, |
||||
|
{3, &IAudioDevice::GetActiveAudioDeviceName, "GetActiveAudioDeviceName"}, |
||||
|
{4, &IAudioDevice::QueryAudioDeviceSystemEvent, "QueryAudioDeviceSystemEvent"}, |
||||
|
{5, &IAudioDevice::GetActiveChannelCount, "GetActiveChannelCount"}, |
||||
|
{6, &IAudioDevice::ListAudioDeviceName, "ListAudioDeviceNameAuto"}, |
||||
|
{7, &IAudioDevice::SetAudioDeviceOutputVolume, "SetAudioDeviceOutputVolumeAuto"}, |
||||
|
{8, &IAudioDevice::GetAudioDeviceOutputVolume, "GetAudioDeviceOutputVolumeAuto"}, |
||||
|
{10, &IAudioDevice::GetActiveAudioDeviceName, "GetActiveAudioDeviceNameAuto"}, |
||||
|
{11, &IAudioDevice::QueryAudioDeviceInputEvent, "QueryAudioDeviceInputEvent"}, |
||||
|
{12, &IAudioDevice::QueryAudioDeviceOutputEvent, "QueryAudioDeviceOutputEvent"}, |
||||
|
{13, &IAudioDevice::GetActiveAudioDeviceName, "GetActiveAudioOutputDeviceName"}, |
||||
|
{14, &IAudioDevice::ListAudioOutputDeviceName, "ListAudioOutputDeviceName"}, |
||||
|
}; |
||||
|
RegisterHandlers(functions); |
||||
|
|
||||
|
event->Signal(); |
||||
|
} |
||||
|
|
||||
|
IAudioDevice::~IAudioDevice() { |
||||
|
service_context.CloseEvent(event); |
||||
|
} |
||||
|
|
||||
|
void IAudioDevice::ListAudioDeviceName(HLERequestContext& ctx) { |
||||
|
const size_t in_count = ctx.GetWriteBufferNumElements<AudioDevice::AudioDeviceName>(); |
||||
|
|
||||
|
std::vector<AudioDevice::AudioDeviceName> out_names{}; |
||||
|
|
||||
|
const u32 out_count = impl->ListAudioDeviceName(out_names, in_count); |
||||
|
|
||||
|
std::string out{}; |
||||
|
for (u32 i = 0; i < out_count; i++) { |
||||
|
std::string a{}; |
||||
|
u32 j = 0; |
||||
|
while (out_names[i].name[j] != '\0') { |
||||
|
a += out_names[i].name[j]; |
||||
|
j++; |
||||
|
} |
||||
|
out += "\n\t" + a; |
||||
|
} |
||||
|
|
||||
|
LOG_DEBUG(Service_Audio, "called.\nNames={}", out); |
||||
|
|
||||
|
IPC::ResponseBuilder rb{ctx, 3}; |
||||
|
|
||||
|
ctx.WriteBuffer(out_names); |
||||
|
|
||||
|
rb.Push(ResultSuccess); |
||||
|
rb.Push(out_count); |
||||
|
} |
||||
|
|
||||
|
void IAudioDevice::SetAudioDeviceOutputVolume(HLERequestContext& ctx) { |
||||
|
IPC::RequestParser rp{ctx}; |
||||
|
const f32 volume = rp.Pop<f32>(); |
||||
|
|
||||
|
const auto device_name_buffer = ctx.ReadBuffer(); |
||||
|
const std::string name = Common::StringFromBuffer(device_name_buffer); |
||||
|
|
||||
|
LOG_DEBUG(Service_Audio, "called. name={}, volume={}", name, volume); |
||||
|
|
||||
|
if (name == "AudioTvOutput") { |
||||
|
impl->SetDeviceVolumes(volume); |
||||
|
} |
||||
|
|
||||
|
IPC::ResponseBuilder rb{ctx, 2}; |
||||
|
rb.Push(ResultSuccess); |
||||
|
} |
||||
|
|
||||
|
void IAudioDevice::GetAudioDeviceOutputVolume(HLERequestContext& ctx) { |
||||
|
const auto device_name_buffer = ctx.ReadBuffer(); |
||||
|
const std::string name = Common::StringFromBuffer(device_name_buffer); |
||||
|
|
||||
|
LOG_DEBUG(Service_Audio, "called. Name={}", name); |
||||
|
|
||||
|
f32 volume{1.0f}; |
||||
|
if (name == "AudioTvOutput") { |
||||
|
volume = impl->GetDeviceVolume(name); |
||||
|
} |
||||
|
|
||||
|
IPC::ResponseBuilder rb{ctx, 3}; |
||||
|
rb.Push(ResultSuccess); |
||||
|
rb.Push(volume); |
||||
|
} |
||||
|
|
||||
|
void IAudioDevice::GetActiveAudioDeviceName(HLERequestContext& ctx) { |
||||
|
const auto write_size = ctx.GetWriteBufferSize(); |
||||
|
std::string out_name{"AudioTvOutput"}; |
||||
|
|
||||
|
LOG_DEBUG(Service_Audio, "(STUBBED) called. Name={}", out_name); |
||||
|
|
||||
|
out_name.resize(write_size); |
||||
|
|
||||
|
ctx.WriteBuffer(out_name); |
||||
|
|
||||
|
IPC::ResponseBuilder rb{ctx, 2}; |
||||
|
rb.Push(ResultSuccess); |
||||
|
} |
||||
|
|
||||
|
void IAudioDevice::QueryAudioDeviceSystemEvent(HLERequestContext& ctx) { |
||||
|
LOG_DEBUG(Service_Audio, "(STUBBED) called"); |
||||
|
|
||||
|
event->Signal(); |
||||
|
|
||||
|
IPC::ResponseBuilder rb{ctx, 2, 1}; |
||||
|
rb.Push(ResultSuccess); |
||||
|
rb.PushCopyObjects(event->GetReadableEvent()); |
||||
|
} |
||||
|
|
||||
|
void IAudioDevice::GetActiveChannelCount(HLERequestContext& ctx) { |
||||
|
const auto& sink{system.AudioCore().GetOutputSink()}; |
||||
|
u32 channel_count{sink.GetSystemChannels()}; |
||||
|
|
||||
|
LOG_DEBUG(Service_Audio, "(STUBBED) called. Channels={}", channel_count); |
||||
|
|
||||
|
IPC::ResponseBuilder rb{ctx, 3}; |
||||
|
|
||||
|
rb.Push(ResultSuccess); |
||||
|
rb.Push<u32>(channel_count); |
||||
|
} |
||||
|
|
||||
|
void IAudioDevice::QueryAudioDeviceInputEvent(HLERequestContext& ctx) { |
||||
|
LOG_DEBUG(Service_Audio, "(STUBBED) called"); |
||||
|
|
||||
|
IPC::ResponseBuilder rb{ctx, 2, 1}; |
||||
|
rb.Push(ResultSuccess); |
||||
|
rb.PushCopyObjects(event->GetReadableEvent()); |
||||
|
} |
||||
|
|
||||
|
void IAudioDevice::QueryAudioDeviceOutputEvent(HLERequestContext& ctx) { |
||||
|
LOG_DEBUG(Service_Audio, "called"); |
||||
|
|
||||
|
IPC::ResponseBuilder rb{ctx, 2, 1}; |
||||
|
rb.Push(ResultSuccess); |
||||
|
rb.PushCopyObjects(event->GetReadableEvent()); |
||||
|
} |
||||
|
|
||||
|
void IAudioDevice::ListAudioOutputDeviceName(HLERequestContext& ctx) { |
||||
|
const size_t in_count = ctx.GetWriteBufferNumElements<AudioDevice::AudioDeviceName>(); |
||||
|
|
||||
|
std::vector<AudioDevice::AudioDeviceName> out_names{}; |
||||
|
|
||||
|
const u32 out_count = impl->ListAudioOutputDeviceName(out_names, in_count); |
||||
|
|
||||
|
std::string out{}; |
||||
|
for (u32 i = 0; i < out_count; i++) { |
||||
|
std::string a{}; |
||||
|
u32 j = 0; |
||||
|
while (out_names[i].name[j] != '\0') { |
||||
|
a += out_names[i].name[j]; |
||||
|
j++; |
||||
|
} |
||||
|
out += "\n\t" + a; |
||||
|
} |
||||
|
|
||||
|
LOG_DEBUG(Service_Audio, "called.\nNames={}", out); |
||||
|
|
||||
|
IPC::ResponseBuilder rb{ctx, 3}; |
||||
|
|
||||
|
ctx.WriteBuffer(out_names); |
||||
|
|
||||
|
rb.Push(ResultSuccess); |
||||
|
rb.Push(out_count); |
||||
|
} |
||||
|
|
||||
|
} // namespace Service::Audio
|
||||
@ -0,0 +1,35 @@ |
|||||
|
// SPDX-FileCopyrightText: Copyright 2024 yuzu Emulator Project |
||||
|
// SPDX-License-Identifier: GPL-2.0-or-later |
||||
|
|
||||
|
#pragma once |
||||
|
|
||||
|
#include "audio_core/renderer/audio_device.h" |
||||
|
#include "core/hle/service/kernel_helpers.h" |
||||
|
#include "core/hle/service/service.h" |
||||
|
|
||||
|
namespace Service::Audio { |
||||
|
|
||||
|
class IAudioDevice final : public ServiceFramework<IAudioDevice> { |
||||
|
|
||||
|
public: |
||||
|
explicit IAudioDevice(Core::System& system_, u64 applet_resource_user_id, u32 revision, |
||||
|
u32 device_num); |
||||
|
~IAudioDevice() override; |
||||
|
|
||||
|
private: |
||||
|
void ListAudioDeviceName(HLERequestContext& ctx); |
||||
|
void SetAudioDeviceOutputVolume(HLERequestContext& ctx); |
||||
|
void GetAudioDeviceOutputVolume(HLERequestContext& ctx); |
||||
|
void GetActiveAudioDeviceName(HLERequestContext& ctx); |
||||
|
void QueryAudioDeviceSystemEvent(HLERequestContext& ctx); |
||||
|
void GetActiveChannelCount(HLERequestContext& ctx); |
||||
|
void QueryAudioDeviceInputEvent(HLERequestContext& ctx); |
||||
|
void QueryAudioDeviceOutputEvent(HLERequestContext& ctx); |
||||
|
void ListAudioOutputDeviceName(HLERequestContext& ctx); |
||||
|
|
||||
|
KernelHelpers::ServiceContext service_context; |
||||
|
std::unique_ptr<AudioCore::Renderer::AudioDevice> impl; |
||||
|
Kernel::KEvent* event; |
||||
|
}; |
||||
|
|
||||
|
} // namespace Service::Audio |
||||
@ -0,0 +1,210 @@ |
|||||
|
// SPDX-FileCopyrightText: Copyright 2024 yuzu Emulator Project
|
||||
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||
|
|
||||
|
#include "core/hle/service/audio/audio_renderer.h"
|
||||
|
#include "core/hle/service/ipc_helpers.h"
|
||||
|
|
||||
|
namespace Service::Audio { |
||||
|
using namespace AudioCore::Renderer; |
||||
|
|
||||
|
IAudioRenderer::IAudioRenderer(Core::System& system_, Manager& manager_, |
||||
|
AudioCore::AudioRendererParameterInternal& params, |
||||
|
Kernel::KTransferMemory* transfer_memory, u64 transfer_memory_size, |
||||
|
u32 process_handle, Kernel::KProcess& process_, |
||||
|
u64 applet_resource_user_id, s32 session_id) |
||||
|
: ServiceFramework{system_, "IAudioRenderer"}, service_context{system_, "IAudioRenderer"}, |
||||
|
rendered_event{service_context.CreateEvent("IAudioRendererEvent")}, manager{manager_}, |
||||
|
impl{std::make_unique<Renderer>(system_, manager, rendered_event)}, process{process_} { |
||||
|
// clang-format off
|
||||
|
static const FunctionInfo functions[] = { |
||||
|
{0, &IAudioRenderer::GetSampleRate, "GetSampleRate"}, |
||||
|
{1, &IAudioRenderer::GetSampleCount, "GetSampleCount"}, |
||||
|
{2, &IAudioRenderer::GetMixBufferCount, "GetMixBufferCount"}, |
||||
|
{3, &IAudioRenderer::GetState, "GetState"}, |
||||
|
{4, &IAudioRenderer::RequestUpdate, "RequestUpdate"}, |
||||
|
{5, &IAudioRenderer::Start, "Start"}, |
||||
|
{6, &IAudioRenderer::Stop, "Stop"}, |
||||
|
{7, &IAudioRenderer::QuerySystemEvent, "QuerySystemEvent"}, |
||||
|
{8, &IAudioRenderer::SetRenderingTimeLimit, "SetRenderingTimeLimit"}, |
||||
|
{9, &IAudioRenderer::GetRenderingTimeLimit, "GetRenderingTimeLimit"}, |
||||
|
{10, &IAudioRenderer::RequestUpdate, "RequestUpdateAuto"}, |
||||
|
{11, nullptr, "ExecuteAudioRendererRendering"}, |
||||
|
{12, &IAudioRenderer::SetVoiceDropParameter, "SetVoiceDropParameter"}, |
||||
|
{13, &IAudioRenderer::GetVoiceDropParameter, "GetVoiceDropParameter"}, |
||||
|
}; |
||||
|
// clang-format on
|
||||
|
RegisterHandlers(functions); |
||||
|
|
||||
|
process.Open(); |
||||
|
impl->Initialize(params, transfer_memory, transfer_memory_size, process_handle, process, |
||||
|
applet_resource_user_id, session_id); |
||||
|
} |
||||
|
|
||||
|
IAudioRenderer::~IAudioRenderer() { |
||||
|
impl->Finalize(); |
||||
|
service_context.CloseEvent(rendered_event); |
||||
|
process.Close(); |
||||
|
} |
||||
|
|
||||
|
void IAudioRenderer::GetSampleRate(HLERequestContext& ctx) { |
||||
|
const auto sample_rate{impl->GetSystem().GetSampleRate()}; |
||||
|
|
||||
|
LOG_DEBUG(Service_Audio, "called. Sample rate {}", sample_rate); |
||||
|
|
||||
|
IPC::ResponseBuilder rb{ctx, 3}; |
||||
|
rb.Push(ResultSuccess); |
||||
|
rb.Push(sample_rate); |
||||
|
} |
||||
|
|
||||
|
void IAudioRenderer::GetSampleCount(HLERequestContext& ctx) { |
||||
|
const auto sample_count{impl->GetSystem().GetSampleCount()}; |
||||
|
|
||||
|
LOG_DEBUG(Service_Audio, "called. Sample count {}", sample_count); |
||||
|
|
||||
|
IPC::ResponseBuilder rb{ctx, 3}; |
||||
|
rb.Push(ResultSuccess); |
||||
|
rb.Push(sample_count); |
||||
|
} |
||||
|
|
||||
|
void IAudioRenderer::GetState(HLERequestContext& ctx) { |
||||
|
const u32 state{!impl->GetSystem().IsActive()}; |
||||
|
|
||||
|
LOG_DEBUG(Service_Audio, "called, state {}", state); |
||||
|
|
||||
|
IPC::ResponseBuilder rb{ctx, 3}; |
||||
|
rb.Push(ResultSuccess); |
||||
|
rb.Push(state); |
||||
|
} |
||||
|
|
||||
|
void IAudioRenderer::GetMixBufferCount(HLERequestContext& ctx) { |
||||
|
LOG_DEBUG(Service_Audio, "called"); |
||||
|
|
||||
|
const auto buffer_count{impl->GetSystem().GetMixBufferCount()}; |
||||
|
|
||||
|
IPC::ResponseBuilder rb{ctx, 3}; |
||||
|
rb.Push(ResultSuccess); |
||||
|
rb.Push(buffer_count); |
||||
|
} |
||||
|
|
||||
|
void IAudioRenderer::RequestUpdate(HLERequestContext& ctx) { |
||||
|
LOG_TRACE(Service_Audio, "called"); |
||||
|
|
||||
|
const auto input{ctx.ReadBuffer(0)}; |
||||
|
|
||||
|
// These buffers are written manually to avoid an issue with WriteBuffer throwing errors for
|
||||
|
// checking size 0. Performance size is 0 for most games.
|
||||
|
|
||||
|
auto is_buffer_b{ctx.BufferDescriptorB()[0].Size() != 0}; |
||||
|
if (is_buffer_b) { |
||||
|
const auto buffersB{ctx.BufferDescriptorB()}; |
||||
|
output_buffer.resize_destructive(buffersB[0].Size()); |
||||
|
performance_buffer.resize_destructive(buffersB[1].Size()); |
||||
|
} else { |
||||
|
const auto buffersC{ctx.BufferDescriptorC()}; |
||||
|
output_buffer.resize_destructive(buffersC[0].Size()); |
||||
|
performance_buffer.resize_destructive(buffersC[1].Size()); |
||||
|
} |
||||
|
|
||||
|
auto result = impl->RequestUpdate(input, performance_buffer, output_buffer); |
||||
|
|
||||
|
if (result.IsSuccess()) { |
||||
|
if (is_buffer_b) { |
||||
|
ctx.WriteBufferB(output_buffer.data(), output_buffer.size(), 0); |
||||
|
ctx.WriteBufferB(performance_buffer.data(), performance_buffer.size(), 1); |
||||
|
} else { |
||||
|
ctx.WriteBufferC(output_buffer.data(), output_buffer.size(), 0); |
||||
|
ctx.WriteBufferC(performance_buffer.data(), performance_buffer.size(), 1); |
||||
|
} |
||||
|
} else { |
||||
|
LOG_ERROR(Service_Audio, "RequestUpdate failed error 0x{:02X}!", result.GetDescription()); |
||||
|
} |
||||
|
|
||||
|
IPC::ResponseBuilder rb{ctx, 2}; |
||||
|
rb.Push(result); |
||||
|
} |
||||
|
|
||||
|
void IAudioRenderer::Start(HLERequestContext& ctx) { |
||||
|
LOG_DEBUG(Service_Audio, "called"); |
||||
|
|
||||
|
impl->Start(); |
||||
|
|
||||
|
IPC::ResponseBuilder rb{ctx, 2}; |
||||
|
rb.Push(ResultSuccess); |
||||
|
} |
||||
|
|
||||
|
void IAudioRenderer::Stop(HLERequestContext& ctx) { |
||||
|
LOG_DEBUG(Service_Audio, "called"); |
||||
|
|
||||
|
impl->Stop(); |
||||
|
|
||||
|
IPC::ResponseBuilder rb{ctx, 2}; |
||||
|
rb.Push(ResultSuccess); |
||||
|
} |
||||
|
|
||||
|
void IAudioRenderer::QuerySystemEvent(HLERequestContext& ctx) { |
||||
|
LOG_DEBUG(Service_Audio, "called"); |
||||
|
|
||||
|
if (impl->GetSystem().GetExecutionMode() == AudioCore::ExecutionMode::Manual) { |
||||
|
IPC::ResponseBuilder rb{ctx, 2}; |
||||
|
rb.Push(Audio::ResultNotSupported); |
||||
|
return; |
||||
|
} |
||||
|
|
||||
|
IPC::ResponseBuilder rb{ctx, 2, 1}; |
||||
|
rb.Push(ResultSuccess); |
||||
|
rb.PushCopyObjects(rendered_event->GetReadableEvent()); |
||||
|
} |
||||
|
|
||||
|
void IAudioRenderer::SetRenderingTimeLimit(HLERequestContext& ctx) { |
||||
|
LOG_DEBUG(Service_Audio, "called"); |
||||
|
|
||||
|
IPC::RequestParser rp{ctx}; |
||||
|
auto limit = rp.PopRaw<u32>(); |
||||
|
|
||||
|
auto& system_ = impl->GetSystem(); |
||||
|
system_.SetRenderingTimeLimit(limit); |
||||
|
|
||||
|
IPC::ResponseBuilder rb{ctx, 2}; |
||||
|
rb.Push(ResultSuccess); |
||||
|
} |
||||
|
|
||||
|
void IAudioRenderer::GetRenderingTimeLimit(HLERequestContext& ctx) { |
||||
|
LOG_DEBUG(Service_Audio, "called"); |
||||
|
|
||||
|
auto& system_ = impl->GetSystem(); |
||||
|
auto time = system_.GetRenderingTimeLimit(); |
||||
|
|
||||
|
IPC::ResponseBuilder rb{ctx, 3}; |
||||
|
rb.Push(ResultSuccess); |
||||
|
rb.Push(time); |
||||
|
} |
||||
|
|
||||
|
void IAudioRenderer::ExecuteAudioRendererRendering(HLERequestContext& ctx) { |
||||
|
LOG_DEBUG(Service_Audio, "called"); |
||||
|
} |
||||
|
|
||||
|
void IAudioRenderer::SetVoiceDropParameter(HLERequestContext& ctx) { |
||||
|
LOG_DEBUG(Service_Audio, "called"); |
||||
|
|
||||
|
IPC::RequestParser rp{ctx}; |
||||
|
auto voice_drop_param{rp.Pop<f32>()}; |
||||
|
|
||||
|
auto& system_ = impl->GetSystem(); |
||||
|
system_.SetVoiceDropParameter(voice_drop_param); |
||||
|
|
||||
|
IPC::ResponseBuilder rb{ctx, 2}; |
||||
|
rb.Push(ResultSuccess); |
||||
|
} |
||||
|
|
||||
|
void IAudioRenderer::GetVoiceDropParameter(HLERequestContext& ctx) { |
||||
|
LOG_DEBUG(Service_Audio, "called"); |
||||
|
|
||||
|
auto& system_ = impl->GetSystem(); |
||||
|
auto voice_drop_param{system_.GetVoiceDropParameter()}; |
||||
|
|
||||
|
IPC::ResponseBuilder rb{ctx, 3}; |
||||
|
rb.Push(ResultSuccess); |
||||
|
rb.Push(voice_drop_param); |
||||
|
} |
||||
|
|
||||
|
} // namespace Service::Audio
|
||||
@ -0,0 +1,45 @@ |
|||||
|
// SPDX-FileCopyrightText: Copyright 2024 yuzu Emulator Project |
||||
|
// SPDX-License-Identifier: GPL-2.0-or-later |
||||
|
|
||||
|
#pragma once |
||||
|
|
||||
|
#include "audio_core/renderer/audio_renderer.h" |
||||
|
#include "core/hle/service/kernel_helpers.h" |
||||
|
#include "core/hle/service/service.h" |
||||
|
|
||||
|
namespace Service::Audio { |
||||
|
|
||||
|
class IAudioRenderer final : public ServiceFramework<IAudioRenderer> { |
||||
|
public: |
||||
|
explicit IAudioRenderer(Core::System& system_, AudioCore::Renderer::Manager& manager_, |
||||
|
AudioCore::AudioRendererParameterInternal& params, |
||||
|
Kernel::KTransferMemory* transfer_memory, u64 transfer_memory_size, |
||||
|
u32 process_handle, Kernel::KProcess& process_, |
||||
|
u64 applet_resource_user_id, s32 session_id); |
||||
|
~IAudioRenderer() override; |
||||
|
|
||||
|
private: |
||||
|
void GetSampleRate(HLERequestContext& ctx); |
||||
|
void GetSampleCount(HLERequestContext& ctx); |
||||
|
void GetState(HLERequestContext& ctx); |
||||
|
void GetMixBufferCount(HLERequestContext& ctx); |
||||
|
void RequestUpdate(HLERequestContext& ctx); |
||||
|
void Start(HLERequestContext& ctx); |
||||
|
void Stop(HLERequestContext& ctx); |
||||
|
void QuerySystemEvent(HLERequestContext& ctx); |
||||
|
void SetRenderingTimeLimit(HLERequestContext& ctx); |
||||
|
void GetRenderingTimeLimit(HLERequestContext& ctx); |
||||
|
void ExecuteAudioRendererRendering(HLERequestContext& ctx); |
||||
|
void SetVoiceDropParameter(HLERequestContext& ctx); |
||||
|
void GetVoiceDropParameter(HLERequestContext& ctx); |
||||
|
|
||||
|
KernelHelpers::ServiceContext service_context; |
||||
|
Kernel::KEvent* rendered_event; |
||||
|
AudioCore::Renderer::Manager& manager; |
||||
|
std::unique_ptr<AudioCore::Renderer::Renderer> impl; |
||||
|
Kernel::KProcess& process; |
||||
|
Common::ScratchBuffer<u8> output_buffer; |
||||
|
Common::ScratchBuffer<u8> performance_buffer; |
||||
|
}; |
||||
|
|
||||
|
} // namespace Service::Audio |
||||
@ -0,0 +1,143 @@ |
|||||
|
// SPDX-FileCopyrightText: Copyright 2018 yuzu Emulator Project
|
||||
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||
|
|
||||
|
#include "audio_core/audio_render_manager.h"
|
||||
|
#include "audio_core/common/feature_support.h"
|
||||
|
#include "core/hle/kernel/k_process.h"
|
||||
|
#include "core/hle/kernel/k_transfer_memory.h"
|
||||
|
#include "core/hle/service/audio/audio_device.h"
|
||||
|
#include "core/hle/service/audio/audio_renderer.h"
|
||||
|
#include "core/hle/service/audio/audio_renderer_manager.h"
|
||||
|
#include "core/hle/service/ipc_helpers.h"
|
||||
|
|
||||
|
namespace Service::Audio { |
||||
|
|
||||
|
using namespace AudioCore::Renderer; |
||||
|
|
||||
|
IAudioRendererManager::IAudioRendererManager(Core::System& system_) |
||||
|
: ServiceFramework{system_, "audren:u"}, service_context{system_, "audren:u"}, |
||||
|
impl{std::make_unique<Manager>(system_)} { |
||||
|
// clang-format off
|
||||
|
static const FunctionInfo functions[] = { |
||||
|
{0, &IAudioRendererManager::OpenAudioRenderer, "OpenAudioRenderer"}, |
||||
|
{1, &IAudioRendererManager::GetWorkBufferSize, "GetWorkBufferSize"}, |
||||
|
{2, &IAudioRendererManager::GetAudioDeviceService, "GetAudioDeviceService"}, |
||||
|
{3, nullptr, "OpenAudioRendererForManualExecution"}, |
||||
|
{4, &IAudioRendererManager::GetAudioDeviceServiceWithRevisionInfo, "GetAudioDeviceServiceWithRevisionInfo"}, |
||||
|
}; |
||||
|
// clang-format on
|
||||
|
|
||||
|
RegisterHandlers(functions); |
||||
|
} |
||||
|
|
||||
|
IAudioRendererManager::~IAudioRendererManager() = default; |
||||
|
|
||||
|
void IAudioRendererManager::OpenAudioRenderer(HLERequestContext& ctx) { |
||||
|
IPC::RequestParser rp{ctx}; |
||||
|
|
||||
|
AudioCore::AudioRendererParameterInternal params; |
||||
|
rp.PopRaw<AudioCore::AudioRendererParameterInternal>(params); |
||||
|
rp.Skip(1, false); |
||||
|
auto transfer_memory_size = rp.Pop<u64>(); |
||||
|
auto applet_resource_user_id = rp.Pop<u64>(); |
||||
|
auto transfer_memory_handle = ctx.GetCopyHandle(0); |
||||
|
auto process_handle = ctx.GetCopyHandle(1); |
||||
|
|
||||
|
if (impl->GetSessionCount() + 1 > AudioCore::MaxRendererSessions) { |
||||
|
LOG_ERROR(Service_Audio, "Too many AudioRenderer sessions open!"); |
||||
|
IPC::ResponseBuilder rb{ctx, 2}; |
||||
|
rb.Push(Audio::ResultOutOfSessions); |
||||
|
return; |
||||
|
} |
||||
|
|
||||
|
auto process{ctx.GetObjectFromHandle<Kernel::KProcess>(process_handle).GetPointerUnsafe()}; |
||||
|
auto transfer_memory{ctx.GetObjectFromHandle<Kernel::KTransferMemory>(transfer_memory_handle)}; |
||||
|
|
||||
|
const auto session_id{impl->GetSessionId()}; |
||||
|
if (session_id == -1) { |
||||
|
LOG_ERROR(Service_Audio, "Tried to open a session that's already in use!"); |
||||
|
IPC::ResponseBuilder rb{ctx, 2}; |
||||
|
rb.Push(Audio::ResultOutOfSessions); |
||||
|
return; |
||||
|
} |
||||
|
|
||||
|
LOG_DEBUG(Service_Audio, "Opened new AudioRenderer session {} sessions open {}", session_id, |
||||
|
impl->GetSessionCount()); |
||||
|
|
||||
|
IPC::ResponseBuilder rb{ctx, 2, 0, 1}; |
||||
|
rb.Push(ResultSuccess); |
||||
|
rb.PushIpcInterface<IAudioRenderer>(system, *impl, params, transfer_memory.GetPointerUnsafe(), |
||||
|
transfer_memory_size, process_handle, *process, |
||||
|
applet_resource_user_id, session_id); |
||||
|
} |
||||
|
|
||||
|
void IAudioRendererManager::GetWorkBufferSize(HLERequestContext& ctx) { |
||||
|
AudioCore::AudioRendererParameterInternal params; |
||||
|
|
||||
|
IPC::RequestParser rp{ctx}; |
||||
|
rp.PopRaw<AudioCore::AudioRendererParameterInternal>(params); |
||||
|
|
||||
|
u64 size{0}; |
||||
|
auto result = impl->GetWorkBufferSize(params, size); |
||||
|
|
||||
|
std::string output_info{}; |
||||
|
output_info += fmt::format("\tRevision {}", AudioCore::GetRevisionNum(params.revision)); |
||||
|
output_info += |
||||
|
fmt::format("\n\tSample Rate {}, Sample Count {}", params.sample_rate, params.sample_count); |
||||
|
output_info += fmt::format("\n\tExecution Mode {}, Voice Drop Enabled {}", |
||||
|
static_cast<u32>(params.execution_mode), params.voice_drop_enabled); |
||||
|
output_info += fmt::format( |
||||
|
"\n\tSizes: Effects {:04X}, Mixes {:04X}, Sinks {:04X}, Submixes {:04X}, Splitter Infos " |
||||
|
"{:04X}, Splitter Destinations {:04X}, Voices {:04X}, Performance Frames {:04X} External " |
||||
|
"Context {:04X}", |
||||
|
params.effects, params.mixes, params.sinks, params.sub_mixes, params.splitter_infos, |
||||
|
params.splitter_destinations, params.voices, params.perf_frames, |
||||
|
params.external_context_size); |
||||
|
|
||||
|
LOG_DEBUG(Service_Audio, "called.\nInput params:\n{}\nOutput params:\n\tWorkbuffer size {:08X}", |
||||
|
output_info, size); |
||||
|
|
||||
|
IPC::ResponseBuilder rb{ctx, 4}; |
||||
|
rb.Push(result); |
||||
|
rb.Push<u64>(size); |
||||
|
} |
||||
|
|
||||
|
void IAudioRendererManager::GetAudioDeviceService(HLERequestContext& ctx) { |
||||
|
IPC::RequestParser rp{ctx}; |
||||
|
|
||||
|
const auto applet_resource_user_id = rp.Pop<u64>(); |
||||
|
|
||||
|
LOG_DEBUG(Service_Audio, "called. Applet resource id {}", applet_resource_user_id); |
||||
|
|
||||
|
IPC::ResponseBuilder rb{ctx, 2, 0, 1}; |
||||
|
|
||||
|
rb.Push(ResultSuccess); |
||||
|
rb.PushIpcInterface<IAudioDevice>(system, applet_resource_user_id, |
||||
|
::Common::MakeMagic('R', 'E', 'V', '1'), num_audio_devices++); |
||||
|
} |
||||
|
|
||||
|
void IAudioRendererManager::OpenAudioRendererForManualExecution(HLERequestContext& ctx) { |
||||
|
LOG_ERROR(Service_Audio, "called. Implement me!"); |
||||
|
} |
||||
|
|
||||
|
void IAudioRendererManager::GetAudioDeviceServiceWithRevisionInfo(HLERequestContext& ctx) { |
||||
|
struct Parameters { |
||||
|
u32 revision; |
||||
|
u64 applet_resource_user_id; |
||||
|
}; |
||||
|
|
||||
|
IPC::RequestParser rp{ctx}; |
||||
|
|
||||
|
const auto [revision, applet_resource_user_id] = rp.PopRaw<Parameters>(); |
||||
|
|
||||
|
LOG_DEBUG(Service_Audio, "called. Revision {} Applet resource id {}", |
||||
|
AudioCore::GetRevisionNum(revision), applet_resource_user_id); |
||||
|
|
||||
|
IPC::ResponseBuilder rb{ctx, 2, 0, 1}; |
||||
|
|
||||
|
rb.Push(ResultSuccess); |
||||
|
rb.PushIpcInterface<IAudioDevice>(system, applet_resource_user_id, revision, |
||||
|
num_audio_devices++); |
||||
|
} |
||||
|
|
||||
|
} // namespace Service::Audio
|
||||
@ -1,552 +0,0 @@ |
|||||
// SPDX-FileCopyrightText: Copyright 2018 yuzu Emulator Project
|
|
||||
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
||||
|
|
||||
#include <array>
|
|
||||
#include <memory>
|
|
||||
|
|
||||
#include "audio_core/audio_core.h"
|
|
||||
#include "audio_core/common/audio_renderer_parameter.h"
|
|
||||
#include "audio_core/common/feature_support.h"
|
|
||||
#include "audio_core/renderer/audio_device.h"
|
|
||||
#include "audio_core/renderer/audio_renderer.h"
|
|
||||
#include "audio_core/renderer/voice/voice_info.h"
|
|
||||
#include "common/alignment.h"
|
|
||||
#include "common/bit_util.h"
|
|
||||
#include "common/common_funcs.h"
|
|
||||
#include "common/logging/log.h"
|
|
||||
#include "common/polyfill_ranges.h"
|
|
||||
#include "common/scratch_buffer.h"
|
|
||||
#include "common/string_util.h"
|
|
||||
#include "core/core.h"
|
|
||||
#include "core/hle/kernel/k_event.h"
|
|
||||
#include "core/hle/kernel/k_process.h"
|
|
||||
#include "core/hle/kernel/k_transfer_memory.h"
|
|
||||
#include "core/hle/service/audio/audren_u.h"
|
|
||||
#include "core/hle/service/audio/errors.h"
|
|
||||
#include "core/hle/service/ipc_helpers.h"
|
|
||||
#include "core/memory.h"
|
|
||||
|
|
||||
using namespace AudioCore::Renderer; |
|
||||
|
|
||||
namespace Service::Audio { |
|
||||
|
|
||||
class IAudioRenderer final : public ServiceFramework<IAudioRenderer> { |
|
||||
public: |
|
||||
explicit IAudioRenderer(Core::System& system_, Manager& manager_, |
|
||||
AudioCore::AudioRendererParameterInternal& params, |
|
||||
Kernel::KTransferMemory* transfer_memory, u64 transfer_memory_size, |
|
||||
u32 process_handle, Kernel::KProcess& process_, |
|
||||
u64 applet_resource_user_id, s32 session_id) |
|
||||
: ServiceFramework{system_, "IAudioRenderer"}, service_context{system_, "IAudioRenderer"}, |
|
||||
rendered_event{service_context.CreateEvent("IAudioRendererEvent")}, manager{manager_}, |
|
||||
impl{std::make_unique<Renderer>(system_, manager, rendered_event)}, process{process_} { |
|
||||
// clang-format off
|
|
||||
static const FunctionInfo functions[] = { |
|
||||
{0, &IAudioRenderer::GetSampleRate, "GetSampleRate"}, |
|
||||
{1, &IAudioRenderer::GetSampleCount, "GetSampleCount"}, |
|
||||
{2, &IAudioRenderer::GetMixBufferCount, "GetMixBufferCount"}, |
|
||||
{3, &IAudioRenderer::GetState, "GetState"}, |
|
||||
{4, &IAudioRenderer::RequestUpdate, "RequestUpdate"}, |
|
||||
{5, &IAudioRenderer::Start, "Start"}, |
|
||||
{6, &IAudioRenderer::Stop, "Stop"}, |
|
||||
{7, &IAudioRenderer::QuerySystemEvent, "QuerySystemEvent"}, |
|
||||
{8, &IAudioRenderer::SetRenderingTimeLimit, "SetRenderingTimeLimit"}, |
|
||||
{9, &IAudioRenderer::GetRenderingTimeLimit, "GetRenderingTimeLimit"}, |
|
||||
{10, &IAudioRenderer::RequestUpdate, "RequestUpdateAuto"}, |
|
||||
{11, nullptr, "ExecuteAudioRendererRendering"}, |
|
||||
{12, &IAudioRenderer::SetVoiceDropParameter, "SetVoiceDropParameter"}, |
|
||||
{13, &IAudioRenderer::GetVoiceDropParameter, "GetVoiceDropParameter"}, |
|
||||
}; |
|
||||
// clang-format on
|
|
||||
RegisterHandlers(functions); |
|
||||
|
|
||||
process.Open(); |
|
||||
impl->Initialize(params, transfer_memory, transfer_memory_size, process_handle, process, |
|
||||
applet_resource_user_id, session_id); |
|
||||
} |
|
||||
|
|
||||
~IAudioRenderer() override { |
|
||||
impl->Finalize(); |
|
||||
service_context.CloseEvent(rendered_event); |
|
||||
process.Close(); |
|
||||
} |
|
||||
|
|
||||
private: |
|
||||
void GetSampleRate(HLERequestContext& ctx) { |
|
||||
const auto sample_rate{impl->GetSystem().GetSampleRate()}; |
|
||||
|
|
||||
LOG_DEBUG(Service_Audio, "called. Sample rate {}", sample_rate); |
|
||||
|
|
||||
IPC::ResponseBuilder rb{ctx, 3}; |
|
||||
rb.Push(ResultSuccess); |
|
||||
rb.Push(sample_rate); |
|
||||
} |
|
||||
|
|
||||
void GetSampleCount(HLERequestContext& ctx) { |
|
||||
const auto sample_count{impl->GetSystem().GetSampleCount()}; |
|
||||
|
|
||||
LOG_DEBUG(Service_Audio, "called. Sample count {}", sample_count); |
|
||||
|
|
||||
IPC::ResponseBuilder rb{ctx, 3}; |
|
||||
rb.Push(ResultSuccess); |
|
||||
rb.Push(sample_count); |
|
||||
} |
|
||||
|
|
||||
void GetState(HLERequestContext& ctx) { |
|
||||
const u32 state{!impl->GetSystem().IsActive()}; |
|
||||
|
|
||||
LOG_DEBUG(Service_Audio, "called, state {}", state); |
|
||||
|
|
||||
IPC::ResponseBuilder rb{ctx, 3}; |
|
||||
rb.Push(ResultSuccess); |
|
||||
rb.Push(state); |
|
||||
} |
|
||||
|
|
||||
void GetMixBufferCount(HLERequestContext& ctx) { |
|
||||
LOG_DEBUG(Service_Audio, "called"); |
|
||||
|
|
||||
const auto buffer_count{impl->GetSystem().GetMixBufferCount()}; |
|
||||
|
|
||||
IPC::ResponseBuilder rb{ctx, 3}; |
|
||||
rb.Push(ResultSuccess); |
|
||||
rb.Push(buffer_count); |
|
||||
} |
|
||||
|
|
||||
void RequestUpdate(HLERequestContext& ctx) { |
|
||||
LOG_TRACE(Service_Audio, "called"); |
|
||||
|
|
||||
const auto input{ctx.ReadBuffer(0)}; |
|
||||
|
|
||||
// These buffers are written manually to avoid an issue with WriteBuffer throwing errors for
|
|
||||
// checking size 0. Performance size is 0 for most games.
|
|
||||
|
|
||||
auto is_buffer_b{ctx.BufferDescriptorB()[0].Size() != 0}; |
|
||||
if (is_buffer_b) { |
|
||||
const auto buffersB{ctx.BufferDescriptorB()}; |
|
||||
output_buffer.resize_destructive(buffersB[0].Size()); |
|
||||
performance_buffer.resize_destructive(buffersB[1].Size()); |
|
||||
} else { |
|
||||
const auto buffersC{ctx.BufferDescriptorC()}; |
|
||||
output_buffer.resize_destructive(buffersC[0].Size()); |
|
||||
performance_buffer.resize_destructive(buffersC[1].Size()); |
|
||||
} |
|
||||
|
|
||||
auto result = impl->RequestUpdate(input, performance_buffer, output_buffer); |
|
||||
|
|
||||
if (result.IsSuccess()) { |
|
||||
if (is_buffer_b) { |
|
||||
ctx.WriteBufferB(output_buffer.data(), output_buffer.size(), 0); |
|
||||
ctx.WriteBufferB(performance_buffer.data(), performance_buffer.size(), 1); |
|
||||
} else { |
|
||||
ctx.WriteBufferC(output_buffer.data(), output_buffer.size(), 0); |
|
||||
ctx.WriteBufferC(performance_buffer.data(), performance_buffer.size(), 1); |
|
||||
} |
|
||||
} else { |
|
||||
LOG_ERROR(Service_Audio, "RequestUpdate failed error 0x{:02X}!", |
|
||||
result.GetDescription()); |
|
||||
} |
|
||||
|
|
||||
IPC::ResponseBuilder rb{ctx, 2}; |
|
||||
rb.Push(result); |
|
||||
} |
|
||||
|
|
||||
void Start(HLERequestContext& ctx) { |
|
||||
LOG_DEBUG(Service_Audio, "called"); |
|
||||
|
|
||||
impl->Start(); |
|
||||
|
|
||||
IPC::ResponseBuilder rb{ctx, 2}; |
|
||||
rb.Push(ResultSuccess); |
|
||||
} |
|
||||
|
|
||||
void Stop(HLERequestContext& ctx) { |
|
||||
LOG_DEBUG(Service_Audio, "called"); |
|
||||
|
|
||||
impl->Stop(); |
|
||||
|
|
||||
IPC::ResponseBuilder rb{ctx, 2}; |
|
||||
rb.Push(ResultSuccess); |
|
||||
} |
|
||||
|
|
||||
void QuerySystemEvent(HLERequestContext& ctx) { |
|
||||
LOG_DEBUG(Service_Audio, "called"); |
|
||||
|
|
||||
if (impl->GetSystem().GetExecutionMode() == AudioCore::ExecutionMode::Manual) { |
|
||||
IPC::ResponseBuilder rb{ctx, 2}; |
|
||||
rb.Push(Audio::ResultNotSupported); |
|
||||
return; |
|
||||
} |
|
||||
|
|
||||
IPC::ResponseBuilder rb{ctx, 2, 1}; |
|
||||
rb.Push(ResultSuccess); |
|
||||
rb.PushCopyObjects(rendered_event->GetReadableEvent()); |
|
||||
} |
|
||||
|
|
||||
void SetRenderingTimeLimit(HLERequestContext& ctx) { |
|
||||
LOG_DEBUG(Service_Audio, "called"); |
|
||||
|
|
||||
IPC::RequestParser rp{ctx}; |
|
||||
auto limit = rp.PopRaw<u32>(); |
|
||||
|
|
||||
auto& system_ = impl->GetSystem(); |
|
||||
system_.SetRenderingTimeLimit(limit); |
|
||||
|
|
||||
IPC::ResponseBuilder rb{ctx, 2}; |
|
||||
rb.Push(ResultSuccess); |
|
||||
} |
|
||||
|
|
||||
void GetRenderingTimeLimit(HLERequestContext& ctx) { |
|
||||
LOG_DEBUG(Service_Audio, "called"); |
|
||||
|
|
||||
auto& system_ = impl->GetSystem(); |
|
||||
auto time = system_.GetRenderingTimeLimit(); |
|
||||
|
|
||||
IPC::ResponseBuilder rb{ctx, 3}; |
|
||||
rb.Push(ResultSuccess); |
|
||||
rb.Push(time); |
|
||||
} |
|
||||
|
|
||||
void ExecuteAudioRendererRendering(HLERequestContext& ctx) { |
|
||||
LOG_DEBUG(Service_Audio, "called"); |
|
||||
} |
|
||||
|
|
||||
void SetVoiceDropParameter(HLERequestContext& ctx) { |
|
||||
LOG_DEBUG(Service_Audio, "called"); |
|
||||
|
|
||||
IPC::RequestParser rp{ctx}; |
|
||||
auto voice_drop_param{rp.Pop<f32>()}; |
|
||||
|
|
||||
auto& system_ = impl->GetSystem(); |
|
||||
system_.SetVoiceDropParameter(voice_drop_param); |
|
||||
|
|
||||
IPC::ResponseBuilder rb{ctx, 2}; |
|
||||
rb.Push(ResultSuccess); |
|
||||
} |
|
||||
|
|
||||
void GetVoiceDropParameter(HLERequestContext& ctx) { |
|
||||
LOG_DEBUG(Service_Audio, "called"); |
|
||||
|
|
||||
auto& system_ = impl->GetSystem(); |
|
||||
auto voice_drop_param{system_.GetVoiceDropParameter()}; |
|
||||
|
|
||||
IPC::ResponseBuilder rb{ctx, 3}; |
|
||||
rb.Push(ResultSuccess); |
|
||||
rb.Push(voice_drop_param); |
|
||||
} |
|
||||
|
|
||||
KernelHelpers::ServiceContext service_context; |
|
||||
Kernel::KEvent* rendered_event; |
|
||||
Manager& manager; |
|
||||
std::unique_ptr<Renderer> impl; |
|
||||
Kernel::KProcess& process; |
|
||||
Common::ScratchBuffer<u8> output_buffer; |
|
||||
Common::ScratchBuffer<u8> performance_buffer; |
|
||||
}; |
|
||||
|
|
||||
class IAudioDevice final : public ServiceFramework<IAudioDevice> { |
|
||||
|
|
||||
public: |
|
||||
explicit IAudioDevice(Core::System& system_, u64 applet_resource_user_id, u32 revision, |
|
||||
u32 device_num) |
|
||||
: ServiceFramework{system_, "IAudioDevice"}, service_context{system_, "IAudioDevice"}, |
|
||||
impl{std::make_unique<AudioDevice>(system_, applet_resource_user_id, revision)}, |
|
||||
event{service_context.CreateEvent(fmt::format("IAudioDeviceEvent-{}", device_num))} { |
|
||||
static const FunctionInfo functions[] = { |
|
||||
{0, &IAudioDevice::ListAudioDeviceName, "ListAudioDeviceName"}, |
|
||||
{1, &IAudioDevice::SetAudioDeviceOutputVolume, "SetAudioDeviceOutputVolume"}, |
|
||||
{2, &IAudioDevice::GetAudioDeviceOutputVolume, "GetAudioDeviceOutputVolume"}, |
|
||||
{3, &IAudioDevice::GetActiveAudioDeviceName, "GetActiveAudioDeviceName"}, |
|
||||
{4, &IAudioDevice::QueryAudioDeviceSystemEvent, "QueryAudioDeviceSystemEvent"}, |
|
||||
{5, &IAudioDevice::GetActiveChannelCount, "GetActiveChannelCount"}, |
|
||||
{6, &IAudioDevice::ListAudioDeviceName, "ListAudioDeviceNameAuto"}, |
|
||||
{7, &IAudioDevice::SetAudioDeviceOutputVolume, "SetAudioDeviceOutputVolumeAuto"}, |
|
||||
{8, &IAudioDevice::GetAudioDeviceOutputVolume, "GetAudioDeviceOutputVolumeAuto"}, |
|
||||
{10, &IAudioDevice::GetActiveAudioDeviceName, "GetActiveAudioDeviceNameAuto"}, |
|
||||
{11, &IAudioDevice::QueryAudioDeviceInputEvent, "QueryAudioDeviceInputEvent"}, |
|
||||
{12, &IAudioDevice::QueryAudioDeviceOutputEvent, "QueryAudioDeviceOutputEvent"}, |
|
||||
{13, &IAudioDevice::GetActiveAudioDeviceName, "GetActiveAudioOutputDeviceName"}, |
|
||||
{14, &IAudioDevice::ListAudioOutputDeviceName, "ListAudioOutputDeviceName"}, |
|
||||
}; |
|
||||
RegisterHandlers(functions); |
|
||||
|
|
||||
event->Signal(); |
|
||||
} |
|
||||
|
|
||||
~IAudioDevice() override { |
|
||||
service_context.CloseEvent(event); |
|
||||
} |
|
||||
|
|
||||
private: |
|
||||
void ListAudioDeviceName(HLERequestContext& ctx) { |
|
||||
const size_t in_count = ctx.GetWriteBufferNumElements<AudioDevice::AudioDeviceName>(); |
|
||||
|
|
||||
std::vector<AudioDevice::AudioDeviceName> out_names{}; |
|
||||
|
|
||||
const u32 out_count = impl->ListAudioDeviceName(out_names, in_count); |
|
||||
|
|
||||
std::string out{}; |
|
||||
for (u32 i = 0; i < out_count; i++) { |
|
||||
std::string a{}; |
|
||||
u32 j = 0; |
|
||||
while (out_names[i].name[j] != '\0') { |
|
||||
a += out_names[i].name[j]; |
|
||||
j++; |
|
||||
} |
|
||||
out += "\n\t" + a; |
|
||||
} |
|
||||
|
|
||||
LOG_DEBUG(Service_Audio, "called.\nNames={}", out); |
|
||||
|
|
||||
IPC::ResponseBuilder rb{ctx, 3}; |
|
||||
|
|
||||
ctx.WriteBuffer(out_names); |
|
||||
|
|
||||
rb.Push(ResultSuccess); |
|
||||
rb.Push(out_count); |
|
||||
} |
|
||||
|
|
||||
void SetAudioDeviceOutputVolume(HLERequestContext& ctx) { |
|
||||
IPC::RequestParser rp{ctx}; |
|
||||
const f32 volume = rp.Pop<f32>(); |
|
||||
|
|
||||
const auto device_name_buffer = ctx.ReadBuffer(); |
|
||||
const std::string name = Common::StringFromBuffer(device_name_buffer); |
|
||||
|
|
||||
LOG_DEBUG(Service_Audio, "called. name={}, volume={}", name, volume); |
|
||||
|
|
||||
if (name == "AudioTvOutput") { |
|
||||
impl->SetDeviceVolumes(volume); |
|
||||
} |
|
||||
|
|
||||
IPC::ResponseBuilder rb{ctx, 2}; |
|
||||
rb.Push(ResultSuccess); |
|
||||
} |
|
||||
|
|
||||
void GetAudioDeviceOutputVolume(HLERequestContext& ctx) { |
|
||||
const auto device_name_buffer = ctx.ReadBuffer(); |
|
||||
const std::string name = Common::StringFromBuffer(device_name_buffer); |
|
||||
|
|
||||
LOG_DEBUG(Service_Audio, "called. Name={}", name); |
|
||||
|
|
||||
f32 volume{1.0f}; |
|
||||
if (name == "AudioTvOutput") { |
|
||||
volume = impl->GetDeviceVolume(name); |
|
||||
} |
|
||||
|
|
||||
IPC::ResponseBuilder rb{ctx, 3}; |
|
||||
rb.Push(ResultSuccess); |
|
||||
rb.Push(volume); |
|
||||
} |
|
||||
|
|
||||
void GetActiveAudioDeviceName(HLERequestContext& ctx) { |
|
||||
const auto write_size = ctx.GetWriteBufferSize(); |
|
||||
std::string out_name{"AudioTvOutput"}; |
|
||||
|
|
||||
LOG_DEBUG(Service_Audio, "(STUBBED) called. Name={}", out_name); |
|
||||
|
|
||||
out_name.resize(write_size); |
|
||||
|
|
||||
ctx.WriteBuffer(out_name); |
|
||||
|
|
||||
IPC::ResponseBuilder rb{ctx, 2}; |
|
||||
rb.Push(ResultSuccess); |
|
||||
} |
|
||||
|
|
||||
void QueryAudioDeviceSystemEvent(HLERequestContext& ctx) { |
|
||||
LOG_DEBUG(Service_Audio, "(STUBBED) called"); |
|
||||
|
|
||||
event->Signal(); |
|
||||
|
|
||||
IPC::ResponseBuilder rb{ctx, 2, 1}; |
|
||||
rb.Push(ResultSuccess); |
|
||||
rb.PushCopyObjects(event->GetReadableEvent()); |
|
||||
} |
|
||||
|
|
||||
void GetActiveChannelCount(HLERequestContext& ctx) { |
|
||||
const auto& sink{system.AudioCore().GetOutputSink()}; |
|
||||
u32 channel_count{sink.GetSystemChannels()}; |
|
||||
|
|
||||
LOG_DEBUG(Service_Audio, "(STUBBED) called. Channels={}", channel_count); |
|
||||
|
|
||||
IPC::ResponseBuilder rb{ctx, 3}; |
|
||||
|
|
||||
rb.Push(ResultSuccess); |
|
||||
rb.Push<u32>(channel_count); |
|
||||
} |
|
||||
|
|
||||
void QueryAudioDeviceInputEvent(HLERequestContext& ctx) { |
|
||||
LOG_DEBUG(Service_Audio, "(STUBBED) called"); |
|
||||
|
|
||||
IPC::ResponseBuilder rb{ctx, 2, 1}; |
|
||||
rb.Push(ResultSuccess); |
|
||||
rb.PushCopyObjects(event->GetReadableEvent()); |
|
||||
} |
|
||||
|
|
||||
void QueryAudioDeviceOutputEvent(HLERequestContext& ctx) { |
|
||||
LOG_DEBUG(Service_Audio, "called"); |
|
||||
|
|
||||
IPC::ResponseBuilder rb{ctx, 2, 1}; |
|
||||
rb.Push(ResultSuccess); |
|
||||
rb.PushCopyObjects(event->GetReadableEvent()); |
|
||||
} |
|
||||
|
|
||||
void ListAudioOutputDeviceName(HLERequestContext& ctx) { |
|
||||
const size_t in_count = ctx.GetWriteBufferNumElements<AudioDevice::AudioDeviceName>(); |
|
||||
|
|
||||
std::vector<AudioDevice::AudioDeviceName> out_names{}; |
|
||||
|
|
||||
const u32 out_count = impl->ListAudioOutputDeviceName(out_names, in_count); |
|
||||
|
|
||||
std::string out{}; |
|
||||
for (u32 i = 0; i < out_count; i++) { |
|
||||
std::string a{}; |
|
||||
u32 j = 0; |
|
||||
while (out_names[i].name[j] != '\0') { |
|
||||
a += out_names[i].name[j]; |
|
||||
j++; |
|
||||
} |
|
||||
out += "\n\t" + a; |
|
||||
} |
|
||||
|
|
||||
LOG_DEBUG(Service_Audio, "called.\nNames={}", out); |
|
||||
|
|
||||
IPC::ResponseBuilder rb{ctx, 3}; |
|
||||
|
|
||||
ctx.WriteBuffer(out_names); |
|
||||
|
|
||||
rb.Push(ResultSuccess); |
|
||||
rb.Push(out_count); |
|
||||
} |
|
||||
|
|
||||
KernelHelpers::ServiceContext service_context; |
|
||||
std::unique_ptr<AudioDevice> impl; |
|
||||
Kernel::KEvent* event; |
|
||||
}; |
|
||||
|
|
||||
AudRenU::AudRenU(Core::System& system_) |
|
||||
: ServiceFramework{system_, "audren:u"}, |
|
||||
service_context{system_, "audren:u"}, impl{std::make_unique<Manager>(system_)} { |
|
||||
// clang-format off
|
|
||||
static const FunctionInfo functions[] = { |
|
||||
{0, &AudRenU::OpenAudioRenderer, "OpenAudioRenderer"}, |
|
||||
{1, &AudRenU::GetWorkBufferSize, "GetWorkBufferSize"}, |
|
||||
{2, &AudRenU::GetAudioDeviceService, "GetAudioDeviceService"}, |
|
||||
{3, nullptr, "OpenAudioRendererForManualExecution"}, |
|
||||
{4, &AudRenU::GetAudioDeviceServiceWithRevisionInfo, "GetAudioDeviceServiceWithRevisionInfo"}, |
|
||||
}; |
|
||||
// clang-format on
|
|
||||
|
|
||||
RegisterHandlers(functions); |
|
||||
} |
|
||||
|
|
||||
AudRenU::~AudRenU() = default; |
|
||||
|
|
||||
void AudRenU::OpenAudioRenderer(HLERequestContext& ctx) { |
|
||||
IPC::RequestParser rp{ctx}; |
|
||||
|
|
||||
AudioCore::AudioRendererParameterInternal params; |
|
||||
rp.PopRaw<AudioCore::AudioRendererParameterInternal>(params); |
|
||||
rp.Skip(1, false); |
|
||||
auto transfer_memory_size = rp.Pop<u64>(); |
|
||||
auto applet_resource_user_id = rp.Pop<u64>(); |
|
||||
auto transfer_memory_handle = ctx.GetCopyHandle(0); |
|
||||
auto process_handle = ctx.GetCopyHandle(1); |
|
||||
|
|
||||
if (impl->GetSessionCount() + 1 > AudioCore::MaxRendererSessions) { |
|
||||
LOG_ERROR(Service_Audio, "Too many AudioRenderer sessions open!"); |
|
||||
IPC::ResponseBuilder rb{ctx, 2}; |
|
||||
rb.Push(Audio::ResultOutOfSessions); |
|
||||
return; |
|
||||
} |
|
||||
|
|
||||
auto process{ctx.GetObjectFromHandle<Kernel::KProcess>(process_handle).GetPointerUnsafe()}; |
|
||||
auto transfer_memory{ctx.GetObjectFromHandle<Kernel::KTransferMemory>(transfer_memory_handle)}; |
|
||||
|
|
||||
const auto session_id{impl->GetSessionId()}; |
|
||||
if (session_id == -1) { |
|
||||
LOG_ERROR(Service_Audio, "Tried to open a session that's already in use!"); |
|
||||
IPC::ResponseBuilder rb{ctx, 2}; |
|
||||
rb.Push(Audio::ResultOutOfSessions); |
|
||||
return; |
|
||||
} |
|
||||
|
|
||||
LOG_DEBUG(Service_Audio, "Opened new AudioRenderer session {} sessions open {}", session_id, |
|
||||
impl->GetSessionCount()); |
|
||||
|
|
||||
IPC::ResponseBuilder rb{ctx, 2, 0, 1}; |
|
||||
rb.Push(ResultSuccess); |
|
||||
rb.PushIpcInterface<IAudioRenderer>(system, *impl, params, transfer_memory.GetPointerUnsafe(), |
|
||||
transfer_memory_size, process_handle, *process, |
|
||||
applet_resource_user_id, session_id); |
|
||||
} |
|
||||
|
|
||||
void AudRenU::GetWorkBufferSize(HLERequestContext& ctx) { |
|
||||
AudioCore::AudioRendererParameterInternal params; |
|
||||
|
|
||||
IPC::RequestParser rp{ctx}; |
|
||||
rp.PopRaw<AudioCore::AudioRendererParameterInternal>(params); |
|
||||
|
|
||||
u64 size{0}; |
|
||||
auto result = impl->GetWorkBufferSize(params, size); |
|
||||
|
|
||||
std::string output_info{}; |
|
||||
output_info += fmt::format("\tRevision {}", AudioCore::GetRevisionNum(params.revision)); |
|
||||
output_info += |
|
||||
fmt::format("\n\tSample Rate {}, Sample Count {}", params.sample_rate, params.sample_count); |
|
||||
output_info += fmt::format("\n\tExecution Mode {}, Voice Drop Enabled {}", |
|
||||
static_cast<u32>(params.execution_mode), params.voice_drop_enabled); |
|
||||
output_info += fmt::format( |
|
||||
"\n\tSizes: Effects {:04X}, Mixes {:04X}, Sinks {:04X}, Submixes {:04X}, Splitter Infos " |
|
||||
"{:04X}, Splitter Destinations {:04X}, Voices {:04X}, Performance Frames {:04X} External " |
|
||||
"Context {:04X}", |
|
||||
params.effects, params.mixes, params.sinks, params.sub_mixes, params.splitter_infos, |
|
||||
params.splitter_destinations, params.voices, params.perf_frames, |
|
||||
params.external_context_size); |
|
||||
|
|
||||
LOG_DEBUG(Service_Audio, "called.\nInput params:\n{}\nOutput params:\n\tWorkbuffer size {:08X}", |
|
||||
output_info, size); |
|
||||
|
|
||||
IPC::ResponseBuilder rb{ctx, 4}; |
|
||||
rb.Push(result); |
|
||||
rb.Push<u64>(size); |
|
||||
} |
|
||||
|
|
||||
void AudRenU::GetAudioDeviceService(HLERequestContext& ctx) { |
|
||||
IPC::RequestParser rp{ctx}; |
|
||||
|
|
||||
const auto applet_resource_user_id = rp.Pop<u64>(); |
|
||||
|
|
||||
LOG_DEBUG(Service_Audio, "called. Applet resource id {}", applet_resource_user_id); |
|
||||
|
|
||||
IPC::ResponseBuilder rb{ctx, 2, 0, 1}; |
|
||||
|
|
||||
rb.Push(ResultSuccess); |
|
||||
rb.PushIpcInterface<IAudioDevice>(system, applet_resource_user_id, |
|
||||
::Common::MakeMagic('R', 'E', 'V', '1'), num_audio_devices++); |
|
||||
} |
|
||||
|
|
||||
void AudRenU::OpenAudioRendererForManualExecution(HLERequestContext& ctx) { |
|
||||
LOG_ERROR(Service_Audio, "called. Implement me!"); |
|
||||
} |
|
||||
|
|
||||
void AudRenU::GetAudioDeviceServiceWithRevisionInfo(HLERequestContext& ctx) { |
|
||||
struct Parameters { |
|
||||
u32 revision; |
|
||||
u64 applet_resource_user_id; |
|
||||
}; |
|
||||
|
|
||||
IPC::RequestParser rp{ctx}; |
|
||||
|
|
||||
const auto [revision, applet_resource_user_id] = rp.PopRaw<Parameters>(); |
|
||||
|
|
||||
LOG_DEBUG(Service_Audio, "called. Revision {} Applet resource id {}", |
|
||||
AudioCore::GetRevisionNum(revision), applet_resource_user_id); |
|
||||
|
|
||||
IPC::ResponseBuilder rb{ctx, 2, 0, 1}; |
|
||||
|
|
||||
rb.Push(ResultSuccess); |
|
||||
rb.PushIpcInterface<IAudioDevice>(system, applet_resource_user_id, revision, |
|
||||
num_audio_devices++); |
|
||||
} |
|
||||
|
|
||||
} // namespace Service::Audio
|
|
||||
Write
Preview
Loading…
Cancel
Save
Reference in new issue