msm: adsprpc : Fix use after free in fastrpc_update_ramdump_status

Thread1 can free up the fl->init memory in
fastrpc_init_create_dynamic_process  with fl spin lock, same time thread2
adding fl->init_mem to chan->initmems list with global spin lock in
fastrpc_update_ramdump_status can lead to use after free in
fastrpc_ramdump_collection. Fix is to use global spin lock while
handling fl->init_mem.

Change-Id: I7a497dc962b6967a4d594a3acce55f8ce0eb3a55
Signed-off-by: rnallago <quic_rnallago@quicinc.com>
This commit is contained in:
Ramesh Nallagopu
2024-02-27 15:54:03 +05:30
committed by rnallago
parent 941b3f835f
commit cc9738786a

View File

@@ -4043,7 +4043,7 @@ bail:
static int fastrpc_init_create_dynamic_process(struct fastrpc_file *fl, static int fastrpc_init_create_dynamic_process(struct fastrpc_file *fl,
struct fastrpc_ioctl_init_attrs *uproc) struct fastrpc_ioctl_init_attrs *uproc)
{ {
int err = 0, memlen = 0, mflags = 0, locked = 0; int err = 0, memlen = 0, mflags = 0, locked = 0, glocked = 0;
struct fastrpc_ioctl_invoke_async ioctl; struct fastrpc_ioctl_invoke_async ioctl;
struct fastrpc_ioctl_init *init = &uproc->init; struct fastrpc_ioctl_init *init = &uproc->init;
/* First page for init-mem and second page for proc-attrs */ /* First page for init-mem and second page for proc-attrs */
@@ -4057,6 +4057,8 @@ static int fastrpc_init_create_dynamic_process(struct fastrpc_file *fl,
unsigned int dsp_userpd_memlen = 0; unsigned int dsp_userpd_memlen = 0;
struct fastrpc_buf *init_mem; struct fastrpc_buf *init_mem;
struct fastrpc_mmap *sharedbuf_map = NULL; struct fastrpc_mmap *sharedbuf_map = NULL;
struct fastrpc_apps *me = &gfa;
unsigned long irq_flags = 0;
struct { struct {
int pgid; int pgid;
@@ -4161,20 +4163,6 @@ static int fastrpc_init_create_dynamic_process(struct fastrpc_file *fl,
ADSPRPC_ERR("donated memory allocated in userspace\n"); ADSPRPC_ERR("donated memory allocated in userspace\n");
goto bail; goto bail;
} }
/* Free any previous donated memory */
spin_lock(&fl->hlock);
locked = 1;
if (fl->init_mem) {
init_mem = fl->init_mem;
fl->init_mem = NULL;
spin_unlock(&fl->hlock);
locked = 0;
fastrpc_buf_free(init_mem, 0);
}
if (locked) {
spin_unlock(&fl->hlock);
locked = 0;
}
/* Allocate DMA buffer in kernel for donating to remote process /* Allocate DMA buffer in kernel for donating to remote process
* Unsigned PD requires additional memory because of the * Unsigned PD requires additional memory because of the
@@ -4278,13 +4266,21 @@ bail:
if (err) { if (err) {
ADSPRPC_ERR("failed with err %d\n", err); ADSPRPC_ERR("failed with err %d\n", err);
fl->dsp_process_state = PROCESS_CREATE_DEFAULT; fl->dsp_process_state = PROCESS_CREATE_DEFAULT;
spin_unlock(&fl->hlock);
locked = 0;
spin_lock_irqsave(&me->hlock, irq_flags);
glocked = 1;
if (!IS_ERR_OR_NULL(fl->init_mem)) { if (!IS_ERR_OR_NULL(fl->init_mem)) {
init_mem = fl->init_mem; init_mem = fl->init_mem;
fl->init_mem = NULL; fl->init_mem = NULL;
spin_unlock(&fl->hlock); spin_unlock_irqrestore(&me->hlock, irq_flags);
locked = 0; glocked = 0;
fastrpc_buf_free(init_mem, 0); fastrpc_buf_free(init_mem, 0);
} }
if (glocked) {
spin_unlock_irqrestore(&me->hlock, irq_flags);
glocked = 0;
}
} else { } else {
fl->dsp_process_state = PROCESS_CREATE_SUCCESS; fl->dsp_process_state = PROCESS_CREATE_SUCCESS;
} }
@@ -5939,6 +5935,7 @@ static int fastrpc_file_free(struct fastrpc_file *fl)
unsigned long irq_flags = 0; unsigned long irq_flags = 0;
bool is_locked = false; bool is_locked = false;
int i; int i;
struct fastrpc_buf *init_mem = NULL;
if (!fl) if (!fl)
return 0; return 0;
@@ -6010,8 +6007,22 @@ skip_dump_wait:
wake_up_interruptible(&fl->proc_state_notif.notif_wait_queue); wake_up_interruptible(&fl->proc_state_notif.notif_wait_queue);
spin_unlock_irqrestore(&fl->proc_state_notif.nqlock, flags); spin_unlock_irqrestore(&fl->proc_state_notif.nqlock, flags);
if (!IS_ERR_OR_NULL(fl->init_mem)) if (!is_locked) {
fastrpc_buf_free(fl->init_mem, 0); spin_lock_irqsave(&fl->apps->hlock, irq_flags);
is_locked = true;
}
if (!IS_ERR_OR_NULL(fl->init_mem)) {
init_mem = fl->init_mem;
fl->init_mem = NULL;
is_locked = false;
spin_unlock_irqrestore(&fl->apps->hlock, irq_flags);
fastrpc_buf_free(init_mem, 0);
}
if (is_locked) {
is_locked = false;
spin_unlock_irqrestore(&fl->apps->hlock, irq_flags);
}
fastrpc_context_list_dtor(fl); fastrpc_context_list_dtor(fl);
fastrpc_cached_buf_list_free(fl); fastrpc_cached_buf_list_free(fl);
if (!IS_ERR_OR_NULL(fl->hdr_bufs)) if (!IS_ERR_OR_NULL(fl->hdr_bufs))