mirror of
https://github.com/Divested-Mobile/DivestOS-Build.git
synced 2025-01-20 04:11:32 -05:00
176 lines
6.0 KiB
Diff
176 lines
6.0 KiB
Diff
|
From 01673e148223c10782b03c5485aff2a82b1900c4 Mon Sep 17 00:00:00 2001
|
||
|
From: Abdulla Anam <abdullahanam@codeaurora.org>
|
||
|
Date: Wed, 22 Mar 2017 14:00:10 +0530
|
||
|
Subject: msm: vidc: Protect debug_buffer access in core_info_read with lock.
|
||
|
|
||
|
Serialize core_info_read with lock so that multiple concurrent
|
||
|
threads do not cause the write to overflow. Also have the bound
|
||
|
check to avoid overflow in write_str function.
|
||
|
|
||
|
CRs-Fixed: 2013361
|
||
|
Change-Id: Ia18a4b94cafd69af1d367861f2499fc202f18e9f
|
||
|
Signed-off-by: Abdulla Anam <abdullahanam@codeaurora.org>
|
||
|
---
|
||
|
drivers/media/platform/msm/vidc/msm_v4l2_vidc.c | 3 ++
|
||
|
drivers/media/platform/msm/vidc/msm_vidc_debug.c | 37 ++++++++++++++++++++----
|
||
|
drivers/media/platform/msm/vidc/msm_vidc_debug.h | 1 +
|
||
|
3 files changed, 35 insertions(+), 6 deletions(-)
|
||
|
|
||
|
diff --git a/drivers/media/platform/msm/vidc/msm_v4l2_vidc.c b/drivers/media/platform/msm/vidc/msm_v4l2_vidc.c
|
||
|
index 319df5e..f77b943 100644
|
||
|
--- a/drivers/media/platform/msm/vidc/msm_v4l2_vidc.c
|
||
|
+++ b/drivers/media/platform/msm/vidc/msm_v4l2_vidc.c
|
||
|
@@ -789,6 +789,8 @@ static int __init msm_vidc_init(void)
|
||
|
if (rc) {
|
||
|
dprintk(VIDC_ERR,
|
||
|
"Failed to register platform driver\n");
|
||
|
+ msm_vidc_debugfs_deinit_drv();
|
||
|
+ debugfs_remove_recursive(vidc_driver->debugfs_root);
|
||
|
kfree(vidc_driver);
|
||
|
vidc_driver = NULL;
|
||
|
}
|
||
|
@@ -799,6 +801,7 @@ static int __init msm_vidc_init(void)
|
||
|
static void __exit msm_vidc_exit(void)
|
||
|
{
|
||
|
platform_driver_unregister(&msm_vidc_driver);
|
||
|
+ msm_vidc_debugfs_deinit_drv();
|
||
|
debugfs_remove_recursive(vidc_driver->debugfs_root);
|
||
|
kfree(vidc_driver);
|
||
|
vidc_driver = NULL;
|
||
|
diff --git a/drivers/media/platform/msm/vidc/msm_vidc_debug.c b/drivers/media/platform/msm/vidc/msm_vidc_debug.c
|
||
|
index d34da57..7780990 100644
|
||
|
--- a/drivers/media/platform/msm/vidc/msm_vidc_debug.c
|
||
|
+++ b/drivers/media/platform/msm/vidc/msm_vidc_debug.c
|
||
|
@@ -1,4 +1,4 @@
|
||
|
-/* Copyright (c) 2012-2016, The Linux Foundation. All rights reserved.
|
||
|
+/* Copyright (c) 2012-2017, The Linux Foundation. All rights reserved.
|
||
|
*
|
||
|
* This program is free software; you can redistribute it and/or modify
|
||
|
* it under the terms of the GNU General Public License version 2 and
|
||
|
@@ -34,6 +34,7 @@ int msm_vidc_debug_timeout = 0;
|
||
|
#define MAX_DBG_BUF_SIZE 4096
|
||
|
|
||
|
struct debug_buffer {
|
||
|
+ struct mutex lock;
|
||
|
char ptr[MAX_DBG_BUF_SIZE];
|
||
|
char *curr;
|
||
|
u32 filled_size;
|
||
|
@@ -60,8 +61,12 @@ static u32 write_str(struct debug_buffer *buffer, const char *fmt, ...)
|
||
|
{
|
||
|
va_list args;
|
||
|
u32 size;
|
||
|
+
|
||
|
+ char *curr = buffer->curr;
|
||
|
+ char *end = buffer->ptr + MAX_DBG_BUF_SIZE;
|
||
|
+
|
||
|
va_start(args, fmt);
|
||
|
- size = vscnprintf(buffer->curr, MAX_DBG_BUF_SIZE - 1, fmt, args);
|
||
|
+ size = vscnprintf(curr, end - curr, fmt, args);
|
||
|
va_end(args);
|
||
|
buffer->curr += size;
|
||
|
buffer->filled_size += size;
|
||
|
@@ -75,12 +80,15 @@ static ssize_t core_info_read(struct file *file, char __user *buf,
|
||
|
struct hfi_device *hdev;
|
||
|
struct hal_fw_info fw_info = { {0} };
|
||
|
int i = 0, rc = 0;
|
||
|
+ ssize_t len = 0;
|
||
|
|
||
|
if (!core || !core->device) {
|
||
|
dprintk(VIDC_ERR, "Invalid params, core: %pK\n", core);
|
||
|
return 0;
|
||
|
}
|
||
|
hdev = core->device;
|
||
|
+
|
||
|
+ mutex_lock(&dbg_buf.lock);
|
||
|
INIT_DBG_BUF(dbg_buf);
|
||
|
write_str(&dbg_buf, "===============================\n");
|
||
|
write_str(&dbg_buf, "CORE %d: %pK\n", core->id, core);
|
||
|
@@ -104,8 +112,11 @@ err_fw_info:
|
||
|
completion_done(&core->completions[SYS_MSG_INDEX(i)]) ?
|
||
|
"pending" : "done");
|
||
|
}
|
||
|
- return simple_read_from_buffer(buf, count, ppos,
|
||
|
+ len = simple_read_from_buffer(buf, count, ppos,
|
||
|
dbg_buf.ptr, dbg_buf.filled_size);
|
||
|
+
|
||
|
+ mutex_unlock(&dbg_buf.lock);
|
||
|
+ return len;
|
||
|
}
|
||
|
|
||
|
static const struct file_operations core_info_fops = {
|
||
|
@@ -143,7 +154,10 @@ static const struct file_operations ssr_fops = {
|
||
|
struct dentry *msm_vidc_debugfs_init_drv(void)
|
||
|
{
|
||
|
bool ok = false;
|
||
|
- struct dentry *dir = debugfs_create_dir("msm_vidc", NULL);
|
||
|
+ struct dentry *dir = NULL;
|
||
|
+
|
||
|
+ mutex_init(&dbg_buf.lock);
|
||
|
+ dir = debugfs_create_dir("msm_vidc", NULL);
|
||
|
if (IS_ERR_OR_NULL(dir)) {
|
||
|
dir = NULL;
|
||
|
goto failed_create_dir;
|
||
|
@@ -212,6 +226,7 @@ struct dentry *msm_vidc_debugfs_init_core(struct msm_vidc_core *core,
|
||
|
dprintk(VIDC_ERR, "Failed to create debugfs for msm_vidc\n");
|
||
|
goto failed_create_dir;
|
||
|
}
|
||
|
+
|
||
|
if (!debugfs_create_file("info", S_IRUGO, dir, core, &core_info_fops)) {
|
||
|
dprintk(VIDC_ERR, "debugfs_create_file: fail\n");
|
||
|
goto failed_create_dir;
|
||
|
@@ -266,6 +281,7 @@ static ssize_t inst_info_read(struct file *file, char __user *buf,
|
||
|
struct msm_vidc_inst *inst = file->private_data;
|
||
|
struct msm_vidc_core *core = inst ? inst->core : NULL;
|
||
|
int i, j;
|
||
|
+ ssize_t len = 0;
|
||
|
|
||
|
if (!inst || !core) {
|
||
|
dprintk(VIDC_ERR, "Invalid params, core: %pK inst %pK\n",
|
||
|
@@ -277,6 +293,7 @@ static ssize_t inst_info_read(struct file *file, char __user *buf,
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
+ mutex_lock(&dbg_buf.lock);
|
||
|
INIT_DBG_BUF(dbg_buf);
|
||
|
write_str(&dbg_buf, "===============================\n");
|
||
|
write_str(&dbg_buf, "INSTANCE: %pK (%s)\n", inst,
|
||
|
@@ -333,9 +350,12 @@ static ssize_t inst_info_read(struct file *file, char __user *buf,
|
||
|
write_str(&dbg_buf, "FBD Count: %d\n", inst->count.fbd);
|
||
|
|
||
|
publish_unreleased_reference(inst);
|
||
|
- put_inst(inst);
|
||
|
- return simple_read_from_buffer(buf, count, ppos,
|
||
|
+ len = simple_read_from_buffer(buf, count, ppos,
|
||
|
dbg_buf.ptr, dbg_buf.filled_size);
|
||
|
+
|
||
|
+ mutex_unlock(&dbg_buf.lock);
|
||
|
+ put_inst(inst);
|
||
|
+ return len;
|
||
|
}
|
||
|
|
||
|
static const struct file_operations inst_info_fops = {
|
||
|
@@ -416,3 +436,8 @@ void msm_vidc_debugfs_update(struct msm_vidc_inst *inst,
|
||
|
}
|
||
|
}
|
||
|
|
||
|
+void msm_vidc_debugfs_deinit_drv(void)
|
||
|
+{
|
||
|
+ mutex_destroy(&dbg_buf.lock);
|
||
|
+}
|
||
|
+
|
||
|
diff --git a/drivers/media/platform/msm/vidc/msm_vidc_debug.h b/drivers/media/platform/msm/vidc/msm_vidc_debug.h
|
||
|
index 55485c6..abf8d3a 100644
|
||
|
--- a/drivers/media/platform/msm/vidc/msm_vidc_debug.h
|
||
|
+++ b/drivers/media/platform/msm/vidc/msm_vidc_debug.h
|
||
|
@@ -126,6 +126,7 @@ struct dentry *msm_vidc_debugfs_init_inst(struct msm_vidc_inst *inst,
|
||
|
struct dentry *parent);
|
||
|
void msm_vidc_debugfs_update(struct msm_vidc_inst *inst,
|
||
|
enum msm_vidc_debugfs_event e);
|
||
|
+void msm_vidc_debugfs_deinit_drv(void);
|
||
|
|
||
|
static inline void tic(struct msm_vidc_inst *i, enum profiling_points p,
|
||
|
char *b)
|
||
|
--
|
||
|
cgit v1.1
|
||
|
|