cs_ioctl_reserve_signals() makes the encapsulated signal handle visible
in the context IDR before the SOB pointer and pre-reserve SOB value are
set. Concurrent unreserve and wait paths dereference those fields after
IDR lookup.

Reserve the IDR slot with a NULL entry, initialize the handle including
the SOB fields, and replace the slot with the handle only after the
visible state is ready.

Fixes: dadf17abb724 ("habanalabs: add support for encapsulated signals 
reservation")
Signed-off-by: Ruoyu Wang <[email protected]>
---
 .../habanalabs/common/command_submission.c    | 25 +++++++++++++------
 1 file changed, 17 insertions(+), 8 deletions(-)

diff --git a/drivers/accel/habanalabs/common/command_submission.c 
b/drivers/accel/habanalabs/common/command_submission.c
index ba4257bda77b..791fc01e24c5 100644
--- a/drivers/accel/habanalabs/common/command_submission.c
+++ b/drivers/accel/habanalabs/common/command_submission.c
@@ -2009,6 +2009,7 @@ static int cs_ioctl_reserve_signals(struct hl_fpriv 
*hpriv,
        struct hl_cs_encaps_sig_handle *handle;
        struct hl_encaps_signals_mgr *mgr;
        struct hl_hw_sob *hw_sob;
+       void *old;
        int hdl_id;
        int rc = 0;
 
@@ -2045,13 +2046,19 @@ static int cs_ioctl_reserve_signals(struct hl_fpriv 
*hpriv,
        }
 
        handle->count = count;
+       handle->q_idx = q_idx;
+       handle->hdev = hdev;
+       handle->cs_seq = ULLONG_MAX;
+       kref_init(&handle->refcount);
 
        hl_ctx_get(hpriv->ctx);
        handle->ctx = hpriv->ctx;
        mgr = &hpriv->ctx->sig_mgr;
 
        spin_lock(&mgr->lock);
-       hdl_id = idr_alloc(&mgr->handles, handle, 1, 0, GFP_ATOMIC);
+       hdl_id = idr_alloc(&mgr->handles, NULL, 1, 0, GFP_ATOMIC);
+       if (hdl_id >= 0)
+               handle->id = hdl_id;
        spin_unlock(&mgr->lock);
 
        if (hdl_id < 0) {
@@ -2060,11 +2067,6 @@ static int cs_ioctl_reserve_signals(struct hl_fpriv 
*hpriv,
                goto put_ctx;
        }
 
-       handle->id = hdl_id;
-       handle->q_idx = q_idx;
-       handle->hdev = hdev;
-       kref_init(&handle->refcount);
-
        hdev->asic_funcs->hw_queues_lock(hdev);
 
        hw_sob = &prop->hw_sob[prop->curr_sob_offset];
@@ -2093,11 +2095,18 @@ static int cs_ioctl_reserve_signals(struct hl_fpriv 
*hpriv,
         */
        handle->pre_sob_val = prop->next_sob_val - handle->count;
 
-       handle->cs_seq = ULLONG_MAX;
-
        *signals_count = prop->next_sob_val;
        hdev->asic_funcs->hw_queues_unlock(hdev);
 
+       spin_lock(&mgr->lock);
+       old = idr_replace(&mgr->handles, handle, hdl_id);
+       spin_unlock(&mgr->lock);
+
+       if (WARN_ON(IS_ERR(old))) {
+               rc = PTR_ERR(old);
+               goto remove_idr;
+       }
+
        *sob_addr = handle->hw_sob->sob_addr;
        *handle_id = hdl_id;
 

Reply via email to