diff options
83 files changed, 713 insertions, 370 deletions
| diff --git a/src/audio_core/audio_in_manager.cpp b/src/audio_core/audio_in_manager.cpp index f39fb4002..3dfb613cb 100644 --- a/src/audio_core/audio_in_manager.cpp +++ b/src/audio_core/audio_in_manager.cpp @@ -20,7 +20,7 @@ Manager::Manager(Core::System& system_) : system{system_} {  Result Manager::AcquireSessionId(size_t& session_id) {      if (num_free_sessions == 0) {          LOG_ERROR(Service_Audio, "All 4 AudioIn sessions are in use, cannot create any more"); -        return Service::Audio::ERR_MAXIMUM_SESSIONS_REACHED; +        return Service::Audio::ResultOutOfSessions;      }      session_id = session_ids[next_session_id];      next_session_id = (next_session_id + 1) % MaxInSessions; diff --git a/src/audio_core/audio_manager.cpp b/src/audio_core/audio_manager.cpp index 2acde668e..10b56f214 100644 --- a/src/audio_core/audio_manager.cpp +++ b/src/audio_core/audio_manager.cpp @@ -19,7 +19,7 @@ void AudioManager::Shutdown() {  Result AudioManager::SetOutManager(BufferEventFunc buffer_func) {      if (!running) { -        return Service::Audio::ERR_OPERATION_FAILED; +        return Service::Audio::ResultOperationFailed;      }      std::scoped_lock l{lock}; @@ -35,7 +35,7 @@ Result AudioManager::SetOutManager(BufferEventFunc buffer_func) {  Result AudioManager::SetInManager(BufferEventFunc buffer_func) {      if (!running) { -        return Service::Audio::ERR_OPERATION_FAILED; +        return Service::Audio::ResultOperationFailed;      }      std::scoped_lock l{lock}; diff --git a/src/audio_core/audio_out_manager.cpp b/src/audio_core/audio_out_manager.cpp index 1766efde1..f22821360 100644 --- a/src/audio_core/audio_out_manager.cpp +++ b/src/audio_core/audio_out_manager.cpp @@ -19,7 +19,7 @@ Manager::Manager(Core::System& system_) : system{system_} {  Result Manager::AcquireSessionId(size_t& session_id) {      if (num_free_sessions == 0) {          LOG_ERROR(Service_Audio, "All 12 Audio Out sessions are in use, cannot create any more"); -        return Service::Audio::ERR_MAXIMUM_SESSIONS_REACHED; +        return Service::Audio::ResultOutOfSessions;      }      session_id = session_ids[next_session_id];      next_session_id = (next_session_id + 1) % MaxOutSessions; diff --git a/src/audio_core/audio_render_manager.cpp b/src/audio_core/audio_render_manager.cpp index 7aba2b423..320715727 100644 --- a/src/audio_core/audio_render_manager.cpp +++ b/src/audio_core/audio_render_manager.cpp @@ -28,7 +28,7 @@ SystemManager& Manager::GetSystemManager() {  Result Manager::GetWorkBufferSize(const AudioRendererParameterInternal& params,                                    u64& out_count) const {      if (!CheckValidRevision(params.revision)) { -        return Service::Audio::ERR_INVALID_REVISION; +        return Service::Audio::ResultInvalidRevision;      }      out_count = System::GetWorkBufferSize(params); diff --git a/src/audio_core/in/audio_in.cpp b/src/audio_core/in/audio_in.cpp index 91ccd5ad7..df8c44d1f 100644 --- a/src/audio_core/in/audio_in.cpp +++ b/src/audio_core/in/audio_in.cpp @@ -46,7 +46,7 @@ Result In::AppendBuffer(const AudioInBuffer& buffer, u64 tag) {      if (system.AppendBuffer(buffer, tag)) {          return ResultSuccess;      } -    return Service::Audio::ERR_BUFFER_COUNT_EXCEEDED; +    return Service::Audio::ResultBufferCountReached;  }  void In::ReleaseAndRegisterBuffers() { diff --git a/src/audio_core/in/audio_in_system.cpp b/src/audio_core/in/audio_in_system.cpp index 934ef8c1c..e23e51758 100644 --- a/src/audio_core/in/audio_in_system.cpp +++ b/src/audio_core/in/audio_in_system.cpp @@ -45,11 +45,11 @@ Result System::IsConfigValid(const std::string_view device_name,                               const AudioInParameter& in_params) const {      if ((device_name.size() > 0) &&          (device_name != GetDefaultDeviceName() && device_name != GetDefaultUacDeviceName())) { -        return Service::Audio::ERR_INVALID_DEVICE_NAME; +        return Service::Audio::ResultNotFound;      }      if (in_params.sample_rate != TargetSampleRate && in_params.sample_rate > 0) { -        return Service::Audio::ERR_INVALID_SAMPLE_RATE; +        return Service::Audio::ResultInvalidSampleRate;      }      return ResultSuccess; @@ -80,7 +80,7 @@ Result System::Initialize(std::string device_name, const AudioInParameter& in_pa  Result System::Start() {      if (state != State::Stopped) { -        return Service::Audio::ERR_OPERATION_FAILED; +        return Service::Audio::ResultOperationFailed;      }      session->Initialize(name, sample_format, channel_count, session_id, handle, diff --git a/src/audio_core/out/audio_out.cpp b/src/audio_core/out/audio_out.cpp index d3ee4f0eb..b7ea13405 100644 --- a/src/audio_core/out/audio_out.cpp +++ b/src/audio_core/out/audio_out.cpp @@ -46,7 +46,7 @@ Result Out::AppendBuffer(const AudioOutBuffer& buffer, const u64 tag) {      if (system.AppendBuffer(buffer, tag)) {          return ResultSuccess;      } -    return Service::Audio::ERR_BUFFER_COUNT_EXCEEDED; +    return Service::Audio::ResultBufferCountReached;  }  void Out::ReleaseAndRegisterBuffers() { diff --git a/src/audio_core/out/audio_out_system.cpp b/src/audio_core/out/audio_out_system.cpp index e096a1dac..bd13f7219 100644 --- a/src/audio_core/out/audio_out_system.cpp +++ b/src/audio_core/out/audio_out_system.cpp @@ -33,11 +33,11 @@ std::string_view System::GetDefaultOutputDeviceName() const {  Result System::IsConfigValid(std::string_view device_name,                               const AudioOutParameter& in_params) const {      if ((device_name.size() > 0) && (device_name != GetDefaultOutputDeviceName())) { -        return Service::Audio::ERR_INVALID_DEVICE_NAME; +        return Service::Audio::ResultNotFound;      }      if (in_params.sample_rate != TargetSampleRate && in_params.sample_rate > 0) { -        return Service::Audio::ERR_INVALID_SAMPLE_RATE; +        return Service::Audio::ResultInvalidSampleRate;      }      if (in_params.channel_count == 0 || in_params.channel_count == 2 || @@ -45,7 +45,7 @@ Result System::IsConfigValid(std::string_view device_name,          return ResultSuccess;      } -    return Service::Audio::ERR_INVALID_CHANNEL_COUNT; +    return Service::Audio::ResultInvalidChannelCount;  }  Result System::Initialize(std::string device_name, const AudioOutParameter& in_params, u32 handle_, @@ -80,7 +80,7 @@ size_t System::GetSessionId() const {  Result System::Start() {      if (state != State::Stopped) { -        return Service::Audio::ERR_OPERATION_FAILED; +        return Service::Audio::ResultOperationFailed;      }      session->Initialize(name, sample_format, channel_count, session_id, handle, diff --git a/src/audio_core/renderer/adsp/audio_renderer.cpp b/src/audio_core/renderer/adsp/audio_renderer.cpp index 78c15629b..0e437e779 100644 --- a/src/audio_core/renderer/adsp/audio_renderer.cpp +++ b/src/audio_core/renderer/adsp/audio_renderer.cpp @@ -135,7 +135,7 @@ void AudioRenderer::ThreadFunc() {      static constexpr char name[]{"AudioRenderer"};      MicroProfileOnThreadCreate(name);      Common::SetCurrentThreadName(name); -    Common::SetCurrentThreadPriority(Common::ThreadPriority::Critical); +    Common::SetCurrentThreadPriority(Common::ThreadPriority::High);      if (mailbox->ADSPWaitMessage() != RenderMessage::AudioRenderer_InitializeOK) {          LOG_ERROR(Service_Audio,                    "ADSP Audio Renderer -- Failed to receive initialize message from host!"); diff --git a/src/audio_core/renderer/audio_renderer.cpp b/src/audio_core/renderer/audio_renderer.cpp index 51aa17599..a8257eb2e 100644 --- a/src/audio_core/renderer/audio_renderer.cpp +++ b/src/audio_core/renderer/audio_renderer.cpp @@ -22,7 +22,7 @@ Result Renderer::Initialize(const AudioRendererParameterInternal& params,          if (!manager.AddSystem(system)) {              LOG_ERROR(Service_Audio,                        "Both Audio Render sessions are in use, cannot create any more"); -            return Service::Audio::ERR_MAXIMUM_SESSIONS_REACHED; +            return Service::Audio::ResultOutOfSessions;          }          system_registered = true;      } diff --git a/src/audio_core/renderer/behavior/info_updater.cpp b/src/audio_core/renderer/behavior/info_updater.cpp index 574cf0982..e312eb166 100644 --- a/src/audio_core/renderer/behavior/info_updater.cpp +++ b/src/audio_core/renderer/behavior/info_updater.cpp @@ -48,7 +48,7 @@ Result InfoUpdater::UpdateVoiceChannelResources(VoiceContext& voice_context) {          LOG_ERROR(Service_Audio,                    "Consumed an incorrect voice resource size, header size={}, consumed={}",                    in_header->voice_resources_size, consumed_input_size); -        return Service::Audio::ERR_INVALID_UPDATE_DATA; +        return Service::Audio::ResultInvalidUpdateInfo;      }      input += consumed_input_size; @@ -123,7 +123,7 @@ Result InfoUpdater::UpdateVoices(VoiceContext& voice_context,      if (consumed_input_size != in_header->voices_size) {          LOG_ERROR(Service_Audio, "Consumed an incorrect voices size, header size={}, consumed={}",                    in_header->voices_size, consumed_input_size); -        return Service::Audio::ERR_INVALID_UPDATE_DATA; +        return Service::Audio::ResultInvalidUpdateInfo;      }      out_header->voices_size = consumed_output_size; @@ -184,7 +184,7 @@ Result InfoUpdater::UpdateEffectsVersion1(EffectContext& effect_context, const b      if (consumed_input_size != in_header->effects_size) {          LOG_ERROR(Service_Audio, "Consumed an incorrect effects size, header size={}, consumed={}",                    in_header->effects_size, consumed_input_size); -        return Service::Audio::ERR_INVALID_UPDATE_DATA; +        return Service::Audio::ResultInvalidUpdateInfo;      }      out_header->effects_size = consumed_output_size; @@ -239,7 +239,7 @@ Result InfoUpdater::UpdateEffectsVersion2(EffectContext& effect_context, const b      if (consumed_input_size != in_header->effects_size) {          LOG_ERROR(Service_Audio, "Consumed an incorrect effects size, header size={}, consumed={}",                    in_header->effects_size, consumed_input_size); -        return Service::Audio::ERR_INVALID_UPDATE_DATA; +        return Service::Audio::ResultInvalidUpdateInfo;      }      out_header->effects_size = consumed_output_size; @@ -267,7 +267,7 @@ Result InfoUpdater::UpdateMixes(MixContext& mix_context, const u32 mix_buffer_co      }      if (mix_buffer_count == 0) { -        return Service::Audio::ERR_INVALID_UPDATE_DATA; +        return Service::Audio::ResultInvalidUpdateInfo;      }      std::span<const MixInfo::InParameter> in_params{ @@ -281,13 +281,13 @@ Result InfoUpdater::UpdateMixes(MixContext& mix_context, const u32 mix_buffer_co              total_buffer_count += params.buffer_count;              if (params.dest_mix_id > static_cast<s32>(mix_context.GetCount()) &&                  params.dest_mix_id != UnusedMixId && params.mix_id != FinalMixId) { -                return Service::Audio::ERR_INVALID_UPDATE_DATA; +                return Service::Audio::ResultInvalidUpdateInfo;              }          }      }      if (total_buffer_count > mix_buffer_count) { -        return Service::Audio::ERR_INVALID_UPDATE_DATA; +        return Service::Audio::ResultInvalidUpdateInfo;      }      bool mix_dirty{false}; @@ -317,7 +317,7 @@ Result InfoUpdater::UpdateMixes(MixContext& mix_context, const u32 mix_buffer_co      if (mix_dirty) {          if (behaviour.IsSplitterSupported() && splitter_context.UsingSplitter()) {              if (!mix_context.TSortInfo(splitter_context)) { -                return Service::Audio::ERR_INVALID_UPDATE_DATA; +                return Service::Audio::ResultInvalidUpdateInfo;              }          } else {              mix_context.SortInfo(); @@ -327,7 +327,7 @@ Result InfoUpdater::UpdateMixes(MixContext& mix_context, const u32 mix_buffer_co      if (consumed_input_size != in_header->mix_size) {          LOG_ERROR(Service_Audio, "Consumed an incorrect mixes size, header size={}, consumed={}",                    in_header->mix_size, consumed_input_size); -        return Service::Audio::ERR_INVALID_UPDATE_DATA; +        return Service::Audio::ResultInvalidUpdateInfo;      }      input += mix_count * sizeof(MixInfo::InParameter); @@ -384,7 +384,7 @@ Result InfoUpdater::UpdateSinks(SinkContext& sink_context, std::span<MemoryPoolI      if (consumed_input_size != in_header->sinks_size) {          LOG_ERROR(Service_Audio, "Consumed an incorrect sinks size, header size={}, consumed={}",                    in_header->sinks_size, consumed_input_size); -        return Service::Audio::ERR_INVALID_UPDATE_DATA; +        return Service::Audio::ResultInvalidUpdateInfo;      }      input += consumed_input_size; @@ -411,7 +411,7 @@ Result InfoUpdater::UpdateMemoryPools(std::span<MemoryPoolInfo> memory_pools,              state != MemoryPoolInfo::ResultState::MapFailed &&              state != MemoryPoolInfo::ResultState::InUse) {              LOG_WARNING(Service_Audio, "Invalid ResultState from updating memory pools"); -            return Service::Audio::ERR_INVALID_UPDATE_DATA; +            return Service::Audio::ResultInvalidUpdateInfo;          }      } @@ -423,7 +423,7 @@ Result InfoUpdater::UpdateMemoryPools(std::span<MemoryPoolInfo> memory_pools,          LOG_ERROR(Service_Audio,                    "Consumed an incorrect memory pool size, header size={}, consumed={}",                    in_header->memory_pool_size, consumed_input_size); -        return Service::Audio::ERR_INVALID_UPDATE_DATA; +        return Service::Audio::ResultInvalidUpdateInfo;      }      input += consumed_input_size; @@ -453,7 +453,7 @@ Result InfoUpdater::UpdatePerformanceBuffer(std::span<u8> performance_output,          LOG_ERROR(Service_Audio,                    "Consumed an incorrect performance size, header size={}, consumed={}",                    in_header->performance_buffer_size, consumed_input_size); -        return Service::Audio::ERR_INVALID_UPDATE_DATA; +        return Service::Audio::ResultInvalidUpdateInfo;      }      input += consumed_input_size; @@ -467,18 +467,18 @@ Result InfoUpdater::UpdateBehaviorInfo(BehaviorInfo& behaviour_) {      const auto in_params{reinterpret_cast<const BehaviorInfo::InParameter*>(input)};      if (!CheckValidRevision(in_params->revision)) { -        return Service::Audio::ERR_INVALID_UPDATE_DATA; +        return Service::Audio::ResultInvalidUpdateInfo;      }      if (in_params->revision != behaviour_.GetUserRevision()) { -        return Service::Audio::ERR_INVALID_UPDATE_DATA; +        return Service::Audio::ResultInvalidUpdateInfo;      }      behaviour_.ClearError();      behaviour_.UpdateFlags(in_params->flags);      if (in_header->behaviour_size != sizeof(BehaviorInfo::InParameter)) { -        return Service::Audio::ERR_INVALID_UPDATE_DATA; +        return Service::Audio::ResultInvalidUpdateInfo;      }      input += sizeof(BehaviorInfo::InParameter); @@ -500,7 +500,7 @@ Result InfoUpdater::UpdateErrorInfo(const BehaviorInfo& behaviour_) {  Result InfoUpdater::UpdateSplitterInfo(SplitterContext& splitter_context) {      u32 consumed_size{0};      if (!splitter_context.Update(input, consumed_size)) { -        return Service::Audio::ERR_INVALID_UPDATE_DATA; +        return Service::Audio::ResultInvalidUpdateInfo;      }      input += consumed_size; @@ -529,9 +529,9 @@ Result InfoUpdater::UpdateRendererInfo(const u64 elapsed_frames) {  Result InfoUpdater::CheckConsumedSize() {      if (CpuAddr(input) - CpuAddr(input_origin.data()) != expected_input_size) { -        return Service::Audio::ERR_INVALID_UPDATE_DATA; +        return Service::Audio::ResultInvalidUpdateInfo;      } else if (CpuAddr(output) - CpuAddr(output_origin.data()) != expected_output_size) { -        return Service::Audio::ERR_INVALID_UPDATE_DATA; +        return Service::Audio::ResultInvalidUpdateInfo;      }      return ResultSuccess;  } diff --git a/src/audio_core/renderer/memory/pool_mapper.cpp b/src/audio_core/renderer/memory/pool_mapper.cpp index 2baf2ce08..7fd2b5f47 100644 --- a/src/audio_core/renderer/memory/pool_mapper.cpp +++ b/src/audio_core/renderer/memory/pool_mapper.cpp @@ -92,7 +92,7 @@ bool PoolMapper::TryAttachBuffer(BehaviorInfo::ErrorInfo& error_info, AddressInf      address_info.Setup(address, size);      if (!FillDspAddr(address_info)) { -        error_info.error_code = Service::Audio::ERR_POOL_MAPPING_FAILED; +        error_info.error_code = Service::Audio::ResultInvalidAddressInfo;          error_info.address = address;          return force_map;      } diff --git a/src/audio_core/renderer/system.cpp b/src/audio_core/renderer/system.cpp index 31cbee282..28f063641 100644 --- a/src/audio_core/renderer/system.cpp +++ b/src/audio_core/renderer/system.cpp @@ -101,15 +101,15 @@ Result System::Initialize(const AudioRendererParameterInternal& params,                            Kernel::KTransferMemory* transfer_memory, u64 transfer_memory_size,                            u32 process_handle_, u64 applet_resource_user_id_, s32 session_id_) {      if (!CheckValidRevision(params.revision)) { -        return Service::Audio::ERR_INVALID_REVISION; +        return Service::Audio::ResultInvalidRevision;      }      if (GetWorkBufferSize(params) > transfer_memory_size) { -        return Service::Audio::ERR_INSUFFICIENT_BUFFER_SIZE; +        return Service::Audio::ResultInsufficientBuffer;      }      if (process_handle_ == 0) { -        return Service::Audio::ERR_INVALID_PROCESS_HANDLE; +        return Service::Audio::ResultInvalidHandle;      }      behavior.SetUserLibRevision(params.revision); @@ -143,19 +143,19 @@ Result System::Initialize(const AudioRendererParameterInternal& params,      samples_workbuffer =          allocator.Allocate<s32>((voice_channels + mix_buffer_count) * sample_count, 0x10);      if (samples_workbuffer.empty()) { -        return Service::Audio::ERR_INSUFFICIENT_BUFFER_SIZE; +        return Service::Audio::ResultInsufficientBuffer;      }      auto upsampler_workbuffer{allocator.Allocate<s32>(          (voice_channels + mix_buffer_count) * TargetSampleCount * upsampler_count, 0x10)};      if (upsampler_workbuffer.empty()) { -        return Service::Audio::ERR_INSUFFICIENT_BUFFER_SIZE; +        return Service::Audio::ResultInsufficientBuffer;      }      depop_buffer =          allocator.Allocate<s32>(Common::AlignUp(static_cast<u32>(mix_buffer_count), 0x40), 0x40);      if (depop_buffer.empty()) { -        return Service::Audio::ERR_INSUFFICIENT_BUFFER_SIZE; +        return Service::Audio::ResultInsufficientBuffer;      }      // invalidate samples_workbuffer DSP cache @@ -166,12 +166,12 @@ Result System::Initialize(const AudioRendererParameterInternal& params,      }      if (voice_infos.empty()) { -        return Service::Audio::ERR_INSUFFICIENT_BUFFER_SIZE; +        return Service::Audio::ResultInsufficientBuffer;      }      auto sorted_voice_infos{allocator.Allocate<VoiceInfo*>(params.voices, 0x10)};      if (sorted_voice_infos.empty()) { -        return Service::Audio::ERR_INSUFFICIENT_BUFFER_SIZE; +        return Service::Audio::ResultInsufficientBuffer;      }      std::memset(sorted_voice_infos.data(), 0, sorted_voice_infos.size_bytes()); @@ -183,12 +183,12 @@ Result System::Initialize(const AudioRendererParameterInternal& params,      }      if (voice_channel_resources.empty()) { -        return Service::Audio::ERR_INSUFFICIENT_BUFFER_SIZE; +        return Service::Audio::ResultInsufficientBuffer;      }      auto voice_cpu_states{allocator.Allocate<VoiceState>(params.voices, 0x10)};      if (voice_cpu_states.empty()) { -        return Service::Audio::ERR_INSUFFICIENT_BUFFER_SIZE; +        return Service::Audio::ResultInsufficientBuffer;      }      for (auto& voice_state : voice_cpu_states) { @@ -198,7 +198,7 @@ Result System::Initialize(const AudioRendererParameterInternal& params,      auto mix_infos{allocator.Allocate<MixInfo>(params.sub_mixes + 1, 0x10)};      if (mix_infos.empty()) { -        return Service::Audio::ERR_INSUFFICIENT_BUFFER_SIZE; +        return Service::Audio::ResultInsufficientBuffer;      }      u32 effect_process_order_count{0}; @@ -208,7 +208,7 @@ Result System::Initialize(const AudioRendererParameterInternal& params,          effect_process_order_count = params.effects * (params.sub_mixes + 1);          effect_process_order_buffer = allocator.Allocate<s32>(effect_process_order_count, 0x10);          if (effect_process_order_buffer.empty()) { -            return Service::Audio::ERR_INSUFFICIENT_BUFFER_SIZE; +            return Service::Audio::ResultInsufficientBuffer;          }      } @@ -222,7 +222,7 @@ Result System::Initialize(const AudioRendererParameterInternal& params,      auto sorted_mix_infos{allocator.Allocate<MixInfo*>(params.sub_mixes + 1, 0x10)};      if (sorted_mix_infos.empty()) { -        return Service::Audio::ERR_INSUFFICIENT_BUFFER_SIZE; +        return Service::Audio::ResultInsufficientBuffer;      }      std::memset(sorted_mix_infos.data(), 0, sorted_mix_infos.size_bytes()); @@ -235,7 +235,7 @@ Result System::Initialize(const AudioRendererParameterInternal& params,          auto edge_matrix_workbuffer{allocator.Allocate<u8>(edge_matrix_size, 1)};          if (node_states_workbuffer.empty() || edge_matrix_workbuffer.size() == 0) { -            return Service::Audio::ERR_INSUFFICIENT_BUFFER_SIZE; +            return Service::Audio::ResultInsufficientBuffer;          }          mix_context.Initialize(sorted_mix_infos, mix_infos, params.sub_mixes + 1, @@ -250,7 +250,7 @@ Result System::Initialize(const AudioRendererParameterInternal& params,      upsampler_manager = allocator.Allocate<UpsamplerManager>(1, 0x10).data();      if (upsampler_manager == nullptr) { -        return Service::Audio::ERR_INSUFFICIENT_BUFFER_SIZE; +        return Service::Audio::ResultInsufficientBuffer;      }      memory_pool_workbuffer = allocator.Allocate<MemoryPoolInfo>(memory_pool_count, 0x10); @@ -259,18 +259,18 @@ Result System::Initialize(const AudioRendererParameterInternal& params,      }      if (memory_pool_workbuffer.empty() && memory_pool_count > 0) { -        return Service::Audio::ERR_INSUFFICIENT_BUFFER_SIZE; +        return Service::Audio::ResultInsufficientBuffer;      }      if (!splitter_context.Initialize(behavior, params, allocator)) { -        return Service::Audio::ERR_INSUFFICIENT_BUFFER_SIZE; +        return Service::Audio::ResultInsufficientBuffer;      }      std::span<EffectResultState> effect_result_states_cpu{};      if (behavior.IsEffectInfoVersion2Supported() && params.effects > 0) {          effect_result_states_cpu = allocator.Allocate<EffectResultState>(params.effects, 0x10);          if (effect_result_states_cpu.empty()) { -            return Service::Audio::ERR_INSUFFICIENT_BUFFER_SIZE; +            return Service::Audio::ResultInsufficientBuffer;          }          std::memset(effect_result_states_cpu.data(), 0, effect_result_states_cpu.size_bytes());      } @@ -289,7 +289,7 @@ Result System::Initialize(const AudioRendererParameterInternal& params,                                          upsampler_workbuffer);      if (upsampler_infos.empty()) { -        return Service::Audio::ERR_INSUFFICIENT_BUFFER_SIZE; +        return Service::Audio::ResultInsufficientBuffer;      }      auto effect_infos{allocator.Allocate<EffectInfoBase>(params.effects, 0x40)}; @@ -298,14 +298,14 @@ Result System::Initialize(const AudioRendererParameterInternal& params,      }      if (effect_infos.empty() && params.effects > 0) { -        return Service::Audio::ERR_INSUFFICIENT_BUFFER_SIZE; +        return Service::Audio::ResultInsufficientBuffer;      }      std::span<EffectResultState> effect_result_states_dsp{};      if (behavior.IsEffectInfoVersion2Supported() && params.effects > 0) {          effect_result_states_dsp = allocator.Allocate<EffectResultState>(params.effects, 0x40);          if (effect_result_states_dsp.empty()) { -            return Service::Audio::ERR_INSUFFICIENT_BUFFER_SIZE; +            return Service::Audio::ResultInsufficientBuffer;          }          std::memset(effect_result_states_dsp.data(), 0, effect_result_states_dsp.size_bytes());      } @@ -319,14 +319,14 @@ Result System::Initialize(const AudioRendererParameterInternal& params,      }      if (sinks.empty()) { -        return Service::Audio::ERR_INSUFFICIENT_BUFFER_SIZE; +        return Service::Audio::ResultInsufficientBuffer;      }      sink_context.Initialize(sinks, params.sinks);      auto voice_dsp_states{allocator.Allocate<VoiceState>(params.voices, 0x40)};      if (voice_dsp_states.empty()) { -        return Service::Audio::ERR_INSUFFICIENT_BUFFER_SIZE; +        return Service::Audio::ResultInsufficientBuffer;      }      for (auto& voice_state : voice_dsp_states) { @@ -344,7 +344,7 @@ Result System::Initialize(const AudioRendererParameterInternal& params,              0xC};          performance_workbuffer = allocator.Allocate<u8>(perf_workbuffer_size, 0x40);          if (performance_workbuffer.empty()) { -            return Service::Audio::ERR_INSUFFICIENT_BUFFER_SIZE; +            return Service::Audio::ResultInsufficientBuffer;          }          std::memset(performance_workbuffer.data(), 0, performance_workbuffer.size_bytes());          performance_manager.Initialize(performance_workbuffer, performance_workbuffer.size_bytes(), @@ -360,7 +360,7 @@ Result System::Initialize(const AudioRendererParameterInternal& params,      command_workbuffer_size = allocator.GetRemainingSize();      command_workbuffer = allocator.Allocate<u8>(command_workbuffer_size, 0x40);      if (command_workbuffer.empty()) { -        return Service::Audio::ERR_INSUFFICIENT_BUFFER_SIZE; +        return Service::Audio::ResultInsufficientBuffer;      }      command_buffer_size = 0; diff --git a/src/audio_core/renderer/voice/voice_info.cpp b/src/audio_core/renderer/voice/voice_info.cpp index 1849eeb57..c0bfb23fc 100644 --- a/src/audio_core/renderer/voice/voice_info.cpp +++ b/src/audio_core/renderer/voice/voice_info.cpp @@ -181,7 +181,7 @@ void VoiceInfo::UpdateWaveBuffer(std::span<BehaviorInfo::ErrorInfo> error_info,          if (wave_buffer_internal.start_offset * byte_size > wave_buffer_internal.size ||              wave_buffer_internal.end_offset * byte_size > wave_buffer_internal.size) {              LOG_ERROR(Service_Audio, "Invalid PCM16 start/end wavebuffer sizes!"); -            error_info[0].error_code = Service::Audio::ERR_INVALID_UPDATE_DATA; +            error_info[0].error_code = Service::Audio::ResultInvalidUpdateInfo;              error_info[0].address = wave_buffer_internal.address;              return;          } @@ -192,7 +192,7 @@ void VoiceInfo::UpdateWaveBuffer(std::span<BehaviorInfo::ErrorInfo> error_info,          if (wave_buffer_internal.start_offset * byte_size > wave_buffer_internal.size ||              wave_buffer_internal.end_offset * byte_size > wave_buffer_internal.size) {              LOG_ERROR(Service_Audio, "Invalid PCMFloat start/end wavebuffer sizes!"); -            error_info[0].error_code = Service::Audio::ERR_INVALID_UPDATE_DATA; +            error_info[0].error_code = Service::Audio::ResultInvalidUpdateInfo;              error_info[0].address = wave_buffer_internal.address;              return;          } @@ -216,7 +216,7 @@ void VoiceInfo::UpdateWaveBuffer(std::span<BehaviorInfo::ErrorInfo> error_info,          if (start > static_cast<s64>(wave_buffer_internal.size) ||              end > static_cast<s64>(wave_buffer_internal.size)) {              LOG_ERROR(Service_Audio, "Invalid ADPCM start/end wavebuffer sizes!"); -            error_info[0].error_code = Service::Audio::ERR_INVALID_UPDATE_DATA; +            error_info[0].error_code = Service::Audio::ResultInvalidUpdateInfo;              error_info[0].address = wave_buffer_internal.address;              return;          } @@ -228,7 +228,7 @@ void VoiceInfo::UpdateWaveBuffer(std::span<BehaviorInfo::ErrorInfo> error_info,      if (wave_buffer_internal.start_offset < 0 || wave_buffer_internal.end_offset < 0) {          LOG_ERROR(Service_Audio, "Invalid input start/end wavebuffer sizes!"); -        error_info[0].error_code = Service::Audio::ERR_INVALID_UPDATE_DATA; +        error_info[0].error_code = Service::Audio::ResultInvalidUpdateInfo;          error_info[0].address = wave_buffer_internal.address;          return;      } diff --git a/src/common/CMakeLists.txt b/src/common/CMakeLists.txt index 58ff5f2f3..61ab68864 100644 --- a/src/common/CMakeLists.txt +++ b/src/common/CMakeLists.txt @@ -91,6 +91,7 @@ add_library(common STATIC      multi_level_page_table.h      nvidia_flags.cpp      nvidia_flags.h +    overflow.h      page_table.cpp      page_table.h      param_package.cpp diff --git a/src/common/bit_cast.h b/src/common/bit_cast.h index 535148b4d..c6110c542 100644 --- a/src/common/bit_cast.h +++ b/src/common/bit_cast.h @@ -3,19 +3,21 @@  #pragma once -#include <cstring> -#include <type_traits> +#include <version> + +#ifdef __cpp_lib_bit_cast +#include <bit> +#endif  namespace Common {  template <typename To, typename From> -[[nodiscard]] std::enable_if_t<sizeof(To) == sizeof(From) && std::is_trivially_copyable_v<From> && -                                   std::is_trivially_copyable_v<To>, -                               To> -BitCast(const From& src) noexcept { -    To dst; -    std::memcpy(&dst, &src, sizeof(To)); -    return dst; +constexpr inline To BitCast(const From& from) { +#ifdef __cpp_lib_bit_cast +    return std::bit_cast<To>(from); +#else +    return __builtin_bit_cast(To, from); +#endif  }  } // namespace Common diff --git a/src/common/input.h b/src/common/input.h index b5748a6c8..98e934685 100644 --- a/src/common/input.h +++ b/src/common/input.h @@ -46,7 +46,7 @@ enum class PollingMode {      // Constant polling of buttons, analogs and motion data      Active,      // Only update on button change, digital analogs -    Pasive, +    Passive,      // Enable near field communication polling      NFC,      // Enable infrared camera polling diff --git a/src/common/overflow.h b/src/common/overflow.h new file mode 100644 index 000000000..44d8e7e73 --- /dev/null +++ b/src/common/overflow.h @@ -0,0 +1,22 @@ +// SPDX-FileCopyrightText: Copyright 2023 yuzu Emulator Project +// SPDX-License-Identifier: GPL-2.0-or-later + +#pragma once + +#include <type_traits> +#include "bit_cast.h" + +namespace Common { + +template <typename T> +    requires(std::is_integral_v<T> && std::is_signed_v<T>) +inline T WrappingAdd(T lhs, T rhs) { +    using U = std::make_unsigned_t<T>; + +    U lhs_u = BitCast<U>(lhs); +    U rhs_u = BitCast<U>(rhs); + +    return BitCast<T>(lhs_u + rhs_u); +} + +} // namespace Common diff --git a/src/common/settings.h b/src/common/settings.h index 1ae28ce93..b77a1580a 100644 --- a/src/common/settings.h +++ b/src/common/settings.h @@ -503,7 +503,7 @@ struct Values {      Setting<bool> tas_loop{false, "tas_loop"};      Setting<bool> mouse_panning{false, "mouse_panning"}; -    Setting<u8, true> mouse_panning_sensitivity{10, 1, 100, "mouse_panning_sensitivity"}; +    Setting<u8, true> mouse_panning_sensitivity{50, 1, 100, "mouse_panning_sensitivity"};      Setting<bool> mouse_enabled{false, "mouse_enabled"};      Setting<bool> emulate_analog_keyboard{false, "emulate_analog_keyboard"}; diff --git a/src/common/steady_clock.cpp b/src/common/steady_clock.cpp index 0d5908aa7..782859196 100644 --- a/src/common/steady_clock.cpp +++ b/src/common/steady_clock.cpp @@ -23,6 +23,19 @@ static s64 WindowsQueryPerformanceCounter() {      QueryPerformanceCounter(&counter);      return counter.QuadPart;  } + +static s64 GetSystemTimeNS() { +    // GetSystemTimePreciseAsFileTime returns the file time in 100ns units. +    static constexpr s64 Multiplier = 100; +    // Convert Windows epoch to Unix epoch. +    static constexpr s64 WindowsEpochToUnixEpochNS = 0x19DB1DED53E8000LL; + +    FILETIME filetime; +    GetSystemTimePreciseAsFileTime(&filetime); +    return Multiplier * ((static_cast<s64>(filetime.dwHighDateTime) << 32) + +                         static_cast<s64>(filetime.dwLowDateTime)) - +           WindowsEpochToUnixEpochNS; +}  #endif  SteadyClock::time_point SteadyClock::Now() noexcept { @@ -53,4 +66,16 @@ SteadyClock::time_point SteadyClock::Now() noexcept {  #endif  } +RealTimeClock::time_point RealTimeClock::Now() noexcept { +#if defined(_WIN32) +    return time_point{duration{GetSystemTimeNS()}}; +#elif defined(__APPLE__) +    return time_point{duration{clock_gettime_nsec_np(CLOCK_REALTIME)}}; +#else +    timespec ts; +    clock_gettime(CLOCK_REALTIME, &ts); +    return time_point{std::chrono::seconds{ts.tv_sec} + std::chrono::nanoseconds{ts.tv_nsec}}; +#endif +} +  }; // namespace Common diff --git a/src/common/steady_clock.h b/src/common/steady_clock.h index 9497cf865..dbd0e2513 100644 --- a/src/common/steady_clock.h +++ b/src/common/steady_clock.h @@ -20,4 +20,15 @@ struct SteadyClock {      [[nodiscard]] static time_point Now() noexcept;  }; +struct RealTimeClock { +    using rep = s64; +    using period = std::nano; +    using duration = std::chrono::nanoseconds; +    using time_point = std::chrono::time_point<RealTimeClock>; + +    static constexpr bool is_steady = false; + +    [[nodiscard]] static time_point Now() noexcept; +}; +  } // namespace Common diff --git a/src/common/x64/native_clock.cpp b/src/common/x64/native_clock.cpp index bc1a973b0..76c66e7ee 100644 --- a/src/common/x64/native_clock.cpp +++ b/src/common/x64/native_clock.cpp @@ -53,11 +53,11 @@ u64 EstimateRDTSCFrequency() {      FencedRDTSC();      // Get the current time. -    const auto start_time = Common::SteadyClock::Now(); +    const auto start_time = Common::RealTimeClock::Now();      const u64 tsc_start = FencedRDTSC();      // Wait for 250 milliseconds.      std::this_thread::sleep_for(std::chrono::milliseconds{250}); -    const auto end_time = Common::SteadyClock::Now(); +    const auto end_time = Common::RealTimeClock::Now();      const u64 tsc_end = FencedRDTSC();      // Calculate differences.      const u64 timer_diff = static_cast<u64>( @@ -72,13 +72,29 @@ NativeClock::NativeClock(u64 emulated_cpu_frequency_, u64 emulated_clock_frequen                           u64 rtsc_frequency_)      : WallClock(emulated_cpu_frequency_, emulated_clock_frequency_, true), rtsc_frequency{                                                                                 rtsc_frequency_} { +    // Thread to re-adjust the RDTSC frequency after 10 seconds has elapsed. +    time_sync_thread = std::jthread{[this](std::stop_token token) { +        // Get the current time. +        const auto start_time = Common::RealTimeClock::Now(); +        const u64 tsc_start = FencedRDTSC(); +        // Wait for 10 seconds. +        if (!Common::StoppableTimedWait(token, std::chrono::seconds{10})) { +            return; +        } +        const auto end_time = Common::RealTimeClock::Now(); +        const u64 tsc_end = FencedRDTSC(); +        // Calculate differences. +        const u64 timer_diff = static_cast<u64>( +            std::chrono::duration_cast<std::chrono::nanoseconds>(end_time - start_time).count()); +        const u64 tsc_diff = tsc_end - tsc_start; +        const u64 tsc_freq = MultiplyAndDivide64(tsc_diff, 1000000000ULL, timer_diff); +        rtsc_frequency = tsc_freq; +        CalculateAndSetFactors(); +    }}; +      time_point.inner.last_measure = FencedRDTSC();      time_point.inner.accumulated_ticks = 0U; -    ns_rtsc_factor = GetFixedPoint64Factor(NS_RATIO, rtsc_frequency); -    us_rtsc_factor = GetFixedPoint64Factor(US_RATIO, rtsc_frequency); -    ms_rtsc_factor = GetFixedPoint64Factor(MS_RATIO, rtsc_frequency); -    clock_rtsc_factor = GetFixedPoint64Factor(emulated_clock_frequency, rtsc_frequency); -    cpu_rtsc_factor = GetFixedPoint64Factor(emulated_cpu_frequency, rtsc_frequency); +    CalculateAndSetFactors();  }  u64 NativeClock::GetRTSC() { @@ -138,6 +154,14 @@ u64 NativeClock::GetCPUCycles() {      return MultiplyHigh(rtsc_value, cpu_rtsc_factor);  } +void NativeClock::CalculateAndSetFactors() { +    ns_rtsc_factor = GetFixedPoint64Factor(NS_RATIO, rtsc_frequency); +    us_rtsc_factor = GetFixedPoint64Factor(US_RATIO, rtsc_frequency); +    ms_rtsc_factor = GetFixedPoint64Factor(MS_RATIO, rtsc_frequency); +    clock_rtsc_factor = GetFixedPoint64Factor(emulated_clock_frequency, rtsc_frequency); +    cpu_rtsc_factor = GetFixedPoint64Factor(emulated_cpu_frequency, rtsc_frequency); +} +  } // namespace X64  } // namespace Common diff --git a/src/common/x64/native_clock.h b/src/common/x64/native_clock.h index 38ae7a462..03ca291d8 100644 --- a/src/common/x64/native_clock.h +++ b/src/common/x64/native_clock.h @@ -3,6 +3,7 @@  #pragma once +#include "common/polyfill_thread.h"  #include "common/wall_clock.h"  namespace Common { @@ -28,6 +29,8 @@ public:  private:      u64 GetRTSC(); +    void CalculateAndSetFactors(); +      union alignas(16) TimePoint {          TimePoint() : pack{} {}          u128 pack{}; @@ -47,6 +50,8 @@ private:      u64 ms_rtsc_factor{};      u64 rtsc_frequency; + +    std::jthread time_sync_thread;  };  } // namespace X64 diff --git a/src/core/CMakeLists.txt b/src/core/CMakeLists.txt index 4a1a8bb43..75e0c4f38 100644 --- a/src/core/CMakeLists.txt +++ b/src/core/CMakeLists.txt @@ -454,7 +454,6 @@ add_library(core STATIC      hle/service/filesystem/fsp_srv.h      hle/service/fgm/fgm.cpp      hle/service/fgm/fgm.h -    hle/service/friend/errors.h      hle/service/friend/friend.cpp      hle/service/friend/friend.h      hle/service/friend/friend_interface.cpp diff --git a/src/core/core_timing.cpp b/src/core/core_timing.cpp index 742cfb996..cd4df4522 100644 --- a/src/core/core_timing.cpp +++ b/src/core/core_timing.cpp @@ -53,7 +53,7 @@ void CoreTiming::ThreadEntry(CoreTiming& instance) {      static constexpr char name[] = "HostTiming";      MicroProfileOnThreadCreate(name);      Common::SetCurrentThreadName(name); -    Common::SetCurrentThreadPriority(Common::ThreadPriority::Critical); +    Common::SetCurrentThreadPriority(Common::ThreadPriority::High);      instance.on_thread_init();      instance.ThreadLoop();      MicroProfileOnThreadExit(); diff --git a/src/core/cpu_manager.cpp b/src/core/cpu_manager.cpp index 04a11f444..980bb97f9 100644 --- a/src/core/cpu_manager.cpp +++ b/src/core/cpu_manager.cpp @@ -192,7 +192,7 @@ void CpuManager::RunThread(std::stop_token token, std::size_t core) {      }      MicroProfileOnThreadCreate(name.c_str());      Common::SetCurrentThreadName(name.c_str()); -    Common::SetCurrentThreadPriority(Common::ThreadPriority::High); +    Common::SetCurrentThreadPriority(Common::ThreadPriority::Critical);      auto& data = core_data[core];      data.host_context = Common::Fiber::ThreadToFiber(); diff --git a/src/core/hle/kernel/k_address_space_info.cpp b/src/core/hle/kernel/k_address_space_info.cpp index 97972ebae..c36eb5dc4 100644 --- a/src/core/hle/kernel/k_address_space_info.cpp +++ b/src/core/hle/kernel/k_address_space_info.cpp @@ -44,11 +44,11 @@ const KAddressSpaceInfo& GetAddressSpaceInfo(size_t width, KAddressSpaceInfo::Ty  } // namespace -uintptr_t KAddressSpaceInfo::GetAddressSpaceStart(size_t width, KAddressSpaceInfo::Type type) { +std::size_t KAddressSpaceInfo::GetAddressSpaceStart(size_t width, KAddressSpaceInfo::Type type) {      return GetAddressSpaceInfo(width, type).address;  } -size_t KAddressSpaceInfo::GetAddressSpaceSize(size_t width, KAddressSpaceInfo::Type type) { +std::size_t KAddressSpaceInfo::GetAddressSpaceSize(size_t width, KAddressSpaceInfo::Type type) {      return GetAddressSpaceInfo(width, type).size;  } diff --git a/src/core/hle/kernel/k_address_space_info.h b/src/core/hle/kernel/k_address_space_info.h index 69e9d77f2..9a26f6b90 100644 --- a/src/core/hle/kernel/k_address_space_info.h +++ b/src/core/hle/kernel/k_address_space_info.h @@ -18,7 +18,7 @@ struct KAddressSpaceInfo final {          Count,      }; -    static u64 GetAddressSpaceStart(std::size_t width, Type type); +    static std::size_t GetAddressSpaceStart(std::size_t width, Type type);      static std::size_t GetAddressSpaceSize(std::size_t width, Type type);      const std::size_t bit_width{}; diff --git a/src/core/hle/kernel/k_device_address_space.h b/src/core/hle/kernel/k_device_address_space.h index 4709df995..b4a014c38 100644 --- a/src/core/hle/kernel/k_device_address_space.h +++ b/src/core/hle/kernel/k_device_address_space.h @@ -21,9 +21,9 @@ public:      ~KDeviceAddressSpace();      Result Initialize(u64 address, u64 size); -    void Finalize(); +    void Finalize() override; -    bool IsInitialized() const { +    bool IsInitialized() const override {          return m_is_initialized;      }      static void PostDestroy(uintptr_t arg) {} diff --git a/src/core/hle/kernel/k_process.h b/src/core/hle/kernel/k_process.h index 09bf2f1d0..549809000 100644 --- a/src/core/hle/kernel/k_process.h +++ b/src/core/hle/kernel/k_process.h @@ -310,10 +310,10 @@ public:      /// Clears the signaled state of the process if and only if it's signaled.      ///      /// @pre The process must not be already terminated. If this is called on a -    ///      terminated process, then ERR_INVALID_STATE will be returned. +    ///      terminated process, then ResultInvalidState will be returned.      ///      /// @pre The process must be in a signaled state. If this is called on a -    ///      process instance that is not signaled, ERR_INVALID_STATE will be +    ///      process instance that is not signaled, ResultInvalidState will be      ///      returned.      Result Reset(); diff --git a/src/core/hle/kernel/k_resource_limit.cpp b/src/core/hle/kernel/k_resource_limit.cpp index b9d22b414..626517619 100644 --- a/src/core/hle/kernel/k_resource_limit.cpp +++ b/src/core/hle/kernel/k_resource_limit.cpp @@ -2,6 +2,7 @@  // SPDX-License-Identifier: GPL-2.0-or-later  #include "common/assert.h" +#include "common/overflow.h"  #include "core/core.h"  #include "core/core_timing.h"  #include "core/hle/kernel/k_resource_limit.h" @@ -104,7 +105,7 @@ bool KResourceLimit::Reserve(LimitableResource which, s64 value, s64 timeout) {          ASSERT(current_hints[index] <= current_values[index]);          // If we would overflow, don't allow to succeed. -        if (current_values[index] + value <= current_values[index]) { +        if (Common::WrappingAdd(current_values[index], value) <= current_values[index]) {              break;          } diff --git a/src/core/hle/kernel/k_thread.cpp b/src/core/hle/kernel/k_thread.cpp index 96b90ffef..26e3700e4 100644 --- a/src/core/hle/kernel/k_thread.cpp +++ b/src/core/hle/kernel/k_thread.cpp @@ -49,6 +49,7 @@ static void ResetThreadContext32(Core::ARM_Interface::ThreadContext32& context,      context.cpu_registers[0] = arg;      context.cpu_registers[15] = entry_point;      context.cpu_registers[13] = stack_top; +    context.fpscr = 0;  }  static void ResetThreadContext64(Core::ARM_Interface::ThreadContext64& context, VAddr stack_top, @@ -58,8 +59,8 @@ static void ResetThreadContext64(Core::ARM_Interface::ThreadContext64& context,      context.cpu_registers[18] = Kernel::KSystemControl::GenerateRandomU64() | 1;      context.pc = entry_point;      context.sp = stack_top; -    // TODO(merry): Perform a hardware test to determine the below value.      context.fpcr = 0; +    context.fpsr = 0;  }  } // namespace @@ -815,6 +816,27 @@ void KThread::Continue() {      KScheduler::OnThreadStateChanged(kernel, this, old_state);  } +void KThread::CloneFpuStatus() { +    // We shouldn't reach here when starting kernel threads. +    ASSERT(this->GetOwnerProcess() != nullptr); +    ASSERT(this->GetOwnerProcess() == GetCurrentProcessPointer(kernel)); + +    if (this->GetOwnerProcess()->Is64BitProcess()) { +        // Clone FPSR and FPCR. +        ThreadContext64 cur_ctx{}; +        kernel.System().CurrentArmInterface().SaveContext(cur_ctx); + +        this->GetContext64().fpcr = cur_ctx.fpcr; +        this->GetContext64().fpsr = cur_ctx.fpsr; +    } else { +        // Clone FPSCR. +        ThreadContext32 cur_ctx{}; +        kernel.System().CurrentArmInterface().SaveContext(cur_ctx); + +        this->GetContext32().fpscr = cur_ctx.fpscr; +    } +} +  Result KThread::SetActivity(Svc::ThreadActivity activity) {      // Lock ourselves.      KScopedLightLock lk(activity_pause_lock); diff --git a/src/core/hle/kernel/k_thread.h b/src/core/hle/kernel/k_thread.h index bd125f5f1..9423f08ca 100644 --- a/src/core/hle/kernel/k_thread.h +++ b/src/core/hle/kernel/k_thread.h @@ -254,6 +254,8 @@ public:          thread_context_32.tpidr = static_cast<u32>(value);      } +    void CloneFpuStatus(); +      [[nodiscard]] ThreadContext32& GetContext32() {          return thread_context_32;      } diff --git a/src/core/hle/kernel/svc/svc_synchronization.cpp b/src/core/hle/kernel/svc/svc_synchronization.cpp index 1a8f7e191..9e7bf9530 100644 --- a/src/core/hle/kernel/svc/svc_synchronization.cpp +++ b/src/core/hle/kernel/svc/svc_synchronization.cpp @@ -48,19 +48,15 @@ Result ResetSignal(Core::System& system, Handle handle) {      return ResultInvalidHandle;  } -/// Wait for the given handles to synchronize, timeout after the specified nanoseconds -Result WaitSynchronization(Core::System& system, s32* index, VAddr handles_address, s32 num_handles, -                           s64 nano_seconds) { -    LOG_TRACE(Kernel_SVC, "called handles_address=0x{:X}, num_handles={}, nano_seconds={}", -              handles_address, num_handles, nano_seconds); - +static Result WaitSynchronization(Core::System& system, int32_t* out_index, const Handle* handles, +                                  int32_t num_handles, int64_t timeout_ns) {      // Ensure number of handles is valid. -    R_UNLESS(0 <= num_handles && num_handles <= ArgumentHandleCountMax, ResultOutOfRange); +    R_UNLESS(0 <= num_handles && num_handles <= Svc::ArgumentHandleCountMax, ResultOutOfRange); +    // Get the synchronization context.      auto& kernel = system.Kernel(); +    auto& handle_table = GetCurrentProcess(kernel).GetHandleTable();      std::vector<KSynchronizationObject*> objs(num_handles); -    const auto& handle_table = GetCurrentProcess(kernel).GetHandleTable(); -    Handle* handles = system.Memory().GetPointer<Handle>(handles_address);      // Copy user handles.      if (num_handles > 0) { @@ -68,21 +64,38 @@ Result WaitSynchronization(Core::System& system, s32* index, VAddr handles_addre          R_UNLESS(handle_table.GetMultipleObjects<KSynchronizationObject>(objs.data(), handles,                                                                           num_handles),                   ResultInvalidHandle); -        for (const auto& obj : objs) { -            kernel.RegisterInUseObject(obj); -        }      }      // Ensure handles are closed when we're done.      SCOPE_EXIT({ -        for (s32 i = 0; i < num_handles; ++i) { -            kernel.UnregisterInUseObject(objs[i]); +        for (auto i = 0; i < num_handles; ++i) {              objs[i]->Close();          }      }); -    return KSynchronizationObject::Wait(kernel, index, objs.data(), static_cast<s32>(objs.size()), -                                        nano_seconds); +    // Wait on the objects. +    Result res = KSynchronizationObject::Wait(kernel, out_index, objs.data(), +                                              static_cast<s32>(objs.size()), timeout_ns); + +    R_SUCCEED_IF(res == ResultSessionClosed); +    R_RETURN(res); +} + +/// Wait for the given handles to synchronize, timeout after the specified nanoseconds +Result WaitSynchronization(Core::System& system, int32_t* out_index, VAddr user_handles, +                           int32_t num_handles, int64_t timeout_ns) { +    LOG_TRACE(Kernel_SVC, "called user_handles={:#x}, num_handles={}, timeout_ns={}", user_handles, +              num_handles, timeout_ns); + +    // Ensure number of handles is valid. +    R_UNLESS(0 <= num_handles && num_handles <= Svc::ArgumentHandleCountMax, ResultOutOfRange); + +    std::vector<Handle> handles(num_handles); +    if (num_handles > 0) { +        system.Memory().ReadBlock(user_handles, handles.data(), num_handles * sizeof(Handle)); +    } + +    R_RETURN(WaitSynchronization(system, out_index, handles.data(), num_handles, timeout_ns));  }  /// Resumes a thread waiting on WaitSynchronization diff --git a/src/core/hle/kernel/svc/svc_thread.cpp b/src/core/hle/kernel/svc/svc_thread.cpp index b39807841..9bc1ebe74 100644 --- a/src/core/hle/kernel/svc/svc_thread.cpp +++ b/src/core/hle/kernel/svc/svc_thread.cpp @@ -82,6 +82,9 @@ Result CreateThread(Core::System& system, Handle* out_handle, VAddr entry_point,      // Commit the thread reservation.      thread_reservation.Commit(); +    // Clone the current fpu status to the new thread. +    thread->CloneFpuStatus(); +      // Register the new thread.      KThread::Register(kernel, thread); diff --git a/src/core/hle/service/acc/acc.cpp b/src/core/hle/service/acc/acc.cpp index ddc3a6dbe..120282aa4 100644 --- a/src/core/hle/service/acc/acc.cpp +++ b/src/core/hle/service/acc/acc.cpp @@ -30,12 +30,6 @@  namespace Service::Account { -constexpr Result ERR_INVALID_USER_ID{ErrorModule::Account, 20}; -constexpr Result ERR_INVALID_APPLICATION_ID{ErrorModule::Account, 22}; -constexpr Result ERR_INVALID_BUFFER{ErrorModule::Account, 30}; -constexpr Result ERR_INVALID_BUFFER_SIZE{ErrorModule::Account, 31}; -constexpr Result ERR_FAILED_SAVE_DATA{ErrorModule::Account, 100}; -  // Thumbnails are hard coded to be at least this size  constexpr std::size_t THUMBNAIL_SIZE = 0x24000; @@ -384,7 +378,7 @@ protected:          if (user_data.size() < sizeof(UserData)) {              LOG_ERROR(Service_ACC, "UserData buffer too small!");              IPC::ResponseBuilder rb{ctx, 2}; -            rb.Push(ERR_INVALID_BUFFER); +            rb.Push(Account::ResultInvalidArrayLength);              return;          } @@ -394,7 +388,7 @@ protected:          if (!profile_manager.SetProfileBaseAndData(user_id, base, data)) {              LOG_ERROR(Service_ACC, "Failed to update user data and base!");              IPC::ResponseBuilder rb{ctx, 2}; -            rb.Push(ERR_FAILED_SAVE_DATA); +            rb.Push(Account::ResultAccountUpdateFailed);              return;          } @@ -417,7 +411,7 @@ protected:          if (user_data.size() < sizeof(UserData)) {              LOG_ERROR(Service_ACC, "UserData buffer too small!");              IPC::ResponseBuilder rb{ctx, 2}; -            rb.Push(ERR_INVALID_BUFFER); +            rb.Push(Account::ResultInvalidArrayLength);              return;          } @@ -432,7 +426,7 @@ protected:              !profile_manager.SetProfileBaseAndData(user_id, base, data)) {              LOG_ERROR(Service_ACC, "Failed to update profile data, base, and image!");              IPC::ResponseBuilder rb{ctx, 2}; -            rb.Push(ERR_FAILED_SAVE_DATA); +            rb.Push(Account::ResultAccountUpdateFailed);              return;          } @@ -764,7 +758,7 @@ void Module::Interface::InitializeApplicationInfoRestricted(HLERequestContext& c  Result Module::Interface::InitializeApplicationInfoBase() {      if (application_info) {          LOG_ERROR(Service_ACC, "Application already initialized"); -        return ERR_ACCOUNTINFO_ALREADY_INITIALIZED; +        return Account::ResultApplicationInfoAlreadyInitialized;      }      // TODO(ogniK): This should be changed to reflect the target process for when we have multiple @@ -775,7 +769,7 @@ Result Module::Interface::InitializeApplicationInfoBase() {      if (launch_property.Failed()) {          LOG_ERROR(Service_ACC, "Failed to get launch property"); -        return ERR_ACCOUNTINFO_BAD_APPLICATION; +        return Account::ResultInvalidApplication;      }      switch (launch_property->base_game_storage_id) { @@ -791,7 +785,7 @@ Result Module::Interface::InitializeApplicationInfoBase() {      default:          LOG_ERROR(Service_ACC, "Invalid game storage ID! storage_id={}",                    launch_property->base_game_storage_id); -        return ERR_ACCOUNTINFO_BAD_APPLICATION; +        return Account::ResultInvalidApplication;      }      LOG_WARNING(Service_ACC, "ApplicationInfo init required"); @@ -899,20 +893,20 @@ void Module::Interface::StoreSaveDataThumbnail(HLERequestContext& ctx, const Com      if (tid == 0) {          LOG_ERROR(Service_ACC, "TitleID is not valid!"); -        rb.Push(ERR_INVALID_APPLICATION_ID); +        rb.Push(Account::ResultInvalidApplication);          return;      }      if (uuid.IsInvalid()) {          LOG_ERROR(Service_ACC, "User ID is not valid!"); -        rb.Push(ERR_INVALID_USER_ID); +        rb.Push(Account::ResultInvalidUserId);          return;      }      const auto thumbnail_size = ctx.GetReadBufferSize();      if (thumbnail_size != THUMBNAIL_SIZE) {          LOG_ERROR(Service_ACC, "Buffer size is empty! size={:X} expecting {:X}", thumbnail_size,                    THUMBNAIL_SIZE); -        rb.Push(ERR_INVALID_BUFFER_SIZE); +        rb.Push(Account::ResultInvalidArrayLength);          return;      } diff --git a/src/core/hle/service/acc/errors.h b/src/core/hle/service/acc/errors.h index e9c16b951..433ebfe9d 100644 --- a/src/core/hle/service/acc/errors.h +++ b/src/core/hle/service/acc/errors.h @@ -7,7 +7,13 @@  namespace Service::Account { -constexpr Result ERR_ACCOUNTINFO_BAD_APPLICATION{ErrorModule::Account, 22}; -constexpr Result ERR_ACCOUNTINFO_ALREADY_INITIALIZED{ErrorModule::Account, 41}; +constexpr Result ResultCancelledByUser{ErrorModule::Account, 1}; +constexpr Result ResultNoNotifications{ErrorModule::Account, 15}; +constexpr Result ResultInvalidUserId{ErrorModule::Account, 20}; +constexpr Result ResultInvalidApplication{ErrorModule::Account, 22}; +constexpr Result ResultNullptr{ErrorModule::Account, 30}; +constexpr Result ResultInvalidArrayLength{ErrorModule::Account, 32}; +constexpr Result ResultApplicationInfoAlreadyInitialized{ErrorModule::Account, 41}; +constexpr Result ResultAccountUpdateFailed{ErrorModule::Account, 100};  } // namespace Service::Account diff --git a/src/core/hle/service/am/am.cpp b/src/core/hle/service/am/am.cpp index f74c7b550..f17df5124 100644 --- a/src/core/hle/service/am/am.cpp +++ b/src/core/hle/service/am/am.cpp @@ -39,9 +39,9 @@  namespace Service::AM { -constexpr Result ERR_NO_DATA_IN_CHANNEL{ErrorModule::AM, 2}; -constexpr Result ERR_NO_MESSAGES{ErrorModule::AM, 3}; -constexpr Result ERR_SIZE_OUT_OF_BOUNDS{ErrorModule::AM, 503}; +constexpr Result ResultNoDataInChannel{ErrorModule::AM, 2}; +constexpr Result ResultNoMessages{ErrorModule::AM, 3}; +constexpr Result ResultInvalidOffset{ErrorModule::AM, 503};  enum class LaunchParameterKind : u32 {      ApplicationSpecific = 1, @@ -758,7 +758,7 @@ void ICommonStateGetter::ReceiveMessage(HLERequestContext& ctx) {      if (message == AppletMessageQueue::AppletMessage::None) {          LOG_ERROR(Service_AM, "Message queue is empty"); -        rb.Push(ERR_NO_MESSAGES); +        rb.Push(AM::ResultNoMessages);          rb.PushEnum<AppletMessageQueue::AppletMessage>(message);          return;      } @@ -1028,7 +1028,7 @@ private:              LOG_DEBUG(Service_AM,                        "storage is a nullptr. There is no data in the current normal channel");              IPC::ResponseBuilder rb{ctx, 2}; -            rb.Push(ERR_NO_DATA_IN_CHANNEL); +            rb.Push(AM::ResultNoDataInChannel);              return;          } @@ -1059,7 +1059,7 @@ private:              LOG_DEBUG(Service_AM,                        "storage is a nullptr. There is no data in the current interactive channel");              IPC::ResponseBuilder rb{ctx, 2}; -            rb.Push(ERR_NO_DATA_IN_CHANNEL); +            rb.Push(AM::ResultNoDataInChannel);              return;          } @@ -1138,7 +1138,7 @@ void IStorageAccessor::Write(HLERequestContext& ctx) {                    backing.GetSize(), size, offset);          IPC::ResponseBuilder rb{ctx, 2}; -        rb.Push(ERR_SIZE_OUT_OF_BOUNDS); +        rb.Push(AM::ResultInvalidOffset);          return;      } @@ -1161,7 +1161,7 @@ void IStorageAccessor::Read(HLERequestContext& ctx) {                    backing.GetSize(), size, offset);          IPC::ResponseBuilder rb{ctx, 2}; -        rb.Push(ERR_SIZE_OUT_OF_BOUNDS); +        rb.Push(AM::ResultInvalidOffset);          return;      } @@ -1502,7 +1502,7 @@ void IApplicationFunctions::PopLaunchParameter(HLERequestContext& ctx) {      LOG_ERROR(Service_AM, "Attempted to load launch parameter but none was found!");      IPC::ResponseBuilder rb{ctx, 2}; -    rb.Push(ERR_NO_DATA_IN_CHANNEL); +    rb.Push(AM::ResultNoDataInChannel);  }  void IApplicationFunctions::CreateApplicationAndRequestToStartForQuest(HLERequestContext& ctx) { @@ -1799,7 +1799,7 @@ void IApplicationFunctions::TryPopFromFriendInvitationStorageChannel(HLERequestC      LOG_WARNING(Service_AM, "(STUBBED) called");      IPC::ResponseBuilder rb{ctx, 2}; -    rb.Push(ERR_NO_DATA_IN_CHANNEL); +    rb.Push(AM::ResultNoDataInChannel);  }  void IApplicationFunctions::GetNotificationStorageChannelEvent(HLERequestContext& ctx) { diff --git a/src/core/hle/service/am/applets/applet_cabinet.cpp b/src/core/hle/service/am/applets/applet_cabinet.cpp index d0969b0f1..162687b29 100644 --- a/src/core/hle/service/am/applets/applet_cabinet.cpp +++ b/src/core/hle/service/am/applets/applet_cabinet.cpp @@ -119,7 +119,7 @@ void Cabinet::DisplayCompleted(bool apply_changes, std::string_view amiibo_name)      case Service::NFP::CabinetMode::StartNicknameAndOwnerSettings: {          Service::NFP::AmiiboName name{};          std::memcpy(name.data(), amiibo_name.data(), std::min(amiibo_name.size(), name.size() - 1)); -        nfp_device->SetNicknameAndOwner(name); +        nfp_device->SetRegisterInfoPrivate(name);          break;      }      case Service::NFP::CabinetMode::StartGameDataEraser: @@ -129,7 +129,7 @@ void Cabinet::DisplayCompleted(bool apply_changes, std::string_view amiibo_name)          nfp_device->RestoreAmiibo();          break;      case Service::NFP::CabinetMode::StartFormatter: -        nfp_device->DeleteAllData(); +        nfp_device->Format();          break;      default:          UNIMPLEMENTED_MSG("Unknown CabinetMode={}", applet_input_common.applet_mode); diff --git a/src/core/hle/service/am/applets/applet_controller.cpp b/src/core/hle/service/am/applets/applet_controller.cpp index b418031de..58484519b 100644 --- a/src/core/hle/service/am/applets/applet_controller.cpp +++ b/src/core/hle/service/am/applets/applet_controller.cpp @@ -19,10 +19,9 @@  namespace Service::AM::Applets { -// This error code (0x183ACA) is thrown when the applet fails to initialize. -[[maybe_unused]] constexpr Result ERR_CONTROLLER_APPLET_3101{ErrorModule::HID, 3101}; -// This error code (0x183CCA) is thrown when the u32 result in ControllerSupportResultInfo is 2. -[[maybe_unused]] constexpr Result ERR_CONTROLLER_APPLET_3102{ErrorModule::HID, 3102}; +[[maybe_unused]] constexpr Result ResultControllerSupportCanceled{ErrorModule::HID, 3101}; +[[maybe_unused]] constexpr Result ResultControllerSupportNotSupportedNpadStyle{ErrorModule::HID, +                                                                               3102};  static Core::Frontend::ControllerParameters ConvertToFrontendParameters(      ControllerSupportArgPrivate private_arg, ControllerSupportArgHeader header, bool enable_text, diff --git a/src/core/hle/service/am/applets/applet_profile_select.cpp b/src/core/hle/service/am/applets/applet_profile_select.cpp index c738db028..1d69f5447 100644 --- a/src/core/hle/service/am/applets/applet_profile_select.cpp +++ b/src/core/hle/service/am/applets/applet_profile_select.cpp @@ -7,13 +7,12 @@  #include "common/string_util.h"  #include "core/core.h"  #include "core/frontend/applets/profile_select.h" +#include "core/hle/service/acc/errors.h"  #include "core/hle/service/am/am.h"  #include "core/hle/service/am/applets/applet_profile_select.h"  namespace Service::AM::Applets { -constexpr Result ERR_USER_CANCELLED_SELECTION{ErrorModule::Account, 1}; -  ProfileSelect::ProfileSelect(Core::System& system_, LibraryAppletMode applet_mode_,                               const Core::Frontend::ProfileSelectApplet& frontend_)      : Applet{system_, applet_mode_}, frontend{frontend_}, system{system_} {} @@ -63,8 +62,8 @@ void ProfileSelect::SelectionComplete(std::optional<Common::UUID> uuid) {          output.result = 0;          output.uuid_selected = *uuid;      } else { -        status = ERR_USER_CANCELLED_SELECTION; -        output.result = ERR_USER_CANCELLED_SELECTION.raw; +        status = Account::ResultCancelledByUser; +        output.result = Account::ResultCancelledByUser.raw;          output.uuid_selected = Common::InvalidUUID;      } diff --git a/src/core/hle/service/audio/audren_u.cpp b/src/core/hle/service/audio/audren_u.cpp index 0a6830ffa..7086d4750 100644 --- a/src/core/hle/service/audio/audren_u.cpp +++ b/src/core/hle/service/audio/audren_u.cpp @@ -170,7 +170,7 @@ private:          if (impl->GetSystem().GetExecutionMode() == AudioCore::ExecutionMode::Manual) {              IPC::ResponseBuilder rb{ctx, 2}; -            rb.Push(ERR_NOT_SUPPORTED); +            rb.Push(Audio::ResultNotSupported);              return;          } @@ -448,7 +448,7 @@ void AudRenU::OpenAudioRenderer(HLERequestContext& ctx) {      if (impl->GetSessionCount() + 1 > AudioCore::MaxRendererSessions) {          LOG_ERROR(Service_Audio, "Too many AudioRenderer sessions open!");          IPC::ResponseBuilder rb{ctx, 2}; -        rb.Push(ERR_MAXIMUM_SESSIONS_REACHED); +        rb.Push(Audio::ResultOutOfSessions);          return;      } @@ -461,7 +461,7 @@ void AudRenU::OpenAudioRenderer(HLERequestContext& ctx) {      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(ERR_MAXIMUM_SESSIONS_REACHED); +        rb.Push(Audio::ResultOutOfSessions);          return;      } diff --git a/src/core/hle/service/audio/errors.h b/src/core/hle/service/audio/errors.h index d706978cb..3d3d3d97a 100644 --- a/src/core/hle/service/audio/errors.h +++ b/src/core/hle/service/audio/errors.h @@ -7,17 +7,17 @@  namespace Service::Audio { -constexpr Result ERR_INVALID_DEVICE_NAME{ErrorModule::Audio, 1}; -constexpr Result ERR_OPERATION_FAILED{ErrorModule::Audio, 2}; -constexpr Result ERR_INVALID_SAMPLE_RATE{ErrorModule::Audio, 3}; -constexpr Result ERR_INSUFFICIENT_BUFFER_SIZE{ErrorModule::Audio, 4}; -constexpr Result ERR_MAXIMUM_SESSIONS_REACHED{ErrorModule::Audio, 5}; -constexpr Result ERR_BUFFER_COUNT_EXCEEDED{ErrorModule::Audio, 8}; -constexpr Result ERR_INVALID_CHANNEL_COUNT{ErrorModule::Audio, 10}; -constexpr Result ERR_INVALID_UPDATE_DATA{ErrorModule::Audio, 41}; -constexpr Result ERR_POOL_MAPPING_FAILED{ErrorModule::Audio, 42}; -constexpr Result ERR_NOT_SUPPORTED{ErrorModule::Audio, 513}; -constexpr Result ERR_INVALID_PROCESS_HANDLE{ErrorModule::Audio, 1536}; -constexpr Result ERR_INVALID_REVISION{ErrorModule::Audio, 1537}; +constexpr Result ResultNotFound{ErrorModule::Audio, 1}; +constexpr Result ResultOperationFailed{ErrorModule::Audio, 2}; +constexpr Result ResultInvalidSampleRate{ErrorModule::Audio, 3}; +constexpr Result ResultInsufficientBuffer{ErrorModule::Audio, 4}; +constexpr Result ResultOutOfSessions{ErrorModule::Audio, 5}; +constexpr Result ResultBufferCountReached{ErrorModule::Audio, 8}; +constexpr Result ResultInvalidChannelCount{ErrorModule::Audio, 10}; +constexpr Result ResultInvalidUpdateInfo{ErrorModule::Audio, 41}; +constexpr Result ResultInvalidAddressInfo{ErrorModule::Audio, 42}; +constexpr Result ResultNotSupported{ErrorModule::Audio, 513}; +constexpr Result ResultInvalidHandle{ErrorModule::Audio, 1536}; +constexpr Result ResultInvalidRevision{ErrorModule::Audio, 1537};  } // namespace Service::Audio diff --git a/src/core/hle/service/friend/errors.h b/src/core/hle/service/friend/errors.h deleted file mode 100644 index ff525d865..000000000 --- a/src/core/hle/service/friend/errors.h +++ /dev/null @@ -1,11 +0,0 @@ -// SPDX-FileCopyrightText: Copyright 2019 yuzu Emulator Project -// SPDX-License-Identifier: GPL-2.0-or-later - -#pragma once - -#include "core/hle/result.h" - -namespace Service::Friend { - -constexpr Result ERR_NO_NOTIFICATIONS{ErrorModule::Account, 15}; -} diff --git a/src/core/hle/service/friend/friend.cpp b/src/core/hle/service/friend/friend.cpp index 447deab8b..9d05f9801 100644 --- a/src/core/hle/service/friend/friend.cpp +++ b/src/core/hle/service/friend/friend.cpp @@ -6,7 +6,7 @@  #include "common/uuid.h"  #include "core/core.h"  #include "core/hle/kernel/k_event.h" -#include "core/hle/service/friend/errors.h" +#include "core/hle/service/acc/errors.h"  #include "core/hle/service/friend/friend.h"  #include "core/hle/service/friend/friend_interface.h"  #include "core/hle/service/ipc_helpers.h" @@ -259,7 +259,7 @@ private:          if (notifications.empty()) {              LOG_ERROR(Service_Friend, "No notifications in queue!");              IPC::ResponseBuilder rb{ctx, 2}; -            rb.Push(ERR_NO_NOTIFICATIONS); +            rb.Push(Account::ResultNoNotifications);              return;          } diff --git a/src/core/hle/service/glue/arp.cpp b/src/core/hle/service/glue/arp.cpp index 9db136bac..929dcca0d 100644 --- a/src/core/hle/service/glue/arp.cpp +++ b/src/core/hle/service/glue/arp.cpp @@ -61,7 +61,7 @@ void ARP_R::GetApplicationLaunchProperty(HLERequestContext& ctx) {      if (!title_id.has_value()) {          LOG_ERROR(Service_ARP, "Failed to get title ID for process ID!");          IPC::ResponseBuilder rb{ctx, 2}; -        rb.Push(ERR_NOT_REGISTERED); +        rb.Push(Glue::ResultProcessIdNotRegistered);          return;      } @@ -109,7 +109,7 @@ void ARP_R::GetApplicationControlProperty(HLERequestContext& ctx) {      if (!title_id.has_value()) {          LOG_ERROR(Service_ARP, "Failed to get title ID for process ID!");          IPC::ResponseBuilder rb{ctx, 2}; -        rb.Push(ERR_NOT_REGISTERED); +        rb.Push(Glue::ResultProcessIdNotRegistered);          return;      } @@ -178,7 +178,7 @@ private:          if (process_id == 0) {              LOG_ERROR(Service_ARP, "Must have non-zero process ID!");              IPC::ResponseBuilder rb{ctx, 2}; -            rb.Push(ERR_INVALID_PROCESS_ID); +            rb.Push(Glue::ResultInvalidProcessId);              return;          } @@ -186,7 +186,7 @@ private:              LOG_ERROR(Service_ARP,                        "Attempted to issue registrar, but registrar is already issued!");              IPC::ResponseBuilder rb{ctx, 2}; -            rb.Push(ERR_INVALID_ACCESS); +            rb.Push(Glue::ResultAlreadyBound);              return;          } @@ -205,7 +205,7 @@ private:                  Service_ARP,                  "Attempted to set application launch property, but registrar is already issued!");              IPC::ResponseBuilder rb{ctx, 2}; -            rb.Push(ERR_INVALID_ACCESS); +            rb.Push(Glue::ResultAlreadyBound);              return;          } @@ -224,7 +224,7 @@ private:                  Service_ARP,                  "Attempted to set application control property, but registrar is already issued!");              IPC::ResponseBuilder rb{ctx, 2}; -            rb.Push(ERR_INVALID_ACCESS); +            rb.Push(Glue::ResultAlreadyBound);              return;          } @@ -263,7 +263,7 @@ void ARP_W::AcquireRegistrar(HLERequestContext& ctx) {          system, [this](u64 process_id, ApplicationLaunchProperty launch, std::vector<u8> control) {              const auto res = GetTitleIDForProcessID(system, process_id);              if (!res.has_value()) { -                return ERR_NOT_REGISTERED; +                return Glue::ResultProcessIdNotRegistered;              }              return manager.Register(*res, launch, std::move(control)); @@ -283,7 +283,7 @@ void ARP_W::UnregisterApplicationInstance(HLERequestContext& ctx) {      if (process_id == 0) {          LOG_ERROR(Service_ARP, "Must have non-zero process ID!");          IPC::ResponseBuilder rb{ctx, 2}; -        rb.Push(ERR_INVALID_PROCESS_ID); +        rb.Push(Glue::ResultInvalidProcessId);          return;      } @@ -292,7 +292,7 @@ void ARP_W::UnregisterApplicationInstance(HLERequestContext& ctx) {      if (!title_id.has_value()) {          LOG_ERROR(Service_ARP, "No title ID for process ID!");          IPC::ResponseBuilder rb{ctx, 2}; -        rb.Push(ERR_NOT_REGISTERED); +        rb.Push(Glue::ResultProcessIdNotRegistered);          return;      } diff --git a/src/core/hle/service/glue/errors.h b/src/core/hle/service/glue/errors.h index d4ce7f44e..30feaa5c0 100644 --- a/src/core/hle/service/glue/errors.h +++ b/src/core/hle/service/glue/errors.h @@ -7,9 +7,8 @@  namespace Service::Glue { -constexpr Result ERR_INVALID_RESOURCE{ErrorModule::ARP, 30}; -constexpr Result ERR_INVALID_PROCESS_ID{ErrorModule::ARP, 31}; -constexpr Result ERR_INVALID_ACCESS{ErrorModule::ARP, 42}; -constexpr Result ERR_NOT_REGISTERED{ErrorModule::ARP, 102}; +constexpr Result ResultInvalidProcessId{ErrorModule::ARP, 31}; +constexpr Result ResultAlreadyBound{ErrorModule::ARP, 42}; +constexpr Result ResultProcessIdNotRegistered{ErrorModule::ARP, 102};  } // namespace Service::Glue diff --git a/src/core/hle/service/glue/glue_manager.cpp b/src/core/hle/service/glue/glue_manager.cpp index 8a654cdca..4bf67921b 100644 --- a/src/core/hle/service/glue/glue_manager.cpp +++ b/src/core/hle/service/glue/glue_manager.cpp @@ -17,12 +17,12 @@ ARPManager::~ARPManager() = default;  ResultVal<ApplicationLaunchProperty> ARPManager::GetLaunchProperty(u64 title_id) const {      if (title_id == 0) { -        return ERR_INVALID_PROCESS_ID; +        return Glue::ResultInvalidProcessId;      }      const auto iter = entries.find(title_id);      if (iter == entries.end()) { -        return ERR_NOT_REGISTERED; +        return Glue::ResultProcessIdNotRegistered;      }      return iter->second.launch; @@ -30,12 +30,12 @@ ResultVal<ApplicationLaunchProperty> ARPManager::GetLaunchProperty(u64 title_id)  ResultVal<std::vector<u8>> ARPManager::GetControlProperty(u64 title_id) const {      if (title_id == 0) { -        return ERR_INVALID_PROCESS_ID; +        return Glue::ResultInvalidProcessId;      }      const auto iter = entries.find(title_id);      if (iter == entries.end()) { -        return ERR_NOT_REGISTERED; +        return Glue::ResultProcessIdNotRegistered;      }      return iter->second.control; @@ -44,12 +44,12 @@ ResultVal<std::vector<u8>> ARPManager::GetControlProperty(u64 title_id) const {  Result ARPManager::Register(u64 title_id, ApplicationLaunchProperty launch,                              std::vector<u8> control) {      if (title_id == 0) { -        return ERR_INVALID_PROCESS_ID; +        return Glue::ResultInvalidProcessId;      }      const auto iter = entries.find(title_id);      if (iter != entries.end()) { -        return ERR_INVALID_ACCESS; +        return Glue::ResultAlreadyBound;      }      entries.insert_or_assign(title_id, MapEntry{launch, std::move(control)}); @@ -58,12 +58,12 @@ Result ARPManager::Register(u64 title_id, ApplicationLaunchProperty launch,  Result ARPManager::Unregister(u64 title_id) {      if (title_id == 0) { -        return ERR_INVALID_PROCESS_ID; +        return Glue::ResultInvalidProcessId;      }      const auto iter = entries.find(title_id);      if (iter == entries.end()) { -        return ERR_NOT_REGISTERED; +        return Glue::ResultProcessIdNotRegistered;      }      entries.erase(iter); diff --git a/src/core/hle/service/glue/glue_manager.h b/src/core/hle/service/glue/glue_manager.h index cd0b092ac..1cf53d9d9 100644 --- a/src/core/hle/service/glue/glue_manager.h +++ b/src/core/hle/service/glue/glue_manager.h @@ -30,23 +30,23 @@ public:      ~ARPManager();      // Returns the ApplicationLaunchProperty corresponding to the provided title ID if it was -    // previously registered, otherwise ERR_NOT_REGISTERED if it was never registered or -    // ERR_INVALID_PROCESS_ID if the title ID is 0. +    // previously registered, otherwise ResultProcessIdNotRegistered if it was never registered or +    // ResultInvalidProcessId if the title ID is 0.      ResultVal<ApplicationLaunchProperty> GetLaunchProperty(u64 title_id) const;      // Returns a vector of the raw bytes of NACP data (necessarily 0x4000 in size) corresponding to -    // the provided title ID if it was previously registered, otherwise ERR_NOT_REGISTERED if it was -    // never registered or ERR_INVALID_PROCESS_ID if the title ID is 0. +    // the provided title ID if it was previously registered, otherwise ResultProcessIdNotRegistered +    // if it was never registered or ResultInvalidProcessId if the title ID is 0.      ResultVal<std::vector<u8>> GetControlProperty(u64 title_id) const;      // Adds a new entry to the internal database with the provided parameters, returning -    // ERR_INVALID_ACCESS if attempting to re-register a title ID without an intermediate Unregister -    // step, and ERR_INVALID_PROCESS_ID if the title ID is 0. +    // ResultProcessIdNotRegistered if attempting to re-register a title ID without an intermediate +    // Unregister step, and ResultInvalidProcessId if the title ID is 0.      Result Register(u64 title_id, ApplicationLaunchProperty launch, std::vector<u8> control);      // Removes the registration for the provided title ID from the database, returning -    // ERR_NOT_REGISTERED if it doesn't exist in the database and ERR_INVALID_PROCESS_ID if the -    // title ID is 0. +    // ResultProcessIdNotRegistered if it doesn't exist in the database and ResultInvalidProcessId +    // if the title ID is 0.      Result Unregister(u64 title_id);      // Removes all entries from the database, always succeeds. Should only be used when resetting diff --git a/src/core/hle/service/hid/controllers/stubbed.cpp b/src/core/hle/service/hid/controllers/stubbed.cpp index df9ee0c3f..9e2f3ab21 100644 --- a/src/core/hle/service/hid/controllers/stubbed.cpp +++ b/src/core/hle/service/hid/controllers/stubbed.cpp @@ -26,7 +26,7 @@ void Controller_Stubbed::OnUpdate(const Core::Timing::CoreTiming& core_timing) {      }      CommonHeader header{}; -    header.timestamp = core_timing.GetCPUTicks(); +    header.timestamp = core_timing.GetGlobalTimeNs().count();      header.total_entry_count = 17;      header.entry_count = 0;      header.last_entry_index = 0; diff --git a/src/core/hle/service/hid/controllers/touchscreen.cpp b/src/core/hle/service/hid/controllers/touchscreen.cpp index d90a4e732..3ef91df4b 100644 --- a/src/core/hle/service/hid/controllers/touchscreen.cpp +++ b/src/core/hle/service/hid/controllers/touchscreen.cpp @@ -32,7 +32,7 @@ void Controller_Touchscreen::OnInit() {}  void Controller_Touchscreen::OnRelease() {}  void Controller_Touchscreen::OnUpdate(const Core::Timing::CoreTiming& core_timing) { -    shared_memory->touch_screen_lifo.timestamp = core_timing.GetCPUTicks(); +    shared_memory->touch_screen_lifo.timestamp = core_timing.GetGlobalTimeNs().count();      if (!IsControllerActivated()) {          shared_memory->touch_screen_lifo.buffer_count = 0; @@ -85,7 +85,7 @@ void Controller_Touchscreen::OnUpdate(const Core::Timing::CoreTiming& core_timin      const auto active_fingers_count =          static_cast<std::size_t>(std::distance(active_fingers.begin(), end_iter)); -    const u64 tick = core_timing.GetCPUTicks(); +    const u64 timestamp = static_cast<u64>(core_timing.GetGlobalTimeNs().count());      const auto& last_entry = shared_memory->touch_screen_lifo.ReadCurrentEntry().state;      next_state.sampling_number = last_entry.sampling_number + 1; @@ -102,8 +102,8 @@ void Controller_Touchscreen::OnUpdate(const Core::Timing::CoreTiming& core_timin              touch_entry.diameter_x = Settings::values.touchscreen.diameter_x;              touch_entry.diameter_y = Settings::values.touchscreen.diameter_y;              touch_entry.rotation_angle = Settings::values.touchscreen.rotation_angle; -            touch_entry.delta_time = tick - active_fingers[id].last_touch; -            fingers[active_fingers[id].id].last_touch = tick; +            touch_entry.delta_time = timestamp - active_fingers[id].last_touch; +            fingers[active_fingers[id].id].last_touch = timestamp;              touch_entry.finger = active_fingers[id].id;              touch_entry.attribute.raw = active_fingers[id].attribute.raw;          } else { diff --git a/src/core/hle/service/ipc_helpers.h b/src/core/hle/service/ipc_helpers.h index 3e67123c7..8703b57ca 100644 --- a/src/core/hle/service/ipc_helpers.h +++ b/src/core/hle/service/ipc_helpers.h @@ -19,7 +19,7 @@  namespace IPC { -constexpr Result ERR_REMOTE_PROCESS_DEAD{ErrorModule::HIPC, 301}; +constexpr Result ResultSessionClosed{ErrorModule::HIPC, 301};  class RequestHelperBase {  protected: diff --git a/src/core/hle/service/nfp/amiibo_crypto.cpp b/src/core/hle/service/nfp/amiibo_crypto.cpp index ffb2f959c..ddf04b1d7 100644 --- a/src/core/hle/service/nfp/amiibo_crypto.cpp +++ b/src/core/hle/service/nfp/amiibo_crypto.cpp @@ -80,13 +80,16 @@ NTAG215File NfcDataToEncodedData(const EncryptedNTAG215File& nfc_data) {      encoded_data.hmac_data = nfc_data.user_memory.hmac_data;      encoded_data.constant_value = nfc_data.user_memory.constant_value;      encoded_data.write_counter = nfc_data.user_memory.write_counter; +    encoded_data.amiibo_version = nfc_data.user_memory.amiibo_version;      encoded_data.settings = nfc_data.user_memory.settings;      encoded_data.owner_mii = nfc_data.user_memory.owner_mii; -    encoded_data.title_id = nfc_data.user_memory.title_id; -    encoded_data.applicaton_write_counter = nfc_data.user_memory.applicaton_write_counter; +    encoded_data.application_id = nfc_data.user_memory.application_id; +    encoded_data.application_write_counter = nfc_data.user_memory.application_write_counter;      encoded_data.application_area_id = nfc_data.user_memory.application_area_id; +    encoded_data.application_id_byte = nfc_data.user_memory.application_id_byte;      encoded_data.unknown = nfc_data.user_memory.unknown;      encoded_data.unknown2 = nfc_data.user_memory.unknown2; +    encoded_data.application_area_crc = nfc_data.user_memory.application_area_crc;      encoded_data.application_area = nfc_data.user_memory.application_area;      encoded_data.hmac_tag = nfc_data.user_memory.hmac_tag;      encoded_data.lock_bytes = nfc_data.uuid.lock_bytes; @@ -111,13 +114,16 @@ EncryptedNTAG215File EncodedDataToNfcData(const NTAG215File& encoded_data) {      nfc_data.user_memory.hmac_data = encoded_data.hmac_data;      nfc_data.user_memory.constant_value = encoded_data.constant_value;      nfc_data.user_memory.write_counter = encoded_data.write_counter; +    nfc_data.user_memory.amiibo_version = encoded_data.amiibo_version;      nfc_data.user_memory.settings = encoded_data.settings;      nfc_data.user_memory.owner_mii = encoded_data.owner_mii; -    nfc_data.user_memory.title_id = encoded_data.title_id; -    nfc_data.user_memory.applicaton_write_counter = encoded_data.applicaton_write_counter; +    nfc_data.user_memory.application_id = encoded_data.application_id; +    nfc_data.user_memory.application_write_counter = encoded_data.application_write_counter;      nfc_data.user_memory.application_area_id = encoded_data.application_area_id; +    nfc_data.user_memory.application_id_byte = encoded_data.application_id_byte;      nfc_data.user_memory.unknown = encoded_data.unknown;      nfc_data.user_memory.unknown2 = encoded_data.unknown2; +    nfc_data.user_memory.application_area_crc = encoded_data.application_area_crc;      nfc_data.user_memory.application_area = encoded_data.application_area;      nfc_data.user_memory.hmac_tag = encoded_data.hmac_tag;      nfc_data.user_memory.model_info = encoded_data.model_info; diff --git a/src/core/hle/service/nfp/nfp_device.cpp b/src/core/hle/service/nfp/nfp_device.cpp index 1bdc42741..ddff90d6a 100644 --- a/src/core/hle/service/nfp/nfp_device.cpp +++ b/src/core/hle/service/nfp/nfp_device.cpp @@ -174,8 +174,8 @@ Result NfpDevice::StopDetection() {      if (device_state == DeviceState::TagFound || device_state == DeviceState::TagMounted) {          CloseAmiibo(); -        return ResultSuccess;      } +      if (device_state == DeviceState::SearchingForTag || device_state == DeviceState::TagRemoved) {          device_state = DeviceState::Initialized;          return ResultSuccess; @@ -204,9 +204,7 @@ Result NfpDevice::Flush() {      const auto& current_date = GetAmiiboDate(current_posix_time);      if (settings.write_date.raw_date != current_date.raw_date) {          settings.write_date = current_date; -        settings.crc_counter++; -        // TODO: Find how to calculate the crc check -        // settings.crc = CalculateCRC(settings); +        UpdateSettingsCrc();      }      tag_data.write_counter++; @@ -318,7 +316,7 @@ Result NfpDevice::GetCommonInfo(CommonInfo& common_info) const {      common_info = {          .last_write_date = settings.write_date.GetWriteDate(),          .write_counter = tag_data.write_counter, -        .version = 0, +        .version = tag_data.amiibo_version,          .application_area_size = sizeof(ApplicationArea),      };      return ResultSuccess; @@ -370,13 +368,95 @@ Result NfpDevice::GetRegisterInfo(RegisterInfo& register_info) const {          .mii_char_info = manager.ConvertV3ToCharInfo(tag_data.owner_mii),          .creation_date = settings.init_date.GetWriteDate(),          .amiibo_name = GetAmiiboName(settings), -        .font_region = {}, +        .font_region = settings.settings.font_region, +    }; + +    return ResultSuccess; +} + +Result NfpDevice::GetAdminInfo(AdminInfo& admin_info) const { +    if (device_state != DeviceState::TagMounted) { +        LOG_ERROR(Service_NFC, "Wrong device state {}", device_state); +        if (device_state == DeviceState::TagRemoved) { +            return TagRemoved; +        } +        return WrongDeviceState; +    } + +    if (mount_target == MountTarget::None || mount_target == MountTarget::Rom) { +        LOG_ERROR(Service_NFC, "Amiibo is read only", device_state); +        return WrongDeviceState; +    } + +    u8 flags = static_cast<u8>(tag_data.settings.settings.raw >> 0x4); +    if (tag_data.settings.settings.amiibo_initialized == 0) { +        flags = flags & 0xfe; +    } + +    u64 application_id = 0; +    u32 application_area_id = 0; +    AppAreaVersion app_area_version = AppAreaVersion::NotSet; +    if (tag_data.settings.settings.appdata_initialized != 0) { +        application_id = tag_data.application_id; +        app_area_version = +            static_cast<AppAreaVersion>(application_id >> application_id_version_offset & 0xf); + +        // Restore application id to original value +        if (application_id >> 0x38 != 0) { +            const u8 application_byte = tag_data.application_id_byte & 0xf; +            application_id = RemoveVersionByte(application_id) | +                             (static_cast<u64>(application_byte) << application_id_version_offset); +        } + +        application_area_id = tag_data.application_area_id; +    } + +    // TODO: Validate this data +    admin_info = { +        .application_id = application_id, +        .application_area_id = application_area_id, +        .crc_change_counter = tag_data.settings.crc_counter, +        .flags = flags, +        .tag_type = PackedTagType::Type2, +        .app_area_version = app_area_version,      };      return ResultSuccess;  } -Result NfpDevice::SetNicknameAndOwner(const AmiiboName& amiibo_name) { +Result NfpDevice::DeleteRegisterInfo() { +    if (device_state != DeviceState::TagMounted) { +        LOG_ERROR(Service_NFC, "Wrong device state {}", device_state); +        if (device_state == DeviceState::TagRemoved) { +            return TagRemoved; +        } +        return WrongDeviceState; +    } + +    if (mount_target == MountTarget::None || mount_target == MountTarget::Rom) { +        LOG_ERROR(Service_NFC, "Amiibo is read only", device_state); +        return WrongDeviceState; +    } + +    if (tag_data.settings.settings.amiibo_initialized == 0) { +        return RegistrationIsNotInitialized; +    } + +    Common::TinyMT rng{}; +    rng.GenerateRandomBytes(&tag_data.owner_mii, sizeof(tag_data.owner_mii)); +    rng.GenerateRandomBytes(&tag_data.settings.amiibo_name, sizeof(tag_data.settings.amiibo_name)); +    rng.GenerateRandomBytes(&tag_data.unknown, sizeof(u8)); +    rng.GenerateRandomBytes(&tag_data.unknown2[0], sizeof(u32)); +    rng.GenerateRandomBytes(&tag_data.unknown2[1], sizeof(u32)); +    rng.GenerateRandomBytes(&tag_data.application_area_crc, sizeof(u32)); +    rng.GenerateRandomBytes(&tag_data.settings.init_date, sizeof(u32)); +    tag_data.settings.settings.font_region.Assign(0); +    tag_data.settings.settings.amiibo_initialized.Assign(0); + +    return Flush(); +} + +Result NfpDevice::SetRegisterInfoPrivate(const AmiiboName& amiibo_name) {      if (device_state != DeviceState::TagMounted) {          LOG_ERROR(Service_NFP, "Wrong device state {}", device_state);          if (device_state == DeviceState::TagRemoved) { @@ -393,16 +473,23 @@ Result NfpDevice::SetNicknameAndOwner(const AmiiboName& amiibo_name) {      Service::Mii::MiiManager manager;      auto& settings = tag_data.settings; -    settings.init_date = GetAmiiboDate(current_posix_time); -    settings.write_date = GetAmiiboDate(current_posix_time); -    settings.crc_counter++; -    // TODO: Find how to calculate the crc check -    // settings.crc = CalculateCRC(settings); +    if (tag_data.settings.settings.amiibo_initialized == 0) { +        settings.init_date = GetAmiiboDate(current_posix_time); +        settings.write_date.raw_date = 0; +    }      SetAmiiboName(settings, amiibo_name);      tag_data.owner_mii = manager.ConvertCharInfoToV3(manager.BuildDefault(0)); +    tag_data.unknown = 0; +    tag_data.unknown2[6] = 0; +    settings.country_code_id = 0; +    settings.settings.font_region.Assign(0);      settings.settings.amiibo_initialized.Assign(1); +    // TODO: this is a mix of tag.file input +    std::array<u8, 0x7e> unknown_input{}; +    tag_data.application_area_crc = CalculateCrc(unknown_input); +      return Flush();  } @@ -425,23 +512,17 @@ Result NfpDevice::RestoreAmiibo() {      return ResultSuccess;  } -Result NfpDevice::DeleteAllData() { -    const auto result = DeleteApplicationArea(); -    if (result.IsError()) { -        return result; -    } +Result NfpDevice::Format() { +    auto result1 = DeleteApplicationArea(); +    auto result2 = DeleteRegisterInfo(); -    if (device_state != DeviceState::TagMounted) { -        LOG_ERROR(Service_NFP, "Wrong device state {}", device_state); -        if (device_state == DeviceState::TagRemoved) { -            return TagRemoved; -        } -        return WrongDeviceState; +    if (result1.IsError()) { +        return result1;      } -    Common::TinyMT rng{}; -    rng.GenerateRandomBytes(&tag_data.owner_mii, sizeof(tag_data.owner_mii)); -    tag_data.settings.settings.amiibo_initialized.Assign(0); +    if (result2.IsError()) { +        return result2; +    }      return Flush();  } @@ -569,7 +650,10 @@ Result NfpDevice::SetApplicationArea(std::span<const u8> data) {      rng.GenerateRandomBytes(tag_data.application_area.data() + data.size(),                              sizeof(ApplicationArea) - data.size()); -    tag_data.applicaton_write_counter++; +    if (tag_data.application_write_counter != counter_limit) { +        tag_data.application_write_counter++; +    } +      is_data_moddified = true;      return ResultSuccess; @@ -617,14 +701,25 @@ Result NfpDevice::RecreateApplicationArea(u32 access_id, std::span<const u8> dat      rng.GenerateRandomBytes(tag_data.application_area.data() + data.size(),                              sizeof(ApplicationArea) - data.size()); -    // TODO: Investigate why the title id needs to be moddified -    tag_data.title_id = system.GetApplicationProcessProgramID(); -    tag_data.title_id = tag_data.title_id | 0x30000000ULL; +    if (tag_data.application_write_counter != counter_limit) { +        tag_data.application_write_counter++; +    } + +    const u64 application_id = system.GetApplicationProcessProgramID(); + +    tag_data.application_id_byte = +        static_cast<u8>(application_id >> application_id_version_offset & 0xf); +    tag_data.application_id = +        RemoveVersionByte(application_id) | +        (static_cast<u64>(AppAreaVersion::NintendoSwitch) << application_id_version_offset);      tag_data.settings.settings.appdata_initialized.Assign(1);      tag_data.application_area_id = access_id; -    tag_data.applicaton_write_counter++;      tag_data.unknown = {}; +    // TODO: this is a mix of tag_data input +    std::array<u8, 0x7e> unknown_input{}; +    tag_data.application_area_crc = CalculateCrc(unknown_input); +      return Flush();  } @@ -642,12 +737,20 @@ Result NfpDevice::DeleteApplicationArea() {          return WrongDeviceState;      } +    if (tag_data.settings.settings.appdata_initialized == 0) { +        return ApplicationAreaIsNotInitialized; +    } + +    if (tag_data.application_write_counter != counter_limit) { +        tag_data.application_write_counter++; +    } +      Common::TinyMT rng{};      rng.GenerateRandomBytes(tag_data.application_area.data(), sizeof(ApplicationArea)); -    rng.GenerateRandomBytes(&tag_data.title_id, sizeof(u64)); +    rng.GenerateRandomBytes(&tag_data.application_id, sizeof(u64));      rng.GenerateRandomBytes(&tag_data.application_area_id, sizeof(u32)); +    rng.GenerateRandomBytes(&tag_data.application_id_byte, sizeof(u8));      tag_data.settings.settings.appdata_initialized.Assign(0); -    tag_data.applicaton_write_counter++;      tag_data.unknown = {};      return Flush(); @@ -719,4 +822,45 @@ AmiiboDate NfpDevice::GetAmiiboDate(s64 posix_time) const {      return amiibo_date;  } +u64 NfpDevice::RemoveVersionByte(u64 application_id) const { +    return application_id & ~(0xfULL << application_id_version_offset); +} + +void NfpDevice::UpdateSettingsCrc() { +    auto& settings = tag_data.settings; + +    if (settings.crc_counter != counter_limit) { +        settings.crc_counter++; +    } + +    // TODO: this reads data from a global, find what it is +    std::array<u8, 8> unknown_input{}; +    settings.crc = CalculateCrc(unknown_input); +} + +u32 NfpDevice::CalculateCrc(std::span<const u8> data) { +    constexpr u32 magic = 0xedb88320; +    u32 crc = 0xffffffff; + +    if (data.size() == 0) { +        return 0; +    } + +    for (u8 input : data) { +        u32 temp = (crc ^ input) >> 1; +        if (((crc ^ input) & 1) != 0) { +            temp = temp ^ magic; +        } + +        for (std::size_t step = 0; step < 7; ++step) { +            crc = temp >> 1; +            if ((temp & 1) != 0) { +                crc = temp >> 1 ^ magic; +            } +        } +    } + +    return ~crc; +} +  } // namespace Service::NFP diff --git a/src/core/hle/service/nfp/nfp_device.h b/src/core/hle/service/nfp/nfp_device.h index b6a46f2ac..06386401d 100644 --- a/src/core/hle/service/nfp/nfp_device.h +++ b/src/core/hle/service/nfp/nfp_device.h @@ -47,10 +47,12 @@ public:      Result GetCommonInfo(CommonInfo& common_info) const;      Result GetModelInfo(ModelInfo& model_info) const;      Result GetRegisterInfo(RegisterInfo& register_info) const; +    Result GetAdminInfo(AdminInfo& admin_info) const; -    Result SetNicknameAndOwner(const AmiiboName& amiibo_name); +    Result DeleteRegisterInfo(); +    Result SetRegisterInfoPrivate(const AmiiboName& amiibo_name);      Result RestoreAmiibo(); -    Result DeleteAllData(); +    Result Format();      Result OpenApplicationArea(u32 access_id);      Result GetApplicationAreaId(u32& application_area_id) const; @@ -76,6 +78,9 @@ private:      AmiiboName GetAmiiboName(const AmiiboSettings& settings) const;      void SetAmiiboName(AmiiboSettings& settings, const AmiiboName& amiibo_name);      AmiiboDate GetAmiiboDate(s64 posix_time) const; +    u64 RemoveVersionByte(u64 application_id) const; +    void UpdateSettingsCrc(); +    u32 CalculateCrc(std::span<const u8>);      bool is_controller_set{};      int callback_key; diff --git a/src/core/hle/service/nfp/nfp_types.h b/src/core/hle/service/nfp/nfp_types.h index fc228c2b2..142343d6e 100644 --- a/src/core/hle/service/nfp/nfp_types.h +++ b/src/core/hle/service/nfp/nfp_types.h @@ -10,6 +10,8 @@  namespace Service::NFP {  static constexpr std::size_t amiibo_name_length = 0xA; +static constexpr std::size_t application_id_version_offset = 0x1c; +static constexpr std::size_t counter_limit = 0xffff;  enum class ServiceType : u32 {      User, @@ -99,6 +101,14 @@ enum class TagProtocol : u32 {      All = 0xFFFFFFFFU,  }; +enum class AppAreaVersion : u8 { +    Nintendo3DS = 0, +    NintendoWiiU = 1, +    Nintendo3DSv2 = 2, +    NintendoSwitch = 3, +    NotSet = 0xFF, +}; +  enum class CabinetMode : u8 {      StartNicknameAndOwnerSettings,      StartGameDataEraser, @@ -197,6 +207,7 @@ struct Settings {      union {          u8 raw{}; +        BitField<0, 4, u8> font_region;          BitField<4, 1, u8> amiibo_initialized;          BitField<5, 1, u8> appdata_initialized;      }; @@ -236,18 +247,20 @@ static_assert(sizeof(NTAG215Password) == 0x8, "NTAG215Password is an invalid siz  struct EncryptedAmiiboFile {      u8 constant_value;                     // Must be A5      u16_be write_counter;                  // Number of times the amiibo has been written? -    INSERT_PADDING_BYTES(0x1);             // Unknown 1 +    u8 amiibo_version;                     // Amiibo file version      AmiiboSettings settings;               // Encrypted amiibo settings      HashData hmac_tag;                     // Hash      AmiiboModelInfo model_info;            // Encrypted amiibo model info      HashData keygen_salt;                  // Salt      HashData hmac_data;                    // Hash      Service::Mii::Ver3StoreData owner_mii; // Encrypted Mii data -    u64_be title_id;                       // Encrypted Game id -    u16_be applicaton_write_counter;       // Encrypted Counter +    u64_be application_id;                 // Encrypted Game id +    u16_be application_write_counter;      // Encrypted Counter      u32_be application_area_id;            // Encrypted Game id -    std::array<u8, 0x2> unknown; -    std::array<u32, 0x8> unknown2; +    u8 application_id_byte; +    u8 unknown; +    std::array<u32, 0x7> unknown2; +    u32_be application_area_crc;      ApplicationArea application_area; // Encrypted Game data  };  static_assert(sizeof(EncryptedAmiiboFile) == 0x1F8, "AmiiboFile is an invalid size"); @@ -259,14 +272,16 @@ struct NTAG215File {      HashData hmac_data;        // Hash      u8 constant_value;         // Must be A5      u16_be write_counter;      // Number of times the amiibo has been written? -    INSERT_PADDING_BYTES(0x1); // Unknown 1 +    u8 amiibo_version;         // Amiibo file version      AmiiboSettings settings; -    Service::Mii::Ver3StoreData owner_mii; // Encrypted Mii data -    u64_be title_id; -    u16_be applicaton_write_counter; // Encrypted Counter +    Service::Mii::Ver3StoreData owner_mii; // Mii data +    u64_be application_id;                 // Game id +    u16_be application_write_counter;      // Counter      u32_be application_area_id; -    std::array<u8, 0x2> unknown; -    std::array<u32, 0x8> unknown2; +    u8 application_id_byte; +    u8 unknown; +    std::array<u32, 0x7> unknown2; +    u32_be application_area_crc;      ApplicationArea application_area; // Encrypted Game data      HashData hmac_tag;                // Hash      UniqueSerialNumber uid;           // Unique serial number @@ -336,6 +351,18 @@ struct RegisterInfo {  };  static_assert(sizeof(RegisterInfo) == 0x100, "RegisterInfo is an invalid size"); +struct AdminInfo { +    u64 application_id; +    u32 application_area_id; +    u16 crc_change_counter; +    u8 flags; +    PackedTagType tag_type; +    AppAreaVersion app_area_version; +    INSERT_PADDING_BYTES(0x7); +    INSERT_PADDING_BYTES(0x28); +}; +static_assert(sizeof(AdminInfo) == 0x40, "AdminInfo is an invalid size"); +  struct SectorKey {      MifareCmd command;      u8 unknown; // Usually 1 diff --git a/src/core/hle/service/ns/errors.h b/src/core/hle/service/ns/errors.h index 8a7621798..16d2ea6f7 100644 --- a/src/core/hle/service/ns/errors.h +++ b/src/core/hle/service/ns/errors.h @@ -7,5 +7,6 @@  namespace Service::NS { -constexpr Result ERR_APPLICATION_LANGUAGE_NOT_FOUND{ErrorModule::NS, 300}; -}
\ No newline at end of file +constexpr Result ResultApplicationLanguageNotFound{ErrorModule::NS, 300}; + +} diff --git a/src/core/hle/service/ns/ns.cpp b/src/core/hle/service/ns/ns.cpp index d6f0faea2..376067a95 100644 --- a/src/core/hle/service/ns/ns.cpp +++ b/src/core/hle/service/ns/ns.cpp @@ -416,14 +416,14 @@ ResultVal<u8> IApplicationManagerInterface::GetApplicationDesiredLanguage(      if (application_language == std::nullopt) {          LOG_ERROR(Service_NS, "Could not convert application language! language_code={}",                    language_code); -        return ERR_APPLICATION_LANGUAGE_NOT_FOUND; +        return Service::NS::ResultApplicationLanguageNotFound;      }      const auto priority_list = GetApplicationLanguagePriorityList(*application_language);      if (!priority_list) {          LOG_ERROR(Service_NS,                    "Could not find application language priorities! application_language={}",                    *application_language); -        return ERR_APPLICATION_LANGUAGE_NOT_FOUND; +        return Service::NS::ResultApplicationLanguageNotFound;      }      // Try to find a valid language. @@ -436,7 +436,7 @@ ResultVal<u8> IApplicationManagerInterface::GetApplicationDesiredLanguage(      LOG_ERROR(Service_NS, "Could not find a valid language! supported_languages={:08X}",                supported_languages); -    return ERR_APPLICATION_LANGUAGE_NOT_FOUND; +    return Service::NS::ResultApplicationLanguageNotFound;  }  void IApplicationManagerInterface::ConvertApplicationLanguageToLanguageCode( @@ -461,7 +461,7 @@ ResultVal<u64> IApplicationManagerInterface::ConvertApplicationLanguageToLanguag          ConvertToLanguageCode(static_cast<ApplicationLanguage>(application_language));      if (language_code == std::nullopt) {          LOG_ERROR(Service_NS, "Language not found! application_language={}", application_language); -        return ERR_APPLICATION_LANGUAGE_NOT_FOUND; +        return Service::NS::ResultApplicationLanguageNotFound;      }      return static_cast<u64>(*language_code); diff --git a/src/core/hle/service/server_manager.cpp b/src/core/hle/service/server_manager.cpp index c91f6d880..bd04cd023 100644 --- a/src/core/hle/service/server_manager.cpp +++ b/src/core/hle/service/server_manager.cpp @@ -404,7 +404,7 @@ Result ServerManager::CompleteSyncRequest(RequestState&& request) {      rc = request.session->SendReplyHLE();      // If the session has been closed, we're done. -    if (rc == Kernel::ResultSessionClosed || service_rc == IPC::ERR_REMOTE_PROCESS_DEAD) { +    if (rc == Kernel::ResultSessionClosed || service_rc == IPC::ResultSessionClosed) {          // Close the session.          request.session->Close(); diff --git a/src/core/hle/service/service.cpp b/src/core/hle/service/service.cpp index eed615377..69cdb5918 100644 --- a/src/core/hle/service/service.cpp +++ b/src/core/hle/service/service.cpp @@ -176,7 +176,7 @@ Result ServiceFrameworkBase::HandleSyncRequest(Kernel::KServerSession& session,      case IPC::CommandType::TIPC_Close: {          IPC::ResponseBuilder rb{ctx, 2};          rb.Push(ResultSuccess); -        result = IPC::ERR_REMOTE_PROCESS_DEAD; +        result = IPC::ResultSessionClosed;          break;      }      case IPC::CommandType::ControlWithContext: diff --git a/src/core/hle/service/set/set.cpp b/src/core/hle/service/set/set.cpp index 88df52331..f5788b481 100644 --- a/src/core/hle/service/set/set.cpp +++ b/src/core/hle/service/set/set.cpp @@ -74,7 +74,7 @@ constexpr std::array<std::pair<LanguageCode, KeyboardLayout>, 18> language_to_la  constexpr std::size_t PRE_4_0_0_MAX_ENTRIES = 0xF;  constexpr std::size_t POST_4_0_0_MAX_ENTRIES = 0x40; -constexpr Result ERR_INVALID_LANGUAGE{ErrorModule::Settings, 625}; +constexpr Result ResultInvalidLanguage{ErrorModule::Settings, 625};  void PushResponseLanguageCode(HLERequestContext& ctx, std::size_t num_language_codes) {      IPC::ResponseBuilder rb{ctx, 3}; @@ -130,7 +130,7 @@ void SET::MakeLanguageCode(HLERequestContext& ctx) {      if (index >= available_language_codes.size()) {          LOG_ERROR(Service_SET, "Invalid language code index! index={}", index);          IPC::ResponseBuilder rb{ctx, 2}; -        rb.Push(ERR_INVALID_LANGUAGE); +        rb.Push(Set::ResultInvalidLanguage);          return;      } diff --git a/src/core/hle/service/sm/sm.cpp b/src/core/hle/service/sm/sm.cpp index a46f47d3e..b4046d3ce 100644 --- a/src/core/hle/service/sm/sm.cpp +++ b/src/core/hle/service/sm/sm.cpp @@ -18,10 +18,10 @@  namespace Service::SM { -constexpr Result ERR_NOT_INITIALIZED(ErrorModule::SM, 2); -constexpr Result ERR_ALREADY_REGISTERED(ErrorModule::SM, 4); -constexpr Result ERR_INVALID_NAME(ErrorModule::SM, 6); -constexpr Result ERR_SERVICE_NOT_REGISTERED(ErrorModule::SM, 7); +constexpr Result ResultInvalidClient(ErrorModule::SM, 2); +constexpr Result ResultAlreadyRegistered(ErrorModule::SM, 4); +constexpr Result ResultInvalidServiceName(ErrorModule::SM, 6); +constexpr Result ResultNotRegistered(ErrorModule::SM, 7);  ServiceManager::ServiceManager(Kernel::KernelCore& kernel_) : kernel{kernel_} {      controller_interface = std::make_unique<Controller>(kernel.System()); @@ -45,7 +45,7 @@ void ServiceManager::InvokeControlRequest(HLERequestContext& context) {  static Result ValidateServiceName(const std::string& name) {      if (name.empty() || name.size() > 8) {          LOG_ERROR(Service_SM, "Invalid service name! service={}", name); -        return ERR_INVALID_NAME; +        return Service::SM::ResultInvalidServiceName;      }      return ResultSuccess;  } @@ -58,7 +58,7 @@ Result ServiceManager::RegisterService(std::string name, u32 max_sessions,      std::scoped_lock lk{lock};      if (registered_services.find(name) != registered_services.end()) {          LOG_ERROR(Service_SM, "Service is already registered! service={}", name); -        return ERR_ALREADY_REGISTERED; +        return Service::SM::ResultAlreadyRegistered;      }      auto* port = Kernel::KPort::Create(kernel); @@ -80,7 +80,7 @@ Result ServiceManager::UnregisterService(const std::string& name) {      const auto iter = registered_services.find(name);      if (iter == registered_services.end()) {          LOG_ERROR(Service_SM, "Server is not registered! service={}", name); -        return ERR_SERVICE_NOT_REGISTERED; +        return Service::SM::ResultNotRegistered;      }      registered_services.erase(iter); @@ -96,7 +96,7 @@ ResultVal<Kernel::KPort*> ServiceManager::GetServicePort(const std::string& name      auto it = service_ports.find(name);      if (it == service_ports.end()) {          LOG_WARNING(Service_SM, "Server is not registered! service={}", name); -        return ERR_SERVICE_NOT_REGISTERED; +        return Service::SM::ResultNotRegistered;      }      return it->second; @@ -160,7 +160,7 @@ static std::string PopServiceName(IPC::RequestParser& rp) {  ResultVal<Kernel::KClientSession*> SM::GetServiceImpl(HLERequestContext& ctx) {      if (!ctx.GetManager()->GetIsInitializedForSm()) { -        return ERR_NOT_INITIALIZED; +        return Service::SM::ResultInvalidClient;      }      IPC::RequestParser rp{ctx}; @@ -168,15 +168,15 @@ ResultVal<Kernel::KClientSession*> SM::GetServiceImpl(HLERequestContext& ctx) {      // Find the named port.      auto port_result = service_manager.GetServicePort(name); -    if (port_result.Code() == ERR_INVALID_NAME) { +    if (port_result.Code() == Service::SM::ResultInvalidServiceName) {          LOG_ERROR(Service_SM, "Invalid service name '{}'", name); -        return ERR_INVALID_NAME; +        return Service::SM::ResultInvalidServiceName;      }      if (port_result.Failed()) {          LOG_INFO(Service_SM, "Waiting for service {} to become available", name);          ctx.SetIsDeferred(); -        return ERR_SERVICE_NOT_REGISTERED; +        return Service::SM::ResultNotRegistered;      }      auto& port = port_result.Unwrap(); diff --git a/src/core/perf_stats.cpp b/src/core/perf_stats.cpp index f09c176f8..1231c0dc8 100644 --- a/src/core/perf_stats.cpp +++ b/src/core/perf_stats.cpp @@ -126,8 +126,8 @@ double PerfStats::GetLastFrameTimeScale() const {  }  void SpeedLimiter::DoSpeedLimiting(microseconds current_system_time_us) { -    if (!Settings::values.use_speed_limit.GetValue() || -        Settings::values.use_multi_core.GetValue()) { +    if (Settings::values.use_multi_core.GetValue() || +        !Settings::values.use_speed_limit.GetValue()) {          return;      } diff --git a/src/input_common/drivers/joycon.cpp b/src/input_common/drivers/joycon.cpp index b4cd39a20..8b57ebe07 100644 --- a/src/input_common/drivers/joycon.cpp +++ b/src/input_common/drivers/joycon.cpp @@ -307,8 +307,8 @@ Common::Input::DriverResult Joycons::SetPollingMode(const PadIdentifier& identif      switch (polling_mode) {      case Common::Input::PollingMode::Active:          return static_cast<Common::Input::DriverResult>(handle->SetActiveMode()); -    case Common::Input::PollingMode::Pasive: -        return static_cast<Common::Input::DriverResult>(handle->SetPasiveMode()); +    case Common::Input::PollingMode::Passive: +        return static_cast<Common::Input::DriverResult>(handle->SetPassiveMode());      case Common::Input::PollingMode::IR:          return static_cast<Common::Input::DriverResult>(handle->SetIrMode());      case Common::Input::PollingMode::NFC: diff --git a/src/input_common/drivers/mouse.cpp b/src/input_common/drivers/mouse.cpp index 8b7f9aee9..94e92c37d 100644 --- a/src/input_common/drivers/mouse.cpp +++ b/src/input_common/drivers/mouse.cpp @@ -3,6 +3,7 @@  #include <thread>  #include <fmt/format.h> +#include <math.h>  #include "common/param_package.h"  #include "common/settings.h" @@ -11,8 +12,9 @@  namespace InputCommon {  constexpr int update_time = 10; -constexpr float default_stick_sensitivity = 0.022f; -constexpr float default_motion_sensitivity = 0.008f; +constexpr float default_stick_sensitivity = 0.0044f; +constexpr float default_motion_sensitivity = 0.0003f; +constexpr float maximum_rotation_speed = 2.0f;  constexpr int mouse_axis_x = 0;  constexpr int mouse_axis_y = 1;  constexpr int wheel_axis_x = 2; @@ -99,11 +101,13 @@ void Mouse::UpdateMotionInput() {      const float sensitivity =          Settings::values.mouse_panning_sensitivity.GetValue() * default_motion_sensitivity; -    // Slow movement by 7% -    if (Settings::values.mouse_panning) { -        last_motion_change *= 0.93f; -    } else { -        last_motion_change.z *= 0.93f; +    const float rotation_velocity = std::sqrt(last_motion_change.x * last_motion_change.x + +                                              last_motion_change.y * last_motion_change.y); + +    if (rotation_velocity > maximum_rotation_speed / sensitivity) { +        const float multiplier = maximum_rotation_speed / rotation_velocity / sensitivity; +        last_motion_change.x = last_motion_change.x * multiplier; +        last_motion_change.y = last_motion_change.y * multiplier;      }      const BasicMotion motion_data{ @@ -116,6 +120,12 @@ void Mouse::UpdateMotionInput() {          .delta_timestamp = update_time * 1000,      }; +    if (Settings::values.mouse_panning) { +        last_motion_change.x = 0; +        last_motion_change.y = 0; +    } +    last_motion_change.z = 0; +      SetMotion(motion_identifier, 0, motion_data);  } @@ -125,7 +135,7 @@ void Mouse::Move(int x, int y, int center_x, int center_y) {          auto mouse_change =              (Common::MakeVec(x, y) - Common::MakeVec(center_x, center_y)).Cast<float>(); -        Common::Vec3<float> motion_change{-mouse_change.y, -mouse_change.x, last_motion_change.z}; +        last_motion_change += {-mouse_change.y, -mouse_change.x, last_motion_change.z};          const auto move_distance = mouse_change.Length();          if (move_distance == 0) { @@ -141,7 +151,6 @@ void Mouse::Move(int x, int y, int center_x, int center_y) {          // Average mouse movements          last_mouse_change = (last_mouse_change * 0.91f) + (mouse_change * 0.09f); -        last_motion_change = (last_motion_change * 0.69f) + (motion_change * 0.31f);          const auto last_move_distance = last_mouse_change.Length(); diff --git a/src/input_common/drivers/virtual_amiibo.h b/src/input_common/drivers/virtual_amiibo.h index 13cacfc0a..488d00b31 100644 --- a/src/input_common/drivers/virtual_amiibo.h +++ b/src/input_common/drivers/virtual_amiibo.h @@ -60,6 +60,6 @@ private:      std::string file_path{};      State state{State::Initialized};      std::vector<u8> nfc_data; -    Common::Input::PollingMode polling_mode{Common::Input::PollingMode::Pasive}; +    Common::Input::PollingMode polling_mode{Common::Input::PollingMode::Passive};  };  } // namespace InputCommon diff --git a/src/input_common/helpers/joycon_driver.cpp b/src/input_common/helpers/joycon_driver.cpp index e65b6b845..78cc5893c 100644 --- a/src/input_common/helpers/joycon_driver.cpp +++ b/src/input_common/helpers/joycon_driver.cpp @@ -410,7 +410,7 @@ DriverResult JoyconDriver::SetIrsConfig(IrsMode mode_, IrsResolution format_) {      return result;  } -DriverResult JoyconDriver::SetPasiveMode() { +DriverResult JoyconDriver::SetPassiveMode() {      std::scoped_lock lock{mutex};      motion_enabled = false;      hidbus_enabled = false; diff --git a/src/input_common/helpers/joycon_driver.h b/src/input_common/helpers/joycon_driver.h index c1e189fa5..b52a13ecf 100644 --- a/src/input_common/helpers/joycon_driver.h +++ b/src/input_common/helpers/joycon_driver.h @@ -44,7 +44,7 @@ public:      DriverResult SetVibration(const VibrationValue& vibration);      DriverResult SetLedConfig(u8 led_pattern);      DriverResult SetIrsConfig(IrsMode mode_, IrsResolution format_); -    DriverResult SetPasiveMode(); +    DriverResult SetPassiveMode();      DriverResult SetActiveMode();      DriverResult SetIrMode();      DriverResult SetNfcMode(); diff --git a/src/input_common/helpers/joycon_protocol/joycon_types.h b/src/input_common/helpers/joycon_protocol/joycon_types.h index 2e50a99a8..dcac0e422 100644 --- a/src/input_common/helpers/joycon_protocol/joycon_types.h +++ b/src/input_common/helpers/joycon_protocol/joycon_types.h @@ -78,7 +78,7 @@ enum class PadButton : u32 {      Capture = 0x200000,  }; -enum class PasivePadButton : u32 { +enum class PassivePadButton : u32 {      Down_A = 0x0001,      Right_X = 0x0002,      Left_B = 0x0004, @@ -95,7 +95,7 @@ enum class PasivePadButton : u32 {      ZL_ZR = 0x8000,  }; -enum class PasivePadStick : u8 { +enum class PassivePadStick : u8 {      Right = 0x00,      RightDown = 0x01,      Down = 0x02, diff --git a/src/input_common/helpers/joycon_protocol/poller.cpp b/src/input_common/helpers/joycon_protocol/poller.cpp index ab48352b8..dca797f7a 100644 --- a/src/input_common/helpers/joycon_protocol/poller.cpp +++ b/src/input_common/helpers/joycon_protocol/poller.cpp @@ -48,13 +48,13 @@ void JoyconPoller::ReadPassiveMode(std::span<u8> buffer) {      switch (device_type) {      case ControllerType::Left: -        UpdatePasiveLeftPadInput(data); +        UpdatePassiveLeftPadInput(data);          break;      case ControllerType::Right: -        UpdatePasiveRightPadInput(data); +        UpdatePassiveRightPadInput(data);          break;      case ControllerType::Pro: -        UpdatePasiveProPadInput(data); +        UpdatePassiveProPadInput(data);          break;      default:          break; @@ -210,12 +210,12 @@ void JoyconPoller::UpdateActiveProPadInput(const InputReportActive& input,      }  } -void JoyconPoller::UpdatePasiveLeftPadInput(const InputReportPassive& input) { -    static constexpr std::array<PasivePadButton, 11> left_buttons{ -        PasivePadButton::Down_A,  PasivePadButton::Right_X, PasivePadButton::Left_B, -        PasivePadButton::Up_Y,    PasivePadButton::SL,      PasivePadButton::SR, -        PasivePadButton::L_R,     PasivePadButton::ZL_ZR,   PasivePadButton::Minus, -        PasivePadButton::Capture, PasivePadButton::StickL, +void JoyconPoller::UpdatePassiveLeftPadInput(const InputReportPassive& input) { +    static constexpr std::array<PassivePadButton, 11> left_buttons{ +        PassivePadButton::Down_A,  PassivePadButton::Right_X, PassivePadButton::Left_B, +        PassivePadButton::Up_Y,    PassivePadButton::SL,      PassivePadButton::SR, +        PassivePadButton::L_R,     PassivePadButton::ZL_ZR,   PassivePadButton::Minus, +        PassivePadButton::Capture, PassivePadButton::StickL,      };      for (auto left_button : left_buttons) { @@ -225,17 +225,17 @@ void JoyconPoller::UpdatePasiveLeftPadInput(const InputReportPassive& input) {      }      const auto [left_axis_x, left_axis_y] = -        GetPassiveAxisValue(static_cast<PasivePadStick>(input.stick_state)); +        GetPassiveAxisValue(static_cast<PassivePadStick>(input.stick_state));      callbacks.on_stick_data(static_cast<int>(PadAxes::LeftStickX), left_axis_x);      callbacks.on_stick_data(static_cast<int>(PadAxes::LeftStickY), left_axis_y);  } -void JoyconPoller::UpdatePasiveRightPadInput(const InputReportPassive& input) { -    static constexpr std::array<PasivePadButton, 11> right_buttons{ -        PasivePadButton::Down_A, PasivePadButton::Right_X, PasivePadButton::Left_B, -        PasivePadButton::Up_Y,   PasivePadButton::SL,      PasivePadButton::SR, -        PasivePadButton::L_R,    PasivePadButton::ZL_ZR,   PasivePadButton::Plus, -        PasivePadButton::Home,   PasivePadButton::StickR, +void JoyconPoller::UpdatePassiveRightPadInput(const InputReportPassive& input) { +    static constexpr std::array<PassivePadButton, 11> right_buttons{ +        PassivePadButton::Down_A, PassivePadButton::Right_X, PassivePadButton::Left_B, +        PassivePadButton::Up_Y,   PassivePadButton::SL,      PassivePadButton::SR, +        PassivePadButton::L_R,    PassivePadButton::ZL_ZR,   PassivePadButton::Plus, +        PassivePadButton::Home,   PassivePadButton::StickR,      };      for (auto right_button : right_buttons) { @@ -245,18 +245,18 @@ void JoyconPoller::UpdatePasiveRightPadInput(const InputReportPassive& input) {      }      const auto [right_axis_x, right_axis_y] = -        GetPassiveAxisValue(static_cast<PasivePadStick>(input.stick_state)); +        GetPassiveAxisValue(static_cast<PassivePadStick>(input.stick_state));      callbacks.on_stick_data(static_cast<int>(PadAxes::RightStickX), right_axis_x);      callbacks.on_stick_data(static_cast<int>(PadAxes::RightStickY), right_axis_y);  } -void JoyconPoller::UpdatePasiveProPadInput(const InputReportPassive& input) { -    static constexpr std::array<PasivePadButton, 14> pro_buttons{ -        PasivePadButton::Down_A, PasivePadButton::Right_X, PasivePadButton::Left_B, -        PasivePadButton::Up_Y,   PasivePadButton::SL,      PasivePadButton::SR, -        PasivePadButton::L_R,    PasivePadButton::ZL_ZR,   PasivePadButton::Minus, -        PasivePadButton::Plus,   PasivePadButton::Capture, PasivePadButton::Home, -        PasivePadButton::StickL, PasivePadButton::StickR, +void JoyconPoller::UpdatePassiveProPadInput(const InputReportPassive& input) { +    static constexpr std::array<PassivePadButton, 14> pro_buttons{ +        PassivePadButton::Down_A, PassivePadButton::Right_X, PassivePadButton::Left_B, +        PassivePadButton::Up_Y,   PassivePadButton::SL,      PassivePadButton::SR, +        PassivePadButton::L_R,    PassivePadButton::ZL_ZR,   PassivePadButton::Minus, +        PassivePadButton::Plus,   PassivePadButton::Capture, PassivePadButton::Home, +        PassivePadButton::StickL, PassivePadButton::StickR,      };      for (auto pro_button : pro_buttons) { @@ -266,9 +266,9 @@ void JoyconPoller::UpdatePasiveProPadInput(const InputReportPassive& input) {      }      const auto [left_axis_x, left_axis_y] = -        GetPassiveAxisValue(static_cast<PasivePadStick>(input.stick_state && 0xf)); +        GetPassiveAxisValue(static_cast<PassivePadStick>(input.stick_state & 0xf));      const auto [right_axis_x, right_axis_y] = -        GetPassiveAxisValue(static_cast<PasivePadStick>(input.stick_state >> 4)); +        GetPassiveAxisValue(static_cast<PassivePadStick>(input.stick_state >> 4));      callbacks.on_stick_data(static_cast<int>(PadAxes::LeftStickX), left_axis_x);      callbacks.on_stick_data(static_cast<int>(PadAxes::LeftStickY), left_axis_y);      callbacks.on_stick_data(static_cast<int>(PadAxes::RightStickX), right_axis_x); @@ -283,25 +283,25 @@ f32 JoyconPoller::GetAxisValue(u16 raw_value, Joycon::JoyStickAxisCalibration ca      return value / calibration.min;  } -std::pair<f32, f32> JoyconPoller::GetPassiveAxisValue(PasivePadStick raw_value) const { +std::pair<f32, f32> JoyconPoller::GetPassiveAxisValue(PassivePadStick raw_value) const {      switch (raw_value) { -    case PasivePadStick::Right: +    case PassivePadStick::Right:          return {1.0f, 0.0f}; -    case PasivePadStick::RightDown: +    case PassivePadStick::RightDown:          return {1.0f, -1.0f}; -    case PasivePadStick::Down: +    case PassivePadStick::Down:          return {0.0f, -1.0f}; -    case PasivePadStick::DownLeft: +    case PassivePadStick::DownLeft:          return {-1.0f, -1.0f}; -    case PasivePadStick::Left: +    case PassivePadStick::Left:          return {-1.0f, 0.0f}; -    case PasivePadStick::LeftUp: +    case PassivePadStick::LeftUp:          return {-1.0f, 1.0f}; -    case PasivePadStick::Up: +    case PassivePadStick::Up:          return {0.0f, 1.0f}; -    case PasivePadStick::UpRight: +    case PassivePadStick::UpRight:          return {1.0f, 1.0f}; -    case PasivePadStick::Neutral: +    case PassivePadStick::Neutral:      default:          return {0.0f, 0.0f};      } diff --git a/src/input_common/helpers/joycon_protocol/poller.h b/src/input_common/helpers/joycon_protocol/poller.h index 5c897f070..0fa72c6db 100644 --- a/src/input_common/helpers/joycon_protocol/poller.h +++ b/src/input_common/helpers/joycon_protocol/poller.h @@ -46,15 +46,15 @@ private:                                     const MotionStatus& motion_status);      void UpdateActiveProPadInput(const InputReportActive& input, const MotionStatus& motion_status); -    void UpdatePasiveLeftPadInput(const InputReportPassive& buffer); -    void UpdatePasiveRightPadInput(const InputReportPassive& buffer); -    void UpdatePasiveProPadInput(const InputReportPassive& buffer); +    void UpdatePassiveLeftPadInput(const InputReportPassive& buffer); +    void UpdatePassiveRightPadInput(const InputReportPassive& buffer); +    void UpdatePassiveProPadInput(const InputReportPassive& buffer);      /// Returns a calibrated joystick axis from raw axis data      f32 GetAxisValue(u16 raw_value, JoyStickAxisCalibration calibration) const;      /// Returns a digital joystick axis from passive axis data -    std::pair<f32, f32> GetPassiveAxisValue(PasivePadStick raw_value) const; +    std::pair<f32, f32> GetPassiveAxisValue(PassivePadStick raw_value) const;      /// Returns a calibrated accelerometer axis from raw motion data      f32 GetAccelerometerValue(s16 raw, const MotionSensorCalibration& cal, diff --git a/src/input_common/input_mapping.cpp b/src/input_common/input_mapping.cpp index 2ff480ff9..9361b00c5 100644 --- a/src/input_common/input_mapping.cpp +++ b/src/input_common/input_mapping.cpp @@ -146,6 +146,7 @@ void MappingFactory::RegisterMotion(const MappingData& data) {      if (data.engine == "mouse") {          new_input.Set("motion", 0);          new_input.Set("pad", 1); +        new_input.Set("threshold", 0.001f);          input_queue.Push(new_input);          return;      } diff --git a/src/shader_recompiler/ir_opt/global_memory_to_storage_buffer_pass.cpp b/src/shader_recompiler/ir_opt/global_memory_to_storage_buffer_pass.cpp index 336338e62..d1e59f22e 100644 --- a/src/shader_recompiler/ir_opt/global_memory_to_storage_buffer_pass.cpp +++ b/src/shader_recompiler/ir_opt/global_memory_to_storage_buffer_pass.cpp @@ -35,6 +35,7 @@ struct Bias {      u32 index;      u32 offset_begin;      u32 offset_end; +    u32 alignment;  };  using boost::container::flat_set; @@ -349,7 +350,8 @@ std::optional<StorageBufferAddr> Track(const IR::Value& value, const Bias* bias)              .index = index.U32(),              .offset = offset.U32(),          }; -        if (!Common::IsAligned(storage_buffer.offset, 16)) { +        const u32 alignment{bias ? bias->alignment : 8U}; +        if (!Common::IsAligned(storage_buffer.offset, alignment)) {              // The SSBO pointer has to be aligned              return std::nullopt;          } @@ -371,6 +373,7 @@ void CollectStorageBuffers(IR::Block& block, IR::Inst& inst, StorageInfo& info)          .index = 0,          .offset_begin = 0x110,          .offset_end = 0x610, +        .alignment = 16,      };      // Track the low address of the instruction      const std::optional<LowAddrInfo> low_addr_info{TrackLowAddress(&inst)}; @@ -386,8 +389,11 @@ void CollectStorageBuffers(IR::Block& block, IR::Inst& inst, StorageInfo& info)          storage_buffer = Track(low_addr, nullptr);          if (!storage_buffer) {              // If that also fails, use NVN fallbacks +            LOG_WARNING(Shader, "Storage buffer failed to track, using global memory fallbacks");              return;          } +        LOG_WARNING(Shader, "Storage buffer tracked without bias, index {} offset {}", +                    storage_buffer->index, storage_buffer->offset);      }      // Collect storage buffer and the instruction      if (IsGlobalMemoryWrite(inst)) { diff --git a/src/video_core/buffer_cache/buffer_cache.h b/src/video_core/buffer_cache/buffer_cache.h index 2a150ccdc..1f656ffa8 100644 --- a/src/video_core/buffer_cache/buffer_cache.h +++ b/src/video_core/buffer_cache/buffer_cache.h @@ -383,7 +383,8 @@ private:      void NotifyBufferDeletion(); -    [[nodiscard]] Binding StorageBufferBinding(GPUVAddr ssbo_addr, bool is_written = false) const; +    [[nodiscard]] Binding StorageBufferBinding(GPUVAddr ssbo_addr, u32 cbuf_index, +                                               bool is_written = false) const;      [[nodiscard]] TextureBufferBinding GetTextureBufferBinding(GPUVAddr gpu_addr, u32 size,                                                                 PixelFormat format); @@ -802,7 +803,7 @@ void BufferCache<P>::BindGraphicsStorageBuffer(size_t stage, size_t ssbo_index,      const auto& cbufs = maxwell3d->state.shader_stages[stage];      const GPUVAddr ssbo_addr = cbufs.const_buffers[cbuf_index].address + cbuf_offset; -    storage_buffers[stage][ssbo_index] = StorageBufferBinding(ssbo_addr, is_written); +    storage_buffers[stage][ssbo_index] = StorageBufferBinding(ssbo_addr, cbuf_index, is_written);  }  template <class P> @@ -842,7 +843,7 @@ void BufferCache<P>::BindComputeStorageBuffer(size_t ssbo_index, u32 cbuf_index,      const auto& cbufs = launch_desc.const_buffer_config;      const GPUVAddr ssbo_addr = cbufs[cbuf_index].Address() + cbuf_offset; -    compute_storage_buffers[ssbo_index] = StorageBufferBinding(ssbo_addr, is_written); +    compute_storage_buffers[ssbo_index] = StorageBufferBinding(ssbo_addr, cbuf_index, is_written);  }  template <class P> @@ -1988,11 +1989,26 @@ void BufferCache<P>::NotifyBufferDeletion() {  template <class P>  typename BufferCache<P>::Binding BufferCache<P>::StorageBufferBinding(GPUVAddr ssbo_addr, +                                                                      u32 cbuf_index,                                                                        bool is_written) const {      const GPUVAddr gpu_addr = gpu_memory->Read<u64>(ssbo_addr); -    const u32 size = gpu_memory->Read<u32>(ssbo_addr + 8); +    const auto size = [&]() { +        const bool is_nvn_cbuf = cbuf_index == 0; +        // The NVN driver buffer (index 0) is known to pack the SSBO address followed by its size. +        if (is_nvn_cbuf) { +            return gpu_memory->Read<u32>(ssbo_addr + 8); +        } +        // Other titles (notably Doom Eternal) may use STG/LDG on buffer addresses in custom defined +        // cbufs, which do not store the sizes adjacent to the addresses, so use the fully +        // mapped buffer size for now. +        const u32 memory_layout_size = static_cast<u32>(gpu_memory->GetMemoryLayoutSize(gpu_addr)); +        LOG_INFO(HW_GPU, "Binding storage buffer for cbuf index {}, MemoryLayoutSize 0x{:X}", +                 cbuf_index, memory_layout_size); +        return memory_layout_size; +    }();      const std::optional<VAddr> cpu_addr = gpu_memory->GpuToCpuAddress(gpu_addr);      if (!cpu_addr || size == 0) { +        LOG_WARNING(HW_GPU, "Failed to find storage buffer for cbuf index {}", cbuf_index);          return NULL_BINDING;      }      const VAddr cpu_end = Common::AlignUp(*cpu_addr + size, Core::Memory::YUZU_PAGESIZE); diff --git a/src/video_core/gpu_thread.cpp b/src/video_core/gpu_thread.cpp index 9c103c0d4..050b11874 100644 --- a/src/video_core/gpu_thread.cpp +++ b/src/video_core/gpu_thread.cpp @@ -25,7 +25,7 @@ static void RunThread(std::stop_token stop_token, Core::System& system,      SCOPE_EXIT({ MicroProfileOnThreadExit(); });      Common::SetCurrentThreadName(name.c_str()); -    Common::SetCurrentThreadPriority(Common::ThreadPriority::High); +    Common::SetCurrentThreadPriority(Common::ThreadPriority::Critical);      system.RegisterHostThread();      auto current_context = context.Acquire(); diff --git a/src/video_core/renderer_opengl/gl_fence_manager.cpp b/src/video_core/renderer_opengl/gl_fence_manager.cpp index 91463f854..5326172af 100644 --- a/src/video_core/renderer_opengl/gl_fence_manager.cpp +++ b/src/video_core/renderer_opengl/gl_fence_manager.cpp @@ -27,9 +27,7 @@ bool GLInnerFence::IsSignaled() const {          return true;      }      ASSERT(sync_object.handle != 0); -    GLint sync_status; -    glGetSynciv(sync_object.handle, GL_SYNC_STATUS, 1, nullptr, &sync_status); -    return sync_status == GL_SIGNALED; +    return sync_object.IsSignaled();  }  void GLInnerFence::Wait() { diff --git a/src/video_core/renderer_opengl/gl_graphics_pipeline.cpp b/src/video_core/renderer_opengl/gl_graphics_pipeline.cpp index 29491e762..89000d6e0 100644 --- a/src/video_core/renderer_opengl/gl_graphics_pipeline.cpp +++ b/src/video_core/renderer_opengl/gl_graphics_pipeline.cpp @@ -621,10 +621,7 @@ bool GraphicsPipeline::IsBuilt() noexcept {      if (built_fence.handle == 0) {          return false;      } -    // Timeout of zero means this is non-blocking -    const auto sync_status = glClientWaitSync(built_fence.handle, 0, 0); -    ASSERT(sync_status != GL_WAIT_FAILED); -    is_built = sync_status != GL_TIMEOUT_EXPIRED; +    is_built = built_fence.IsSignaled();      return is_built;  } diff --git a/src/video_core/renderer_opengl/gl_resource_manager.cpp b/src/video_core/renderer_opengl/gl_resource_manager.cpp index 3a664fdec..eae8fd110 100644 --- a/src/video_core/renderer_opengl/gl_resource_manager.cpp +++ b/src/video_core/renderer_opengl/gl_resource_manager.cpp @@ -3,6 +3,7 @@  #include <string_view>  #include <glad/glad.h> +#include "common/assert.h"  #include "common/microprofile.h"  #include "video_core/renderer_opengl/gl_resource_manager.h"  #include "video_core/renderer_opengl/gl_shader_util.h" @@ -158,6 +159,15 @@ void OGLSync::Release() {      handle = 0;  } +bool OGLSync::IsSignaled() const noexcept { +    // At least on Nvidia, glClientWaitSync with a timeout of 0 +    // is faster than glGetSynciv of GL_SYNC_STATUS. +    // Timeout of 0 means this check is non-blocking. +    const auto sync_status = glClientWaitSync(handle, 0, 0); +    ASSERT(sync_status != GL_WAIT_FAILED); +    return sync_status != GL_TIMEOUT_EXPIRED; +} +  void OGLFramebuffer::Create() {      if (handle != 0)          return; diff --git a/src/video_core/renderer_opengl/gl_resource_manager.h b/src/video_core/renderer_opengl/gl_resource_manager.h index bc05ba4bd..77362acd2 100644 --- a/src/video_core/renderer_opengl/gl_resource_manager.h +++ b/src/video_core/renderer_opengl/gl_resource_manager.h @@ -263,6 +263,9 @@ public:      /// Deletes the internal OpenGL resource      void Release(); +    /// Checks if the sync has been signaled +    bool IsSignaled() const noexcept; +      GLsync handle = 0;  }; diff --git a/src/video_core/renderer_opengl/gl_texture_cache.cpp b/src/video_core/renderer_opengl/gl_texture_cache.cpp index b047e7b3d..9b99125e5 100644 --- a/src/video_core/renderer_opengl/gl_texture_cache.cpp +++ b/src/video_core/renderer_opengl/gl_texture_cache.cpp @@ -112,13 +112,17 @@ GLenum ImageTarget(Shader::TextureType type, int num_samples = 1) {      return GL_NONE;  } -GLenum TextureMode(PixelFormat format, bool is_first) { +GLenum TextureMode(PixelFormat format, std::array<SwizzleSource, 4> swizzle) { +    bool any_r = +        std::ranges::any_of(swizzle, [](SwizzleSource s) { return s == SwizzleSource::R; });      switch (format) {      case PixelFormat::D24_UNORM_S8_UINT:      case PixelFormat::D32_FLOAT_S8_UINT: -        return is_first ? GL_DEPTH_COMPONENT : GL_STENCIL_INDEX; +        // R = depth, G = stencil +        return any_r ? GL_DEPTH_COMPONENT : GL_STENCIL_INDEX;      case PixelFormat::S8_UINT_D24_UNORM: -        return is_first ? GL_STENCIL_INDEX : GL_DEPTH_COMPONENT; +        // R = stencil, G = depth +        return any_r ? GL_STENCIL_INDEX : GL_DEPTH_COMPONENT;      default:          ASSERT(false);          return GL_DEPTH_COMPONENT; @@ -208,8 +212,7 @@ void ApplySwizzle(GLuint handle, PixelFormat format, std::array<SwizzleSource, 4      case PixelFormat::D32_FLOAT_S8_UINT:      case PixelFormat::S8_UINT_D24_UNORM:          UNIMPLEMENTED_IF(swizzle[0] != SwizzleSource::R && swizzle[0] != SwizzleSource::G); -        glTextureParameteri(handle, GL_DEPTH_STENCIL_TEXTURE_MODE, -                            TextureMode(format, swizzle[0] == SwizzleSource::R)); +        glTextureParameteri(handle, GL_DEPTH_STENCIL_TEXTURE_MODE, TextureMode(format, swizzle));          std::ranges::transform(swizzle, swizzle.begin(), ConvertGreenRed);          break;      case PixelFormat::A5B5G5R1_UNORM: { @@ -714,9 +717,7 @@ std::optional<size_t> TextureCacheRuntime::StagingBuffers::FindBuffer(size_t req              continue;          }          if (syncs[index].handle != 0) { -            GLint status; -            glGetSynciv(syncs[index].handle, GL_SYNC_STATUS, 1, nullptr, &status); -            if (status != GL_SIGNALED) { +            if (!syncs[index].IsSignaled()) {                  continue;              }              syncs[index].Release(); diff --git a/src/video_core/renderer_vulkan/vk_buffer_cache.cpp b/src/video_core/renderer_vulkan/vk_buffer_cache.cpp index b0153a502..9cbcb3c8f 100644 --- a/src/video_core/renderer_vulkan/vk_buffer_cache.cpp +++ b/src/video_core/renderer_vulkan/vk_buffer_cache.cpp @@ -238,7 +238,7 @@ private:          return indices;      } -    void MakeAndUpdateIndices(u8* staging_data, size_t quad_size, u32 quad, u32 first) { +    void MakeAndUpdateIndices(u8* staging_data, size_t quad_size, u32 quad, u32 first) override {          switch (index_type) {          case VK_INDEX_TYPE_UINT8_EXT:              std::memcpy(staging_data, MakeIndices<u8>(quad, first).data(), quad_size); @@ -278,7 +278,7 @@ private:          return indices;      } -    void MakeAndUpdateIndices(u8* staging_data, size_t quad_size, u32 quad, u32 first) { +    void MakeAndUpdateIndices(u8* staging_data, size_t quad_size, u32 quad, u32 first) override {          switch (index_type) {          case VK_INDEX_TYPE_UINT8_EXT:              std::memcpy(staging_data, MakeIndices<u8>(quad, first).data(), quad_size); diff --git a/src/video_core/renderer_vulkan/vk_rasterizer.cpp b/src/video_core/renderer_vulkan/vk_rasterizer.cpp index f085d53a1..25965b684 100644 --- a/src/video_core/renderer_vulkan/vk_rasterizer.cpp +++ b/src/video_core/renderer_vulkan/vk_rasterizer.cpp @@ -1294,7 +1294,7 @@ void RasterizerVulkan::UpdateDepthBoundsTestEnable(Tegra::Engines::Maxwell3D::Re          LOG_WARNING(Render_Vulkan, "Depth bounds is enabled but not supported");          enabled = false;      } -    scheduler.Record([enable = regs.depth_bounds_enable](vk::CommandBuffer cmdbuf) { +    scheduler.Record([enable = enabled](vk::CommandBuffer cmdbuf) {          cmdbuf.SetDepthBoundsTestEnableEXT(enable);      });  } diff --git a/src/video_core/renderer_vulkan/vk_texture_cache.cpp b/src/video_core/renderer_vulkan/vk_texture_cache.cpp index 8a204f93f..e013d1c60 100644 --- a/src/video_core/renderer_vulkan/vk_texture_cache.cpp +++ b/src/video_core/renderer_vulkan/vk_texture_cache.cpp @@ -189,13 +189,16 @@ constexpr VkBorderColor ConvertBorderColor(const std::array<float, 4>& color) {      if (info.IsRenderTarget()) {          return ImageAspectMask(info.format);      } -    const bool is_first = info.Swizzle()[0] == SwizzleSource::R; +    bool any_r = +        std::ranges::any_of(info.Swizzle(), [](SwizzleSource s) { return s == SwizzleSource::R; });      switch (info.format) {      case PixelFormat::D24_UNORM_S8_UINT:      case PixelFormat::D32_FLOAT_S8_UINT: -        return is_first ? VK_IMAGE_ASPECT_DEPTH_BIT : VK_IMAGE_ASPECT_STENCIL_BIT; +        // R = depth, G = stencil +        return any_r ? VK_IMAGE_ASPECT_DEPTH_BIT : VK_IMAGE_ASPECT_STENCIL_BIT;      case PixelFormat::S8_UINT_D24_UNORM: -        return is_first ? VK_IMAGE_ASPECT_STENCIL_BIT : VK_IMAGE_ASPECT_DEPTH_BIT; +        // R = stencil, G = depth +        return any_r ? VK_IMAGE_ASPECT_STENCIL_BIT : VK_IMAGE_ASPECT_DEPTH_BIT;      case PixelFormat::D16_UNORM:      case PixelFormat::D32_FLOAT:          return VK_IMAGE_ASPECT_DEPTH_BIT; @@ -1769,7 +1772,7 @@ Sampler::Sampler(TextureCacheRuntime& runtime, const Tegra::Texture::TSCEntry& t          .minLod = tsc.mipmap_filter == TextureMipmapFilter::None ? 0.0f : tsc.MinLod(),          .maxLod = tsc.mipmap_filter == TextureMipmapFilter::None ? 0.25f : tsc.MaxLod(),          .borderColor = -            arbitrary_borders ? VK_BORDER_COLOR_INT_CUSTOM_EXT : ConvertBorderColor(color), +            arbitrary_borders ? VK_BORDER_COLOR_FLOAT_CUSTOM_EXT : ConvertBorderColor(color),          .unnormalizedCoordinates = VK_FALSE,      });  } | 
