diff options
Diffstat (limited to 'src')
| -rw-r--r-- | src/core/hle/service/am/am.cpp | 202 | ||||
| -rw-r--r-- | src/core/hle/service/am/am.h | 1 | ||||
| -rw-r--r-- | src/core/hle/service/nvdrv/devices/nvhost_gpu.h | 47 | ||||
| -rw-r--r-- | src/video_core/engines/maxwell_3d.h | 40 | ||||
| -rw-r--r-- | src/video_core/renderer_opengl/gl_rasterizer.cpp | 4 | ||||
| -rw-r--r-- | src/video_core/renderer_opengl/gl_rasterizer_cache.cpp | 3 | ||||
| -rw-r--r-- | src/yuzu/main.cpp | 26 | ||||
| -rw-r--r-- | src/yuzu/main.h | 1 | ||||
| -rw-r--r-- | src/yuzu_cmd/emu_window/emu_window_sdl2.cpp | 23 | ||||
| -rw-r--r-- | src/yuzu_cmd/emu_window/emu_window_sdl2.h | 3 | 
10 files changed, 265 insertions, 85 deletions
diff --git a/src/core/hle/service/am/am.cpp b/src/core/hle/service/am/am.cpp index 6b1d6bf97..12954556d 100644 --- a/src/core/hle/service/am/am.cpp +++ b/src/core/hle/service/am/am.cpp @@ -3,6 +3,7 @@  // Refer to the license.txt file included.  #include <cinttypes> +#include <stack>  #include "core/file_sys/filesystem.h"  #include "core/hle/ipc_helpers.h"  #include "core/hle/kernel/event.h" @@ -154,7 +155,7 @@ ISelfController::ISelfController(std::shared_ptr<NVFlinger::NVFlinger> nvflinger      RegisterHandlers(functions);      launchable_event = -        Kernel::Event::Create(Kernel::ResetType::OneShot, "ISelfController:LaunchableEvent"); +        Kernel::Event::Create(Kernel::ResetType::Sticky, "ISelfController:LaunchableEvent");  }  void ISelfController::SetFocusHandlingMode(Kernel::HLERequestContext& ctx) { @@ -348,19 +349,100 @@ void ICommonStateGetter::GetPerformanceMode(Kernel::HLERequestContext& ctx) {      NGLOG_WARNING(Service_AM, "(STUBBED) called");  } +class IStorageAccessor final : public ServiceFramework<IStorageAccessor> { +public: +    explicit IStorageAccessor(std::vector<u8> buffer) +        : ServiceFramework("IStorageAccessor"), buffer(std::move(buffer)) { +        static const FunctionInfo functions[] = { +            {0, &IStorageAccessor::GetSize, "GetSize"}, +            {10, &IStorageAccessor::Write, "Write"}, +            {11, &IStorageAccessor::Read, "Read"}, +        }; +        RegisterHandlers(functions); +    } + +private: +    std::vector<u8> buffer; + +    void GetSize(Kernel::HLERequestContext& ctx) { +        IPC::ResponseBuilder rb{ctx, 4}; + +        rb.Push(RESULT_SUCCESS); +        rb.Push(static_cast<u64>(buffer.size())); + +        NGLOG_DEBUG(Service_AM, "called"); +    } + +    void Write(Kernel::HLERequestContext& ctx) { +        IPC::RequestParser rp{ctx}; + +        const u64 offset{rp.Pop<u64>()}; +        const std::vector<u8> data{ctx.ReadBuffer()}; + +        ASSERT(offset + data.size() <= buffer.size()); + +        std::memcpy(&buffer[offset], data.data(), data.size()); + +        IPC::ResponseBuilder rb{rp.MakeBuilder(2, 0, 0)}; +        rb.Push(RESULT_SUCCESS); + +        NGLOG_DEBUG(Service_AM, "called, offset={}", offset); +    } + +    void Read(Kernel::HLERequestContext& ctx) { +        IPC::RequestParser rp{ctx}; + +        const u64 offset{rp.Pop<u64>()}; +        const size_t size{ctx.GetWriteBufferSize()}; + +        ASSERT(offset + size <= buffer.size()); + +        ctx.WriteBuffer(buffer.data() + offset, size); + +        IPC::ResponseBuilder rb{rp.MakeBuilder(2, 0, 0)}; +        rb.Push(RESULT_SUCCESS); + +        NGLOG_DEBUG(Service_AM, "called, offset={}", offset); +    } +}; + +class IStorage final : public ServiceFramework<IStorage> { +public: +    explicit IStorage(std::vector<u8> buffer) +        : ServiceFramework("IStorage"), buffer(std::move(buffer)) { +        static const FunctionInfo functions[] = { +            {0, &IStorage::Open, "Open"}, +            {1, nullptr, "OpenTransferStorage"}, +        }; +        RegisterHandlers(functions); +    } + +private: +    std::vector<u8> buffer; + +    void Open(Kernel::HLERequestContext& ctx) { +        IPC::ResponseBuilder rb{ctx, 2, 0, 1}; + +        rb.Push(RESULT_SUCCESS); +        rb.PushIpcInterface<AM::IStorageAccessor>(buffer); + +        NGLOG_DEBUG(Service_AM, "called"); +    } +}; +  class ILibraryAppletAccessor final : public ServiceFramework<ILibraryAppletAccessor> {  public:      explicit ILibraryAppletAccessor() : ServiceFramework("ILibraryAppletAccessor") {          static const FunctionInfo functions[] = {              {0, &ILibraryAppletAccessor::GetAppletStateChangedEvent, "GetAppletStateChangedEvent"},              {1, nullptr, "IsCompleted"}, -            {10, nullptr, "Start"}, +            {10, &ILibraryAppletAccessor::Start, "Start"},              {20, nullptr, "RequestExit"},              {25, nullptr, "Terminate"}, -            {30, nullptr, "GetResult"}, +            {30, &ILibraryAppletAccessor::GetResult, "GetResult"},              {50, nullptr, "SetOutOfFocusApplicationSuspendingEnabled"}, -            {100, nullptr, "PushInData"}, -            {101, nullptr, "PopOutData"}, +            {100, &ILibraryAppletAccessor::PushInData, "PushInData"}, +            {101, &ILibraryAppletAccessor::PopOutData, "PopOutData"},              {102, nullptr, "PushExtraStorage"},              {103, nullptr, "PushInteractiveInData"},              {104, nullptr, "PopInteractiveOutData"}, @@ -388,6 +470,41 @@ private:          NGLOG_WARNING(Service_AM, "(STUBBED) called");      } +    void GetResult(Kernel::HLERequestContext& ctx) { +        IPC::ResponseBuilder rb{ctx, 2}; +        rb.Push(RESULT_SUCCESS); + +        NGLOG_WARNING(Service_AM, "(STUBBED) called"); +    } + +    void Start(Kernel::HLERequestContext& ctx) { +        IPC::ResponseBuilder rb{ctx, 2}; +        rb.Push(RESULT_SUCCESS); + +        NGLOG_WARNING(Service_AM, "(STUBBED) called"); +    } + +    void PushInData(Kernel::HLERequestContext& ctx) { +        IPC::RequestParser rp{ctx}; +        storage_stack.push(rp.PopIpcInterface<AM::IStorage>()); + +        IPC::ResponseBuilder rb{rp.MakeBuilder(2, 0, 0)}; +        rb.Push(RESULT_SUCCESS); + +        NGLOG_DEBUG(Service_AM, "called"); +    } + +    void PopOutData(Kernel::HLERequestContext& ctx) { +        IPC::ResponseBuilder rb{ctx, 2, 0, 1}; +        rb.Push(RESULT_SUCCESS); +        rb.PushIpcInterface<AM::IStorage>(std::move(storage_stack.top())); + +        storage_stack.pop(); + +        NGLOG_DEBUG(Service_AM, "called"); +    } + +    std::stack<std::shared_ptr<AM::IStorage>> storage_stack;      Kernel::SharedPtr<Kernel::Event> state_changed_event;  }; @@ -396,7 +513,7 @@ ILibraryAppletCreator::ILibraryAppletCreator() : ServiceFramework("ILibraryApple          {0, &ILibraryAppletCreator::CreateLibraryApplet, "CreateLibraryApplet"},          {1, nullptr, "TerminateAllLibraryApplets"},          {2, nullptr, "AreAnyLibraryAppletsLeft"}, -        {10, nullptr, "CreateStorage"}, +        {10, &ILibraryAppletCreator::CreateStorage, "CreateStorage"},          {11, nullptr, "CreateTransferMemoryStorage"},          {12, nullptr, "CreateHandleStorage"},      }; @@ -412,72 +529,17 @@ void ILibraryAppletCreator::CreateLibraryApplet(Kernel::HLERequestContext& ctx)      NGLOG_DEBUG(Service_AM, "called");  } -class IStorageAccessor final : public ServiceFramework<IStorageAccessor> { -public: -    explicit IStorageAccessor(std::vector<u8> buffer) -        : ServiceFramework("IStorageAccessor"), buffer(std::move(buffer)) { -        static const FunctionInfo functions[] = { -            {0, &IStorageAccessor::GetSize, "GetSize"}, -            {10, nullptr, "Write"}, -            {11, &IStorageAccessor::Read, "Read"}, -        }; -        RegisterHandlers(functions); -    } - -private: -    std::vector<u8> buffer; - -    void GetSize(Kernel::HLERequestContext& ctx) { -        IPC::ResponseBuilder rb{ctx, 4}; - -        rb.Push(RESULT_SUCCESS); -        rb.Push(static_cast<u64>(buffer.size())); - -        NGLOG_DEBUG(Service_AM, "called"); -    } - -    void Read(Kernel::HLERequestContext& ctx) { -        IPC::RequestParser rp{ctx}; - -        u64 offset = rp.Pop<u64>(); - -        const size_t size{ctx.GetWriteBufferSize()}; - -        ASSERT(offset + size <= buffer.size()); - -        ctx.WriteBuffer(buffer.data() + offset, size); - -        IPC::ResponseBuilder rb{ctx, 2}; - -        rb.Push(RESULT_SUCCESS); - -        NGLOG_DEBUG(Service_AM, "called"); -    } -}; - -class IStorage final : public ServiceFramework<IStorage> { -public: -    explicit IStorage(std::vector<u8> buffer) -        : ServiceFramework("IStorage"), buffer(std::move(buffer)) { -        static const FunctionInfo functions[] = { -            {0, &IStorage::Open, "Open"}, -            {1, nullptr, "OpenTransferStorage"}, -        }; -        RegisterHandlers(functions); -    } - -private: -    std::vector<u8> buffer; - -    void Open(Kernel::HLERequestContext& ctx) { -        IPC::ResponseBuilder rb{ctx, 2, 0, 1}; +void ILibraryAppletCreator::CreateStorage(Kernel::HLERequestContext& ctx) { +    IPC::RequestParser rp{ctx}; +    const u64 size{rp.Pop<u64>()}; +    std::vector<u8> buffer(size); -        rb.Push(RESULT_SUCCESS); -        rb.PushIpcInterface<AM::IStorageAccessor>(buffer); +    IPC::ResponseBuilder rb{rp.MakeBuilder(2, 0, 1)}; +    rb.Push(RESULT_SUCCESS); +    rb.PushIpcInterface<AM::IStorage>(std::move(buffer)); -        NGLOG_DEBUG(Service_AM, "called"); -    } -}; +    NGLOG_DEBUG(Service_AM, "called, size={}", size); +}  IApplicationFunctions::IApplicationFunctions() : ServiceFramework("IApplicationFunctions") {      static const FunctionInfo functions[] = { diff --git a/src/core/hle/service/am/am.h b/src/core/hle/service/am/am.h index ff8eb14d7..301a6c798 100644 --- a/src/core/hle/service/am/am.h +++ b/src/core/hle/service/am/am.h @@ -121,6 +121,7 @@ public:  private:      void CreateLibraryApplet(Kernel::HLERequestContext& ctx); +    void CreateStorage(Kernel::HLERequestContext& ctx);  };  class IApplicationFunctions final : public ServiceFramework<IApplicationFunctions> { diff --git a/src/core/hle/service/nvdrv/devices/nvhost_gpu.h b/src/core/hle/service/nvdrv/devices/nvhost_gpu.h index 2ecf818f3..56b5ed60d 100644 --- a/src/core/hle/service/nvdrv/devices/nvhost_gpu.h +++ b/src/core/hle/service/nvdrv/devices/nvhost_gpu.h @@ -26,11 +26,19 @@ public:  private:      enum class IoctlCommand : u32_le {          IocSetNVMAPfdCommand = 0x40044801, +        IocAllocGPFIFOCommand = 0x40084805,          IocSetClientDataCommand = 0x40084714,          IocGetClientDataCommand = 0x80084715,          IocZCullBind = 0xc010480b,          IocSetErrorNotifierCommand = 0xC018480C,          IocChannelSetPriorityCommand = 0x4004480D, +        IocEnableCommand = 0x0000480E, +        IocDisableCommand = 0x0000480F, +        IocPreemptCommand = 0x00004810, +        IocForceResetCommand = 0x00004811, +        IocEventIdControlCommand = 0x40084812, +        IocGetErrorNotificationCommand = 0xC0104817, +        IocAllocGPFIFOExCommand = 0x40204818,          IocAllocGPFIFOEx2Command = 0xC020481A,          IocAllocObjCtxCommand = 0xC0104809,          IocChannelGetWaitbaseCommand = 0xC0080003, @@ -56,6 +64,12 @@ private:      };      static_assert(sizeof(IoctlChannelSetTimeout) == 4, "IoctlChannelSetTimeout is incorrect size"); +    struct IoctlAllocGPFIFO { +        u32_le num_entries; +        u32_le flags; +    }; +    static_assert(sizeof(IoctlAllocGPFIFO) == 8, "IoctlAllocGPFIFO is incorrect size"); +      struct IoctlClientData {          u64_le data;      }; @@ -76,12 +90,45 @@ private:      };      static_assert(sizeof(IoctlSetErrorNotifier) == 24, "IoctlSetErrorNotifier is incorrect size"); +    struct IoctlChannelSetPriority { +        u32_le priority; +    }; +    static_assert(sizeof(IoctlChannelSetPriority) == 4, +                  "IoctlChannelSetPriority is incorrect size"); + +    struct IoctlEventIdControl { +        u32_le cmd; // 0=disable, 1=enable, 2=clear +        u32_le id; +    }; +    static_assert(sizeof(IoctlEventIdControl) == 8, "IoctlEventIdControl is incorrect size"); + +    struct IoctlGetErrorNotification { +        u64_le timestamp; +        u32_le info32; +        u16_le info16; +        u16_le status; // always 0xFFFF +    }; +    static_assert(sizeof(IoctlGetErrorNotification) == 16, +                  "IoctlGetErrorNotification is incorrect size"); +      struct IoctlFence {          u32_le id;          u32_le value;      };      static_assert(sizeof(IoctlFence) == 8, "IoctlFence is incorrect size"); +    struct IoctlAllocGpfifoEx { +        u32_le num_entries; +        u32_le flags; +        u32_le unk0; +        u32_le unk1; +        u32_le unk2; +        u32_le unk3; +        u32_le unk4; +        u32_le unk5; +    }; +    static_assert(sizeof(IoctlAllocGpfifoEx) == 32, "IoctlAllocGpfifoEx is incorrect size"); +      struct IoctlAllocGpfifoEx2 {          u32_le num_entries;   // in          u32_le flags;         // in diff --git a/src/video_core/engines/maxwell_3d.h b/src/video_core/engines/maxwell_3d.h index 5cf62fb01..245410c95 100644 --- a/src/video_core/engines/maxwell_3d.h +++ b/src/video_core/engines/maxwell_3d.h @@ -354,10 +354,35 @@ public:                      f32 scale_x;                      f32 scale_y;                      f32 scale_z; -                    u32 translate_x; -                    u32 translate_y; -                    u32 translate_z; +                    f32 translate_x; +                    f32 translate_y; +                    f32 translate_z;                      INSERT_PADDING_WORDS(2); + +                    MathUtil::Rectangle<s32> GetRect() const { +                        return { +                            GetX(),               // left +                            GetY() + GetHeight(), // top +                            GetX() + GetWidth(),  // right +                            GetY()                // bottom +                        }; +                    }; + +                    s32 GetX() const { +                        return static_cast<s32>(std::max(0.0f, translate_x - std::fabs(scale_x))); +                    } + +                    s32 GetY() const { +                        return static_cast<s32>(std::max(0.0f, translate_y - std::fabs(scale_y))); +                    } + +                    s32 GetWidth() const { +                        return static_cast<s32>(translate_x + std::fabs(scale_x)) - GetX(); +                    } + +                    s32 GetHeight() const { +                        return static_cast<s32>(translate_y + std::fabs(scale_y)) - GetY(); +                    }                  } viewport_transform[NumViewports];                  struct { @@ -371,15 +396,6 @@ public:                      };                      float depth_range_near;                      float depth_range_far; - -                    MathUtil::Rectangle<s32> GetRect() const { -                        return { -                            static_cast<s32>(x),          // left -                            static_cast<s32>(y + height), // top -                            static_cast<s32>(x + width),  // right -                            static_cast<s32>(y)           // bottom -                        }; -                    };                  } viewport[NumViewports];                  INSERT_PADDING_WORDS(0x1D); diff --git a/src/video_core/renderer_opengl/gl_rasterizer.cpp b/src/video_core/renderer_opengl/gl_rasterizer.cpp index 35c1b1890..0a33868b7 100644 --- a/src/video_core/renderer_opengl/gl_rasterizer.cpp +++ b/src/video_core/renderer_opengl/gl_rasterizer.cpp @@ -298,7 +298,7 @@ void RasterizerOpenGL::DrawArrays() {      const bool has_stencil = false;      const bool using_color_fb = true;      const bool using_depth_fb = false; -    const MathUtil::Rectangle<s32> viewport_rect{regs.viewport[0].GetRect()}; +    const MathUtil::Rectangle<s32> viewport_rect{regs.viewport_transform[0].GetRect()};      const bool write_color_fb =          state.color_mask.red_enabled == GL_TRUE || state.color_mask.green_enabled == GL_TRUE || @@ -702,7 +702,7 @@ void RasterizerOpenGL::BindFramebufferSurfaces(const Surface& color_surface,  void RasterizerOpenGL::SyncViewport(const MathUtil::Rectangle<u32>& surfaces_rect, u16 res_scale) {      const auto& regs = Core::System().GetInstance().GPU().Maxwell3D().regs; -    const MathUtil::Rectangle<s32> viewport_rect{regs.viewport[0].GetRect()}; +    const MathUtil::Rectangle<s32> viewport_rect{regs.viewport_transform[0].GetRect()};      state.viewport.x = static_cast<GLint>(surfaces_rect.left) + viewport_rect.left * res_scale;      state.viewport.y = static_cast<GLint>(surfaces_rect.bottom) + viewport_rect.bottom * res_scale; diff --git a/src/video_core/renderer_opengl/gl_rasterizer_cache.cpp b/src/video_core/renderer_opengl/gl_rasterizer_cache.cpp index 65d643447..d6048f639 100644 --- a/src/video_core/renderer_opengl/gl_rasterizer_cache.cpp +++ b/src/video_core/renderer_opengl/gl_rasterizer_cache.cpp @@ -933,7 +933,8 @@ Surface RasterizerCacheOpenGL::GetSurface(const SurfaceParams& params, ScaleMatc      // Use GetSurfaceSubRect instead      ASSERT(params.width == params.stride); -    ASSERT(!params.is_tiled || (params.width % 8 == 0 && params.height % 8 == 0)); +    ASSERT(!params.is_tiled || +           (params.GetActualWidth() % 8 == 0 && params.GetActualHeight() % 8 == 0));      // Check for an exact match in existing surfaces      Surface surface = diff --git a/src/yuzu/main.cpp b/src/yuzu/main.cpp index a5d7807e2..3038bd6da 100644 --- a/src/yuzu/main.cpp +++ b/src/yuzu/main.cpp @@ -335,6 +335,24 @@ void GMainWindow::OnDisplayTitleBars(bool show) {      }  } +bool GMainWindow::SupportsRequiredGLExtensions() { +    QStringList unsupported_ext; + +    if (!GLAD_GL_ARB_program_interface_query) +        unsupported_ext.append("ARB_program_interface_query"); +    if (!GLAD_GL_ARB_separate_shader_objects) +        unsupported_ext.append("ARB_separate_shader_objects"); +    if (!GLAD_GL_ARB_shader_storage_buffer_object) +        unsupported_ext.append("ARB_shader_storage_buffer_object"); +    if (!GLAD_GL_ARB_vertex_attrib_binding) +        unsupported_ext.append("ARB_vertex_attrib_binding"); + +    for (const QString& ext : unsupported_ext) +        NGLOG_CRITICAL(Frontend, "Unsupported GL extension: {}", ext.toStdString()); + +    return unsupported_ext.empty(); +} +  bool GMainWindow::LoadROM(const QString& filename) {      // Shutdown previous session if the emu thread is still active...      if (emu_thread != nullptr) @@ -350,6 +368,14 @@ bool GMainWindow::LoadROM(const QString& filename) {          return false;      } +    if (!SupportsRequiredGLExtensions()) { +        QMessageBox::critical( +            this, tr("Error while initializing OpenGL Core!"), +            tr("Your GPU may not support one or more required OpenGL extensions. Please " +               "ensure you have the latest graphics driver. See the log for more details.")); +        return false; +    } +      Core::System& system{Core::System::GetInstance()};      system.SetGPUDebugContext(debug_context); diff --git a/src/yuzu/main.h b/src/yuzu/main.h index 20ff65314..ac3024d8a 100644 --- a/src/yuzu/main.h +++ b/src/yuzu/main.h @@ -79,6 +79,7 @@ private:      void ConnectWidgetEvents();      void ConnectMenuEvents(); +    bool SupportsRequiredGLExtensions();      bool LoadROM(const QString& filename);      void BootGame(const QString& filename);      void ShutdownGame(); diff --git a/src/yuzu_cmd/emu_window/emu_window_sdl2.cpp b/src/yuzu_cmd/emu_window/emu_window_sdl2.cpp index e21de6f21..cfd8eb7e6 100644 --- a/src/yuzu_cmd/emu_window/emu_window_sdl2.cpp +++ b/src/yuzu_cmd/emu_window/emu_window_sdl2.cpp @@ -78,6 +78,24 @@ void EmuWindow_SDL2::Fullscreen() {      SDL_MaximizeWindow(render_window);  } +bool EmuWindow_SDL2::SupportsRequiredGLExtensions() { +    std::vector<std::string> unsupported_ext; + +    if (!GLAD_GL_ARB_program_interface_query) +        unsupported_ext.push_back("ARB_program_interface_query"); +    if (!GLAD_GL_ARB_separate_shader_objects) +        unsupported_ext.push_back("ARB_separate_shader_objects"); +    if (!GLAD_GL_ARB_shader_storage_buffer_object) +        unsupported_ext.push_back("ARB_shader_storage_buffer_object"); +    if (!GLAD_GL_ARB_vertex_attrib_binding) +        unsupported_ext.push_back("ARB_vertex_attrib_binding"); + +    for (const std::string& ext : unsupported_ext) +        NGLOG_CRITICAL(Frontend, "Unsupported GL extension: {}", ext); + +    return unsupported_ext.empty(); +} +  EmuWindow_SDL2::EmuWindow_SDL2(bool fullscreen) {      InputCommon::Init(); @@ -128,6 +146,11 @@ EmuWindow_SDL2::EmuWindow_SDL2(bool fullscreen) {          exit(1);      } +    if (!SupportsRequiredGLExtensions()) { +        NGLOG_CRITICAL(Frontend, "GPU does not support all required OpenGL extensions! Exiting..."); +        exit(1); +    } +      OnResize();      OnMinimalClientAreaChangeRequest(GetActiveConfig().min_client_area_size);      SDL_PumpEvents(); diff --git a/src/yuzu_cmd/emu_window/emu_window_sdl2.h b/src/yuzu_cmd/emu_window/emu_window_sdl2.h index 7d5cfffb6..1d835c3c6 100644 --- a/src/yuzu_cmd/emu_window/emu_window_sdl2.h +++ b/src/yuzu_cmd/emu_window/emu_window_sdl2.h @@ -46,6 +46,9 @@ private:      /// Called when user passes the fullscreen parameter flag      void Fullscreen(); +    /// Whether the GPU and driver supports the OpenGL extension required +    bool SupportsRequiredGLExtensions(); +      /// Called when a configuration change affects the minimal size of the window      void OnMinimalClientAreaChangeRequest(          const std::pair<unsigned, unsigned>& minimal_size) override;  | 
