diff options
| -rw-r--r-- | src/core/hle/kernel/k_server_session.h | 2 | ||||
| -rw-r--r-- | src/core/hle/service/nvdrv/core/nvmap.cpp | 5 | ||||
| -rw-r--r-- | src/core/hle/service/nvdrv/core/nvmap.h | 1 | ||||
| -rw-r--r-- | src/core/hle/service/nvdrv/devices/nvmap.cpp | 10 | ||||
| -rw-r--r-- | src/core/hle/service/nvflinger/buffer_queue_producer.cpp | 7 | ||||
| -rw-r--r-- | src/core/hle/service/nvflinger/nvflinger.cpp | 13 | ||||
| -rw-r--r-- | src/core/hle/service/nvflinger/nvflinger.h | 5 | ||||
| -rw-r--r-- | src/core/hle/service/sm/sm.cpp | 1 | ||||
| -rw-r--r-- | src/core/hle/service/vi/display/vi_display.h | 6 | ||||
| -rw-r--r-- | src/core/hle/service/vi/vi.cpp | 8 | ||||
| -rw-r--r-- | src/video_core/engines/maxwell_3d.cpp | 129 | ||||
| -rw-r--r-- | src/video_core/engines/maxwell_3d.h | 2 | 
12 files changed, 116 insertions, 73 deletions
diff --git a/src/core/hle/kernel/k_server_session.h b/src/core/hle/kernel/k_server_session.h index 32135473b..188aef4af 100644 --- a/src/core/hle/kernel/k_server_session.h +++ b/src/core/hle/kernel/k_server_session.h @@ -91,7 +91,7 @@ private:      /// List of threads which are pending a reply.      boost::intrusive::list<KSessionRequest> m_request_list; -    KSessionRequest* m_current_request; +    KSessionRequest* m_current_request{};      KLightLock m_lock;  }; diff --git a/src/core/hle/service/nvdrv/core/nvmap.cpp b/src/core/hle/service/nvdrv/core/nvmap.cpp index fbd8a74a5..a51ca5444 100644 --- a/src/core/hle/service/nvdrv/core/nvmap.cpp +++ b/src/core/hle/service/nvdrv/core/nvmap.cpp @@ -255,15 +255,16 @@ std::optional<NvMap::FreeInfo> NvMap::FreeHandle(Handle::Id handle, bool interna              .address = handle_description->address,              .size = handle_description->size,              .was_uncached = handle_description->flags.map_uncached.Value() != 0, +            .can_unlock = true,          };      } else {          return std::nullopt;      } -    // Handle hasn't been freed from memory, set address to 0 to mark that the handle wasn't freed +    // If the handle hasn't been freed from memory, mark that      if (!hWeak.expired()) {          LOG_DEBUG(Service_NVDRV, "nvmap handle: {} wasn't freed as it is still in use", handle); -        freeInfo.address = 0; +        freeInfo.can_unlock = false;      }      return freeInfo; diff --git a/src/core/hle/service/nvdrv/core/nvmap.h b/src/core/hle/service/nvdrv/core/nvmap.h index b9dd3801f..a8e573890 100644 --- a/src/core/hle/service/nvdrv/core/nvmap.h +++ b/src/core/hle/service/nvdrv/core/nvmap.h @@ -105,6 +105,7 @@ public:          u64 address;       //!< Address the handle referred to before deletion          u64 size;          //!< Page-aligned handle size          bool was_uncached; //!< If the handle was allocated as uncached +        bool can_unlock;   //!< If the address region is ready to be unlocked      };      explicit NvMap(Tegra::Host1x::Host1x& host1x); diff --git a/src/core/hle/service/nvdrv/devices/nvmap.cpp b/src/core/hle/service/nvdrv/devices/nvmap.cpp index b60679021..44388655d 100644 --- a/src/core/hle/service/nvdrv/devices/nvmap.cpp +++ b/src/core/hle/service/nvdrv/devices/nvmap.cpp @@ -251,10 +251,12 @@ NvResult nvmap::IocFree(const std::vector<u8>& input, std::vector<u8>& output) {      }      if (auto freeInfo{file.FreeHandle(params.handle, false)}) { -        ASSERT(system.CurrentProcess() -                   ->PageTable() -                   .UnlockForDeviceAddressSpace(freeInfo->address, freeInfo->size) -                   .IsSuccess()); +        if (freeInfo->can_unlock) { +            ASSERT(system.CurrentProcess() +                       ->PageTable() +                       .UnlockForDeviceAddressSpace(freeInfo->address, freeInfo->size) +                       .IsSuccess()); +        }          params.address = freeInfo->address;          params.size = static_cast<u32>(freeInfo->size);          params.flags.raw = 0; diff --git a/src/core/hle/service/nvflinger/buffer_queue_producer.cpp b/src/core/hle/service/nvflinger/buffer_queue_producer.cpp index 77ddbb6ef..41ba44b21 100644 --- a/src/core/hle/service/nvflinger/buffer_queue_producer.cpp +++ b/src/core/hle/service/nvflinger/buffer_queue_producer.cpp @@ -742,6 +742,13 @@ Status BufferQueueProducer::Disconnect(NativeWindowApi api) {              return Status::NoError;          } +        // HACK: We are not Android. Remove handle for items in queue, and clear queue. +        // Allows synchronous destruction of nvmap handles. +        for (auto& item : core->queue) { +            nvmap.FreeHandle(item.graphic_buffer->BufferId(), true); +        } +        core->queue.clear(); +          switch (api) {          case NativeWindowApi::Egl:          case NativeWindowApi::Cpu: diff --git a/src/core/hle/service/nvflinger/nvflinger.cpp b/src/core/hle/service/nvflinger/nvflinger.cpp index dad93b38e..c3af12c90 100644 --- a/src/core/hle/service/nvflinger/nvflinger.cpp +++ b/src/core/hle/service/nvflinger/nvflinger.cpp @@ -138,6 +138,19 @@ std::optional<u64> NVFlinger::OpenDisplay(std::string_view name) {      return itr->GetID();  } +bool NVFlinger::CloseDisplay(u64 display_id) { +    const auto lock_guard = Lock(); +    auto* const display = FindDisplay(display_id); + +    if (display == nullptr) { +        return false; +    } + +    display->Reset(); + +    return true; +} +  std::optional<u64> NVFlinger::CreateLayer(u64 display_id) {      const auto lock_guard = Lock();      auto* const display = FindDisplay(display_id); diff --git a/src/core/hle/service/nvflinger/nvflinger.h b/src/core/hle/service/nvflinger/nvflinger.h index b8191c595..460bef976 100644 --- a/src/core/hle/service/nvflinger/nvflinger.h +++ b/src/core/hle/service/nvflinger/nvflinger.h @@ -58,6 +58,11 @@ public:      /// If an invalid display name is provided, then an empty optional is returned.      [[nodiscard]] std::optional<u64> OpenDisplay(std::string_view name); +    /// Closes the specified display by its ID. +    /// +    /// Returns false if an invalid display ID is provided. +    [[nodiscard]] bool CloseDisplay(u64 display_id); +      /// Creates a layer on the specified display and returns the layer ID.      ///      /// If an invalid display ID is specified, then an empty optional is returned. diff --git a/src/core/hle/service/sm/sm.cpp b/src/core/hle/service/sm/sm.cpp index 48e70f93c..e2b8d8720 100644 --- a/src/core/hle/service/sm/sm.cpp +++ b/src/core/hle/service/sm/sm.cpp @@ -80,7 +80,6 @@ ResultVal<Kernel::KPort*> ServiceManager::GetServicePort(const std::string& name      }      auto* port = Kernel::KPort::Create(kernel); -    SCOPE_EXIT({ port->Close(); });      port->Initialize(ServerSessionCountMax, false, name);      auto handler = it->second; diff --git a/src/core/hle/service/vi/display/vi_display.h b/src/core/hle/service/vi/display/vi_display.h index 33d5f398c..0b65a65da 100644 --- a/src/core/hle/service/vi/display/vi_display.h +++ b/src/core/hle/service/vi/display/vi_display.h @@ -106,6 +106,12 @@ public:      ///      void CloseLayer(u64 layer_id); +    /// Resets the display for a new connection. +    void Reset() { +        layers.clear(); +        got_vsync_event = false; +    } +      /// Attempts to find a layer with the given ID.      ///      /// @param layer_id The layer ID. diff --git a/src/core/hle/service/vi/vi.cpp b/src/core/hle/service/vi/vi.cpp index 9c917cacf..bb283e74e 100644 --- a/src/core/hle/service/vi/vi.cpp +++ b/src/core/hle/service/vi/vi.cpp @@ -324,10 +324,10 @@ private:          IPC::RequestParser rp{ctx};          const u64 display = rp.Pop<u64>(); -        LOG_WARNING(Service_VI, "(STUBBED) called. display=0x{:016X}", display); +        const Result rc = nv_flinger.CloseDisplay(display) ? ResultSuccess : ResultUnknown;          IPC::ResponseBuilder rb{ctx, 2}; -        rb.Push(ResultSuccess); +        rb.Push(rc);      }      void CreateManagedLayer(Kernel::HLERequestContext& ctx) { @@ -508,10 +508,10 @@ private:          IPC::RequestParser rp{ctx};          const u64 display_id = rp.Pop<u64>(); -        LOG_WARNING(Service_VI, "(STUBBED) called. display_id=0x{:016X}", display_id); +        const Result rc = nv_flinger.CloseDisplay(display_id) ? ResultSuccess : ResultUnknown;          IPC::ResponseBuilder rb{ctx, 2}; -        rb.Push(ResultSuccess); +        rb.Push(rc);      }      // This literally does nothing internally in the actual service itself, diff --git a/src/video_core/engines/maxwell_3d.cpp b/src/video_core/engines/maxwell_3d.cpp index 5208bea75..f9794dfe4 100644 --- a/src/video_core/engines/maxwell_3d.cpp +++ b/src/video_core/engines/maxwell_3d.cpp @@ -123,9 +123,6 @@ void Maxwell3D::InitializeRegisterDefaults() {      draw_command[MAXWELL3D_REG_INDEX(vertex_buffer.count)] = true;      draw_command[MAXWELL3D_REG_INDEX(index_buffer.first)] = true;      draw_command[MAXWELL3D_REG_INDEX(index_buffer.count)] = true; -    draw_command[MAXWELL3D_REG_INDEX(index_buffer32_first)] = true; -    draw_command[MAXWELL3D_REG_INDEX(index_buffer16_first)] = true; -    draw_command[MAXWELL3D_REG_INDEX(index_buffer8_first)] = true;      draw_command[MAXWELL3D_REG_INDEX(draw_inline_index)] = true;      draw_command[MAXWELL3D_REG_INDEX(inline_index_2x16.even)] = true;      draw_command[MAXWELL3D_REG_INDEX(inline_index_4x8.index0)] = true; @@ -216,6 +213,21 @@ void Maxwell3D::ProcessMethodCall(u32 method, u32 argument, u32 nonshadow_argume          return ProcessCBBind(3);      case MAXWELL3D_REG_INDEX(bind_groups[4].raw_config):          return ProcessCBBind(4); +    case MAXWELL3D_REG_INDEX(index_buffer32_first): +        regs.index_buffer.count = regs.index_buffer32_first.count; +        regs.index_buffer.first = regs.index_buffer32_first.first; +        dirty.flags[VideoCommon::Dirty::IndexBuffer] = true; +        return ProcessDraw(); +    case MAXWELL3D_REG_INDEX(index_buffer16_first): +        regs.index_buffer.count = regs.index_buffer16_first.count; +        regs.index_buffer.first = regs.index_buffer16_first.first; +        dirty.flags[VideoCommon::Dirty::IndexBuffer] = true; +        return ProcessDraw(); +    case MAXWELL3D_REG_INDEX(index_buffer8_first): +        regs.index_buffer.count = regs.index_buffer8_first.count; +        regs.index_buffer.first = regs.index_buffer8_first.first; +        dirty.flags[VideoCommon::Dirty::IndexBuffer] = true; +        return ProcessDraw();      case MAXWELL3D_REG_INDEX(topology_override):          use_topology_override = true;          return; @@ -583,6 +595,31 @@ void Maxwell3D::ProcessClearBuffers() {      rasterizer->Clear();  } +void Maxwell3D::ProcessDraw(u32 instance_count) { +    LOG_TRACE(HW_GPU, "called, topology={}, count={}", regs.draw.topology.Value(), +              regs.vertex_buffer.count); + +    ASSERT_MSG(!(regs.index_buffer.count && regs.vertex_buffer.count), "Both indexed and direct?"); + +    // Both instance configuration registers can not be set at the same time. +    ASSERT_MSG(regs.draw.instance_id == Maxwell3D::Regs::Draw::InstanceId::First || +                   regs.draw.instance_id != Maxwell3D::Regs::Draw::InstanceId::Unchanged, +               "Illegal combination of instancing parameters"); + +    ProcessTopologyOverride(); + +    const bool is_indexed = regs.index_buffer.count && !regs.vertex_buffer.count; +    if (ShouldExecute()) { +        rasterizer->Draw(is_indexed, instance_count); +    } + +    if (is_indexed) { +        regs.index_buffer.count = 0; +    } else { +        regs.vertex_buffer.count = 0; +    } +} +  void Maxwell3D::ProcessDeferredDraw() {      if (deferred_draw_method.empty()) {          return; @@ -596,23 +633,28 @@ void Maxwell3D::ProcessDeferredDraw() {      DrawMode draw_mode{DrawMode::Undefined};      u32 instance_count = 1; -    auto first_method = deferred_draw_method[0]; -    if (MAXWELL3D_REG_INDEX(draw.begin) == first_method) { -        // The minimum number of methods for drawing must be greater than or equal to -        // 3[draw.begin->vertex(index)count->draw.end] to avoid errors in index mode drawing -        if (deferred_draw_method.size() < 3) { -            return; -        } -        draw_mode = (regs.draw.instance_id == Maxwell3D::Regs::Draw::InstanceId::Subsequent) || -                            (regs.draw.instance_id == Maxwell3D::Regs::Draw::InstanceId::Unchanged) -                        ? DrawMode::Instance -                        : DrawMode::General; -    } else if (MAXWELL3D_REG_INDEX(index_buffer32_first) == first_method || -               MAXWELL3D_REG_INDEX(index_buffer16_first) == first_method || -               MAXWELL3D_REG_INDEX(index_buffer8_first) == first_method) { -        draw_mode = DrawMode::General; +    u32 index = 0; +    u32 method = 0; +    u32 method_count = static_cast<u32>(deferred_draw_method.size()); +    for (; index < method_count && +           (method = deferred_draw_method[index]) != MAXWELL3D_REG_INDEX(draw.begin); +         ++index) +        ; + +    if (MAXWELL3D_REG_INDEX(draw.begin) != method) { +        return;      } +    // The minimum number of methods for drawing must be greater than or equal to +    // 3[draw.begin->vertex(index)count(first)->draw.end] to avoid errors in index mode drawing +    if ((method_count - index) < 3) { +        return; +    } +    draw_mode = (regs.draw.instance_id == Maxwell3D::Regs::Draw::InstanceId::Subsequent) || +                        (regs.draw.instance_id == Maxwell3D::Regs::Draw::InstanceId::Unchanged) +                    ? DrawMode::Instance +                    : DrawMode::General; +      // Drawing will only begin with draw.begin or index_buffer method, other methods directly      // clear      if (draw_mode == DrawMode::Undefined) { @@ -622,53 +664,18 @@ void Maxwell3D::ProcessDeferredDraw() {      if (draw_mode == DrawMode::Instance) {          ASSERT_MSG(deferred_draw_method.size() % 4 == 0, "Instance mode method size error"); -        instance_count = static_cast<u32>(deferred_draw_method.size()) / 4; +        instance_count = static_cast<u32>(method_count - index) / 4;      } else { -        if (MAXWELL3D_REG_INDEX(index_buffer32_first) == first_method) { -            regs.index_buffer.count = regs.index_buffer32_first.count; -            regs.index_buffer.first = regs.index_buffer32_first.first; -            dirty.flags[VideoCommon::Dirty::IndexBuffer] = true; -        } else if (MAXWELL3D_REG_INDEX(index_buffer32_first) == first_method) { -            regs.index_buffer.count = regs.index_buffer16_first.count; -            regs.index_buffer.first = regs.index_buffer16_first.first; -            dirty.flags[VideoCommon::Dirty::IndexBuffer] = true; -        } else if (MAXWELL3D_REG_INDEX(index_buffer32_first) == first_method) { -            regs.index_buffer.count = regs.index_buffer8_first.count; -            regs.index_buffer.first = regs.index_buffer8_first.first; -            dirty.flags[VideoCommon::Dirty::IndexBuffer] = true; -        } else { -            auto second_method = deferred_draw_method[1]; -            if (MAXWELL3D_REG_INDEX(draw_inline_index) == second_method || -                MAXWELL3D_REG_INDEX(inline_index_2x16.even) == second_method || -                MAXWELL3D_REG_INDEX(inline_index_4x8.index0) == second_method) { -                regs.index_buffer.count = static_cast<u32>(inline_index_draw_indexes.size() / 4); -                regs.index_buffer.format = Regs::IndexFormat::UnsignedInt; -            } +        method = deferred_draw_method[index + 1]; +        if (MAXWELL3D_REG_INDEX(draw_inline_index) == method || +            MAXWELL3D_REG_INDEX(inline_index_2x16.even) == method || +            MAXWELL3D_REG_INDEX(inline_index_4x8.index0) == method) { +            regs.index_buffer.count = static_cast<u32>(inline_index_draw_indexes.size() / 4); +            regs.index_buffer.format = Regs::IndexFormat::UnsignedInt;          }      } -    LOG_TRACE(HW_GPU, "called, topology={}, count={}", regs.draw.topology.Value(), -              regs.vertex_buffer.count); - -    ASSERT_MSG(!(regs.index_buffer.count && regs.vertex_buffer.count), "Both indexed and direct?"); - -    // Both instance configuration registers can not be set at the same time. -    ASSERT_MSG(regs.draw.instance_id == Maxwell3D::Regs::Draw::InstanceId::First || -                   regs.draw.instance_id != Maxwell3D::Regs::Draw::InstanceId::Unchanged, -               "Illegal combination of instancing parameters"); - -    ProcessTopologyOverride(); - -    const bool is_indexed = regs.index_buffer.count && !regs.vertex_buffer.count; -    if (ShouldExecute()) { -        rasterizer->Draw(is_indexed, instance_count); -    } - -    if (is_indexed) { -        regs.index_buffer.count = 0; -    } else { -        regs.vertex_buffer.count = 0; -    } +    ProcessDraw(instance_count);      deferred_draw_method.clear();      inline_index_draw_indexes.clear(); diff --git a/src/video_core/engines/maxwell_3d.h b/src/video_core/engines/maxwell_3d.h index bd23ebc12..a948fcb14 100644 --- a/src/video_core/engines/maxwell_3d.h +++ b/src/video_core/engines/maxwell_3d.h @@ -3143,6 +3143,8 @@ private:      /// Handles use of topology overrides (e.g., to avoid using a topology assigned from a macro)      void ProcessTopologyOverride(); +    void ProcessDraw(u32 instance_count = 1); +      void ProcessDeferredDraw();      /// Returns a query's value or an empty object if the value will be deferred through a cache.  | 
