diff options
64 files changed, 367 insertions, 366 deletions
| diff --git a/.travis/macos/deps.sh b/.travis/macos/deps.sh index 1a547c060..faeafa216 100755 --- a/.travis/macos/deps.sh +++ b/.travis/macos/deps.sh @@ -1,5 +1,6 @@  #!/bin/sh -ex  brew update -brew install dylibbundler p7zip qt5 sdl2 ccache +brew install p7zip qt5 sdl2 ccache  brew outdated cmake || brew upgrade cmake +pip3 install macpack diff --git a/.travis/macos/upload.sh b/.travis/macos/upload.sh index 9ba95086b..66e3455ff 100755 --- a/.travis/macos/upload.sh +++ b/.travis/macos/upload.sh @@ -11,92 +11,19 @@ mkdir "$REV_NAME"  cp build/bin/yuzu-cmd "$REV_NAME"  cp -r build/bin/yuzu.app "$REV_NAME" -# move qt libs into app bundle for deployment -$(brew --prefix)/opt/qt5/bin/macdeployqt "${REV_NAME}/yuzu.app" +# move libs into folder for deployment +macpack "${REV_NAME}/yuzu.app/Contents/MacOS/yuzu" -d "../Frameworks" +# move qt frameworks into app bundle for deployment +$(brew --prefix)/opt/qt5/bin/macdeployqt "${REV_NAME}/yuzu.app" -executable="${REV_NAME}/yuzu.app/Contents/MacOS/yuzu" -# move SDL2 libs into folder for deployment -dylibbundler -b -x "${REV_NAME}/yuzu-cmd" -cd -d "${REV_NAME}/libs" -p "@executable_path/libs/" - -# Make the changes to make the yuzu app standalone (i.e. not dependent on the current brew installation). -# To do this, the absolute references to each and every QT framework must be re-written to point to the local frameworks -# (in the Contents/Frameworks folder). -# The "install_name_tool" is used to do so. - -# Coreutils is a hack to coerce Homebrew to point to the absolute Cellar path (symlink dereferenced). i.e: -# ls -l /usr/local/opt/qt5:: /usr/local/opt/qt5 -> ../Cellar/qt5/5.6.1-1 -# grealpath ../Cellar/qt5/5.6.1-1:: /usr/local/Cellar/qt5/5.6.1-1 -brew install coreutils || brew upgrade coreutils || true - -REV_NAME_ALT=$REV_NAME/ -# grealpath is located in coreutils, there is no "realpath" for OS X :( -QT_BREWS_PATH=$(grealpath "$(brew --prefix qt5)") -BREW_PATH=$(brew --prefix) -QT_VERSION_NUM=5 - -$BREW_PATH/opt/qt5/bin/macdeployqt "${REV_NAME_ALT}yuzu.app" \ -    -executable="${REV_NAME_ALT}yuzu.app/Contents/MacOS/yuzu" - -# These are the files that macdeployqt packed into Contents/Frameworks/ - we don't want those, so we replace them. -declare -a macos_libs=("QtCore" "QtWidgets" "QtGui" "QtOpenGL" "QtPrintSupport") - -for macos_lib in "${macos_libs[@]}" -do -    SC_FRAMEWORK_PART=$macos_lib.framework/Versions/$QT_VERSION_NUM/$macos_lib -    # Replace macdeployqt versions of the Frameworks with our own (from /usr/local/opt/qt5/lib/) -    cp "$BREW_PATH/opt/qt5/lib/$SC_FRAMEWORK_PART" "${REV_NAME_ALT}yuzu.app/Contents/Frameworks/$SC_FRAMEWORK_PART" - -    # Replace references within the embedded Framework files with "internal" versions. -    for macos_lib2 in "${macos_libs[@]}" -    do -        # Since brew references both the non-symlinked and symlink paths of QT5, it needs to be duplicated. -        # /usr/local/Cellar/qt5/5.6.1-1/lib and /usr/local/opt/qt5/lib both resolve to the same files. -        # So the two lines below are effectively duplicates when resolved as a path, but as strings, they aren't. -        RM_FRAMEWORK_PART=$macos_lib2.framework/Versions/$QT_VERSION_NUM/$macos_lib2 -        install_name_tool -change \ -            $QT_BREWS_PATH/lib/$RM_FRAMEWORK_PART \ -            @executable_path/../Frameworks/$RM_FRAMEWORK_PART \ -            "${REV_NAME_ALT}yuzu.app/Contents/Frameworks/$SC_FRAMEWORK_PART" -        install_name_tool -change \ -            "$BREW_PATH/opt/qt5/lib/$RM_FRAMEWORK_PART" \ -            @executable_path/../Frameworks/$RM_FRAMEWORK_PART \ -            "${REV_NAME_ALT}yuzu.app/Contents/Frameworks/$SC_FRAMEWORK_PART" -    done -done - -# Handles `This application failed to start because it could not find or load the Qt platform plugin "cocoa"` -# Which manifests itself as: -# "Exception Type: EXC_CRASH (SIGABRT) | Exception Codes: 0x0000000000000000, 0x0000000000000000 | Exception Note: EXC_CORPSE_NOTIFY" -# There may be more dylibs needed to be fixed... -declare -a macos_plugins=("Plugins/platforms/libqcocoa.dylib") - -for macos_lib in "${macos_plugins[@]}" -do -    install_name_tool -id @executable_path/../$macos_lib "${REV_NAME_ALT}yuzu.app/Contents/$macos_lib" -    for macos_lib2 in "${macos_libs[@]}" -    do -        RM_FRAMEWORK_PART=$macos_lib2.framework/Versions/$QT_VERSION_NUM/$macos_lib2 -        install_name_tool -change \ -            $QT_BREWS_PATH/lib/$RM_FRAMEWORK_PART \ -            @executable_path/../Frameworks/$RM_FRAMEWORK_PART \ -            "${REV_NAME_ALT}yuzu.app/Contents/$macos_lib" -        install_name_tool -change \ -            "$BREW_PATH/opt/qt5/lib/$RM_FRAMEWORK_PART" \ -            @executable_path/../Frameworks/$RM_FRAMEWORK_PART \ -            "${REV_NAME_ALT}yuzu.app/Contents/$macos_lib" -    done -done - -for macos_lib in "${macos_libs[@]}" -do -    # Debugging info for Travis-CI -    otool -L "${REV_NAME_ALT}yuzu.app/Contents/Frameworks/$macos_lib.framework/Versions/$QT_VERSION_NUM/$macos_lib" -done +# move libs into folder for deployment +macpack "${REV_NAME}/yuzu-cmd" -d "libs"  # Make the yuzu.app application launch a debugging terminal.  # Store away the actual binary -mv ${REV_NAME_ALT}yuzu.app/Contents/MacOS/yuzu ${REV_NAME_ALT}yuzu.app/Contents/MacOS/yuzu-bin +mv ${REV_NAME}/yuzu.app/Contents/MacOS/yuzu ${REV_NAME}/yuzu.app/Contents/MacOS/yuzu-bin -cat > ${REV_NAME_ALT}yuzu.app/Contents/MacOS/yuzu <<EOL +cat > ${REV_NAME}/yuzu.app/Contents/MacOS/yuzu <<EOL  #!/usr/bin/env bash  cd "\`dirname "\$0"\`"  chmod +x yuzu-bin @@ -105,6 +32,9 @@ EOL  # Content that will serve as the launching script for yuzu (within the .app folder)  # Make the launching script executable -chmod +x ${REV_NAME_ALT}yuzu.app/Contents/MacOS/yuzu +chmod +x ${REV_NAME}/yuzu.app/Contents/MacOS/yuzu + +# Verify loader instructions +find "$REV_NAME" -exec otool -L {} \;  . .travis/common/post-upload.sh diff --git a/src/common/detached_tasks.cpp b/src/common/detached_tasks.cpp index a347d9e02..f268d6021 100644 --- a/src/common/detached_tasks.cpp +++ b/src/common/detached_tasks.cpp @@ -16,22 +16,22 @@ DetachedTasks::DetachedTasks() {  }  void DetachedTasks::WaitForAllTasks() { -    std::unique_lock<std::mutex> lock(mutex); +    std::unique_lock lock{mutex};      cv.wait(lock, [this]() { return count == 0; });  }  DetachedTasks::~DetachedTasks() { -    std::unique_lock<std::mutex> lock(mutex); +    std::unique_lock lock{mutex};      ASSERT(count == 0);      instance = nullptr;  }  void DetachedTasks::AddTask(std::function<void()> task) { -    std::unique_lock<std::mutex> lock(instance->mutex); +    std::unique_lock lock{instance->mutex};      ++instance->count;      std::thread([task{std::move(task)}]() {          task(); -        std::unique_lock<std::mutex> lock(instance->mutex); +        std::unique_lock lock{instance->mutex};          --instance->count;          std::notify_all_at_thread_exit(instance->cv, std::move(lock));      }) diff --git a/src/common/logging/backend.cpp b/src/common/logging/backend.cpp index 4462ff3fb..a03179520 100644 --- a/src/common/logging/backend.cpp +++ b/src/common/logging/backend.cpp @@ -46,12 +46,12 @@ public:      }      void AddBackend(std::unique_ptr<Backend> backend) { -        std::lock_guard<std::mutex> lock(writing_mutex); +        std::lock_guard lock{writing_mutex};          backends.push_back(std::move(backend));      }      void RemoveBackend(std::string_view backend_name) { -        std::lock_guard<std::mutex> lock(writing_mutex); +        std::lock_guard lock{writing_mutex};          const auto it =              std::remove_if(backends.begin(), backends.end(),                             [&backend_name](const auto& i) { return backend_name == i->GetName(); }); @@ -80,7 +80,7 @@ private:          backend_thread = std::thread([&] {              Entry entry;              auto write_logs = [&](Entry& e) { -                std::lock_guard<std::mutex> lock(writing_mutex); +                std::lock_guard lock{writing_mutex};                  for (const auto& backend : backends) {                      backend->Write(e);                  } diff --git a/src/common/thread.cpp b/src/common/thread.cpp index 5144c0d9f..fe7a420cc 100644 --- a/src/common/thread.cpp +++ b/src/common/thread.cpp @@ -27,18 +27,6 @@ namespace Common {  #ifdef _MSC_VER -void SetThreadAffinity(std::thread::native_handle_type thread, u32 mask) { -    SetThreadAffinityMask(thread, mask); -} - -void SetCurrentThreadAffinity(u32 mask) { -    SetThreadAffinityMask(GetCurrentThread(), mask); -} - -void SwitchCurrentThread() { -    SwitchToThread(); -} -  // Sets the debugger-visible name of the current thread.  // Uses undocumented (actually, it is now documented) trick.  // http://msdn.microsoft.com/library/default.asp?url=/library/en-us/vsdebug/html/vxtsksettingthreadname.asp @@ -70,31 +58,6 @@ void SetCurrentThreadName(const char* name) {  #else // !MSVC_VER, so must be POSIX threads -void SetThreadAffinity(std::thread::native_handle_type thread, u32 mask) { -#ifdef __APPLE__ -    thread_policy_set(pthread_mach_thread_np(thread), THREAD_AFFINITY_POLICY, (integer_t*)&mask, 1); -#elif (defined __linux__ || defined __FreeBSD__) && !(defined ANDROID) -    cpu_set_t cpu_set; -    CPU_ZERO(&cpu_set); - -    for (int i = 0; i != sizeof(mask) * 8; ++i) -        if ((mask >> i) & 1) -            CPU_SET(i, &cpu_set); - -    pthread_setaffinity_np(thread, sizeof(cpu_set), &cpu_set); -#endif -} - -void SetCurrentThreadAffinity(u32 mask) { -    SetThreadAffinity(pthread_self(), mask); -} - -#ifndef _WIN32 -void SwitchCurrentThread() { -    usleep(1000 * 1); -} -#endif -  // MinGW with the POSIX threading model does not support pthread_setname_np  #if !defined(_WIN32) || defined(_MSC_VER)  void SetCurrentThreadName(const char* name) { diff --git a/src/common/thread.h b/src/common/thread.h index 2cf74452d..0cfd98be6 100644 --- a/src/common/thread.h +++ b/src/common/thread.h @@ -9,14 +9,13 @@  #include <cstddef>  #include <mutex>  #include <thread> -#include "common/common_types.h"  namespace Common {  class Event {  public:      void Set() { -        std::lock_guard<std::mutex> lk(mutex); +        std::lock_guard lk{mutex};          if (!is_set) {              is_set = true;              condvar.notify_one(); @@ -24,14 +23,14 @@ public:      }      void Wait() { -        std::unique_lock<std::mutex> lk(mutex); +        std::unique_lock lk{mutex};          condvar.wait(lk, [&] { return is_set; });          is_set = false;      }      template <class Clock, class Duration>      bool WaitUntil(const std::chrono::time_point<Clock, Duration>& time) { -        std::unique_lock<std::mutex> lk(mutex); +        std::unique_lock lk{mutex};          if (!condvar.wait_until(lk, time, [this] { return is_set; }))              return false;          is_set = false; @@ -39,7 +38,7 @@ public:      }      void Reset() { -        std::unique_lock<std::mutex> lk(mutex); +        std::unique_lock lk{mutex};          // no other action required, since wait loops on the predicate and any lingering signal will          // get cleared on the first iteration          is_set = false; @@ -57,7 +56,7 @@ public:      /// Blocks until all "count" threads have called Sync()      void Sync() { -        std::unique_lock<std::mutex> lk(mutex); +        std::unique_lock lk{mutex};          const std::size_t current_generation = generation;          if (++waiting == count) { @@ -78,9 +77,6 @@ private:      std::size_t generation = 0; // Incremented once each time the barrier is used  }; -void SetThreadAffinity(std::thread::native_handle_type thread, u32 mask); -void SetCurrentThreadAffinity(u32 mask); -void SwitchCurrentThread(); // On Linux, this is equal to sleep 1ms  void SetCurrentThreadName(const char* name);  } // namespace Common diff --git a/src/common/threadsafe_queue.h b/src/common/threadsafe_queue.h index 821e8536a..e714ba5b3 100644 --- a/src/common/threadsafe_queue.h +++ b/src/common/threadsafe_queue.h @@ -78,7 +78,7 @@ public:      T PopWait() {          if (Empty()) { -            std::unique_lock<std::mutex> lock(cv_mutex); +            std::unique_lock lock{cv_mutex};              cv.wait(lock, [this]() { return !Empty(); });          }          T t; @@ -137,7 +137,7 @@ public:      template <typename Arg>      void Push(Arg&& t) { -        std::lock_guard<std::mutex> lock(write_lock); +        std::lock_guard lock{write_lock};          spsc_queue.Push(t);      } diff --git a/src/core/core_cpu.cpp b/src/core/core_cpu.cpp index 1eefed6d0..e75741db0 100644 --- a/src/core/core_cpu.cpp +++ b/src/core/core_cpu.cpp @@ -22,7 +22,7 @@  namespace Core {  void CpuBarrier::NotifyEnd() { -    std::unique_lock<std::mutex> lock(mutex); +    std::unique_lock lock{mutex};      end = true;      condition.notify_all();  } @@ -34,7 +34,7 @@ bool CpuBarrier::Rendezvous() {      }      if (!end) { -        std::unique_lock<std::mutex> lock(mutex); +        std::unique_lock lock{mutex};          --cores_waiting;          if (!cores_waiting) { @@ -131,7 +131,7 @@ void Cpu::Reschedule() {      reschedule_pending = false;      // Lock the global kernel mutex when we manipulate the HLE state -    std::lock_guard<std::recursive_mutex> lock(HLE::g_hle_lock); +    std::lock_guard lock{HLE::g_hle_lock};      scheduler->Reschedule();  } diff --git a/src/core/frontend/emu_window.cpp b/src/core/frontend/emu_window.cpp index e29afd630..1320bbe77 100644 --- a/src/core/frontend/emu_window.cpp +++ b/src/core/frontend/emu_window.cpp @@ -30,7 +30,7 @@ private:          explicit Device(std::weak_ptr<TouchState>&& touch_state) : touch_state(touch_state) {}          std::tuple<float, float, bool> GetStatus() const override {              if (auto state = touch_state.lock()) { -                std::lock_guard<std::mutex> guard(state->mutex); +                std::lock_guard guard{state->mutex};                  return std::make_tuple(state->touch_x, state->touch_y, state->touch_pressed);              }              return std::make_tuple(0.0f, 0.0f, false); @@ -81,7 +81,7 @@ void EmuWindow::TouchPressed(unsigned framebuffer_x, unsigned framebuffer_y) {      if (!IsWithinTouchscreen(framebuffer_layout, framebuffer_x, framebuffer_y))          return; -    std::lock_guard<std::mutex> guard(touch_state->mutex); +    std::lock_guard guard{touch_state->mutex};      touch_state->touch_x = static_cast<float>(framebuffer_x - framebuffer_layout.screen.left) /                             (framebuffer_layout.screen.right - framebuffer_layout.screen.left);      touch_state->touch_y = static_cast<float>(framebuffer_y - framebuffer_layout.screen.top) / @@ -91,7 +91,7 @@ void EmuWindow::TouchPressed(unsigned framebuffer_x, unsigned framebuffer_y) {  }  void EmuWindow::TouchReleased() { -    std::lock_guard<std::mutex> guard(touch_state->mutex); +    std::lock_guard guard{touch_state->mutex};      touch_state->touch_pressed = false;      touch_state->touch_x = 0;      touch_state->touch_y = 0; diff --git a/src/core/hle/kernel/code_set.h b/src/core/hle/kernel/code_set.h index 834fd23d2..879957dcb 100644 --- a/src/core/hle/kernel/code_set.h +++ b/src/core/hle/kernel/code_set.h @@ -5,7 +5,6 @@  #pragma once  #include <cstddef> -#include <memory>  #include <vector>  #include "common/common_types.h" @@ -78,7 +77,7 @@ struct CodeSet final {      }      /// The overall data that backs this code set. -    std::shared_ptr<std::vector<u8>> memory; +    std::vector<u8> memory;      /// The segments that comprise this code set.      std::array<Segment, 3> segments; diff --git a/src/core/hle/kernel/kernel.cpp b/src/core/hle/kernel/kernel.cpp index 3b73be67b..3f14bfa86 100644 --- a/src/core/hle/kernel/kernel.cpp +++ b/src/core/hle/kernel/kernel.cpp @@ -34,7 +34,7 @@ static void ThreadWakeupCallback(u64 thread_handle, [[maybe_unused]] s64 cycles_      const auto& system = Core::System::GetInstance();      // Lock the global kernel mutex when we enter the kernel HLE. -    std::lock_guard<std::recursive_mutex> lock(HLE::g_hle_lock); +    std::lock_guard lock{HLE::g_hle_lock};      SharedPtr<Thread> thread =          system.Kernel().RetrieveThreadFromWakeupCallbackHandleTable(proper_handle); @@ -115,7 +115,7 @@ struct KernelCore::Impl {      // Creates the default system resource limit      void InitializeSystemResourceLimit(KernelCore& kernel) { -        system_resource_limit = ResourceLimit::Create(kernel, "System"); +        system_resource_limit = ResourceLimit::Create(kernel);          // If setting the default system values fails, then something seriously wrong has occurred.          ASSERT(system_resource_limit->SetLimitValue(ResourceType::PhysicalMemory, 0x200000000) @@ -191,6 +191,10 @@ const Process* KernelCore::CurrentProcess() const {      return impl->current_process;  } +const std::vector<SharedPtr<Process>>& KernelCore::GetProcessList() const { +    return impl->process_list; +} +  void KernelCore::AddNamedPort(std::string name, SharedPtr<ClientPort> port) {      impl->named_ports.emplace(std::move(name), std::move(port));  } diff --git a/src/core/hle/kernel/kernel.h b/src/core/hle/kernel/kernel.h index 03ea5b659..6b8738599 100644 --- a/src/core/hle/kernel/kernel.h +++ b/src/core/hle/kernel/kernel.h @@ -72,6 +72,9 @@ public:      /// Retrieves a const pointer to the current process.      const Process* CurrentProcess() const; +    /// Retrieves the list of processes. +    const std::vector<SharedPtr<Process>>& GetProcessList() const; +      /// Adds a port to the named port table      void AddNamedPort(std::string name, SharedPtr<ClientPort> port); diff --git a/src/core/hle/kernel/process.cpp b/src/core/hle/kernel/process.cpp index 0d782e4ba..041267318 100644 --- a/src/core/hle/kernel/process.cpp +++ b/src/core/hle/kernel/process.cpp @@ -5,6 +5,7 @@  #include <algorithm>  #include <memory>  #include <random> +#include "common/alignment.h"  #include "common/assert.h"  #include "common/logging/log.h"  #include "core/core.h" @@ -75,6 +76,18 @@ SharedPtr<ResourceLimit> Process::GetResourceLimit() const {      return resource_limit;  } +u64 Process::GetTotalPhysicalMemoryUsed() const { +    return vm_manager.GetCurrentHeapSize() + main_thread_stack_size + code_memory_size; +} + +void Process::RegisterThread(const Thread* thread) { +    thread_list.push_back(thread); +} + +void Process::UnregisterThread(const Thread* thread) { +    thread_list.remove(thread); +} +  ResultCode Process::ClearSignalState() {      if (status == ProcessStatus::Exited) {          LOG_ERROR(Kernel, "called on a terminated process instance."); @@ -107,14 +120,17 @@ ResultCode Process::LoadFromMetadata(const FileSys::ProgramMetadata& metadata) {      return handle_table.SetSize(capabilities.GetHandleTableSize());  } -void Process::Run(VAddr entry_point, s32 main_thread_priority, u32 stack_size) { +void Process::Run(VAddr entry_point, s32 main_thread_priority, u64 stack_size) { +    // The kernel always ensures that the given stack size is page aligned. +    main_thread_stack_size = Common::AlignUp(stack_size, Memory::PAGE_SIZE); +      // Allocate and map the main thread stack      // TODO(bunnei): This is heap area that should be allocated by the kernel and not mapped as part      // of the user address space. +    const VAddr mapping_address = vm_manager.GetTLSIORegionEndAddress() - main_thread_stack_size;      vm_manager -        .MapMemoryBlock(vm_manager.GetTLSIORegionEndAddress() - stack_size, -                        std::make_shared<std::vector<u8>>(stack_size, 0), 0, stack_size, -                        MemoryState::Stack) +        .MapMemoryBlock(mapping_address, std::make_shared<std::vector<u8>>(main_thread_stack_size), +                        0, main_thread_stack_size, MemoryState::Stack)          .Unwrap();      vm_manager.LogLayout(); @@ -210,11 +226,13 @@ void Process::FreeTLSSlot(VAddr tls_address) {  }  void Process::LoadModule(CodeSet module_, VAddr base_addr) { +    const auto memory = std::make_shared<std::vector<u8>>(std::move(module_.memory)); +      const auto MapSegment = [&](const CodeSet::Segment& segment, VMAPermission permissions,                                  MemoryState memory_state) {          const auto vma = vm_manager -                             .MapMemoryBlock(segment.addr + base_addr, module_.memory, -                                             segment.offset, segment.size, memory_state) +                             .MapMemoryBlock(segment.addr + base_addr, memory, segment.offset, +                                             segment.size, memory_state)                               .Unwrap();          vm_manager.Reprotect(vma, permissions);      }; @@ -224,6 +242,8 @@ void Process::LoadModule(CodeSet module_, VAddr base_addr) {      MapSegment(module_.RODataSegment(), VMAPermission::Read, MemoryState::CodeData);      MapSegment(module_.DataSegment(), VMAPermission::ReadWrite, MemoryState::CodeData); +    code_memory_size += module_.memory.size(); +      // Clear instruction cache in CPU JIT      system.InvalidateCpuInstructionCaches();  } @@ -237,7 +257,7 @@ void Process::Acquire(Thread* thread) {      ASSERT_MSG(!ShouldWait(thread), "Object unavailable!");  } -bool Process::ShouldWait(Thread* thread) const { +bool Process::ShouldWait(const Thread* thread) const {      return !is_signaled;  } diff --git a/src/core/hle/kernel/process.h b/src/core/hle/kernel/process.h index a0217d3d8..f060f2a3b 100644 --- a/src/core/hle/kernel/process.h +++ b/src/core/hle/kernel/process.h @@ -7,6 +7,7 @@  #include <array>  #include <bitset>  #include <cstddef> +#include <list>  #include <string>  #include <vector>  #include <boost/container/static_vector.hpp> @@ -186,6 +187,22 @@ public:          return random_entropy.at(index);      } +    /// Retrieves the total physical memory used by this process in bytes. +    u64 GetTotalPhysicalMemoryUsed() const; + +    /// Gets the list of all threads created with this process as their owner. +    const std::list<const Thread*>& GetThreadList() const { +        return thread_list; +    } + +    /// Registers a thread as being created under this process, +    /// adding it to this process' thread list. +    void RegisterThread(const Thread* thread); + +    /// Unregisters a thread from this process, removing it +    /// from this process' thread list. +    void UnregisterThread(const Thread* thread); +      /// 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 @@ -210,7 +227,7 @@ public:      /**       * Applies address space changes and launches the process main thread.       */ -    void Run(VAddr entry_point, s32 main_thread_priority, u32 stack_size); +    void Run(VAddr entry_point, s32 main_thread_priority, u64 stack_size);      /**       * Prepares a process for termination by stopping all of its threads @@ -234,7 +251,7 @@ private:      ~Process() override;      /// Checks if the specified thread should wait until this process is available. -    bool ShouldWait(Thread* thread) const override; +    bool ShouldWait(const Thread* thread) const override;      /// Acquires/locks this process for the specified thread if it's available.      void Acquire(Thread* thread) override; @@ -247,6 +264,12 @@ private:      /// Memory manager for this process.      Kernel::VMManager vm_manager; +    /// Size of the main thread's stack in bytes. +    u64 main_thread_stack_size = 0; + +    /// Size of the loaded code memory in bytes. +    u64 code_memory_size = 0; +      /// Current status of the process      ProcessStatus status; @@ -299,6 +322,9 @@ private:      /// Random values for svcGetInfo RandomEntropy      std::array<u64, RANDOM_ENTROPY_SIZE> random_entropy; +    /// List of threads that are running with this process as their owner. +    std::list<const Thread*> thread_list; +      /// System context      Core::System& system; diff --git a/src/core/hle/kernel/readable_event.cpp b/src/core/hle/kernel/readable_event.cpp index 0e5083f70..c2b798a4e 100644 --- a/src/core/hle/kernel/readable_event.cpp +++ b/src/core/hle/kernel/readable_event.cpp @@ -14,7 +14,7 @@ namespace Kernel {  ReadableEvent::ReadableEvent(KernelCore& kernel) : WaitObject{kernel} {}  ReadableEvent::~ReadableEvent() = default; -bool ReadableEvent::ShouldWait(Thread* thread) const { +bool ReadableEvent::ShouldWait(const Thread* thread) const {      return !signaled;  } diff --git a/src/core/hle/kernel/readable_event.h b/src/core/hle/kernel/readable_event.h index 77a9c362c..2eb9dcbb7 100644 --- a/src/core/hle/kernel/readable_event.h +++ b/src/core/hle/kernel/readable_event.h @@ -36,7 +36,7 @@ public:          return HANDLE_TYPE;      } -    bool ShouldWait(Thread* thread) const override; +    bool ShouldWait(const Thread* thread) const override;      void Acquire(Thread* thread) override;      /// Unconditionally clears the readable event's state. diff --git a/src/core/hle/kernel/resource_limit.cpp b/src/core/hle/kernel/resource_limit.cpp index 2f9695005..173f69915 100644 --- a/src/core/hle/kernel/resource_limit.cpp +++ b/src/core/hle/kernel/resource_limit.cpp @@ -16,11 +16,8 @@ constexpr std::size_t ResourceTypeToIndex(ResourceType type) {  ResourceLimit::ResourceLimit(KernelCore& kernel) : Object{kernel} {}  ResourceLimit::~ResourceLimit() = default; -SharedPtr<ResourceLimit> ResourceLimit::Create(KernelCore& kernel, std::string name) { -    SharedPtr<ResourceLimit> resource_limit(new ResourceLimit(kernel)); - -    resource_limit->name = std::move(name); -    return resource_limit; +SharedPtr<ResourceLimit> ResourceLimit::Create(KernelCore& kernel) { +    return new ResourceLimit(kernel);  }  s64 ResourceLimit::GetCurrentResourceValue(ResourceType resource) const { diff --git a/src/core/hle/kernel/resource_limit.h b/src/core/hle/kernel/resource_limit.h index 59dc11c22..70e09858a 100644 --- a/src/core/hle/kernel/resource_limit.h +++ b/src/core/hle/kernel/resource_limit.h @@ -31,16 +31,14 @@ constexpr bool IsValidResourceType(ResourceType type) {  class ResourceLimit final : public Object {  public: -    /** -     * Creates a resource limit object. -     */ -    static SharedPtr<ResourceLimit> Create(KernelCore& kernel, std::string name = "Unknown"); +    /// Creates a resource limit object. +    static SharedPtr<ResourceLimit> Create(KernelCore& kernel);      std::string GetTypeName() const override {          return "ResourceLimit";      }      std::string GetName() const override { -        return name; +        return GetTypeName();      }      static const HandleType HANDLE_TYPE = HandleType::ResourceLimit; @@ -95,9 +93,6 @@ private:      ResourceArray limits{};      /// Current resource limit values.      ResourceArray values{}; - -    /// Name of resource limit object. -    std::string name;  };  } // namespace Kernel diff --git a/src/core/hle/kernel/scheduler.cpp b/src/core/hle/kernel/scheduler.cpp index 6d0f13ecf..ac501bf7f 100644 --- a/src/core/hle/kernel/scheduler.cpp +++ b/src/core/hle/kernel/scheduler.cpp @@ -29,7 +29,7 @@ Scheduler::~Scheduler() {  }  bool Scheduler::HaveReadyThreads() const { -    std::lock_guard<std::mutex> lock(scheduler_mutex); +    std::lock_guard lock{scheduler_mutex};      return !ready_queue.empty();  } @@ -132,7 +132,7 @@ void Scheduler::UpdateLastContextSwitchTime(Thread* thread, Process* process) {  }  void Scheduler::Reschedule() { -    std::lock_guard<std::mutex> lock(scheduler_mutex); +    std::lock_guard lock{scheduler_mutex};      Thread* cur = GetCurrentThread();      Thread* next = PopNextReadyThread(); @@ -148,35 +148,35 @@ void Scheduler::Reschedule() {      SwitchContext(next);  } -void Scheduler::AddThread(SharedPtr<Thread> thread, u32 priority) { -    std::lock_guard<std::mutex> lock(scheduler_mutex); +void Scheduler::AddThread(SharedPtr<Thread> thread) { +    std::lock_guard lock{scheduler_mutex};      thread_list.push_back(std::move(thread));  }  void Scheduler::RemoveThread(Thread* thread) { -    std::lock_guard<std::mutex> lock(scheduler_mutex); +    std::lock_guard lock{scheduler_mutex};      thread_list.erase(std::remove(thread_list.begin(), thread_list.end(), thread),                        thread_list.end());  }  void Scheduler::ScheduleThread(Thread* thread, u32 priority) { -    std::lock_guard<std::mutex> lock(scheduler_mutex); +    std::lock_guard lock{scheduler_mutex};      ASSERT(thread->GetStatus() == ThreadStatus::Ready);      ready_queue.add(thread, priority);  }  void Scheduler::UnscheduleThread(Thread* thread, u32 priority) { -    std::lock_guard<std::mutex> lock(scheduler_mutex); +    std::lock_guard lock{scheduler_mutex};      ASSERT(thread->GetStatus() == ThreadStatus::Ready);      ready_queue.remove(thread, priority);  }  void Scheduler::SetThreadPriority(Thread* thread, u32 priority) { -    std::lock_guard<std::mutex> lock(scheduler_mutex); +    std::lock_guard lock{scheduler_mutex};      if (thread->GetPriority() == priority) {          return;      } @@ -187,7 +187,7 @@ void Scheduler::SetThreadPriority(Thread* thread, u32 priority) {  }  Thread* Scheduler::GetNextSuggestedThread(u32 core, u32 maximum_priority) const { -    std::lock_guard<std::mutex> lock(scheduler_mutex); +    std::lock_guard lock{scheduler_mutex};      const u32 mask = 1U << core;      for (auto* thread : ready_queue) { diff --git a/src/core/hle/kernel/scheduler.h b/src/core/hle/kernel/scheduler.h index 44baeb713..b29bf7be8 100644 --- a/src/core/hle/kernel/scheduler.h +++ b/src/core/hle/kernel/scheduler.h @@ -38,7 +38,7 @@ public:      u64 GetLastContextSwitchTicks() const;      /// Adds a new thread to the scheduler -    void AddThread(SharedPtr<Thread> thread, u32 priority); +    void AddThread(SharedPtr<Thread> thread);      /// Removes a thread from the scheduler      void RemoveThread(Thread* thread); diff --git a/src/core/hle/kernel/server_port.cpp b/src/core/hle/kernel/server_port.cpp index 0e1515c89..708fdf9e1 100644 --- a/src/core/hle/kernel/server_port.cpp +++ b/src/core/hle/kernel/server_port.cpp @@ -30,7 +30,7 @@ void ServerPort::AppendPendingSession(SharedPtr<ServerSession> pending_session)      pending_sessions.push_back(std::move(pending_session));  } -bool ServerPort::ShouldWait(Thread* thread) const { +bool ServerPort::ShouldWait(const Thread* thread) const {      // If there are no pending sessions, we wait until a new one is added.      return pending_sessions.empty();  } diff --git a/src/core/hle/kernel/server_port.h b/src/core/hle/kernel/server_port.h index 9bc667cf2..76293cb8b 100644 --- a/src/core/hle/kernel/server_port.h +++ b/src/core/hle/kernel/server_port.h @@ -75,7 +75,7 @@ public:      /// waiting to be accepted by this port.      void AppendPendingSession(SharedPtr<ServerSession> pending_session); -    bool ShouldWait(Thread* thread) const override; +    bool ShouldWait(const Thread* thread) const override;      void Acquire(Thread* thread) override;  private: diff --git a/src/core/hle/kernel/server_session.cpp b/src/core/hle/kernel/server_session.cpp index 4d8a337a7..40cec143e 100644 --- a/src/core/hle/kernel/server_session.cpp +++ b/src/core/hle/kernel/server_session.cpp @@ -46,7 +46,7 @@ ResultVal<SharedPtr<ServerSession>> ServerSession::Create(KernelCore& kernel, st      return MakeResult(std::move(server_session));  } -bool ServerSession::ShouldWait(Thread* thread) const { +bool ServerSession::ShouldWait(const Thread* thread) const {      // Closed sessions should never wait, an error will be returned from svcReplyAndReceive.      if (parent->client == nullptr)          return false; diff --git a/src/core/hle/kernel/server_session.h b/src/core/hle/kernel/server_session.h index aea4ccfeb..79b84bade 100644 --- a/src/core/hle/kernel/server_session.h +++ b/src/core/hle/kernel/server_session.h @@ -82,7 +82,7 @@ public:       */      ResultCode HandleSyncRequest(SharedPtr<Thread> thread); -    bool ShouldWait(Thread* thread) const override; +    bool ShouldWait(const Thread* thread) const override;      void Acquire(Thread* thread) override; diff --git a/src/core/hle/kernel/svc.cpp b/src/core/hle/kernel/svc.cpp index fd98c0825..ab10db3df 100644 --- a/src/core/hle/kernel/svc.cpp +++ b/src/core/hle/kernel/svc.cpp @@ -709,7 +709,7 @@ static ResultCode GetInfo(u64* result, u64 info_id, u64 handle, u64 info_sub_id)          HeapRegionBaseAddr = 4,          HeapRegionSize = 5,          TotalMemoryUsage = 6, -        TotalHeapUsage = 7, +        TotalPhysicalMemoryUsed = 7,          IsCurrentProcessBeingDebugged = 8,          RegisterResourceLimit = 9,          IdleTickCount = 10, @@ -745,7 +745,7 @@ static ResultCode GetInfo(u64* result, u64 info_id, u64 handle, u64 info_sub_id)      case GetInfoType::NewMapRegionBaseAddr:      case GetInfoType::NewMapRegionSize:      case GetInfoType::TotalMemoryUsage: -    case GetInfoType::TotalHeapUsage: +    case GetInfoType::TotalPhysicalMemoryUsed:      case GetInfoType::IsVirtualAddressMemoryEnabled:      case GetInfoType::PersonalMmHeapUsage:      case GetInfoType::TitleId: @@ -805,8 +805,8 @@ static ResultCode GetInfo(u64* result, u64 info_id, u64 handle, u64 info_sub_id)              *result = process->VMManager().GetTotalMemoryUsage();              return RESULT_SUCCESS; -        case GetInfoType::TotalHeapUsage: -            *result = process->VMManager().GetCurrentHeapSize(); +        case GetInfoType::TotalPhysicalMemoryUsed: +            *result = process->GetTotalPhysicalMemoryUsed();              return RESULT_SUCCESS;          case GetInfoType::IsVirtualAddressMemoryEnabled: @@ -1983,6 +1983,83 @@ static ResultCode SetResourceLimitLimitValue(Handle resource_limit, u32 resource      return RESULT_SUCCESS;  } +static ResultCode GetProcessList(u32* out_num_processes, VAddr out_process_ids, +                                 u32 out_process_ids_size) { +    LOG_DEBUG(Kernel_SVC, "called. out_process_ids=0x{:016X}, out_process_ids_size={}", +              out_process_ids, out_process_ids_size); + +    // If the supplied size is negative or greater than INT32_MAX / sizeof(u64), bail. +    if ((out_process_ids_size & 0xF0000000) != 0) { +        LOG_ERROR(Kernel_SVC, +                  "Supplied size outside [0, 0x0FFFFFFF] range. out_process_ids_size={}", +                  out_process_ids_size); +        return ERR_OUT_OF_RANGE; +    } + +    const auto& kernel = Core::System::GetInstance().Kernel(); +    const auto& vm_manager = kernel.CurrentProcess()->VMManager(); +    const auto total_copy_size = out_process_ids_size * sizeof(u64); + +    if (out_process_ids_size > 0 && +        !vm_manager.IsWithinAddressSpace(out_process_ids, total_copy_size)) { +        LOG_ERROR(Kernel_SVC, "Address range outside address space. begin=0x{:016X}, end=0x{:016X}", +                  out_process_ids, out_process_ids + total_copy_size); +        return ERR_INVALID_ADDRESS_STATE; +    } + +    const auto& process_list = kernel.GetProcessList(); +    const auto num_processes = process_list.size(); +    const auto copy_amount = std::min(std::size_t{out_process_ids_size}, num_processes); + +    for (std::size_t i = 0; i < copy_amount; ++i) { +        Memory::Write64(out_process_ids, process_list[i]->GetProcessID()); +        out_process_ids += sizeof(u64); +    } + +    *out_num_processes = static_cast<u32>(num_processes); +    return RESULT_SUCCESS; +} + +ResultCode GetThreadList(u32* out_num_threads, VAddr out_thread_ids, u32 out_thread_ids_size, +                         Handle debug_handle) { +    // TODO: Handle this case when debug events are supported. +    UNIMPLEMENTED_IF(debug_handle != InvalidHandle); + +    LOG_DEBUG(Kernel_SVC, "called. out_thread_ids=0x{:016X}, out_thread_ids_size={}", +              out_thread_ids, out_thread_ids_size); + +    // If the size is negative or larger than INT32_MAX / sizeof(u64) +    if ((out_thread_ids_size & 0xF0000000) != 0) { +        LOG_ERROR(Kernel_SVC, "Supplied size outside [0, 0x0FFFFFFF] range. size={}", +                  out_thread_ids_size); +        return ERR_OUT_OF_RANGE; +    } + +    const auto* const current_process = Core::System::GetInstance().Kernel().CurrentProcess(); +    const auto& vm_manager = current_process->VMManager(); +    const auto total_copy_size = out_thread_ids_size * sizeof(u64); + +    if (out_thread_ids_size > 0 && +        !vm_manager.IsWithinAddressSpace(out_thread_ids, total_copy_size)) { +        LOG_ERROR(Kernel_SVC, "Address range outside address space. begin=0x{:016X}, end=0x{:016X}", +                  out_thread_ids, out_thread_ids + total_copy_size); +        return ERR_INVALID_ADDRESS_STATE; +    } + +    const auto& thread_list = current_process->GetThreadList(); +    const auto num_threads = thread_list.size(); +    const auto copy_amount = std::min(std::size_t{out_thread_ids_size}, num_threads); + +    auto list_iter = thread_list.cbegin(); +    for (std::size_t i = 0; i < copy_amount; ++i, ++list_iter) { +        Memory::Write64(out_thread_ids, (*list_iter)->GetThreadID()); +        out_thread_ids += sizeof(u64); +    } + +    *out_num_threads = static_cast<u32>(num_threads); +    return RESULT_SUCCESS; +} +  namespace {  struct FunctionDef {      using Func = void(); @@ -2095,8 +2172,8 @@ static const FunctionDef SVC_Table[] = {      {0x62, nullptr, "TerminateDebugProcess"},      {0x63, nullptr, "GetDebugEvent"},      {0x64, nullptr, "ContinueDebugEvent"}, -    {0x65, nullptr, "GetProcessList"}, -    {0x66, nullptr, "GetThreadList"}, +    {0x65, SvcWrap<GetProcessList>, "GetProcessList"}, +    {0x66, SvcWrap<GetThreadList>, "GetThreadList"},      {0x67, nullptr, "GetDebugThreadContext"},      {0x68, nullptr, "SetDebugThreadContext"},      {0x69, nullptr, "QueryDebugProcessMemory"}, @@ -2138,7 +2215,7 @@ void CallSVC(u32 immediate) {      MICROPROFILE_SCOPE(Kernel_SVC);      // Lock the global kernel mutex when we enter the kernel HLE. -    std::lock_guard<std::recursive_mutex> lock(HLE::g_hle_lock); +    std::lock_guard lock{HLE::g_hle_lock};      const FunctionDef* info = GetSVCInfo(immediate);      if (info) { diff --git a/src/core/hle/kernel/svc_wrap.h b/src/core/hle/kernel/svc_wrap.h index 2a2c2c5ea..b3733680f 100644 --- a/src/core/hle/kernel/svc_wrap.h +++ b/src/core/hle/kernel/svc_wrap.h @@ -78,6 +78,14 @@ void SvcWrap() {      FuncReturn(retval);  } +template <ResultCode func(u32*, u64, u32)> +void SvcWrap() { +    u32 param_1 = 0; +    const u32 retval = func(¶m_1, Param(1), static_cast<u32>(Param(2))).raw; +    Core::CurrentArmInterface().SetReg(1, param_1); +    FuncReturn(retval); +} +  template <ResultCode func(u64*, u32)>  void SvcWrap() {      u64 param_1 = 0; diff --git a/src/core/hle/kernel/thread.cpp b/src/core/hle/kernel/thread.cpp index e5853c46f..1b891f632 100644 --- a/src/core/hle/kernel/thread.cpp +++ b/src/core/hle/kernel/thread.cpp @@ -28,7 +28,7 @@  namespace Kernel { -bool Thread::ShouldWait(Thread* thread) const { +bool Thread::ShouldWait(const Thread* thread) const {      return status != ThreadStatus::Dead;  } @@ -62,6 +62,8 @@ void Thread::Stop() {      }      wait_objects.clear(); +    owner_process->UnregisterThread(this); +      // Mark the TLS slot in the thread's page as free.      owner_process->FreeTLSSlot(tls_address);  } @@ -199,9 +201,11 @@ ResultVal<SharedPtr<Thread>> Thread::Create(KernelCore& kernel, std::string name      thread->callback_handle = kernel.ThreadWakeupCallbackHandleTable().Create(thread).Unwrap();      thread->owner_process = &owner_process;      thread->scheduler = &system.Scheduler(processor_id); -    thread->scheduler->AddThread(thread, priority); +    thread->scheduler->AddThread(thread);      thread->tls_address = thread->owner_process->MarkNextAvailableTLSSlotAsUsed(*thread); +    thread->owner_process->RegisterThread(thread.get()); +      // TODO(peachum): move to ScheduleThread() when scheduler is added so selected core is used      // to initialize the context      ResetThreadContext(thread->context, stack_top, entry_point, arg); @@ -229,16 +233,16 @@ void Thread::SetWaitSynchronizationOutput(s32 output) {      context.cpu_registers[1] = output;  } -s32 Thread::GetWaitObjectIndex(WaitObject* object) const { +s32 Thread::GetWaitObjectIndex(const WaitObject* object) const {      ASSERT_MSG(!wait_objects.empty(), "Thread is not waiting for anything"); -    auto match = std::find(wait_objects.rbegin(), wait_objects.rend(), object); +    const auto match = std::find(wait_objects.rbegin(), wait_objects.rend(), object);      return static_cast<s32>(std::distance(match, wait_objects.rend()) - 1);  }  VAddr Thread::GetCommandBufferAddress() const {      // Offset from the start of TLS at which the IPC command buffer begins. -    static constexpr int CommandHeaderOffset = 0x80; -    return GetTLSAddress() + CommandHeaderOffset; +    constexpr u64 command_header_offset = 0x80; +    return GetTLSAddress() + command_header_offset;  }  void Thread::SetStatus(ThreadStatus new_status) { @@ -352,7 +356,7 @@ void Thread::ChangeScheduler() {      if (*new_processor_id != processor_id) {          // Remove thread from previous core's scheduler          scheduler->RemoveThread(this); -        next_scheduler.AddThread(this, current_priority); +        next_scheduler.AddThread(this);      }      processor_id = *new_processor_id; @@ -367,7 +371,7 @@ void Thread::ChangeScheduler() {      system.CpuCore(processor_id).PrepareReschedule();  } -bool Thread::AllWaitObjectsReady() { +bool Thread::AllWaitObjectsReady() const {      return std::none_of(          wait_objects.begin(), wait_objects.end(),          [this](const SharedPtr<WaitObject>& object) { return object->ShouldWait(this); }); diff --git a/src/core/hle/kernel/thread.h b/src/core/hle/kernel/thread.h index 9c684758c..73e5d1bb4 100644 --- a/src/core/hle/kernel/thread.h +++ b/src/core/hle/kernel/thread.h @@ -111,7 +111,7 @@ public:          return HANDLE_TYPE;      } -    bool ShouldWait(Thread* thread) const override; +    bool ShouldWait(const Thread* thread) const override;      void Acquire(Thread* thread) override;      /** @@ -205,7 +205,7 @@ public:       * object in the list.       * @param object Object to query the index of.       */ -    s32 GetWaitObjectIndex(WaitObject* object) const; +    s32 GetWaitObjectIndex(const WaitObject* object) const;      /**       * Stops a thread, invalidating it from further use @@ -299,7 +299,7 @@ public:      }      /// Determines whether all the objects this thread is waiting on are ready. -    bool AllWaitObjectsReady(); +    bool AllWaitObjectsReady() const;      const MutexWaitingThreads& GetMutexWaitingThreads() const {          return wait_mutex_threads; diff --git a/src/core/hle/kernel/wait_object.h b/src/core/hle/kernel/wait_object.h index 5987fb971..04464a51a 100644 --- a/src/core/hle/kernel/wait_object.h +++ b/src/core/hle/kernel/wait_object.h @@ -24,7 +24,7 @@ public:       * @param thread The thread about which we're deciding.       * @return True if the current thread should wait due to this object being unavailable       */ -    virtual bool ShouldWait(Thread* thread) const = 0; +    virtual bool ShouldWait(const Thread* thread) const = 0;      /// Acquire/lock the object for the specified thread if it is available      virtual void Acquire(Thread* thread) = 0; diff --git a/src/core/hle/service/fatal/fatal.cpp b/src/core/hle/service/fatal/fatal.cpp index 770590d0b..2c229bcad 100644 --- a/src/core/hle/service/fatal/fatal.cpp +++ b/src/core/hle/service/fatal/fatal.cpp @@ -25,21 +25,34 @@ Module::Interface::Interface(std::shared_ptr<Module> module, const char* name)  Module::Interface::~Interface() = default;  struct FatalInfo { -    std::array<u64_le, 31> registers{}; // TODO(ogniK): See if this actually is registers or -                                        // not(find a game which has non zero valeus) -    u64_le unk0{}; -    u64_le unk1{}; -    u64_le unk2{}; -    u64_le unk3{}; -    u64_le unk4{}; -    u64_le unk5{}; -    u64_le unk6{}; +    enum class Architecture : s32 { +        AArch64, +        AArch32, +    }; + +    const char* ArchAsString() const { +        return arch == Architecture::AArch64 ? "AArch64" : "AArch32"; +    } + +    std::array<u64_le, 31> registers{}; +    u64_le sp{}; +    u64_le pc{}; +    u64_le pstate{}; +    u64_le afsr0{}; +    u64_le afsr1{}; +    u64_le esr{}; +    u64_le far{};      std::array<u64_le, 32> backtrace{}; -    u64_le unk7{}; -    u64_le unk8{}; +    u64_le program_entry_point{}; + +    // Bit flags that indicate which registers have been set with values +    // for this context. The service itself uses these to determine which +    // registers to specifically print out. +    u64_le set_flags{}; +      u32_le backtrace_size{}; -    u32_le unk9{}; +    Architecture arch{};      u32_le unk10{}; // TODO(ogniK): Is this even used or is it just padding?  };  static_assert(sizeof(FatalInfo) == 0x250, "FatalInfo is an invalid size"); @@ -52,36 +65,36 @@ enum class FatalType : u32 {  static void GenerateErrorReport(ResultCode error_code, const FatalInfo& info) {      const auto title_id = Core::CurrentProcess()->GetTitleID(); -    std::string crash_report = -        fmt::format("Yuzu {}-{} crash report\n" -                    "Title ID:                        {:016x}\n" -                    "Result:                          0x{:X} ({:04}-{:04d})\n" -                    "\n", -                    Common::g_scm_branch, Common::g_scm_desc, title_id, error_code.raw, -                    2000 + static_cast<u32>(error_code.module.Value()), -                    static_cast<u32>(error_code.description.Value()), info.unk8, info.unk7); +    std::string crash_report = fmt::format( +        "Yuzu {}-{} crash report\n" +        "Title ID:                        {:016x}\n" +        "Result:                          0x{:X} ({:04}-{:04d})\n" +        "Set flags:                       0x{:16X}\n" +        "Program entry point:             0x{:16X}\n" +        "\n", +        Common::g_scm_branch, Common::g_scm_desc, title_id, error_code.raw, +        2000 + static_cast<u32>(error_code.module.Value()), +        static_cast<u32>(error_code.description.Value()), info.set_flags, info.program_entry_point);      if (info.backtrace_size != 0x0) {          crash_report += "Registers:\n"; -        // TODO(ogniK): This is just a guess, find a game which actually has non zero values          for (size_t i = 0; i < info.registers.size(); i++) {              crash_report +=                  fmt::format("    X[{:02d}]:                       {:016x}\n", i, info.registers[i]);          } -        crash_report += fmt::format("    Unknown 0:                   {:016x}\n", info.unk0); -        crash_report += fmt::format("    Unknown 1:                   {:016x}\n", info.unk1); -        crash_report += fmt::format("    Unknown 2:                   {:016x}\n", info.unk2); -        crash_report += fmt::format("    Unknown 3:                   {:016x}\n", info.unk3); -        crash_report += fmt::format("    Unknown 4:                   {:016x}\n", info.unk4); -        crash_report += fmt::format("    Unknown 5:                   {:016x}\n", info.unk5); -        crash_report += fmt::format("    Unknown 6:                   {:016x}\n", info.unk6); +        crash_report += fmt::format("    SP:                          {:016x}\n", info.sp); +        crash_report += fmt::format("    PC:                          {:016x}\n", info.pc); +        crash_report += fmt::format("    PSTATE:                      {:016x}\n", info.pstate); +        crash_report += fmt::format("    AFSR0:                       {:016x}\n", info.afsr0); +        crash_report += fmt::format("    AFSR1:                       {:016x}\n", info.afsr1); +        crash_report += fmt::format("    ESR:                         {:016x}\n", info.esr); +        crash_report += fmt::format("    FAR:                         {:016x}\n", info.far);          crash_report += "\nBacktrace:\n";          for (size_t i = 0; i < info.backtrace_size; i++) {              crash_report +=                  fmt::format("    Backtrace[{:02d}]:               {:016x}\n", i, info.backtrace[i]);          } -        crash_report += fmt::format("\nUnknown 7:                       0x{:016x}\n", info.unk7); -        crash_report += fmt::format("Unknown 8:                       0x{:016x}\n", info.unk8); -        crash_report += fmt::format("Unknown 9:                       0x{:016x}\n", info.unk9); + +        crash_report += fmt::format("Architecture:                    {}\n", info.ArchAsString());          crash_report += fmt::format("Unknown 10:                      0x{:016x}\n", info.unk10);      } @@ -125,13 +138,13 @@ static void ThrowFatalError(ResultCode error_code, FatalType fatal_type, const F      case FatalType::ErrorReport:          GenerateErrorReport(error_code, info);          break; -    }; +    }  }  void Module::Interface::ThrowFatal(Kernel::HLERequestContext& ctx) {      LOG_ERROR(Service_Fatal, "called");      IPC::RequestParser rp{ctx}; -    auto error_code = rp.Pop<ResultCode>(); +    const auto error_code = rp.Pop<ResultCode>();      ThrowFatalError(error_code, FatalType::ErrorScreen, {});      IPC::ResponseBuilder rb{ctx, 2}; @@ -141,8 +154,8 @@ void Module::Interface::ThrowFatal(Kernel::HLERequestContext& ctx) {  void Module::Interface::ThrowFatalWithPolicy(Kernel::HLERequestContext& ctx) {      LOG_ERROR(Service_Fatal, "called");      IPC::RequestParser rp(ctx); -    auto error_code = rp.Pop<ResultCode>(); -    auto fatal_type = rp.PopEnum<FatalType>(); +    const auto error_code = rp.Pop<ResultCode>(); +    const auto fatal_type = rp.PopEnum<FatalType>();      ThrowFatalError(error_code, fatal_type, {}); // No info is passed with ThrowFatalWithPolicy      IPC::ResponseBuilder rb{ctx, 2}; @@ -152,9 +165,9 @@ void Module::Interface::ThrowFatalWithPolicy(Kernel::HLERequestContext& ctx) {  void Module::Interface::ThrowFatalWithCpuContext(Kernel::HLERequestContext& ctx) {      LOG_ERROR(Service_Fatal, "called");      IPC::RequestParser rp(ctx); -    auto error_code = rp.Pop<ResultCode>(); -    auto fatal_type = rp.PopEnum<FatalType>(); -    auto fatal_info = ctx.ReadBuffer(); +    const auto error_code = rp.Pop<ResultCode>(); +    const auto fatal_type = rp.PopEnum<FatalType>(); +    const auto fatal_info = ctx.ReadBuffer();      FatalInfo info{};      ASSERT_MSG(fatal_info.size() == sizeof(FatalInfo), "Invalid fatal info buffer size!"); diff --git a/src/core/hle/service/nfc/nfc.cpp b/src/core/hle/service/nfc/nfc.cpp index 5c62d42ba..ca88bf97f 100644 --- a/src/core/hle/service/nfc/nfc.cpp +++ b/src/core/hle/service/nfc/nfc.cpp @@ -150,7 +150,7 @@ private:          IPC::ResponseBuilder rb{ctx, 3};          rb.Push(RESULT_SUCCESS); -        rb.PushRaw<u8>(Settings::values.enable_nfc); +        rb.PushRaw<u8>(true);      }      void GetStateOld(Kernel::HLERequestContext& ctx) { diff --git a/src/core/hle/service/nfp/nfp.cpp b/src/core/hle/service/nfp/nfp.cpp index 1c4482e47..c6babdd4d 100644 --- a/src/core/hle/service/nfp/nfp.cpp +++ b/src/core/hle/service/nfp/nfp.cpp @@ -335,7 +335,7 @@ void Module::Interface::CreateUserInterface(Kernel::HLERequestContext& ctx) {  }  bool Module::Interface::LoadAmiibo(const std::vector<u8>& buffer) { -    std::lock_guard<std::recursive_mutex> lock(HLE::g_hle_lock); +    std::lock_guard lock{HLE::g_hle_lock};      if (buffer.size() < sizeof(AmiiboFile)) {          return false;      } diff --git a/src/core/loader/elf.cpp b/src/core/loader/elf.cpp index 8b1920f22..46ac372f6 100644 --- a/src/core/loader/elf.cpp +++ b/src/core/loader/elf.cpp @@ -341,7 +341,7 @@ Kernel::CodeSet ElfReader::LoadInto(VAddr vaddr) {      }      codeset.entrypoint = base_addr + header->e_entry; -    codeset.memory = std::make_shared<std::vector<u8>>(std::move(program_image)); +    codeset.memory = std::move(program_image);      LOG_DEBUG(Loader, "Done loading."); diff --git a/src/core/loader/nro.cpp b/src/core/loader/nro.cpp index 5de02a94b..31e4a0c84 100644 --- a/src/core/loader/nro.cpp +++ b/src/core/loader/nro.cpp @@ -187,7 +187,7 @@ static bool LoadNroImpl(Kernel::Process& process, const std::vector<u8>& data,      program_image.resize(static_cast<u32>(program_image.size()) + bss_size);      // Load codeset for current process -    codeset.memory = std::make_shared<std::vector<u8>>(std::move(program_image)); +    codeset.memory = std::move(program_image);      process.LoadModule(std::move(codeset), load_base);      // Register module with GDBStub diff --git a/src/core/loader/nso.cpp b/src/core/loader/nso.cpp index 714d85a59..babc7e646 100644 --- a/src/core/loader/nso.cpp +++ b/src/core/loader/nso.cpp @@ -161,7 +161,7 @@ std::optional<VAddr> AppLoader_NSO::LoadModule(Kernel::Process& process,      }      // Load codeset for current process -    codeset.memory = std::make_shared<std::vector<u8>>(std::move(program_image)); +    codeset.memory = std::move(program_image);      process.LoadModule(std::move(codeset), load_base);      // Register module with GDBStub diff --git a/src/core/perf_stats.cpp b/src/core/perf_stats.cpp index c716a462b..4afd6c8a3 100644 --- a/src/core/perf_stats.cpp +++ b/src/core/perf_stats.cpp @@ -18,13 +18,13 @@ using std::chrono::microseconds;  namespace Core {  void PerfStats::BeginSystemFrame() { -    std::lock_guard<std::mutex> lock(object_mutex); +    std::lock_guard lock{object_mutex};      frame_begin = Clock::now();  }  void PerfStats::EndSystemFrame() { -    std::lock_guard<std::mutex> lock(object_mutex); +    std::lock_guard lock{object_mutex};      auto frame_end = Clock::now();      accumulated_frametime += frame_end - frame_begin; @@ -35,13 +35,13 @@ void PerfStats::EndSystemFrame() {  }  void PerfStats::EndGameFrame() { -    std::lock_guard<std::mutex> lock(object_mutex); +    std::lock_guard lock{object_mutex};      game_frames += 1;  }  PerfStatsResults PerfStats::GetAndResetStats(microseconds current_system_time_us) { -    std::lock_guard<std::mutex> lock(object_mutex); +    std::lock_guard lock{object_mutex};      const auto now = Clock::now();      // Walltime elapsed since stats were reset @@ -67,7 +67,7 @@ PerfStatsResults PerfStats::GetAndResetStats(microseconds current_system_time_us  }  double PerfStats::GetLastFrameTimeScale() { -    std::lock_guard<std::mutex> lock(object_mutex); +    std::lock_guard lock{object_mutex};      constexpr double FRAME_LENGTH = 1.0 / 60;      return duration_cast<DoubleSecs>(previous_frame_length).count() / FRAME_LENGTH; diff --git a/src/core/settings.cpp b/src/core/settings.cpp index 6dd3139cc..6d32ebea3 100644 --- a/src/core/settings.cpp +++ b/src/core/settings.cpp @@ -82,7 +82,6 @@ void LogSetting(const std::string& name, const T& value) {  void LogSettings() {      LOG_INFO(Config, "yuzu Configuration:");      LogSetting("System_UseDockedMode", Settings::values.use_docked_mode); -    LogSetting("System_EnableNfc", Settings::values.enable_nfc);      LogSetting("System_RngSeed", Settings::values.rng_seed.value_or(0));      LogSetting("System_CurrentUser", Settings::values.current_user);      LogSetting("System_LanguageIndex", Settings::values.language_index); diff --git a/src/core/settings.h b/src/core/settings.h index cdfb2f742..d543eb32f 100644 --- a/src/core/settings.h +++ b/src/core/settings.h @@ -349,7 +349,6 @@ struct TouchscreenInput {  struct Values {      // System      bool use_docked_mode; -    bool enable_nfc;      std::optional<u32> rng_seed;      // Measured in seconds since epoch      std::optional<std::chrono::seconds> custom_rtc; diff --git a/src/input_common/keyboard.cpp b/src/input_common/keyboard.cpp index 525fe6abc..078374be5 100644 --- a/src/input_common/keyboard.cpp +++ b/src/input_common/keyboard.cpp @@ -36,18 +36,18 @@ struct KeyButtonPair {  class KeyButtonList {  public:      void AddKeyButton(int key_code, KeyButton* key_button) { -        std::lock_guard<std::mutex> guard(mutex); +        std::lock_guard guard{mutex};          list.push_back(KeyButtonPair{key_code, key_button});      }      void RemoveKeyButton(const KeyButton* key_button) { -        std::lock_guard<std::mutex> guard(mutex); +        std::lock_guard guard{mutex};          list.remove_if(              [key_button](const KeyButtonPair& pair) { return pair.key_button == key_button; });      }      void ChangeKeyStatus(int key_code, bool pressed) { -        std::lock_guard<std::mutex> guard(mutex); +        std::lock_guard guard{mutex};          for (const KeyButtonPair& pair : list) {              if (pair.key_code == key_code)                  pair.key_button->status.store(pressed); @@ -55,7 +55,7 @@ public:      }      void ChangeAllKeyStatus(bool pressed) { -        std::lock_guard<std::mutex> guard(mutex); +        std::lock_guard guard{mutex};          for (const KeyButtonPair& pair : list) {              pair.key_button->status.store(pressed);          } diff --git a/src/input_common/motion_emu.cpp b/src/input_common/motion_emu.cpp index 6d96d4019..868251628 100644 --- a/src/input_common/motion_emu.cpp +++ b/src/input_common/motion_emu.cpp @@ -39,7 +39,7 @@ public:      void Tilt(int x, int y) {          auto mouse_move = Common::MakeVec(x, y) - mouse_origin;          if (is_tilting) { -            std::lock_guard<std::mutex> guard(tilt_mutex); +            std::lock_guard guard{tilt_mutex};              if (mouse_move.x == 0 && mouse_move.y == 0) {                  tilt_angle = 0;              } else { @@ -51,13 +51,13 @@ public:      }      void EndTilt() { -        std::lock_guard<std::mutex> guard(tilt_mutex); +        std::lock_guard guard{tilt_mutex};          tilt_angle = 0;          is_tilting = false;      }      std::tuple<Common::Vec3<float>, Common::Vec3<float>> GetStatus() { -        std::lock_guard<std::mutex> guard(status_mutex); +        std::lock_guard guard{status_mutex};          return status;      } @@ -93,7 +93,7 @@ private:              old_q = q;              { -                std::lock_guard<std::mutex> guard(tilt_mutex); +                std::lock_guard guard{tilt_mutex};                  // Find the quaternion describing current 3DS tilting                  q = Common::MakeQuaternion( @@ -115,7 +115,7 @@ private:              // Update the sensor state              { -                std::lock_guard<std::mutex> guard(status_mutex); +                std::lock_guard guard{status_mutex};                  status = std::make_tuple(gravity, angular_rate);              }          } diff --git a/src/input_common/sdl/sdl_impl.cpp b/src/input_common/sdl/sdl_impl.cpp index b132d77f5..5949ecbae 100644 --- a/src/input_common/sdl/sdl_impl.cpp +++ b/src/input_common/sdl/sdl_impl.cpp @@ -55,22 +55,22 @@ public:          : guid{std::move(guid_)}, port{port_}, sdl_joystick{joystick, deleter} {}      void SetButton(int button, bool value) { -        std::lock_guard<std::mutex> lock(mutex); +        std::lock_guard lock{mutex};          state.buttons[button] = value;      }      bool GetButton(int button) const { -        std::lock_guard<std::mutex> lock(mutex); +        std::lock_guard lock{mutex};          return state.buttons.at(button);      }      void SetAxis(int axis, Sint16 value) { -        std::lock_guard<std::mutex> lock(mutex); +        std::lock_guard lock{mutex};          state.axes[axis] = value;      }      float GetAxis(int axis) const { -        std::lock_guard<std::mutex> lock(mutex); +        std::lock_guard lock{mutex};          return state.axes.at(axis) / 32767.0f;      } @@ -92,12 +92,12 @@ public:      }      void SetHat(int hat, Uint8 direction) { -        std::lock_guard<std::mutex> lock(mutex); +        std::lock_guard lock{mutex};          state.hats[hat] = direction;      }      bool GetHatDirection(int hat, Uint8 direction) const { -        std::lock_guard<std::mutex> lock(mutex); +        std::lock_guard lock{mutex};          return (state.hats.at(hat) & direction) != 0;      }      /** @@ -140,7 +140,7 @@ private:   * Get the nth joystick with the corresponding GUID   */  std::shared_ptr<SDLJoystick> SDLState::GetSDLJoystickByGUID(const std::string& guid, int port) { -    std::lock_guard<std::mutex> lock(joystick_map_mutex); +    std::lock_guard lock{joystick_map_mutex};      const auto it = joystick_map.find(guid);      if (it != joystick_map.end()) {          while (it->second.size() <= port) { @@ -161,7 +161,8 @@ std::shared_ptr<SDLJoystick> SDLState::GetSDLJoystickByGUID(const std::string& g  std::shared_ptr<SDLJoystick> SDLState::GetSDLJoystickBySDLID(SDL_JoystickID sdl_id) {      auto sdl_joystick = SDL_JoystickFromInstanceID(sdl_id);      const std::string guid = GetGUID(sdl_joystick); -    std::lock_guard<std::mutex> lock(joystick_map_mutex); + +    std::lock_guard lock{joystick_map_mutex};      auto map_it = joystick_map.find(guid);      if (map_it != joystick_map.end()) {          auto vec_it = std::find_if(map_it->second.begin(), map_it->second.end(), @@ -198,8 +199,9 @@ void SDLState::InitJoystick(int joystick_index) {          LOG_ERROR(Input, "failed to open joystick {}", joystick_index);          return;      } -    std::string guid = GetGUID(sdl_joystick); -    std::lock_guard<std::mutex> lock(joystick_map_mutex); +    const std::string guid = GetGUID(sdl_joystick); + +    std::lock_guard lock{joystick_map_mutex};      if (joystick_map.find(guid) == joystick_map.end()) {          auto joystick = std::make_shared<SDLJoystick>(guid, 0, sdl_joystick);          joystick_map[guid].emplace_back(std::move(joystick)); @@ -221,7 +223,7 @@ void SDLState::CloseJoystick(SDL_Joystick* sdl_joystick) {      std::string guid = GetGUID(sdl_joystick);      std::shared_ptr<SDLJoystick> joystick;      { -        std::lock_guard<std::mutex> lock(joystick_map_mutex); +        std::lock_guard lock{joystick_map_mutex};          // This call to guid is safe since the joystick is guaranteed to be in the map          auto& joystick_guid_list = joystick_map[guid];          const auto joystick_it = @@ -274,7 +276,7 @@ void SDLState::HandleGameControllerEvent(const SDL_Event& event) {  }  void SDLState::CloseJoysticks() { -    std::lock_guard<std::mutex> lock(joystick_map_mutex); +    std::lock_guard lock{joystick_map_mutex};      joystick_map.clear();  } diff --git a/src/video_core/debug_utils/debug_utils.cpp b/src/video_core/debug_utils/debug_utils.cpp index 5ffb492ea..f0ef67535 100644 --- a/src/video_core/debug_utils/debug_utils.cpp +++ b/src/video_core/debug_utils/debug_utils.cpp @@ -10,7 +10,7 @@ namespace Tegra {  void DebugContext::DoOnEvent(Event event, void* data) {      { -        std::unique_lock<std::mutex> lock(breakpoint_mutex); +        std::unique_lock lock{breakpoint_mutex};          // TODO(Subv): Commit the rasterizer's caches so framebuffers, render targets, etc. will          // show on debug widgets @@ -32,7 +32,7 @@ void DebugContext::DoOnEvent(Event event, void* data) {  void DebugContext::Resume() {      { -        std::lock_guard<std::mutex> lock(breakpoint_mutex); +        std::lock_guard lock{breakpoint_mutex};          // Tell all observers that we are about to resume          for (auto& breakpoint_observer : breakpoint_observers) { diff --git a/src/video_core/debug_utils/debug_utils.h b/src/video_core/debug_utils/debug_utils.h index c235faf46..ac3a2eb01 100644 --- a/src/video_core/debug_utils/debug_utils.h +++ b/src/video_core/debug_utils/debug_utils.h @@ -40,7 +40,7 @@ public:          /// Constructs the object such that it observes events of the given DebugContext.          explicit BreakPointObserver(std::shared_ptr<DebugContext> debug_context)              : context_weak(debug_context) { -            std::unique_lock<std::mutex> lock(debug_context->breakpoint_mutex); +            std::unique_lock lock{debug_context->breakpoint_mutex};              debug_context->breakpoint_observers.push_back(this);          } @@ -48,7 +48,7 @@ public:              auto context = context_weak.lock();              if (context) {                  { -                    std::unique_lock<std::mutex> lock(context->breakpoint_mutex); +                    std::unique_lock lock{context->breakpoint_mutex};                      context->breakpoint_observers.remove(this);                  } diff --git a/src/video_core/gpu_thread.cpp b/src/video_core/gpu_thread.cpp index 086b2f625..c5dc199c5 100644 --- a/src/video_core/gpu_thread.cpp +++ b/src/video_core/gpu_thread.cpp @@ -52,8 +52,8 @@ static void RunThread(VideoCore::RendererBase& renderer, Tegra::DmaPusher& dma_p  }  ThreadManager::ThreadManager(VideoCore::RendererBase& renderer, Tegra::DmaPusher& dma_pusher) -    : renderer{renderer}, dma_pusher{dma_pusher}, thread{RunThread, std::ref(renderer), -                                                         std::ref(dma_pusher), std::ref(state)} {} +    : renderer{renderer}, thread{RunThread, std::ref(renderer), std::ref(dma_pusher), +                                 std::ref(state)} {}  ThreadManager::~ThreadManager() {      // Notify GPU thread that a shutdown is pending diff --git a/src/video_core/gpu_thread.h b/src/video_core/gpu_thread.h index 8cd7db1c6..70acb2e79 100644 --- a/src/video_core/gpu_thread.h +++ b/src/video_core/gpu_thread.h @@ -4,10 +4,8 @@  #pragma once -#include <array>  #include <atomic>  #include <condition_variable> -#include <memory>  #include <mutex>  #include <optional>  #include <thread> @@ -97,13 +95,13 @@ struct SynchState final {      std::condition_variable frames_condition;      void IncrementFramesCounter() { -        std::lock_guard<std::mutex> lock{frames_mutex}; +        std::lock_guard lock{frames_mutex};          ++queued_frame_count;      }      void DecrementFramesCounter() {          { -            std::lock_guard<std::mutex> lock{frames_mutex}; +            std::lock_guard lock{frames_mutex};              --queued_frame_count;              if (queued_frame_count) { @@ -115,7 +113,7 @@ struct SynchState final {      void WaitForFrames() {          { -            std::lock_guard<std::mutex> lock{frames_mutex}; +            std::lock_guard lock{frames_mutex};              if (!queued_frame_count) {                  return;              } @@ -123,14 +121,14 @@ struct SynchState final {          // Wait for the GPU to be idle (all commands to be executed)          { -            std::unique_lock<std::mutex> lock{frames_mutex}; +            std::unique_lock lock{frames_mutex};              frames_condition.wait(lock, [this] { return !queued_frame_count; });          }      }      void SignalCommands() {          { -            std::unique_lock<std::mutex> lock{commands_mutex}; +            std::unique_lock lock{commands_mutex};              if (queue.Empty()) {                  return;              } @@ -140,7 +138,7 @@ struct SynchState final {      }      void WaitForCommands() { -        std::unique_lock<std::mutex> lock{commands_mutex}; +        std::unique_lock lock{commands_mutex};          commands_condition.wait(lock, [this] { return !queue.Empty(); });      } @@ -177,7 +175,6 @@ private:  private:      SynchState state;      VideoCore::RendererBase& renderer; -    Tegra::DmaPusher& dma_pusher;      std::thread thread;      std::thread::id thread_id;  }; diff --git a/src/video_core/rasterizer_cache.h b/src/video_core/rasterizer_cache.h index 9fc9f3056..291772186 100644 --- a/src/video_core/rasterizer_cache.h +++ b/src/video_core/rasterizer_cache.h @@ -71,8 +71,8 @@ private:      bool is_registered{};      ///< Whether the object is currently registered with the cache      bool is_dirty{};           ///< Whether the object is dirty (out of sync with guest memory)      u64 last_modified_ticks{}; ///< When the object was last modified, used for in-order flushing -    CacheAddr cache_addr{};    ///< Cache address memory, unique from emulated virtual address space      const u8* host_ptr{};      ///< Pointer to the memory backing this cached region +    CacheAddr cache_addr{};    ///< Cache address memory, unique from emulated virtual address space  };  template <class T> @@ -84,7 +84,7 @@ public:      /// Write any cached resources overlapping the specified region back to memory      void FlushRegion(CacheAddr addr, std::size_t size) { -        std::lock_guard<std::recursive_mutex> lock{mutex}; +        std::lock_guard lock{mutex};          const auto& objects{GetSortedObjectsFromRegion(addr, size)};          for (auto& object : objects) { @@ -94,7 +94,7 @@ public:      /// Mark the specified region as being invalidated      void InvalidateRegion(CacheAddr addr, u64 size) { -        std::lock_guard<std::recursive_mutex> lock{mutex}; +        std::lock_guard lock{mutex};          const auto& objects{GetSortedObjectsFromRegion(addr, size)};          for (auto& object : objects) { @@ -108,7 +108,7 @@ public:      /// Invalidates everything in the cache      void InvalidateAll() { -        std::lock_guard<std::recursive_mutex> lock{mutex}; +        std::lock_guard lock{mutex};          while (interval_cache.begin() != interval_cache.end()) {              Unregister(*interval_cache.begin()->second.begin()); @@ -133,7 +133,7 @@ protected:      /// Register an object into the cache      virtual void Register(const T& object) { -        std::lock_guard<std::recursive_mutex> lock{mutex}; +        std::lock_guard lock{mutex};          object->SetIsRegistered(true);          interval_cache.add({GetInterval(object), ObjectSet{object}}); @@ -143,7 +143,7 @@ protected:      /// Unregisters an object from the cache      virtual void Unregister(const T& object) { -        std::lock_guard<std::recursive_mutex> lock{mutex}; +        std::lock_guard lock{mutex};          object->SetIsRegistered(false);          rasterizer.UpdatePagesCachedCount(object->GetCpuAddr(), object->GetSizeInBytes(), -1); @@ -153,14 +153,14 @@ protected:      /// Returns a ticks counter used for tracking when cached objects were last modified      u64 GetModifiedTicks() { -        std::lock_guard<std::recursive_mutex> lock{mutex}; +        std::lock_guard lock{mutex};          return ++modified_ticks;      }      /// Flushes the specified object, updating appropriate cache state as needed      void FlushObject(const T& object) { -        std::lock_guard<std::recursive_mutex> lock{mutex}; +        std::lock_guard lock{mutex};          if (!object->IsDirty()) {              return; diff --git a/src/video_core/renderer_opengl/gl_buffer_cache.cpp b/src/video_core/renderer_opengl/gl_buffer_cache.cpp index f75c65825..fd091c84c 100644 --- a/src/video_core/renderer_opengl/gl_buffer_cache.cpp +++ b/src/video_core/renderer_opengl/gl_buffer_cache.cpp @@ -15,8 +15,8 @@ namespace OpenGL {  CachedBufferEntry::CachedBufferEntry(VAddr cpu_addr, std::size_t size, GLintptr offset,                                       std::size_t alignment, u8* host_ptr) -    : cpu_addr{cpu_addr}, size{size}, offset{offset}, alignment{alignment}, RasterizerCacheObject{ -                                                                                host_ptr} {} +    : RasterizerCacheObject{host_ptr}, cpu_addr{cpu_addr}, size{size}, offset{offset}, +      alignment{alignment} {}  OGLBufferCache::OGLBufferCache(RasterizerOpenGL& rasterizer, std::size_t size)      : RasterizerCache{rasterizer}, stream_buffer(size, true) {} diff --git a/src/video_core/renderer_opengl/gl_global_cache.cpp b/src/video_core/renderer_opengl/gl_global_cache.cpp index 0fbfbad55..da9326253 100644 --- a/src/video_core/renderer_opengl/gl_global_cache.cpp +++ b/src/video_core/renderer_opengl/gl_global_cache.cpp @@ -15,7 +15,7 @@  namespace OpenGL {  CachedGlobalRegion::CachedGlobalRegion(VAddr cpu_addr, u32 size, u8* host_ptr) -    : cpu_addr{cpu_addr}, size{size}, RasterizerCacheObject{host_ptr} { +    : RasterizerCacheObject{host_ptr}, cpu_addr{cpu_addr}, size{size} {      buffer.Create();      // Bind and unbind the buffer so it gets allocated by the driver      glBindBuffer(GL_SHADER_STORAGE_BUFFER, buffer.handle); diff --git a/src/video_core/renderer_opengl/gl_rasterizer.cpp b/src/video_core/renderer_opengl/gl_rasterizer.cpp index e06dfe43f..046fc935b 100644 --- a/src/video_core/renderer_opengl/gl_rasterizer.cpp +++ b/src/video_core/renderer_opengl/gl_rasterizer.cpp @@ -100,11 +100,9 @@ struct FramebufferCacheKey {      }  }; -RasterizerOpenGL::RasterizerOpenGL(Core::Frontend::EmuWindow& window, Core::System& system, -                                   ScreenInfo& info) -    : res_cache{*this}, shader_cache{*this, system}, global_cache{*this}, -      emu_window{window}, system{system}, screen_info{info}, -      buffer_cache(*this, STREAM_BUFFER_SIZE) { +RasterizerOpenGL::RasterizerOpenGL(Core::System& system, ScreenInfo& info) +    : res_cache{*this}, shader_cache{*this, system}, global_cache{*this}, system{system}, +      screen_info{info}, buffer_cache(*this, STREAM_BUFFER_SIZE) {      // Create sampler objects      for (std::size_t i = 0; i < texture_samplers.size(); ++i) {          texture_samplers[i].Create(); diff --git a/src/video_core/renderer_opengl/gl_rasterizer.h b/src/video_core/renderer_opengl/gl_rasterizer.h index 30f3e8acb..4de565321 100644 --- a/src/video_core/renderer_opengl/gl_rasterizer.h +++ b/src/video_core/renderer_opengl/gl_rasterizer.h @@ -50,8 +50,7 @@ struct FramebufferCacheKey;  class RasterizerOpenGL : public VideoCore::RasterizerInterface {  public: -    explicit RasterizerOpenGL(Core::Frontend::EmuWindow& window, Core::System& system, -                              ScreenInfo& info); +    explicit RasterizerOpenGL(Core::System& system, ScreenInfo& info);      ~RasterizerOpenGL() override;      void DrawArrays() override; @@ -214,7 +213,6 @@ private:      ShaderCacheOpenGL shader_cache;      GlobalRegionCacheOpenGL global_cache; -    Core::Frontend::EmuWindow& emu_window;      Core::System& system;      ScreenInfo& screen_info; diff --git a/src/video_core/renderer_opengl/gl_rasterizer_cache.cpp b/src/video_core/renderer_opengl/gl_rasterizer_cache.cpp index 0235317c0..aba6ce731 100644 --- a/src/video_core/renderer_opengl/gl_rasterizer_cache.cpp +++ b/src/video_core/renderer_opengl/gl_rasterizer_cache.cpp @@ -562,8 +562,8 @@ void RasterizerCacheOpenGL::CopySurface(const Surface& src_surface, const Surfac  }  CachedSurface::CachedSurface(const SurfaceParams& params) -    : params{params}, gl_target{SurfaceTargetToGL(params.target)}, -      cached_size_in_bytes{params.size_in_bytes}, RasterizerCacheObject{params.host_ptr} { +    : RasterizerCacheObject{params.host_ptr}, params{params}, +      gl_target{SurfaceTargetToGL(params.target)}, cached_size_in_bytes{params.size_in_bytes} {      const auto optional_cpu_addr{          Core::System::GetInstance().GPU().MemoryManager().GpuToCpuAddress(params.gpu_addr)}; diff --git a/src/video_core/renderer_opengl/gl_shader_cache.cpp b/src/video_core/renderer_opengl/gl_shader_cache.cpp index 1f8eca6f0..290e654bc 100644 --- a/src/video_core/renderer_opengl/gl_shader_cache.cpp +++ b/src/video_core/renderer_opengl/gl_shader_cache.cpp @@ -215,9 +215,9 @@ CachedShader::CachedShader(VAddr cpu_addr, u64 unique_identifier,                             Maxwell::ShaderProgram program_type, ShaderDiskCacheOpenGL& disk_cache,                             const PrecompiledPrograms& precompiled_programs,                             ProgramCode&& program_code, ProgramCode&& program_code_b, u8* host_ptr) -    : host_ptr{host_ptr}, cpu_addr{cpu_addr}, unique_identifier{unique_identifier}, -      program_type{program_type}, disk_cache{disk_cache}, -      precompiled_programs{precompiled_programs}, RasterizerCacheObject{host_ptr} { +    : RasterizerCacheObject{host_ptr}, host_ptr{host_ptr}, cpu_addr{cpu_addr}, +      unique_identifier{unique_identifier}, program_type{program_type}, disk_cache{disk_cache}, +      precompiled_programs{precompiled_programs} {      const std::size_t code_size = CalculateProgramSize(program_code);      const std::size_t code_size_b = @@ -245,9 +245,9 @@ CachedShader::CachedShader(VAddr cpu_addr, u64 unique_identifier,                             Maxwell::ShaderProgram program_type, ShaderDiskCacheOpenGL& disk_cache,                             const PrecompiledPrograms& precompiled_programs,                             GLShader::ProgramResult result, u8* host_ptr) -    : cpu_addr{cpu_addr}, unique_identifier{unique_identifier}, program_type{program_type}, -      disk_cache{disk_cache}, precompiled_programs{precompiled_programs}, RasterizerCacheObject{ -                                                                              host_ptr} { +    : RasterizerCacheObject{host_ptr}, cpu_addr{cpu_addr}, unique_identifier{unique_identifier}, +      program_type{program_type}, disk_cache{disk_cache}, precompiled_programs{ +                                                              precompiled_programs} {      code = std::move(result.first);      entries = result.second; diff --git a/src/video_core/renderer_opengl/renderer_opengl.cpp b/src/video_core/renderer_opengl/renderer_opengl.cpp index 5e3d862c6..a01efeb05 100644 --- a/src/video_core/renderer_opengl/renderer_opengl.cpp +++ b/src/video_core/renderer_opengl/renderer_opengl.cpp @@ -266,7 +266,7 @@ void RendererOpenGL::CreateRasterizer() {      }      // Initialize sRGB Usage      OpenGLState::ClearsRGBUsed(); -    rasterizer = std::make_unique<RasterizerOpenGL>(render_window, system, screen_info); +    rasterizer = std::make_unique<RasterizerOpenGL>(system, screen_info);  }  void RendererOpenGL::ConfigureFramebufferTexture(TextureInfo& texture, diff --git a/src/video_core/renderer_vulkan/vk_buffer_cache.cpp b/src/video_core/renderer_vulkan/vk_buffer_cache.cpp index eac51ecb3..388b5ffd5 100644 --- a/src/video_core/renderer_vulkan/vk_buffer_cache.cpp +++ b/src/video_core/renderer_vulkan/vk_buffer_cache.cpp @@ -19,8 +19,8 @@ namespace Vulkan {  CachedBufferEntry::CachedBufferEntry(VAddr cpu_addr, std::size_t size, u64 offset,                                       std::size_t alignment, u8* host_ptr) -    : cpu_addr{cpu_addr}, size{size}, offset{offset}, alignment{alignment}, RasterizerCacheObject{ -                                                                                host_ptr} {} +    : RasterizerCacheObject{host_ptr}, cpu_addr{cpu_addr}, size{size}, offset{offset}, +      alignment{alignment} {}  VKBufferCache::VKBufferCache(Tegra::MemoryManager& tegra_memory_manager,                               VideoCore::RasterizerInterface& rasterizer, const VKDevice& device, diff --git a/src/web_service/web_backend.cpp b/src/web_service/web_backend.cpp index 40da1a4e2..dc149d2ed 100644 --- a/src/web_service/web_backend.cpp +++ b/src/web_service/web_backend.cpp @@ -24,7 +24,7 @@ constexpr u32 TIMEOUT_SECONDS = 30;  struct Client::Impl {      Impl(std::string host, std::string username, std::string token)          : host{std::move(host)}, username{std::move(username)}, token{std::move(token)} { -        std::lock_guard<std::mutex> lock(jwt_cache.mutex); +        std::lock_guard lock{jwt_cache.mutex};          if (this->username == jwt_cache.username && this->token == jwt_cache.token) {              jwt = jwt_cache.jwt;          } @@ -151,7 +151,7 @@ struct Client::Impl {          if (result.result_code != Common::WebResult::Code::Success) {              LOG_ERROR(WebService, "UpdateJWT failed");          } else { -            std::lock_guard<std::mutex> lock(jwt_cache.mutex); +            std::lock_guard lock{jwt_cache.mutex};              jwt_cache.username = username;              jwt_cache.token = token;              jwt_cache.jwt = jwt = result.returned_data; diff --git a/src/yuzu/applets/profile_select.cpp b/src/yuzu/applets/profile_select.cpp index 5c1b65a2c..f95f7fe3c 100644 --- a/src/yuzu/applets/profile_select.cpp +++ b/src/yuzu/applets/profile_select.cpp @@ -58,10 +58,7 @@ QtProfileSelectionDialog::QtProfileSelectionDialog(QWidget* parent)      scroll_area = new QScrollArea; -    buttons = new QDialogButtonBox; -    buttons->addButton(tr("Cancel"), QDialogButtonBox::RejectRole); -    buttons->addButton(tr("OK"), QDialogButtonBox::AcceptRole); - +    buttons = new QDialogButtonBox(QDialogButtonBox::Cancel | QDialogButtonBox::Ok);      connect(buttons, &QDialogButtonBox::accepted, this, &QtProfileSelectionDialog::accept);      connect(buttons, &QDialogButtonBox::rejected, this, &QtProfileSelectionDialog::reject); @@ -163,6 +160,6 @@ void QtProfileSelector::SelectProfile(  void QtProfileSelector::MainWindowFinishedSelection(std::optional<Service::Account::UUID> uuid) {      // Acquire the HLE mutex -    std::lock_guard<std::recursive_mutex> lock(HLE::g_hle_lock); +    std::lock_guard lock{HLE::g_hle_lock};      callback(uuid);  } diff --git a/src/yuzu/applets/software_keyboard.cpp b/src/yuzu/applets/software_keyboard.cpp index 8a26fdff1..f3eb29b25 100644 --- a/src/yuzu/applets/software_keyboard.cpp +++ b/src/yuzu/applets/software_keyboard.cpp @@ -75,13 +75,13 @@ QtSoftwareKeyboardDialog::QtSoftwareKeyboardDialog(          length_label->setText(QStringLiteral("%1/%2").arg(text.size()).arg(parameters.max_length));      }); -    buttons = new QDialogButtonBox; -    buttons->addButton(tr("Cancel"), QDialogButtonBox::RejectRole); -    buttons->addButton(parameters.submit_text.empty() -                           ? tr("OK") -                           : QString::fromStdU16String(parameters.submit_text), -                       QDialogButtonBox::AcceptRole); - +    buttons = new QDialogButtonBox(QDialogButtonBox::Cancel); +    if (parameters.submit_text.empty()) { +        buttons->addButton(QDialogButtonBox::Ok); +    } else { +        buttons->addButton(QString::fromStdU16String(parameters.submit_text), +                           QDialogButtonBox::AcceptRole); +    }      connect(buttons, &QDialogButtonBox::accepted, this, &QtSoftwareKeyboardDialog::accept);      connect(buttons, &QDialogButtonBox::rejected, this, &QtSoftwareKeyboardDialog::reject);      layout->addWidget(header_label); @@ -141,12 +141,12 @@ void QtSoftwareKeyboard::SendTextCheckDialog(std::u16string error_message,  void QtSoftwareKeyboard::MainWindowFinishedText(std::optional<std::u16string> text) {      // Acquire the HLE mutex -    std::lock_guard<std::recursive_mutex> lock(HLE::g_hle_lock); +    std::lock_guard lock{HLE::g_hle_lock};      text_output(text);  }  void QtSoftwareKeyboard::MainWindowFinishedCheckDialog() {      // Acquire the HLE mutex -    std::lock_guard<std::recursive_mutex> lock(HLE::g_hle_lock); +    std::lock_guard lock{HLE::g_hle_lock};      finished_check();  } diff --git a/src/yuzu/applets/web_browser.cpp b/src/yuzu/applets/web_browser.cpp index 979b9ec14..ac80b2fa2 100644 --- a/src/yuzu/applets/web_browser.cpp +++ b/src/yuzu/applets/web_browser.cpp @@ -104,12 +104,12 @@ void QtWebBrowser::OpenPage(std::string_view url, std::function<void()> unpack_r  void QtWebBrowser::MainWindowUnpackRomFS() {      // Acquire the HLE mutex -    std::lock_guard<std::recursive_mutex> lock(HLE::g_hle_lock); +    std::lock_guard lock{HLE::g_hle_lock};      unpack_romfs_callback();  }  void QtWebBrowser::MainWindowFinishedBrowsing() {      // Acquire the HLE mutex -    std::lock_guard<std::recursive_mutex> lock(HLE::g_hle_lock); +    std::lock_guard lock{HLE::g_hle_lock};      finished_callback();  } diff --git a/src/yuzu/bootmanager.cpp b/src/yuzu/bootmanager.cpp index 05ad19e1d..7438fbc0a 100644 --- a/src/yuzu/bootmanager.cpp +++ b/src/yuzu/bootmanager.cpp @@ -67,7 +67,7 @@ void EmuThread::run() {              was_active = false;          } else { -            std::unique_lock<std::mutex> lock(running_mutex); +            std::unique_lock lock{running_mutex};              running_cv.wait(lock, [this] { return IsRunning() || exec_step || stop_run; });          }      } diff --git a/src/yuzu/bootmanager.h b/src/yuzu/bootmanager.h index 7226e690e..3183621bc 100644 --- a/src/yuzu/bootmanager.h +++ b/src/yuzu/bootmanager.h @@ -53,7 +53,7 @@ public:       * @note This function is thread-safe       */      void SetRunning(bool running) { -        std::unique_lock<std::mutex> lock(running_mutex); +        std::unique_lock lock{running_mutex};          this->running = running;          lock.unlock();          running_cv.notify_all(); diff --git a/src/yuzu/configuration/config.cpp b/src/yuzu/configuration/config.cpp index 4650f96a3..dead9f807 100644 --- a/src/yuzu/configuration/config.cpp +++ b/src/yuzu/configuration/config.cpp @@ -413,7 +413,6 @@ void Config::ReadValues() {      qt_config->beginGroup("System");      Settings::values.use_docked_mode = ReadSetting("use_docked_mode", false).toBool(); -    Settings::values.enable_nfc = ReadSetting("enable_nfc", true).toBool();      Settings::values.current_user =          std::clamp<int>(ReadSetting("current_user", 0).toInt(), 0, Service::Account::MAX_USERS - 1); @@ -675,7 +674,6 @@ void Config::SaveValues() {      qt_config->beginGroup("System");      WriteSetting("use_docked_mode", Settings::values.use_docked_mode, false); -    WriteSetting("enable_nfc", Settings::values.enable_nfc, true);      WriteSetting("current_user", Settings::values.current_user, 0);      WriteSetting("language_index", Settings::values.language_index, 1); diff --git a/src/yuzu/configuration/configure_general.cpp b/src/yuzu/configuration/configure_general.cpp index 4116b6cd7..389fcf667 100644 --- a/src/yuzu/configuration/configure_general.cpp +++ b/src/yuzu/configuration/configure_general.cpp @@ -33,7 +33,6 @@ void ConfigureGeneral::setConfiguration() {      ui->toggle_user_on_boot->setChecked(UISettings::values.select_user_on_boot);      ui->theme_combobox->setCurrentIndex(ui->theme_combobox->findData(UISettings::values.theme));      ui->use_cpu_jit->setChecked(Settings::values.use_cpu_jit); -    ui->enable_nfc->setChecked(Settings::values.enable_nfc);  }  void ConfigureGeneral::PopulateHotkeyList(const HotkeyRegistry& registry) { @@ -48,5 +47,4 @@ void ConfigureGeneral::applyConfiguration() {          ui->theme_combobox->itemData(ui->theme_combobox->currentIndex()).toString();      Settings::values.use_cpu_jit = ui->use_cpu_jit->isChecked(); -    Settings::values.enable_nfc = ui->enable_nfc->isChecked();  } diff --git a/src/yuzu/configuration/configure_general.ui b/src/yuzu/configuration/configure_general.ui index dff0ad5d0..01d1c0b8e 100644 --- a/src/yuzu/configuration/configure_general.ui +++ b/src/yuzu/configuration/configure_general.ui @@ -71,26 +71,6 @@        </widget>       </item>       <item> -      <widget class="QGroupBox" name="EmulationGroupBox"> -       <property name="title"> -        <string>Emulation</string> -       </property> -       <layout class="QHBoxLayout" name="EmulationHorizontalLayout"> -        <item> -         <layout class="QVBoxLayout" name="EmulationVerticalLayout"> -          <item> -           <widget class="QCheckBox" name="enable_nfc"> -            <property name="text"> -             <string>Enable NFC</string> -            </property> -           </widget> -          </item> -         </layout> -        </item> -       </layout> -      </widget> -     </item> -     <item>        <widget class="QGroupBox" name="theme_group_box">         <property name="title">          <string>Theme</string> diff --git a/src/yuzu_cmd/config.cpp b/src/yuzu_cmd/config.cpp index 32e78049c..f24cc77fe 100644 --- a/src/yuzu_cmd/config.cpp +++ b/src/yuzu_cmd/config.cpp @@ -319,7 +319,6 @@ void Config::ReadValues() {      // System      Settings::values.use_docked_mode = sdl2_config->GetBoolean("System", "use_docked_mode", false); -    Settings::values.enable_nfc = sdl2_config->GetBoolean("System", "enable_nfc", true);      const auto size = sdl2_config->GetInteger("System", "users_size", 0);      Settings::values.current_user = std::clamp<int>( | 
