mirror of https://gitee.com/openkylin/linux.git
staging: tidspbridge: fix bridge_open memory leaks
There are two members of pr_ctxt allocated during bridge_open that are never freed resulting in memory leaks, these are stream_id and node_id, they are now freed on release of the handle (bridge_release) right before freeing pr_ctxt. Error path for bridge_open was also fixed since the same variables could result in memory leaking due to missing handling of failure scenarios. While at it, the indentation changes were introduced to avoid interleaved goto statements inside big if blocks. Signed-off-by: Omar Ramirez Luna <omar.ramirez@ti.com> Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
This commit is contained in:
parent
00da8eddf9
commit
2573897816
|
@ -139,7 +139,9 @@ static int bridge_open(struct inode *ip, struct file *filp)
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
pr_ctxt = kzalloc(sizeof(struct process_context), GFP_KERNEL);
|
pr_ctxt = kzalloc(sizeof(struct process_context), GFP_KERNEL);
|
||||||
if (pr_ctxt) {
|
if (!pr_ctxt)
|
||||||
|
return -ENOMEM;
|
||||||
|
|
||||||
pr_ctxt->res_state = PROC_RES_ALLOCATED;
|
pr_ctxt->res_state = PROC_RES_ALLOCATED;
|
||||||
spin_lock_init(&pr_ctxt->dmm_map_lock);
|
spin_lock_init(&pr_ctxt->dmm_map_lock);
|
||||||
INIT_LIST_HEAD(&pr_ctxt->dmm_map_list);
|
INIT_LIST_HEAD(&pr_ctxt->dmm_map_list);
|
||||||
|
@ -147,27 +149,32 @@ static int bridge_open(struct inode *ip, struct file *filp)
|
||||||
INIT_LIST_HEAD(&pr_ctxt->dmm_rsv_list);
|
INIT_LIST_HEAD(&pr_ctxt->dmm_rsv_list);
|
||||||
|
|
||||||
pr_ctxt->node_id = kzalloc(sizeof(struct idr), GFP_KERNEL);
|
pr_ctxt->node_id = kzalloc(sizeof(struct idr), GFP_KERNEL);
|
||||||
if (pr_ctxt->node_id) {
|
if (!pr_ctxt->node_id) {
|
||||||
idr_init(pr_ctxt->node_id);
|
|
||||||
} else {
|
|
||||||
status = -ENOMEM;
|
status = -ENOMEM;
|
||||||
goto err;
|
goto err1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
idr_init(pr_ctxt->node_id);
|
||||||
|
|
||||||
pr_ctxt->stream_id = kzalloc(sizeof(struct idr), GFP_KERNEL);
|
pr_ctxt->stream_id = kzalloc(sizeof(struct idr), GFP_KERNEL);
|
||||||
if (pr_ctxt->stream_id)
|
if (!pr_ctxt->stream_id) {
|
||||||
idr_init(pr_ctxt->stream_id);
|
|
||||||
else
|
|
||||||
status = -ENOMEM;
|
|
||||||
} else {
|
|
||||||
status = -ENOMEM;
|
status = -ENOMEM;
|
||||||
|
goto err2;
|
||||||
}
|
}
|
||||||
err:
|
|
||||||
|
idr_init(pr_ctxt->stream_id);
|
||||||
|
|
||||||
filp->private_data = pr_ctxt;
|
filp->private_data = pr_ctxt;
|
||||||
|
|
||||||
#ifdef CONFIG_TIDSPBRIDGE_RECOVERY
|
#ifdef CONFIG_TIDSPBRIDGE_RECOVERY
|
||||||
if (!status)
|
|
||||||
atomic_inc(&bridge_cref);
|
atomic_inc(&bridge_cref);
|
||||||
#endif
|
#endif
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
err2:
|
||||||
|
kfree(pr_ctxt->node_id);
|
||||||
|
err1:
|
||||||
|
kfree(pr_ctxt);
|
||||||
return status;
|
return status;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -189,6 +196,8 @@ static int bridge_release(struct inode *ip, struct file *filp)
|
||||||
flush_signals(current);
|
flush_signals(current);
|
||||||
drv_remove_all_resources(pr_ctxt);
|
drv_remove_all_resources(pr_ctxt);
|
||||||
proc_detach(pr_ctxt);
|
proc_detach(pr_ctxt);
|
||||||
|
kfree(pr_ctxt->node_id);
|
||||||
|
kfree(pr_ctxt->stream_id);
|
||||||
kfree(pr_ctxt);
|
kfree(pr_ctxt);
|
||||||
|
|
||||||
filp->private_data = NULL;
|
filp->private_data = NULL;
|
||||||
|
|
Loading…
Reference in New Issue