Commit 2988cf02 authored by Niranjana Vishwanathapura's avatar Niranjana Vishwanathapura Committed by Rodrigo Vivi

drm/xe: Fix memory use after free

The wait_event_timeout() on g2h_fence.wq which is declared on
stack can return before the wake_up() gets called, resulting in a
stack out of bound access when wake_up() accesses the g2h_fene.wq.

Do not declare g2h_fence related wait_queue_head_t on stack.

Fixes the below KASAN BUG and associated kernel crashes.

BUG: KASAN: stack-out-of-bounds in do_raw_spin_lock+0x6f/0x1e0
Read of size 4 at addr ffff88826252f4ac by task kworker/u128:5/467

CPU: 25 PID: 467 Comm: kworker/u128:5 Tainted: G  U 6.3.0-rc4-xe #1
Workqueue: events_unbound g2h_worker_func [xe]
Call Trace:
 <TASK>
 dump_stack_lvl+0x64/0xb0
 print_report+0xc2/0x600
 kasan_report+0x96/0xc0
 do_raw_spin_lock+0x6f/0x1e0
 _raw_spin_lock_irqsave+0x47/0x60
 __wake_up_common_lock+0xc0/0x150
 dequeue_one_g2h+0x20f/0x6a0 [xe]
 g2h_worker_func+0xa9/0x180 [xe]
 process_one_work+0x527/0x990
 worker_thread+0x2d1/0x640
 kthread+0x174/0x1b0
 ret_from_fork+0x29/0x50
 </TASK>
Tested-by: default avatarMatt Roper <matthew.d.roper@intel.com>
Reviewed-by: default avatarBruce Chang <yu.bruce.chang@intel.com>
Signed-off-by: default avatarNiranjana Vishwanathapura <niranjana.vishwanathapura@intel.com>
Signed-off-by: default avatarRodrigo Vivi <rodrigo.vivi@intel.com>
parent 36919ebe
...@@ -23,7 +23,6 @@ ...@@ -23,7 +23,6 @@
/* Used when a CT send wants to block and / or receive data */ /* Used when a CT send wants to block and / or receive data */
struct g2h_fence { struct g2h_fence {
wait_queue_head_t wq;
u32 *response_buffer; u32 *response_buffer;
u32 seqno; u32 seqno;
u16 response_len; u16 response_len;
...@@ -142,6 +141,7 @@ int xe_guc_ct_init(struct xe_guc_ct *ct) ...@@ -142,6 +141,7 @@ int xe_guc_ct_init(struct xe_guc_ct *ct)
ct->fence_context = dma_fence_context_alloc(1); ct->fence_context = dma_fence_context_alloc(1);
INIT_WORK(&ct->g2h_worker, g2h_worker_func); INIT_WORK(&ct->g2h_worker, g2h_worker_func);
init_waitqueue_head(&ct->wq); init_waitqueue_head(&ct->wq);
init_waitqueue_head(&ct->g2h_fence_wq);
primelockdep(ct); primelockdep(ct);
...@@ -484,7 +484,6 @@ static int __guc_ct_send_locked(struct xe_guc_ct *ct, const u32 *action, ...@@ -484,7 +484,6 @@ static int __guc_ct_send_locked(struct xe_guc_ct *ct, const u32 *action,
void *ptr; void *ptr;
g2h_fence->seqno = (ct->fence_seqno++ & 0xffff); g2h_fence->seqno = (ct->fence_seqno++ & 0xffff);
init_waitqueue_head(&g2h_fence->wq);
ptr = xa_store(&ct->fence_lookup, ptr = xa_store(&ct->fence_lookup,
g2h_fence->seqno, g2h_fence->seqno,
g2h_fence, GFP_ATOMIC); g2h_fence, GFP_ATOMIC);
...@@ -709,7 +708,7 @@ static int guc_ct_send_recv(struct xe_guc_ct *ct, const u32 *action, u32 len, ...@@ -709,7 +708,7 @@ static int guc_ct_send_recv(struct xe_guc_ct *ct, const u32 *action, u32 len,
return ret; return ret;
} }
ret = wait_event_timeout(g2h_fence.wq, g2h_fence.done, HZ); ret = wait_event_timeout(ct->g2h_fence_wq, g2h_fence.done, HZ);
if (!ret) { if (!ret) {
drm_err(&xe->drm, "Timed out wait for G2H, fence %u, action %04x", drm_err(&xe->drm, "Timed out wait for G2H, fence %u, action %04x",
g2h_fence.seqno, action[0]); g2h_fence.seqno, action[0]);
...@@ -801,7 +800,7 @@ static int parse_g2h_response(struct xe_guc_ct *ct, u32 *msg, u32 len) ...@@ -801,7 +800,7 @@ static int parse_g2h_response(struct xe_guc_ct *ct, u32 *msg, u32 len)
g2h_fence->done = true; g2h_fence->done = true;
smp_mb(); smp_mb();
wake_up(&g2h_fence->wq); wake_up_all(&ct->g2h_fence_wq);
return 0; return 0;
} }
......
...@@ -74,6 +74,8 @@ struct xe_guc_ct { ...@@ -74,6 +74,8 @@ struct xe_guc_ct {
struct xarray fence_lookup; struct xarray fence_lookup;
/** @wq: wait queue used for reliable CT sends and freeing G2H credits */ /** @wq: wait queue used for reliable CT sends and freeing G2H credits */
wait_queue_head_t wq; wait_queue_head_t wq;
/** @g2h_fence_wq: wait queue used for G2H fencing */
wait_queue_head_t g2h_fence_wq;
#ifdef XE_GUC_CT_SELFTEST #ifdef XE_GUC_CT_SELFTEST
/** @suppress_irq_handler: force flow control to sender */ /** @suppress_irq_handler: force flow control to sender */
bool suppress_irq_handler; bool suppress_irq_handler;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment