Merge pull request #2747 from lioncash/audio
service/audren_u: Unstub ListAudioDeviceName
This commit is contained in:
		
						commit
						ef584f1a3a
					
				
					 7 changed files with 180 additions and 109 deletions
				
			
		|  | @ -19,16 +19,16 @@ | ||||||
| 
 | 
 | ||||||
| namespace Service::Audio { | namespace Service::Audio { | ||||||
| 
 | 
 | ||||||
| void InstallInterfaces(SM::ServiceManager& service_manager) { | void InstallInterfaces(SM::ServiceManager& service_manager, Core::System& system) { | ||||||
|     std::make_shared<AudCtl>()->InstallAsService(service_manager); |     std::make_shared<AudCtl>()->InstallAsService(service_manager); | ||||||
|     std::make_shared<AudOutA>()->InstallAsService(service_manager); |     std::make_shared<AudOutA>()->InstallAsService(service_manager); | ||||||
|     std::make_shared<AudOutU>()->InstallAsService(service_manager); |     std::make_shared<AudOutU>(system)->InstallAsService(service_manager); | ||||||
|     std::make_shared<AudInA>()->InstallAsService(service_manager); |     std::make_shared<AudInA>()->InstallAsService(service_manager); | ||||||
|     std::make_shared<AudInU>()->InstallAsService(service_manager); |     std::make_shared<AudInU>()->InstallAsService(service_manager); | ||||||
|     std::make_shared<AudRecA>()->InstallAsService(service_manager); |     std::make_shared<AudRecA>()->InstallAsService(service_manager); | ||||||
|     std::make_shared<AudRecU>()->InstallAsService(service_manager); |     std::make_shared<AudRecU>()->InstallAsService(service_manager); | ||||||
|     std::make_shared<AudRenA>()->InstallAsService(service_manager); |     std::make_shared<AudRenA>()->InstallAsService(service_manager); | ||||||
|     std::make_shared<AudRenU>()->InstallAsService(service_manager); |     std::make_shared<AudRenU>(system)->InstallAsService(service_manager); | ||||||
|     std::make_shared<CodecCtl>()->InstallAsService(service_manager); |     std::make_shared<CodecCtl>()->InstallAsService(service_manager); | ||||||
|     std::make_shared<HwOpus>()->InstallAsService(service_manager); |     std::make_shared<HwOpus>()->InstallAsService(service_manager); | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -4,6 +4,10 @@ | ||||||
| 
 | 
 | ||||||
| #pragma once | #pragma once | ||||||
| 
 | 
 | ||||||
|  | namespace Core { | ||||||
|  | class System; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| namespace Service::SM { | namespace Service::SM { | ||||||
| class ServiceManager; | class ServiceManager; | ||||||
| } | } | ||||||
|  | @ -11,6 +15,6 @@ class ServiceManager; | ||||||
| namespace Service::Audio { | namespace Service::Audio { | ||||||
| 
 | 
 | ||||||
| /// Registers all Audio services with the specified service manager.
 | /// Registers all Audio services with the specified service manager.
 | ||||||
| void InstallInterfaces(SM::ServiceManager& service_manager); | void InstallInterfaces(SM::ServiceManager& service_manager, Core::System& system); | ||||||
| 
 | 
 | ||||||
| } // namespace Service::Audio
 | } // namespace Service::Audio
 | ||||||
|  |  | ||||||
|  | @ -40,8 +40,8 @@ enum class AudioState : u32 { | ||||||
| 
 | 
 | ||||||
| class IAudioOut final : public ServiceFramework<IAudioOut> { | class IAudioOut final : public ServiceFramework<IAudioOut> { | ||||||
| public: | public: | ||||||
|     IAudioOut(AudoutParams audio_params, AudioCore::AudioOut& audio_core, std::string&& device_name, |     IAudioOut(Core::System& system, AudoutParams audio_params, AudioCore::AudioOut& audio_core, | ||||||
|               std::string&& unique_name) |               std::string&& device_name, std::string&& unique_name) | ||||||
|         : ServiceFramework("IAudioOut"), audio_core(audio_core), |         : ServiceFramework("IAudioOut"), audio_core(audio_core), | ||||||
|           device_name(std::move(device_name)), audio_params(audio_params) { |           device_name(std::move(device_name)), audio_params(audio_params) { | ||||||
|         // clang-format off
 |         // clang-format off
 | ||||||
|  | @ -65,7 +65,6 @@ public: | ||||||
|         RegisterHandlers(functions); |         RegisterHandlers(functions); | ||||||
| 
 | 
 | ||||||
|         // This is the event handle used to check if the audio buffer was released
 |         // This is the event handle used to check if the audio buffer was released
 | ||||||
|         auto& system = Core::System::GetInstance(); |  | ||||||
|         buffer_event = Kernel::WritableEvent::CreateEventPair( |         buffer_event = Kernel::WritableEvent::CreateEventPair( | ||||||
|             system.Kernel(), Kernel::ResetType::Manual, "IAudioOutBufferReleased"); |             system.Kernel(), Kernel::ResetType::Manual, "IAudioOutBufferReleased"); | ||||||
| 
 | 
 | ||||||
|  | @ -212,6 +211,22 @@ private: | ||||||
|     Kernel::EventPair buffer_event; |     Kernel::EventPair buffer_event; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  | AudOutU::AudOutU(Core::System& system_) : ServiceFramework("audout:u"), system{system_} { | ||||||
|  |     // clang-format off
 | ||||||
|  |     static const FunctionInfo functions[] = { | ||||||
|  |         {0, &AudOutU::ListAudioOutsImpl, "ListAudioOuts"}, | ||||||
|  |         {1, &AudOutU::OpenAudioOutImpl, "OpenAudioOut"}, | ||||||
|  |         {2, &AudOutU::ListAudioOutsImpl, "ListAudioOutsAuto"}, | ||||||
|  |         {3, &AudOutU::OpenAudioOutImpl, "OpenAudioOutAuto"}, | ||||||
|  |     }; | ||||||
|  |     // clang-format on
 | ||||||
|  | 
 | ||||||
|  |     RegisterHandlers(functions); | ||||||
|  |     audio_core = std::make_unique<AudioCore::AudioOut>(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | AudOutU::~AudOutU() = default; | ||||||
|  | 
 | ||||||
| void AudOutU::ListAudioOutsImpl(Kernel::HLERequestContext& ctx) { | void AudOutU::ListAudioOutsImpl(Kernel::HLERequestContext& ctx) { | ||||||
|     LOG_DEBUG(Service_Audio, "called"); |     LOG_DEBUG(Service_Audio, "called"); | ||||||
| 
 | 
 | ||||||
|  | @ -248,7 +263,7 @@ void AudOutU::OpenAudioOutImpl(Kernel::HLERequestContext& ctx) { | ||||||
| 
 | 
 | ||||||
|     std::string unique_name{fmt::format("{}-{}", device_name, audio_out_interfaces.size())}; |     std::string unique_name{fmt::format("{}-{}", device_name, audio_out_interfaces.size())}; | ||||||
|     auto audio_out_interface = std::make_shared<IAudioOut>( |     auto audio_out_interface = std::make_shared<IAudioOut>( | ||||||
|         params, *audio_core, std::move(device_name), std::move(unique_name)); |         system, params, *audio_core, std::move(device_name), std::move(unique_name)); | ||||||
| 
 | 
 | ||||||
|     IPC::ResponseBuilder rb{ctx, 6, 0, 1}; |     IPC::ResponseBuilder rb{ctx, 6, 0, 1}; | ||||||
|     rb.Push(RESULT_SUCCESS); |     rb.Push(RESULT_SUCCESS); | ||||||
|  | @ -256,20 +271,9 @@ void AudOutU::OpenAudioOutImpl(Kernel::HLERequestContext& ctx) { | ||||||
|     rb.Push<u32>(params.channel_count); |     rb.Push<u32>(params.channel_count); | ||||||
|     rb.Push<u32>(static_cast<u32>(AudioCore::Codec::PcmFormat::Int16)); |     rb.Push<u32>(static_cast<u32>(AudioCore::Codec::PcmFormat::Int16)); | ||||||
|     rb.Push<u32>(static_cast<u32>(AudioState::Stopped)); |     rb.Push<u32>(static_cast<u32>(AudioState::Stopped)); | ||||||
|     rb.PushIpcInterface<Audio::IAudioOut>(audio_out_interface); |     rb.PushIpcInterface<IAudioOut>(audio_out_interface); | ||||||
| 
 | 
 | ||||||
|     audio_out_interfaces.push_back(std::move(audio_out_interface)); |     audio_out_interfaces.push_back(std::move(audio_out_interface)); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| AudOutU::AudOutU() : ServiceFramework("audout:u") { |  | ||||||
|     static const FunctionInfo functions[] = {{0, &AudOutU::ListAudioOutsImpl, "ListAudioOuts"}, |  | ||||||
|                                              {1, &AudOutU::OpenAudioOutImpl, "OpenAudioOut"}, |  | ||||||
|                                              {2, &AudOutU::ListAudioOutsImpl, "ListAudioOutsAuto"}, |  | ||||||
|                                              {3, &AudOutU::OpenAudioOutImpl, "OpenAudioOutAuto"}}; |  | ||||||
|     RegisterHandlers(functions); |  | ||||||
|     audio_core = std::make_unique<AudioCore::AudioOut>(); |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| AudOutU::~AudOutU() = default; |  | ||||||
| 
 |  | ||||||
| } // namespace Service::Audio
 | } // namespace Service::Audio
 | ||||||
|  |  | ||||||
|  | @ -11,6 +11,10 @@ namespace AudioCore { | ||||||
| class AudioOut; | class AudioOut; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | namespace Core { | ||||||
|  | class System; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| namespace Kernel { | namespace Kernel { | ||||||
| class HLERequestContext; | class HLERequestContext; | ||||||
| } | } | ||||||
|  | @ -21,15 +25,17 @@ class IAudioOut; | ||||||
| 
 | 
 | ||||||
| class AudOutU final : public ServiceFramework<AudOutU> { | class AudOutU final : public ServiceFramework<AudOutU> { | ||||||
| public: | public: | ||||||
|     AudOutU(); |     explicit AudOutU(Core::System& system_); | ||||||
|     ~AudOutU() override; |     ~AudOutU() override; | ||||||
| 
 | 
 | ||||||
| private: | private: | ||||||
|  |     void ListAudioOutsImpl(Kernel::HLERequestContext& ctx); | ||||||
|  |     void OpenAudioOutImpl(Kernel::HLERequestContext& ctx); | ||||||
|  | 
 | ||||||
|     std::vector<std::shared_ptr<IAudioOut>> audio_out_interfaces; |     std::vector<std::shared_ptr<IAudioOut>> audio_out_interfaces; | ||||||
|     std::unique_ptr<AudioCore::AudioOut> audio_core; |     std::unique_ptr<AudioCore::AudioOut> audio_core; | ||||||
| 
 | 
 | ||||||
|     void ListAudioOutsImpl(Kernel::HLERequestContext& ctx); |     Core::System& system; | ||||||
|     void OpenAudioOutImpl(Kernel::HLERequestContext& ctx); |  | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| } // namespace Service::Audio
 | } // namespace Service::Audio
 | ||||||
|  |  | ||||||
|  | @ -5,6 +5,7 @@ | ||||||
| #include <algorithm> | #include <algorithm> | ||||||
| #include <array> | #include <array> | ||||||
| #include <memory> | #include <memory> | ||||||
|  | #include <string_view> | ||||||
| 
 | 
 | ||||||
| #include "audio_core/audio_renderer.h" | #include "audio_core/audio_renderer.h" | ||||||
| #include "common/alignment.h" | #include "common/alignment.h" | ||||||
|  | @ -25,7 +26,7 @@ namespace Service::Audio { | ||||||
| 
 | 
 | ||||||
| class IAudioRenderer final : public ServiceFramework<IAudioRenderer> { | class IAudioRenderer final : public ServiceFramework<IAudioRenderer> { | ||||||
| public: | public: | ||||||
|     explicit IAudioRenderer(AudioCore::AudioRendererParameter audren_params, |     explicit IAudioRenderer(Core::System& system, AudioCore::AudioRendererParameter audren_params, | ||||||
|                             const std::size_t instance_number) |                             const std::size_t instance_number) | ||||||
|         : ServiceFramework("IAudioRenderer") { |         : ServiceFramework("IAudioRenderer") { | ||||||
|         // clang-format off
 |         // clang-format off
 | ||||||
|  | @ -46,7 +47,6 @@ public: | ||||||
|         // clang-format on
 |         // clang-format on
 | ||||||
|         RegisterHandlers(functions); |         RegisterHandlers(functions); | ||||||
| 
 | 
 | ||||||
|         auto& system = Core::System::GetInstance(); |  | ||||||
|         system_event = Kernel::WritableEvent::CreateEventPair( |         system_event = Kernel::WritableEvent::CreateEventPair( | ||||||
|             system.Kernel(), Kernel::ResetType::Manual, "IAudioRenderer:SystemEvent"); |             system.Kernel(), Kernel::ResetType::Manual, "IAudioRenderer:SystemEvent"); | ||||||
|         renderer = std::make_unique<AudioCore::AudioRenderer>( |         renderer = std::make_unique<AudioCore::AudioRenderer>( | ||||||
|  | @ -160,7 +160,8 @@ private: | ||||||
| 
 | 
 | ||||||
| class IAudioDevice final : public ServiceFramework<IAudioDevice> { | class IAudioDevice final : public ServiceFramework<IAudioDevice> { | ||||||
| public: | public: | ||||||
|     IAudioDevice() : ServiceFramework("IAudioDevice") { |     explicit IAudioDevice(Core::System& system, u32_le revision_num) | ||||||
|  |         : ServiceFramework("IAudioDevice"), revision{revision_num} { | ||||||
|         static const FunctionInfo functions[] = { |         static const FunctionInfo functions[] = { | ||||||
|             {0, &IAudioDevice::ListAudioDeviceName, "ListAudioDeviceName"}, |             {0, &IAudioDevice::ListAudioDeviceName, "ListAudioDeviceName"}, | ||||||
|             {1, &IAudioDevice::SetAudioDeviceOutputVolume, "SetAudioDeviceOutputVolume"}, |             {1, &IAudioDevice::SetAudioDeviceOutputVolume, "SetAudioDeviceOutputVolume"}, | ||||||
|  | @ -178,7 +179,7 @@ public: | ||||||
|         }; |         }; | ||||||
|         RegisterHandlers(functions); |         RegisterHandlers(functions); | ||||||
| 
 | 
 | ||||||
|         auto& kernel = Core::System::GetInstance().Kernel(); |         auto& kernel = system.Kernel(); | ||||||
|         buffer_event = Kernel::WritableEvent::CreateEventPair(kernel, Kernel::ResetType::Automatic, |         buffer_event = Kernel::WritableEvent::CreateEventPair(kernel, Kernel::ResetType::Automatic, | ||||||
|                                                               "IAudioOutBufferReleasedEvent"); |                                                               "IAudioOutBufferReleasedEvent"); | ||||||
| 
 | 
 | ||||||
|  | @ -189,15 +190,47 @@ public: | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
| private: | private: | ||||||
|     void ListAudioDeviceName(Kernel::HLERequestContext& ctx) { |     using AudioDeviceName = std::array<char, 256>; | ||||||
|         LOG_WARNING(Service_Audio, "(STUBBED) called"); |     static constexpr std::array<std::string_view, 4> audio_device_names{{ | ||||||
|  |         "AudioStereoJackOutput", | ||||||
|  |         "AudioBuiltInSpeakerOutput", | ||||||
|  |         "AudioTvOutput", | ||||||
|  |         "AudioUsbDeviceOutput", | ||||||
|  |     }}; | ||||||
|  |     enum class DeviceType { | ||||||
|  |         AHUBHeadphones, | ||||||
|  |         AHUBSpeakers, | ||||||
|  |         HDA, | ||||||
|  |         USBOutput, | ||||||
|  |     }; | ||||||
| 
 | 
 | ||||||
|         constexpr std::array<char, 15> audio_interface{{"AudioInterface"}}; |     void ListAudioDeviceName(Kernel::HLERequestContext& ctx) { | ||||||
|         ctx.WriteBuffer(audio_interface); |         LOG_DEBUG(Service_Audio, "called"); | ||||||
|  | 
 | ||||||
|  |         const bool usb_output_supported = | ||||||
|  |             IsFeatureSupported(AudioFeatures::AudioUSBDeviceOutput, revision); | ||||||
|  |         const std::size_t count = ctx.GetWriteBufferSize() / sizeof(AudioDeviceName); | ||||||
|  | 
 | ||||||
|  |         std::vector<AudioDeviceName> name_buffer; | ||||||
|  |         name_buffer.reserve(audio_device_names.size()); | ||||||
|  | 
 | ||||||
|  |         for (std::size_t i = 0; i < count && i < audio_device_names.size(); i++) { | ||||||
|  |             const auto type = static_cast<DeviceType>(i); | ||||||
|  | 
 | ||||||
|  |             if (!usb_output_supported && type == DeviceType::USBOutput) { | ||||||
|  |                 continue; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             const auto& device_name = audio_device_names[i]; | ||||||
|  |             auto& entry = name_buffer.emplace_back(); | ||||||
|  |             device_name.copy(entry.data(), device_name.size()); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         ctx.WriteBuffer(name_buffer); | ||||||
| 
 | 
 | ||||||
|         IPC::ResponseBuilder rb{ctx, 3}; |         IPC::ResponseBuilder rb{ctx, 3}; | ||||||
|         rb.Push(RESULT_SUCCESS); |         rb.Push(RESULT_SUCCESS); | ||||||
|         rb.Push<u32>(1); |         rb.Push(static_cast<u32>(name_buffer.size())); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     void SetAudioDeviceOutputVolume(Kernel::HLERequestContext& ctx) { |     void SetAudioDeviceOutputVolume(Kernel::HLERequestContext& ctx) { | ||||||
|  | @ -216,12 +249,16 @@ private: | ||||||
|     void GetActiveAudioDeviceName(Kernel::HLERequestContext& ctx) { |     void GetActiveAudioDeviceName(Kernel::HLERequestContext& ctx) { | ||||||
|         LOG_WARNING(Service_Audio, "(STUBBED) called"); |         LOG_WARNING(Service_Audio, "(STUBBED) called"); | ||||||
| 
 | 
 | ||||||
|         constexpr std::array<char, 12> audio_interface{{"AudioDevice"}}; |         // Currently set to always be TV audio output.
 | ||||||
|         ctx.WriteBuffer(audio_interface); |         const auto& device_name = audio_device_names[2]; | ||||||
| 
 | 
 | ||||||
|         IPC::ResponseBuilder rb{ctx, 3}; |         AudioDeviceName out_device_name{}; | ||||||
|  |         device_name.copy(out_device_name.data(), device_name.size()); | ||||||
|  | 
 | ||||||
|  |         ctx.WriteBuffer(out_device_name); | ||||||
|  | 
 | ||||||
|  |         IPC::ResponseBuilder rb{ctx, 2}; | ||||||
|         rb.Push(RESULT_SUCCESS); |         rb.Push(RESULT_SUCCESS); | ||||||
|         rb.Push<u32>(1); |  | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     void QueryAudioDeviceSystemEvent(Kernel::HLERequestContext& ctx) { |     void QueryAudioDeviceSystemEvent(Kernel::HLERequestContext& ctx) { | ||||||
|  | @ -250,12 +287,13 @@ private: | ||||||
|         rb.PushCopyObjects(audio_output_device_switch_event.readable); |         rb.PushCopyObjects(audio_output_device_switch_event.readable); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     u32_le revision = 0; | ||||||
|     Kernel::EventPair buffer_event; |     Kernel::EventPair buffer_event; | ||||||
|     Kernel::EventPair audio_output_device_switch_event; |     Kernel::EventPair audio_output_device_switch_event; | ||||||
| 
 | 
 | ||||||
| }; // namespace Audio
 | }; // namespace Audio
 | ||||||
| 
 | 
 | ||||||
| AudRenU::AudRenU() : ServiceFramework("audren:u") { | AudRenU::AudRenU(Core::System& system_) : ServiceFramework("audren:u"), system{system_} { | ||||||
|     // clang-format off
 |     // clang-format off
 | ||||||
|     static const FunctionInfo functions[] = { |     static const FunctionInfo functions[] = { | ||||||
|         {0, &AudRenU::OpenAudioRenderer, "OpenAudioRenderer"}, |         {0, &AudRenU::OpenAudioRenderer, "OpenAudioRenderer"}, | ||||||
|  | @ -328,7 +366,7 @@ void AudRenU::GetAudioRendererWorkBufferSize(Kernel::HLERequestContext& ctx) { | ||||||
|     }; |     }; | ||||||
| 
 | 
 | ||||||
|     // Calculates the portion of the size related to the mix data (and the sorting thereof).
 |     // Calculates the portion of the size related to the mix data (and the sorting thereof).
 | ||||||
|     const auto calculate_mix_info_size = [this](const AudioCore::AudioRendererParameter& params) { |     const auto calculate_mix_info_size = [](const AudioCore::AudioRendererParameter& params) { | ||||||
|         // The size of the mixing info data structure.
 |         // The size of the mixing info data structure.
 | ||||||
|         constexpr u64 mix_info_size = 0x940; |         constexpr u64 mix_info_size = 0x940; | ||||||
| 
 | 
 | ||||||
|  | @ -400,7 +438,7 @@ void AudRenU::GetAudioRendererWorkBufferSize(Kernel::HLERequestContext& ctx) { | ||||||
| 
 | 
 | ||||||
|     // Calculates the part of the size related to the splitter context.
 |     // Calculates the part of the size related to the splitter context.
 | ||||||
|     const auto calculate_splitter_context_size = |     const auto calculate_splitter_context_size = | ||||||
|         [this](const AudioCore::AudioRendererParameter& params) -> u64 { |         [](const AudioCore::AudioRendererParameter& params) -> u64 { | ||||||
|         if (!IsFeatureSupported(AudioFeatures::Splitter, params.revision)) { |         if (!IsFeatureSupported(AudioFeatures::Splitter, params.revision)) { | ||||||
|             return 0; |             return 0; | ||||||
|         } |         } | ||||||
|  | @ -447,7 +485,7 @@ void AudRenU::GetAudioRendererWorkBufferSize(Kernel::HLERequestContext& ctx) { | ||||||
|     }; |     }; | ||||||
| 
 | 
 | ||||||
|     // Calculates the part of the size related to performance statistics.
 |     // Calculates the part of the size related to performance statistics.
 | ||||||
|     const auto calculate_perf_size = [this](const AudioCore::AudioRendererParameter& params) { |     const auto calculate_perf_size = [](const AudioCore::AudioRendererParameter& params) { | ||||||
|         // Extra size value appended to the end of the calculation.
 |         // Extra size value appended to the end of the calculation.
 | ||||||
|         constexpr u64 appended = 128; |         constexpr u64 appended = 128; | ||||||
| 
 | 
 | ||||||
|  | @ -474,78 +512,76 @@ void AudRenU::GetAudioRendererWorkBufferSize(Kernel::HLERequestContext& ctx) { | ||||||
|     }; |     }; | ||||||
| 
 | 
 | ||||||
|     // Calculates the part of the size that relates to the audio command buffer.
 |     // Calculates the part of the size that relates to the audio command buffer.
 | ||||||
|     const auto calculate_command_buffer_size = |     const auto calculate_command_buffer_size = [](const AudioCore::AudioRendererParameter& params) { | ||||||
|         [this](const AudioCore::AudioRendererParameter& params) { |         constexpr u64 alignment = (buffer_alignment_size - 1) * 2; | ||||||
|             constexpr u64 alignment = (buffer_alignment_size - 1) * 2; |  | ||||||
| 
 | 
 | ||||||
|             if (!IsFeatureSupported(AudioFeatures::VariadicCommandBuffer, params.revision)) { |         if (!IsFeatureSupported(AudioFeatures::VariadicCommandBuffer, params.revision)) { | ||||||
|                 constexpr u64 command_buffer_size = 0x18000; |             constexpr u64 command_buffer_size = 0x18000; | ||||||
| 
 | 
 | ||||||
|                 return command_buffer_size + alignment; |             return command_buffer_size + alignment; | ||||||
|             } |         } | ||||||
| 
 | 
 | ||||||
|             // When the variadic command buffer is supported, this means
 |         // When the variadic command buffer is supported, this means
 | ||||||
|             // the command generator for the audio renderer can issue commands
 |         // the command generator for the audio renderer can issue commands
 | ||||||
|             // that are (as one would expect), variable in size. So what we need to do
 |         // that are (as one would expect), variable in size. So what we need to do
 | ||||||
|             // is determine the maximum possible size for a few command data structures
 |         // is determine the maximum possible size for a few command data structures
 | ||||||
|             // then multiply them by the amount of present commands indicated by the given
 |         // then multiply them by the amount of present commands indicated by the given
 | ||||||
|             // respective audio parameters.
 |         // respective audio parameters.
 | ||||||
| 
 | 
 | ||||||
|             constexpr u64 max_biquad_filters = 2; |         constexpr u64 max_biquad_filters = 2; | ||||||
|             constexpr u64 max_mix_buffers = 24; |         constexpr u64 max_mix_buffers = 24; | ||||||
| 
 | 
 | ||||||
|             constexpr u64 biquad_filter_command_size = 0x2C; |         constexpr u64 biquad_filter_command_size = 0x2C; | ||||||
| 
 | 
 | ||||||
|             constexpr u64 depop_mix_command_size = 0x24; |         constexpr u64 depop_mix_command_size = 0x24; | ||||||
|             constexpr u64 depop_setup_command_size = 0x50; |         constexpr u64 depop_setup_command_size = 0x50; | ||||||
| 
 | 
 | ||||||
|             constexpr u64 effect_command_max_size = 0x540; |         constexpr u64 effect_command_max_size = 0x540; | ||||||
| 
 | 
 | ||||||
|             constexpr u64 mix_command_size = 0x1C; |         constexpr u64 mix_command_size = 0x1C; | ||||||
|             constexpr u64 mix_ramp_command_size = 0x24; |         constexpr u64 mix_ramp_command_size = 0x24; | ||||||
|             constexpr u64 mix_ramp_grouped_command_size = 0x13C; |         constexpr u64 mix_ramp_grouped_command_size = 0x13C; | ||||||
| 
 | 
 | ||||||
|             constexpr u64 perf_command_size = 0x28; |         constexpr u64 perf_command_size = 0x28; | ||||||
| 
 | 
 | ||||||
|             constexpr u64 sink_command_size = 0x130; |         constexpr u64 sink_command_size = 0x130; | ||||||
| 
 | 
 | ||||||
|             constexpr u64 submix_command_max_size = |         constexpr u64 submix_command_max_size = | ||||||
|                 depop_mix_command_size + (mix_command_size * max_mix_buffers) * max_mix_buffers; |             depop_mix_command_size + (mix_command_size * max_mix_buffers) * max_mix_buffers; | ||||||
| 
 | 
 | ||||||
|             constexpr u64 volume_command_size = 0x1C; |         constexpr u64 volume_command_size = 0x1C; | ||||||
|             constexpr u64 volume_ramp_command_size = 0x20; |         constexpr u64 volume_ramp_command_size = 0x20; | ||||||
| 
 | 
 | ||||||
|             constexpr u64 voice_biquad_filter_command_size = |         constexpr u64 voice_biquad_filter_command_size = | ||||||
|                 biquad_filter_command_size * max_biquad_filters; |             biquad_filter_command_size * max_biquad_filters; | ||||||
|             constexpr u64 voice_data_command_size = 0x9C; |         constexpr u64 voice_data_command_size = 0x9C; | ||||||
|             const u64 voice_command_max_size = |         const u64 voice_command_max_size = | ||||||
|                 (params.splitter_count * depop_setup_command_size) + |             (params.splitter_count * depop_setup_command_size) + | ||||||
|                 (voice_data_command_size + voice_biquad_filter_command_size + |             (voice_data_command_size + voice_biquad_filter_command_size + volume_ramp_command_size + | ||||||
|                  volume_ramp_command_size + mix_ramp_grouped_command_size); |              mix_ramp_grouped_command_size); | ||||||
| 
 | 
 | ||||||
|             // Now calculate the individual elements that comprise the size and add them together.
 |         // Now calculate the individual elements that comprise the size and add them together.
 | ||||||
|             const u64 effect_commands_size = params.effect_count * effect_command_max_size; |         const u64 effect_commands_size = params.effect_count * effect_command_max_size; | ||||||
| 
 | 
 | ||||||
|             const u64 final_mix_commands_size = |         const u64 final_mix_commands_size = | ||||||
|                 depop_mix_command_size + volume_command_size * max_mix_buffers; |             depop_mix_command_size + volume_command_size * max_mix_buffers; | ||||||
| 
 | 
 | ||||||
|             const u64 perf_commands_size = |         const u64 perf_commands_size = | ||||||
|                 perf_command_size * |             perf_command_size * (CalculateNumPerformanceEntries(params) + max_perf_detail_entries); | ||||||
|                 (CalculateNumPerformanceEntries(params) + max_perf_detail_entries); |  | ||||||
| 
 | 
 | ||||||
|             const u64 sink_commands_size = params.sink_count * sink_command_size; |         const u64 sink_commands_size = params.sink_count * sink_command_size; | ||||||
| 
 | 
 | ||||||
|             const u64 splitter_commands_size = |         const u64 splitter_commands_size = | ||||||
|                 params.num_splitter_send_channels * max_mix_buffers * mix_ramp_command_size; |             params.num_splitter_send_channels * max_mix_buffers * mix_ramp_command_size; | ||||||
| 
 | 
 | ||||||
|             const u64 submix_commands_size = params.submix_count * submix_command_max_size; |         const u64 submix_commands_size = params.submix_count * submix_command_max_size; | ||||||
| 
 | 
 | ||||||
|             const u64 voice_commands_size = params.voice_count * voice_command_max_size; |         const u64 voice_commands_size = params.voice_count * voice_command_max_size; | ||||||
| 
 | 
 | ||||||
|             return effect_commands_size + final_mix_commands_size + perf_commands_size + |         return effect_commands_size + final_mix_commands_size + perf_commands_size + | ||||||
|                    sink_commands_size + splitter_commands_size + submix_commands_size + |                sink_commands_size + splitter_commands_size + submix_commands_size + | ||||||
|                    voice_commands_size + alignment; |                voice_commands_size + alignment; | ||||||
|         }; |     }; | ||||||
| 
 | 
 | ||||||
|     IPC::RequestParser rp{ctx}; |     IPC::RequestParser rp{ctx}; | ||||||
|     const auto params = rp.PopRaw<AudioCore::AudioRendererParameter>(); |     const auto params = rp.PopRaw<AudioCore::AudioRendererParameter>(); | ||||||
|  | @ -578,12 +614,16 @@ void AudRenU::GetAudioRendererWorkBufferSize(Kernel::HLERequestContext& ctx) { | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void AudRenU::GetAudioDeviceService(Kernel::HLERequestContext& ctx) { | void AudRenU::GetAudioDeviceService(Kernel::HLERequestContext& ctx) { | ||||||
|     LOG_DEBUG(Service_Audio, "called"); |     IPC::RequestParser rp{ctx}; | ||||||
|  |     const u64 aruid = rp.Pop<u64>(); | ||||||
| 
 | 
 | ||||||
|  |     LOG_DEBUG(Service_Audio, "called. aruid={:016X}", aruid); | ||||||
|  | 
 | ||||||
|  |     // Revisionless variant of GetAudioDeviceServiceWithRevisionInfo that
 | ||||||
|  |     // always assumes the initial release revision (REV1).
 | ||||||
|     IPC::ResponseBuilder rb{ctx, 2, 0, 1}; |     IPC::ResponseBuilder rb{ctx, 2, 0, 1}; | ||||||
| 
 |  | ||||||
|     rb.Push(RESULT_SUCCESS); |     rb.Push(RESULT_SUCCESS); | ||||||
|     rb.PushIpcInterface<Audio::IAudioDevice>(); |     rb.PushIpcInterface<IAudioDevice>(system, Common::MakeMagic('R', 'E', 'V', '1')); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void AudRenU::OpenAudioRendererAuto(Kernel::HLERequestContext& ctx) { | void AudRenU::OpenAudioRendererAuto(Kernel::HLERequestContext& ctx) { | ||||||
|  | @ -593,13 +633,19 @@ void AudRenU::OpenAudioRendererAuto(Kernel::HLERequestContext& ctx) { | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void AudRenU::GetAudioDeviceServiceWithRevisionInfo(Kernel::HLERequestContext& ctx) { | void AudRenU::GetAudioDeviceServiceWithRevisionInfo(Kernel::HLERequestContext& ctx) { | ||||||
|     LOG_WARNING(Service_Audio, "(STUBBED) called"); |     struct Parameters { | ||||||
|  |         u32 revision; | ||||||
|  |         u64 aruid; | ||||||
|  |     }; | ||||||
|  | 
 | ||||||
|  |     IPC::RequestParser rp{ctx}; | ||||||
|  |     const auto [revision, aruid] = rp.PopRaw<Parameters>(); | ||||||
|  | 
 | ||||||
|  |     LOG_DEBUG(Service_Audio, "called. revision={:08X}, aruid={:016X}", revision, aruid); | ||||||
| 
 | 
 | ||||||
|     IPC::ResponseBuilder rb{ctx, 2, 0, 1}; |     IPC::ResponseBuilder rb{ctx, 2, 0, 1}; | ||||||
| 
 |  | ||||||
|     rb.Push(RESULT_SUCCESS); |     rb.Push(RESULT_SUCCESS); | ||||||
|     rb.PushIpcInterface<Audio::IAudioDevice>(); // TODO(ogniK): Figure out what is different
 |     rb.PushIpcInterface<IAudioDevice>(system, revision); | ||||||
|                                                 // based on the current revision
 |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void AudRenU::OpenAudioRendererImpl(Kernel::HLERequestContext& ctx) { | void AudRenU::OpenAudioRendererImpl(Kernel::HLERequestContext& ctx) { | ||||||
|  | @ -608,14 +654,16 @@ void AudRenU::OpenAudioRendererImpl(Kernel::HLERequestContext& ctx) { | ||||||
|     IPC::ResponseBuilder rb{ctx, 2, 0, 1}; |     IPC::ResponseBuilder rb{ctx, 2, 0, 1}; | ||||||
| 
 | 
 | ||||||
|     rb.Push(RESULT_SUCCESS); |     rb.Push(RESULT_SUCCESS); | ||||||
|     rb.PushIpcInterface<IAudioRenderer>(params, audren_instance_count++); |     rb.PushIpcInterface<IAudioRenderer>(system, params, audren_instance_count++); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| bool AudRenU::IsFeatureSupported(AudioFeatures feature, u32_le revision) const { | bool IsFeatureSupported(AudioFeatures feature, u32_le revision) { | ||||||
|     // Byte swap
 |     // Byte swap
 | ||||||
|     const u32_be version_num = revision - Common::MakeMagic('R', 'E', 'V', '0'); |     const u32_be version_num = revision - Common::MakeMagic('R', 'E', 'V', '0'); | ||||||
| 
 | 
 | ||||||
|     switch (feature) { |     switch (feature) { | ||||||
|  |     case AudioFeatures::AudioUSBDeviceOutput: | ||||||
|  |         return version_num >= 4U; | ||||||
|     case AudioFeatures::Splitter: |     case AudioFeatures::Splitter: | ||||||
|         return version_num >= 2U; |         return version_num >= 2U; | ||||||
|     case AudioFeatures::PerformanceMetricsVersion2: |     case AudioFeatures::PerformanceMetricsVersion2: | ||||||
|  |  | ||||||
|  | @ -6,6 +6,10 @@ | ||||||
| 
 | 
 | ||||||
| #include "core/hle/service/service.h" | #include "core/hle/service/service.h" | ||||||
| 
 | 
 | ||||||
|  | namespace Core { | ||||||
|  | class System; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| namespace Kernel { | namespace Kernel { | ||||||
| class HLERequestContext; | class HLERequestContext; | ||||||
| } | } | ||||||
|  | @ -14,7 +18,7 @@ namespace Service::Audio { | ||||||
| 
 | 
 | ||||||
| class AudRenU final : public ServiceFramework<AudRenU> { | class AudRenU final : public ServiceFramework<AudRenU> { | ||||||
| public: | public: | ||||||
|     explicit AudRenU(); |     explicit AudRenU(Core::System& system_); | ||||||
|     ~AudRenU() override; |     ~AudRenU() override; | ||||||
| 
 | 
 | ||||||
| private: | private: | ||||||
|  | @ -26,14 +30,19 @@ private: | ||||||
| 
 | 
 | ||||||
|     void OpenAudioRendererImpl(Kernel::HLERequestContext& ctx); |     void OpenAudioRendererImpl(Kernel::HLERequestContext& ctx); | ||||||
| 
 | 
 | ||||||
|     enum class AudioFeatures : u32 { |  | ||||||
|         Splitter, |  | ||||||
|         PerformanceMetricsVersion2, |  | ||||||
|         VariadicCommandBuffer, |  | ||||||
|     }; |  | ||||||
| 
 |  | ||||||
|     bool IsFeatureSupported(AudioFeatures feature, u32_le revision) const; |  | ||||||
|     std::size_t audren_instance_count = 0; |     std::size_t audren_instance_count = 0; | ||||||
|  |     Core::System& system; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  | // Describes a particular audio feature that may be supported in a particular revision.
 | ||||||
|  | enum class AudioFeatures : u32 { | ||||||
|  |     AudioUSBDeviceOutput, | ||||||
|  |     Splitter, | ||||||
|  |     PerformanceMetricsVersion2, | ||||||
|  |     VariadicCommandBuffer, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | // Tests if a particular audio feature is supported with a given audio revision.
 | ||||||
|  | bool IsFeatureSupported(AudioFeatures feature, u32_le revision); | ||||||
|  | 
 | ||||||
| } // namespace Service::Audio
 | } // namespace Service::Audio
 | ||||||
|  |  | ||||||
|  | @ -206,7 +206,7 @@ void Init(std::shared_ptr<SM::ServiceManager>& sm, Core::System& system) { | ||||||
|     AM::InstallInterfaces(*sm, nv_flinger, system); |     AM::InstallInterfaces(*sm, nv_flinger, system); | ||||||
|     AOC::InstallInterfaces(*sm); |     AOC::InstallInterfaces(*sm); | ||||||
|     APM::InstallInterfaces(system); |     APM::InstallInterfaces(system); | ||||||
|     Audio::InstallInterfaces(*sm); |     Audio::InstallInterfaces(*sm, system); | ||||||
|     BCAT::InstallInterfaces(*sm); |     BCAT::InstallInterfaces(*sm); | ||||||
|     BPC::InstallInterfaces(*sm); |     BPC::InstallInterfaces(*sm); | ||||||
|     BtDrv::InstallInterfaces(*sm); |     BtDrv::InstallInterfaces(*sm); | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 bunnei
						bunnei