diff options
29 files changed, 288 insertions, 144 deletions
| diff --git a/dist/qt_themes/default/icons/256x256/yuzu.png b/dist/qt_themes/default/icons/256x256/yuzu.pngBinary files differ index 1e501d8a6..bd5cf533f 100644 --- a/dist/qt_themes/default/icons/256x256/yuzu.png +++ b/dist/qt_themes/default/icons/256x256/yuzu.png diff --git a/externals/Vulkan-Headers b/externals/Vulkan-Headers -Subproject e005e1f8175d006adc3676b40ac3dd2212961a6 +Subproject 33d4dd987fc8fc6475ff9ca2b4f0c3cc6e79333 diff --git a/src/audio_core/command_generator.cpp b/src/audio_core/command_generator.cpp index ae4efafb6..ff20ed00f 100644 --- a/src/audio_core/command_generator.cpp +++ b/src/audio_core/command_generator.cpp @@ -129,17 +129,17 @@ s32 ToS32(float sample) {      return static_cast<s32>(rescaled_sample);  } -constexpr std::array<std::size_t, 20> REVERB_TAP_INDEX_1CH{0, 0, 0, 0, 0, 0, 0, 0, 0, 0, -                                                           0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; +constexpr std::array<u8, 20> REVERB_TAP_INDEX_1CH{0, 0, 0, 0, 0, 0, 0, 0, 0, 0, +                                                  0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; -constexpr std::array<std::size_t, 20> REVERB_TAP_INDEX_2CH{0, 0, 0, 1, 1, 1, 1, 0, 0, 0, -                                                           1, 1, 1, 0, 0, 0, 0, 1, 1, 1}; +constexpr std::array<u8, 20> REVERB_TAP_INDEX_2CH{0, 0, 0, 1, 1, 1, 1, 0, 0, 0, +                                                  1, 1, 1, 0, 0, 0, 0, 1, 1, 1}; -constexpr std::array<std::size_t, 20> REVERB_TAP_INDEX_4CH{0, 0, 0, 1, 1, 1, 1, 2, 2, 2, -                                                           1, 1, 1, 0, 0, 0, 0, 3, 3, 3}; +constexpr std::array<u8, 20> REVERB_TAP_INDEX_4CH{0, 0, 0, 1, 1, 1, 1, 2, 2, 2, +                                                  1, 1, 1, 0, 0, 0, 0, 3, 3, 3}; -constexpr std::array<std::size_t, 20> REVERB_TAP_INDEX_6CH{4, 0, 0, 1, 1, 1, 1, 2, 2, 2, -                                                           1, 1, 1, 0, 0, 0, 0, 3, 3, 3}; +constexpr std::array<u8, 20> REVERB_TAP_INDEX_6CH{4, 0, 0, 1, 1, 1, 1, 2, 2, 2, +                                                  1, 1, 1, 0, 0, 0, 0, 3, 3, 3};  template <std::size_t CHANNEL_COUNT>  void ApplyReverbGeneric( diff --git a/src/common/fs/path_util.cpp b/src/common/fs/path_util.cpp index 62318e70c..1074f2421 100644 --- a/src/common/fs/path_util.cpp +++ b/src/common/fs/path_util.cpp @@ -232,9 +232,7 @@ void SetYuzuPath(YuzuPath yuzu_path, const fs::path& new_path) {  fs::path GetExeDirectory() {      wchar_t exe_path[MAX_PATH]; -    GetModuleFileNameW(nullptr, exe_path, MAX_PATH); - -    if (!exe_path) { +    if (GetModuleFileNameW(nullptr, exe_path, MAX_PATH) == 0) {          LOG_ERROR(Common_Filesystem,                    "Failed to get the path to the executable of the current process");      } diff --git a/src/common/input.h b/src/common/input.h index 54fcb24b0..bb42aaacc 100644 --- a/src/common/input.h +++ b/src/common/input.h @@ -72,6 +72,7 @@ enum class PollingError {  enum class VibrationAmplificationType {      Linear,      Exponential, +    Test,  };  // Analog properties for calibration diff --git a/src/common/string_util.cpp b/src/common/string_util.cpp index 703aa5db8..7a495bc79 100644 --- a/src/common/string_util.cpp +++ b/src/common/string_util.cpp @@ -178,6 +178,10 @@ std::wstring UTF8ToUTF16W(const std::string& input) {  #endif +std::u16string U16StringFromBuffer(const u16* input, std::size_t length) { +    return std::u16string(reinterpret_cast<const char16_t*>(input), length); +} +  std::string StringFromFixedZeroTerminatedBuffer(std::string_view buffer, std::size_t max_len) {      std::size_t len = 0;      while (len < buffer.length() && len < max_len && buffer[len] != '\0') { diff --git a/src/common/string_util.h b/src/common/string_util.h index a33830aec..ce18a33cf 100644 --- a/src/common/string_util.h +++ b/src/common/string_util.h @@ -44,6 +44,8 @@ bool SplitPath(const std::string& full_path, std::string* _pPath, std::string* _  #endif +[[nodiscard]] std::u16string U16StringFromBuffer(const u16* input, std::size_t length); +  /**   * Compares the string defined by the range [`begin`, `end`) to the null-terminated C-string   * `other` for equality. diff --git a/src/core/hid/emulated_console.cpp b/src/core/hid/emulated_console.cpp index fd220ccb5..aac45907d 100644 --- a/src/core/hid/emulated_console.cpp +++ b/src/core/hid/emulated_console.cpp @@ -27,12 +27,19 @@ void EmulatedConsole::SetTouchParams() {          // We can't use mouse as touch if native mouse is enabled          touch_params[index++] = Common::ParamPackage{"engine:mouse,axis_x:10,axis_y:11,button:0"};      } -    touch_params[index++] = Common::ParamPackage{"engine:touch,axis_x:0,axis_y:1,button:0"}; -    touch_params[index++] = Common::ParamPackage{"engine:touch,axis_x:2,axis_y:3,button:1"}; + +    touch_params[index++] = +        Common::ParamPackage{"engine:touch,axis_x:0,axis_y:1,button:0,touch_id:0"}; +    touch_params[index++] = +        Common::ParamPackage{"engine:touch,axis_x:2,axis_y:3,button:1,touch_id:1"}; +    touch_params[index++] = +        Common::ParamPackage{"engine:touch,axis_x:4,axis_y:5,button:2,touch_id:2"}; +    touch_params[index++] = +        Common::ParamPackage{"engine:touch,axis_x:6,axis_y:7,button:3,touch_id:3"};      touch_params[index++] = -        Common::ParamPackage{"engine:cemuhookudp,axis_x:17,axis_y:18,button:65536"}; +        Common::ParamPackage{"engine:cemuhookudp,axis_x:17,axis_y:18,button:65536,touch_id:0"};      touch_params[index++] = -        Common::ParamPackage{"engine:cemuhookudp,axis_x:19,axis_y:20,button:131072"}; +        Common::ParamPackage{"engine:cemuhookudp,axis_x:19,axis_y:20,button:131072,touch_id:1"};      const auto button_index =          static_cast<u64>(Settings::values.touch_from_button_map_index.GetValue()); diff --git a/src/core/hid/emulated_controller.cpp b/src/core/hid/emulated_controller.cpp index ba1dcd171..bd2384515 100644 --- a/src/core/hid/emulated_controller.cpp +++ b/src/core/hid/emulated_controller.cpp @@ -884,18 +884,42 @@ bool EmulatedController::SetVibration(std::size_t device_index, VibrationValue v  }  bool EmulatedController::TestVibration(std::size_t device_index) { -    static constexpr VibrationValue test_vibration = { +    if (device_index >= output_devices.size()) { +        return false; +    } +    if (!output_devices[device_index]) { +        return false; +    } + +    const auto player_index = NpadIdTypeToIndex(npad_id_type); +    const auto& player = Settings::values.players.GetValue()[player_index]; + +    if (!player.vibration_enabled) { +        return false; +    } + +    const Common::Input::VibrationStatus test_vibration = {          .low_amplitude = 0.001f, -        .low_frequency = 160.0f, +        .low_frequency = DEFAULT_VIBRATION_VALUE.low_frequency,          .high_amplitude = 0.001f, -        .high_frequency = 320.0f, +        .high_frequency = DEFAULT_VIBRATION_VALUE.high_frequency, +        .type = Common::Input::VibrationAmplificationType::Test, +    }; + +    const Common::Input::VibrationStatus zero_vibration = { +        .low_amplitude = DEFAULT_VIBRATION_VALUE.low_amplitude, +        .low_frequency = DEFAULT_VIBRATION_VALUE.low_frequency, +        .high_amplitude = DEFAULT_VIBRATION_VALUE.high_amplitude, +        .high_frequency = DEFAULT_VIBRATION_VALUE.high_frequency, +        .type = Common::Input::VibrationAmplificationType::Test,      };      // Send a slight vibration to test for rumble support -    SetVibration(device_index, test_vibration); +    output_devices[device_index]->SetVibration(test_vibration);      // Stop any vibration and return the result -    return SetVibration(device_index, DEFAULT_VIBRATION_VALUE); +    return output_devices[device_index]->SetVibration(zero_vibration) == +           Common::Input::VibrationError::None;  }  bool EmulatedController::SetPollingMode(Common::Input::PollingMode polling_mode) { diff --git a/src/core/hid/input_converter.cpp b/src/core/hid/input_converter.cpp index 3c26260f3..18d9f042d 100644 --- a/src/core/hid/input_converter.cpp +++ b/src/core/hid/input_converter.cpp @@ -1,6 +1,7 @@  // SPDX-FileCopyrightText: Copyright 2021 yuzu Emulator Project  // SPDX-License-Identifier: GPL-2.0-or-later +#include <algorithm>  #include <random>  #include "common/input.h" @@ -196,6 +197,9 @@ Common::Input::TouchStatus TransformToTouch(const Common::Input::CallbackStatus&      x = std::clamp(x, 0.0f, 1.0f);      y = std::clamp(y, 0.0f, 1.0f); +    // Limit id to maximum number of fingers +    status.id = std::clamp(status.id, 0, 16); +      if (status.pressed.inverted) {          status.pressed.value = !status.pressed.value;      } diff --git a/src/core/hle/service/hid/controllers/touchscreen.cpp b/src/core/hle/service/hid/controllers/touchscreen.cpp index 108ce5a41..1da8d3eb0 100644 --- a/src/core/hle/service/hid/controllers/touchscreen.cpp +++ b/src/core/hle/service/hid/controllers/touchscreen.cpp @@ -44,7 +44,6 @@ void Controller_Touchscreen::OnUpdate(const Core::Timing::CoreTiming& core_timin      for (std::size_t id = 0; id < MAX_FINGERS; id++) {          const auto& current_touch = touch_status[id];          auto& finger = fingers[id]; -        finger.position = current_touch.position;          finger.id = current_touch.id;          if (finger.attribute.start_touch) { @@ -61,13 +60,18 @@ void Controller_Touchscreen::OnUpdate(const Core::Timing::CoreTiming& core_timin          if (!finger.pressed && current_touch.pressed) {              finger.attribute.start_touch.Assign(1);              finger.pressed = true; +            finger.position = current_touch.position;              continue;          }          if (finger.pressed && !current_touch.pressed) {              finger.attribute.raw = 0;              finger.attribute.end_touch.Assign(1); +            continue;          } + +        // Only update position if touch is not on a special frame +        finger.position = current_touch.position;      }      std::array<Core::HID::TouchFinger, MAX_FINGERS> active_fingers; diff --git a/src/core/hle/service/time/time_zone_manager.cpp b/src/core/hle/service/time/time_zone_manager.cpp index 449a5ac96..eeec34436 100644 --- a/src/core/hle/service/time/time_zone_manager.cpp +++ b/src/core/hle/service/time/time_zone_manager.cpp @@ -110,10 +110,9 @@ static constexpr s64 GetLeapDaysFromYear(s64 year) {      }  } -static constexpr int GetMonthLength(bool is_leap_year, int month) { -    constexpr std::array<int, 12> month_lengths{31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31}; -    constexpr std::array<int, 12> month_lengths_leap{31, 29, 31, 30, 31, 30, -                                                     31, 31, 30, 31, 30, 31}; +static constexpr s8 GetMonthLength(bool is_leap_year, int month) { +    constexpr std::array<s8, 12> month_lengths{31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31}; +    constexpr std::array<s8, 12> month_lengths_leap{31, 29, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31};      return is_leap_year ? month_lengths_leap[month] : month_lengths[month];  } diff --git a/src/input_common/drivers/sdl_driver.cpp b/src/input_common/drivers/sdl_driver.cpp index a5c63e74a..1a14ef10b 100644 --- a/src/input_common/drivers/sdl_driver.cpp +++ b/src/input_common/drivers/sdl_driver.cpp @@ -434,6 +434,7 @@ SDLDriver::SDLDriver(std::string input_engine_) : InputEngine(std::move(input_en              using namespace std::chrono_literals;              while (initialized) {                  SDL_PumpEvents(); +                SendVibrations();                  std::this_thread::sleep_for(1ms);              }          }); @@ -531,13 +532,31 @@ Common::Input::VibrationError SDLDriver::SetRumble(          .type = Common::Input::VibrationAmplificationType::Exponential,      }; -    if (!joystick->RumblePlay(new_vibration)) { -        return Common::Input::VibrationError::Unknown; +    if (vibration.type == Common::Input::VibrationAmplificationType::Test) { +        if (!joystick->RumblePlay(new_vibration)) { +            return Common::Input::VibrationError::Unknown; +        } +        return Common::Input::VibrationError::None;      } +    vibration_queue.Push(VibrationRequest{ +        .identifier = identifier, +        .vibration = new_vibration, +    }); +      return Common::Input::VibrationError::None;  } +void SDLDriver::SendVibrations() { +    while (!vibration_queue.Empty()) { +        VibrationRequest request; +        vibration_queue.Pop(request); +        const auto joystick = GetSDLJoystickByGUID(request.identifier.guid.RawString(), +                                                   static_cast<int>(request.identifier.port)); +        joystick->RumblePlay(request.vibration); +    } +} +  Common::ParamPackage SDLDriver::BuildAnalogParamPackageForButton(int port, std::string guid,                                                                   s32 axis, float value) const {      Common::ParamPackage params{}; diff --git a/src/input_common/drivers/sdl_driver.h b/src/input_common/drivers/sdl_driver.h index dcd0d1e64..c82632506 100644 --- a/src/input_common/drivers/sdl_driver.h +++ b/src/input_common/drivers/sdl_driver.h @@ -12,6 +12,7 @@  #include <SDL.h>  #include "common/common_types.h" +#include "common/threadsafe_queue.h"  #include "input_common/input_engine.h"  union SDL_Event; @@ -64,12 +65,20 @@ public:          const PadIdentifier& identifier, const Common::Input::VibrationStatus& vibration) override;  private: +    struct VibrationRequest { +        PadIdentifier identifier; +        Common::Input::VibrationStatus vibration; +    }; +      void InitJoystick(int joystick_index);      void CloseJoystick(SDL_Joystick* sdl_joystick);      /// Needs to be called before SDL_QuitSubSystem.      void CloseJoysticks(); +    /// Takes all vibrations from the queue and sends the command to the controller +    void SendVibrations(); +      Common::ParamPackage BuildAnalogParamPackageForButton(int port, std::string guid, s32 axis,                                                            float value = 0.1f) const;      Common::ParamPackage BuildButtonParamPackageForButton(int port, std::string guid, @@ -107,6 +116,9 @@ private:      /// Returns true if the button is on the left joycon      bool IsButtonOnLeftSide(Settings::NativeButton::Values button) const; +    /// Queue of vibration request to controllers +    Common::SPSCQueue<VibrationRequest> vibration_queue; +      /// Map of GUID of a list of corresponding virtual Joysticks      std::unordered_map<std::string, std::vector<std::shared_ptr<SDLJoystick>>> joystick_map;      std::mutex joystick_map_mutex; diff --git a/src/input_common/drivers/touch_screen.cpp b/src/input_common/drivers/touch_screen.cpp index 8acbe4584..1753e0893 100644 --- a/src/input_common/drivers/touch_screen.cpp +++ b/src/input_common/drivers/touch_screen.cpp @@ -14,38 +14,93 @@ constexpr PadIdentifier identifier = {  TouchScreen::TouchScreen(std::string input_engine_) : InputEngine(std::move(input_engine_)) {      PreSetController(identifier); +    ReleaseAllTouch();  } -void TouchScreen::TouchMoved(float x, float y, std::size_t finger) { -    if (finger >= 16) { +void TouchScreen::TouchMoved(float x, float y, std::size_t finger_id) { +    const auto index = GetIndexFromFingerId(finger_id); +    if (!index) { +        // Touch doesn't exist handle it as a new one +        TouchPressed(x, y, finger_id);          return;      } -    TouchPressed(x, y, finger); +    const auto i = index.value(); +    fingers[i].is_active = true; +    SetButton(identifier, static_cast<int>(i), true); +    SetAxis(identifier, static_cast<int>(i * 2), x); +    SetAxis(identifier, static_cast<int>(i * 2 + 1), y);  } -void TouchScreen::TouchPressed(float x, float y, std::size_t finger) { -    if (finger >= 16) { +void TouchScreen::TouchPressed(float x, float y, std::size_t finger_id) { +    if (GetIndexFromFingerId(finger_id)) { +        // Touch already exist. Just update the data +        TouchMoved(x, y, finger_id);          return;      } -    SetButton(identifier, static_cast<int>(finger), true); -    SetAxis(identifier, static_cast<int>(finger * 2), x); -    SetAxis(identifier, static_cast<int>(finger * 2 + 1), y); +    const auto index = GetNextFreeIndex(); +    if (!index) { +        // No free entries. Ignore input +        return; +    } +    const auto i = index.value(); +    fingers[i].is_enabled = true; +    fingers[i].finger_id = finger_id; +    TouchMoved(x, y, finger_id);  } -void TouchScreen::TouchReleased(std::size_t finger) { -    if (finger >= 16) { +void TouchScreen::TouchReleased(std::size_t finger_id) { +    const auto index = GetIndexFromFingerId(finger_id); +    if (!index) {          return;      } -    SetButton(identifier, static_cast<int>(finger), false); -    SetAxis(identifier, static_cast<int>(finger * 2), 0.0f); -    SetAxis(identifier, static_cast<int>(finger * 2 + 1), 0.0f); +    const auto i = index.value(); +    fingers[i].is_enabled = false; +    SetButton(identifier, static_cast<int>(i), false); +    SetAxis(identifier, static_cast<int>(i * 2), 0.0f); +    SetAxis(identifier, static_cast<int>(i * 2 + 1), 0.0f); +} + +std::optional<std::size_t> TouchScreen::GetIndexFromFingerId(std::size_t finger_id) const { +    for (std::size_t index = 0; index < MAX_FINGER_COUNT; ++index) { +        const auto& finger = fingers[index]; +        if (!finger.is_enabled) { +            continue; +        } +        if (finger.finger_id == finger_id) { +            return index; +        } +    } +    return std::nullopt; +} + +std::optional<std::size_t> TouchScreen::GetNextFreeIndex() const { +    for (std::size_t index = 0; index < MAX_FINGER_COUNT; ++index) { +        if (!fingers[index].is_enabled) { +            return index; +        } +    } +    return std::nullopt; +} + +void TouchScreen::ClearActiveFlag() { +    for (auto& finger : fingers) { +        finger.is_active = false; +    } +} + +void TouchScreen::ReleaseInactiveTouch() { +    for (const auto& finger : fingers) { +        if (!finger.is_active) { +            TouchReleased(finger.finger_id); +        } +    }  }  void TouchScreen::ReleaseAllTouch() { -    for (int index = 0; index < 16; ++index) { -        SetButton(identifier, index, false); -        SetAxis(identifier, index * 2, 0.0f); -        SetAxis(identifier, index * 2 + 1, 0.0f); +    for (const auto& finger : fingers) { +        if (finger.is_enabled) { +            TouchReleased(finger.finger_id); +        }      }  } diff --git a/src/input_common/drivers/touch_screen.h b/src/input_common/drivers/touch_screen.h index 193478ead..f46036ffd 100644 --- a/src/input_common/drivers/touch_screen.h +++ b/src/input_common/drivers/touch_screen.h @@ -3,41 +3,65 @@  #pragma once +#include <optional> +  #include "input_common/input_engine.h"  namespace InputCommon {  /** - * A button device factory representing a keyboard. It receives keyboard events and forward them - * to all button devices it created. + * A touch device factory representing a touch screen. It receives touch events and forward them + * to all touch devices it created.   */  class TouchScreen final : public InputEngine {  public:      explicit TouchScreen(std::string input_engine_);      /** -     * Signals that mouse has moved. -     * @param x the x-coordinate of the cursor -     * @param y the y-coordinate of the cursor -     * @param center_x the x-coordinate of the middle of the screen -     * @param center_y the y-coordinate of the middle of the screen +     * Signals that touch has moved and marks this touch point as active +     * @param x new horizontal position +     * @param y new vertical position +     * @param finger_id of the touch point to be updated       */ -    void TouchMoved(float x, float y, std::size_t finger); +    void TouchMoved(float x, float y, std::size_t finger_id);      /** -     * Sets the status of all buttons bound with the key to pressed -     * @param key_code the code of the key to press +     * Signals and creates a new touch point with this finger id +     * @param x starting horizontal position +     * @param y starting vertical position +     * @param finger_id to be assigned to the new touch point       */ -    void TouchPressed(float x, float y, std::size_t finger); +    void TouchPressed(float x, float y, std::size_t finger_id);      /** -     * Sets the status of all buttons bound with the key to released -     * @param key_code the code of the key to release +     * Signals and resets the touch point related to the this finger id +     * @param finger_id to be released       */ -    void TouchReleased(std::size_t finger); +    void TouchReleased(std::size_t finger_id); + +    /// Resets the active flag for each touch point +    void ClearActiveFlag(); + +    /// Releases all touch that haven't been marked as active +    void ReleaseInactiveTouch();      /// Resets all inputs to their initial value      void ReleaseAllTouch(); + +private: +    static constexpr std::size_t MAX_FINGER_COUNT = 16; + +    struct TouchStatus { +        std::size_t finger_id{}; +        bool is_enabled{}; +        bool is_active{}; +    }; + +    std::optional<std::size_t> GetIndexFromFingerId(std::size_t finger_id) const; + +    std::optional<std::size_t> GetNextFreeIndex() const; + +    std::array<TouchStatus, MAX_FINGER_COUNT> fingers{};  };  } // namespace InputCommon diff --git a/src/video_core/command_classes/codecs/vp9.cpp b/src/video_core/command_classes/codecs/vp9.cpp index a95618913..c01431441 100644 --- a/src/video_core/command_classes/codecs/vp9.cpp +++ b/src/video_core/command_classes/codecs/vp9.cpp @@ -153,7 +153,7 @@ constexpr Vp9EntropyProbs default_probs{      .high_precision{128, 128},  }; -constexpr std::array<s32, 256> norm_lut{ +constexpr std::array<u8, 256> norm_lut{      0, 7, 6, 6, 5, 5, 5, 5, 4, 4, 4, 4, 4, 4, 4, 4, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3,      2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2,      1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, @@ -164,7 +164,7 @@ constexpr std::array<s32, 256> norm_lut{      0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,  }; -constexpr std::array<s32, 254> map_lut{ +constexpr std::array<u8, 254> map_lut{      20,  21,  22,  23,  24,  25,  0,   26,  27,  28,  29,  30,  31,  32,  33,  34,  35,  36,  37,      1,   38,  39,  40,  41,  42,  43,  44,  45,  46,  47,  48,  49,  2,   50,  51,  52,  53,  54,      55,  56,  57,  58,  59,  60,  61,  3,   62,  63,  64,  65,  66,  67,  68,  69,  70,  71,  72, @@ -232,7 +232,7 @@ constexpr std::array<s32, 254> map_lut{              std::max(0, RecenterNonNeg(0xff - 1 - new_prob, 0xff - 1 - old_prob) - 1));      } -    return map_lut[index]; +    return static_cast<s32>(map_lut[index]);  }  } // Anonymous namespace @@ -819,7 +819,7 @@ void VpxRangeEncoder::Write(bool bit, s32 probability) {          local_range = range - split;      } -    s32 shift = norm_lut[local_range]; +    s32 shift = static_cast<s32>(norm_lut[local_range]);      local_range <<= shift;      count += shift; diff --git a/src/video_core/renderer_vulkan/vk_rasterizer.cpp b/src/video_core/renderer_vulkan/vk_rasterizer.cpp index fd27581ce..ce6c853c1 100644 --- a/src/video_core/renderer_vulkan/vk_rasterizer.cpp +++ b/src/video_core/renderer_vulkan/vk_rasterizer.cpp @@ -784,8 +784,8 @@ void RasterizerVulkan::UpdateStencilFaces(Tegra::Engines::Maxwell3D::Regs& regs)              });      } else {          // Front face defines both faces -        scheduler.Record([ref = regs.stencil_back_func_ref, write_mask = regs.stencil_back_mask, -                          test_mask = regs.stencil_back_func_mask](vk::CommandBuffer cmdbuf) { +        scheduler.Record([ref = regs.stencil_front_func_ref, write_mask = regs.stencil_front_mask, +                          test_mask = regs.stencil_front_func_mask](vk::CommandBuffer cmdbuf) {              cmdbuf.SetStencilReference(VK_STENCIL_FACE_FRONT_AND_BACK, ref);              cmdbuf.SetStencilWriteMask(VK_STENCIL_FACE_FRONT_AND_BACK, write_mask);              cmdbuf.SetStencilCompareMask(VK_STENCIL_FACE_FRONT_AND_BACK, test_mask); diff --git a/src/video_core/surface.h b/src/video_core/surface.h index 86fea61ae..75e055592 100644 --- a/src/video_core/surface.h +++ b/src/video_core/surface.h @@ -147,7 +147,7 @@ enum class SurfaceTarget {      TextureCubeArray,  }; -constexpr std::array<u32, MaxPixelFormat> BLOCK_WIDTH_TABLE = {{ +constexpr std::array<u8, MaxPixelFormat> BLOCK_WIDTH_TABLE = {{      1,  // A8B8G8R8_UNORM      1,  // A8B8G8R8_SNORM      1,  // A8B8G8R8_SINT @@ -249,7 +249,7 @@ constexpr u32 DefaultBlockWidth(PixelFormat format) {      return BLOCK_WIDTH_TABLE[static_cast<std::size_t>(format)];  } -constexpr std::array<u32, MaxPixelFormat> BLOCK_HEIGHT_TABLE = {{ +constexpr std::array<u8, MaxPixelFormat> BLOCK_HEIGHT_TABLE = {{      1,  // A8B8G8R8_UNORM      1,  // A8B8G8R8_SNORM      1,  // A8B8G8R8_SINT @@ -351,7 +351,7 @@ constexpr u32 DefaultBlockHeight(PixelFormat format) {      return BLOCK_HEIGHT_TABLE[static_cast<std::size_t>(format)];  } -constexpr std::array<u32, MaxPixelFormat> BITS_PER_BLOCK_TABLE = {{ +constexpr std::array<u8, MaxPixelFormat> BITS_PER_BLOCK_TABLE = {{      32,  // A8B8G8R8_UNORM      32,  // A8B8G8R8_SNORM      32,  // A8B8G8R8_SINT diff --git a/src/video_core/vulkan_common/vulkan_device.cpp b/src/video_core/vulkan_common/vulkan_device.cpp index 7b2ca8046..b3a77e07f 100644 --- a/src/video_core/vulkan_common/vulkan_device.cpp +++ b/src/video_core/vulkan_common/vulkan_device.cpp @@ -566,7 +566,7 @@ Device::Device(VkInstance instance_, vk::PhysicalDevice physical_, VkSurfaceKHR      }      VkPhysicalDeviceWorkgroupMemoryExplicitLayoutFeaturesKHR workgroup_layout; -    if (khr_workgroup_memory_explicit_layout) { +    if (khr_workgroup_memory_explicit_layout && is_shader_int16_supported) {          workgroup_layout = {              .sType =                  VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_WORKGROUP_MEMORY_EXPLICIT_LAYOUT_FEATURES_KHR, @@ -577,6 +577,11 @@ Device::Device(VkInstance instance_, vk::PhysicalDevice physical_, VkSurfaceKHR              .workgroupMemoryExplicitLayout16BitAccess = VK_TRUE,          };          SetNext(next, workgroup_layout); +    } else if (khr_workgroup_memory_explicit_layout) { +        // TODO(lat9nq): Find a proper fix for this +        LOG_WARNING(Render_Vulkan, "Disabling VK_KHR_workgroup_memory_explicit_layout due to a " +                                   "yuzu bug when host driver does not support 16-bit integers"); +        khr_workgroup_memory_explicit_layout = false;      }      VkPhysicalDevicePipelineExecutablePropertiesFeaturesKHR executable_properties; @@ -664,6 +669,17 @@ Device::Device(VkInstance instance_, vk::PhysicalDevice physical_, VkSurfaceKHR      const bool is_amd =          driver_id == VK_DRIVER_ID_AMD_PROPRIETARY || driver_id == VK_DRIVER_ID_AMD_OPEN_SOURCE;      if (is_amd) { +        // TODO(lat9nq): Add an upper bound when AMD fixes their VK_KHR_push_descriptor +        const bool has_broken_push_descriptor = VK_VERSION_MAJOR(properties.driverVersion) == 2 && +                                                VK_VERSION_MINOR(properties.driverVersion) == 0 && +                                                VK_VERSION_PATCH(properties.driverVersion) >= 226; +        if (khr_push_descriptor && has_broken_push_descriptor) { +            LOG_WARNING( +                Render_Vulkan, +                "Disabling AMD driver 2.0.226 and later from broken VK_KHR_push_descriptor"); +            khr_push_descriptor = false; +        } +          // AMD drivers need a higher amount of Sets per Pool in certain circunstances like in XC2.          sets_per_pool = 96;          // Disable VK_IMAGE_CREATE_CUBE_COMPATIBLE_BIT on AMD GCN4 and lower as it is broken. diff --git a/src/video_core/vulkan_common/vulkan_wrapper.cpp b/src/video_core/vulkan_common/vulkan_wrapper.cpp index b1ea6075a..2ad98dcfe 100644 --- a/src/video_core/vulkan_common/vulkan_wrapper.cpp +++ b/src/video_core/vulkan_common/vulkan_wrapper.cpp @@ -325,6 +325,8 @@ const char* ToString(VkResult result) noexcept {          return "VK_PIPELINE_COMPILE_REQUIRED_EXT";      case VkResult::VK_RESULT_MAX_ENUM:          return "VK_RESULT_MAX_ENUM"; +    case VkResult::VK_ERROR_COMPRESSION_EXHAUSTED_EXT: +        return "VK_ERROR_COMPRESSION_EXHAUSTED_EXT";      }      return "Unknown";  } diff --git a/src/yuzu/CMakeLists.txt b/src/yuzu/CMakeLists.txt index 2ee21f751..404acdd05 100644 --- a/src/yuzu/CMakeLists.txt +++ b/src/yuzu/CMakeLists.txt @@ -240,7 +240,7 @@ elseif(WIN32)      if(MSVC)          set_target_properties(yuzu PROPERTIES LINK_FLAGS_RELEASE "/SUBSYSTEM:WINDOWS")      elseif(MINGW) -        set_target_properties(yuzu PROPERTIES LINK_FLAGS_RELEASE "-mwindows") +        set_target_properties(yuzu PROPERTIES LINK_FLAGS_RELEASE "-Wl,--subsystem,windows")      endif()  endif() diff --git a/src/yuzu/about_dialog.cpp b/src/yuzu/about_dialog.cpp index cbcef7b45..eeff54359 100644 --- a/src/yuzu/about_dialog.cpp +++ b/src/yuzu/about_dialog.cpp @@ -19,7 +19,11 @@ AboutDialog::AboutDialog(QWidget* parent)      const auto yuzu_build_version = override_build.empty() ? yuzu_build : override_build;      ui->setupUi(this); -    ui->labelLogo->setPixmap(QIcon::fromTheme(QStringLiteral("yuzu")).pixmap(200)); +    // Try and request the icon from Qt theme (Linux?) +    const QIcon yuzu_logo = QIcon::fromTheme(QStringLiteral("org.yuzu_emu.yuzu")); +    if (!yuzu_logo.isNull()) { +        ui->labelLogo->setPixmap(yuzu_logo.pixmap(200)); +    }      ui->labelBuildInfo->setText(          ui->labelBuildInfo->text().arg(QString::fromStdString(yuzu_build_version),                                         QString::fromUtf8(Common::g_build_date).left(10))); diff --git a/src/yuzu/aboutdialog.ui b/src/yuzu/aboutdialog.ui index 2f7ddc7f3..1dd7b74bf 100644 --- a/src/yuzu/aboutdialog.ui +++ b/src/yuzu/aboutdialog.ui @@ -26,8 +26,20 @@             <verstretch>0</verstretch>            </sizepolicy>           </property> +         <property name="maximumSize"> +          <size> +           <width>200</width> +           <height>200</height> +          </size> +         </property>           <property name="text"> -          <string><html><head/><body><p><img src=":/icons/yuzu.png"/></p></body></html></string> +          <string/> +         </property> +         <property name="pixmap"> +          <pixmap resource="../../dist/qt_themes/default/default.qrc">:/icons/default/256x256/yuzu.png</pixmap> +         </property> +         <property name="scaledContents"> +          <bool>true</bool>           </property>          </widget>         </item> @@ -152,7 +164,7 @@ p, li { white-space: pre-wrap; }    </layout>   </widget>   <resources> -  <include location="../../dist/icons/icons.qrc"/> +  <include location="../../dist/qt_themes_default/default/default.qrc"/>   </resources>   <connections>    <connection> diff --git a/src/yuzu/applets/qt_software_keyboard.cpp b/src/yuzu/applets/qt_software_keyboard.cpp index d3cf0b43b..e8b217d90 100644 --- a/src/yuzu/applets/qt_software_keyboard.cpp +++ b/src/yuzu/applets/qt_software_keyboard.cpp @@ -411,11 +411,11 @@ void QtSoftwareKeyboardDialog::ShowTextCheckDialog(              break;          } -        auto text = ui->topOSK->currentIndex() == 1 -                        ? ui->text_edit_osk->toPlainText().toStdU16String() -                        : ui->line_edit_osk->text().toStdU16String(); +        const auto text = ui->topOSK->currentIndex() == 1 ? ui->text_edit_osk->toPlainText() +                                                          : ui->line_edit_osk->text(); +        auto text_str = Common::U16StringFromBuffer(text.utf16(), text.size()); -        emit SubmitNormalText(SwkbdResult::Ok, std::move(text), true); +        emit SubmitNormalText(SwkbdResult::Ok, std::move(text_str), true);          break;      }      } @@ -562,7 +562,7 @@ void QtSoftwareKeyboardDialog::keyPressEvent(QKeyEvent* event) {          return;      } -    InlineTextInsertString(entered_text.toStdU16String()); +    InlineTextInsertString(Common::U16StringFromBuffer(entered_text.utf16(), entered_text.size()));  }  void QtSoftwareKeyboardDialog::MoveAndResizeWindow(QPoint pos, QSize size) { @@ -1119,11 +1119,11 @@ void QtSoftwareKeyboardDialog::NormalKeyboardButtonClicked(QPushButton* button)      }      if (button == ui->button_ok || button == ui->button_ok_shift || button == ui->button_ok_num) { -        auto text = ui->topOSK->currentIndex() == 1 -                        ? ui->text_edit_osk->toPlainText().toStdU16String() -                        : ui->line_edit_osk->text().toStdU16String(); +        const auto text = ui->topOSK->currentIndex() == 1 ? ui->text_edit_osk->toPlainText() +                                                          : ui->line_edit_osk->text(); +        auto text_str = Common::U16StringFromBuffer(text.utf16(), text.size()); -        emit SubmitNormalText(SwkbdResult::Ok, std::move(text)); +        emit SubmitNormalText(SwkbdResult::Ok, std::move(text_str));          return;      } @@ -1189,7 +1189,8 @@ void QtSoftwareKeyboardDialog::InlineKeyboardButtonClicked(QPushButton* button)          return;      } -    InlineTextInsertString(button->text().toStdU16String()); +    const auto button_text = button->text(); +    InlineTextInsertString(Common::U16StringFromBuffer(button_text.utf16(), button_text.size()));      // Revert the keyboard to lowercase if the shift key is active.      if (bottom_osk_index == BottomOSKIndex::UpperCase && !caps_lock_enabled) { @@ -1282,11 +1283,11 @@ void QtSoftwareKeyboardDialog::TranslateButtonPress(Core::HID::NpadButton button          if (is_inline) {              emit SubmitInlineText(SwkbdReplyType::DecidedCancel, current_text, cursor_position);          } else { -            auto text = ui->topOSK->currentIndex() == 1 -                            ? ui->text_edit_osk->toPlainText().toStdU16String() -                            : ui->line_edit_osk->text().toStdU16String(); +            const auto text = ui->topOSK->currentIndex() == 1 ? ui->text_edit_osk->toPlainText() +                                                              : ui->line_edit_osk->text(); +            auto text_str = Common::U16StringFromBuffer(text.utf16(), text.size()); -            emit SubmitNormalText(SwkbdResult::Cancel, std::move(text)); +            emit SubmitNormalText(SwkbdResult::Cancel, std::move(text_str));          }          break;      case Core::HID::NpadButton::Y: diff --git a/src/yuzu/bootmanager.cpp b/src/yuzu/bootmanager.cpp index a1b819ae0..8f0a6bbb8 100644 --- a/src/yuzu/bootmanager.cpp +++ b/src/yuzu/bootmanager.cpp @@ -772,65 +772,25 @@ void GRenderWindow::wheelEvent(QWheelEvent* event) {  void GRenderWindow::TouchBeginEvent(const QTouchEvent* event) {      QList<QTouchEvent::TouchPoint> touch_points = event->touchPoints();      for (const auto& touch_point : touch_points) { -        if (!TouchUpdate(touch_point)) { -            TouchStart(touch_point); -        } +        const auto [x, y] = ScaleTouch(touch_point.pos()); +        const auto [touch_x, touch_y] = MapToTouchScreen(x, y); +        input_subsystem->GetTouchScreen()->TouchPressed(touch_x, touch_y, touch_point.id());      }  }  void GRenderWindow::TouchUpdateEvent(const QTouchEvent* event) {      QList<QTouchEvent::TouchPoint> touch_points = event->touchPoints(); +    input_subsystem->GetTouchScreen()->ClearActiveFlag();      for (const auto& touch_point : touch_points) { -        if (!TouchUpdate(touch_point)) { -            TouchStart(touch_point); -        } -    } -    // Release all inactive points -    for (std::size_t id = 0; id < touch_ids.size(); ++id) { -        if (!TouchExist(touch_ids[id], touch_points)) { -            touch_ids[id] = 0; -            input_subsystem->GetTouchScreen()->TouchReleased(id); -        } +        const auto [x, y] = ScaleTouch(touch_point.pos()); +        const auto [touch_x, touch_y] = MapToTouchScreen(x, y); +        input_subsystem->GetTouchScreen()->TouchMoved(touch_x, touch_y, touch_point.id());      } +    input_subsystem->GetTouchScreen()->ReleaseInactiveTouch();  }  void GRenderWindow::TouchEndEvent() { -    for (std::size_t id = 0; id < touch_ids.size(); ++id) { -        if (touch_ids[id] != 0) { -            touch_ids[id] = 0; -            input_subsystem->GetTouchScreen()->TouchReleased(id); -        } -    } -} - -void GRenderWindow::TouchStart(const QTouchEvent::TouchPoint& touch_point) { -    for (std::size_t id = 0; id < touch_ids.size(); ++id) { -        if (touch_ids[id] == 0) { -            touch_ids[id] = touch_point.id() + 1; -            const auto [x, y] = ScaleTouch(touch_point.pos()); -            const auto [touch_x, touch_y] = MapToTouchScreen(x, y); -            input_subsystem->GetTouchScreen()->TouchPressed(touch_x, touch_y, id); -        } -    } -} - -bool GRenderWindow::TouchUpdate(const QTouchEvent::TouchPoint& touch_point) { -    for (std::size_t id = 0; id < touch_ids.size(); ++id) { -        if (touch_ids[id] == static_cast<std::size_t>(touch_point.id() + 1)) { -            const auto [x, y] = ScaleTouch(touch_point.pos()); -            const auto [touch_x, touch_y] = MapToTouchScreen(x, y); -            input_subsystem->GetTouchScreen()->TouchMoved(touch_x, touch_y, id); -            return true; -        } -    } -    return false; -} - -bool GRenderWindow::TouchExist(std::size_t id, -                               const QList<QTouchEvent::TouchPoint>& touch_points) const { -    return std::any_of(touch_points.begin(), touch_points.end(), [id](const auto& point) { -        return id == static_cast<std::size_t>(point.id() + 1); -    }); +    input_subsystem->GetTouchScreen()->ReleaseAllTouch();  }  bool GRenderWindow::event(QEvent* event) { diff --git a/src/yuzu/bootmanager.h b/src/yuzu/bootmanager.h index 4b0ce0293..841816564 100644 --- a/src/yuzu/bootmanager.h +++ b/src/yuzu/bootmanager.h @@ -217,10 +217,6 @@ private:      void TouchUpdateEvent(const QTouchEvent* event);      void TouchEndEvent(); -    void TouchStart(const QTouchEvent::TouchPoint& touch_point); -    bool TouchUpdate(const QTouchEvent::TouchPoint& touch_point); -    bool TouchExist(std::size_t id, const QList<QTouchEvent::TouchPoint>& touch_points) const; -      void OnMinimalClientAreaChangeRequest(std::pair<u32, u32> minimal_size) override;      bool InitializeOpenGL(); @@ -246,8 +242,6 @@ private:      bool first_frame = false;      InputCommon::TasInput::TasState last_tas_state; -    std::array<std::size_t, 16> touch_ids{}; -      Core::System& system;  protected: diff --git a/src/yuzu/main.cpp b/src/yuzu/main.cpp index f607f464a..f4a9a7171 100644 --- a/src/yuzu/main.cpp +++ b/src/yuzu/main.cpp @@ -1401,7 +1401,8 @@ void GMainWindow::BootGame(const QString& filename, u64 program_id, std::size_t      if (loader != nullptr && loader->ReadProgramId(title_id) == Loader::ResultStatus::Success &&          type == StartGameType::Normal) {          // Load per game settings -        const auto file_path = std::filesystem::path{filename.toStdU16String()}; +        const auto file_path = +            std::filesystem::path{Common::U16StringFromBuffer(filename.utf16(), filename.size())};          const auto config_file_name = title_id == 0                                            ? Common::FS::PathToUTF8String(file_path.filename())                                            : fmt::format("{:016X}", title_id); @@ -1482,7 +1483,8 @@ void GMainWindow::BootGame(const QString& filename, u64 program_id, std::size_t      }      if (res != Loader::ResultStatus::Success || title_name.empty()) {          title_name = Common::FS::PathToUTF8String( -            std::filesystem::path{filename.toStdU16String()}.filename()); +            std::filesystem::path{Common::U16StringFromBuffer(filename.utf16(), filename.size())} +                .filename());      }      const bool is_64bit = system->Kernel().CurrentProcess()->Is64BitProcess();      const auto instruction_set_suffix = is_64bit ? tr("(64-bit)") : tr("(32-bit)"); diff --git a/src/yuzu_cmd/emu_window/emu_window_sdl2.cpp b/src/yuzu_cmd/emu_window/emu_window_sdl2.cpp index ae2e62dc5..71c413e64 100644 --- a/src/yuzu_cmd/emu_window/emu_window_sdl2.cpp +++ b/src/yuzu_cmd/emu_window/emu_window_sdl2.cpp @@ -93,7 +93,7 @@ void EmuWindow_SDL2::OnFingerMotion(float x, float y, std::size_t id) {  }  void EmuWindow_SDL2::OnFingerUp() { -    input_subsystem->GetTouchScreen()->TouchReleased(0); +    input_subsystem->GetTouchScreen()->ReleaseAllTouch();  }  void EmuWindow_SDL2::OnKeyEvent(int key, u8 state) { | 
