mirror of https://gitee.com/openkylin/linux.git
iwlwifi: runtime: avoid calling debugfs read functions more than once
Upon first calling read() on a debugfs file, invoke iwl_dbgfs_##name##_read and store the response buffer on the heap, so subsequent read() calls don't need to invoke said function again. This is done because cat etc will call read() repeatedly until EOF is reached (or read() returns 0), which in the current implementation will cause said function to be invoked multiple times. With the current implementation this can also cause buggy behavior in some weird edge cases where the first invocation returns a string of length n, and the second of length m>n: The last m-n characters of the second invocation will be printed to screen. Signed-off-by: Naftali Goldstein <naftali.goldstein@intel.com> Signed-off-by: Luca Coelho <luciano.coelho@intel.com>
This commit is contained in:
parent
fe997b00c5
commit
dd2690579f
|
@ -8,6 +8,7 @@
|
||||||
* Copyright(c) 2012 - 2014 Intel Corporation. All rights reserved.
|
* Copyright(c) 2012 - 2014 Intel Corporation. All rights reserved.
|
||||||
* Copyright(c) 2013 - 2015 Intel Mobile Communications GmbH
|
* Copyright(c) 2013 - 2015 Intel Mobile Communications GmbH
|
||||||
* Copyright(c) 2016 - 2017 Intel Deutschland GmbH
|
* Copyright(c) 2016 - 2017 Intel Deutschland GmbH
|
||||||
|
* Copyright (C) 2018 Intel Corporation
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of version 2 of the GNU General Public License as
|
* it under the terms of version 2 of the GNU General Public License as
|
||||||
|
@ -33,6 +34,7 @@
|
||||||
* Copyright(c) 2012 - 2014 Intel Corporation. All rights reserved.
|
* Copyright(c) 2012 - 2014 Intel Corporation. All rights reserved.
|
||||||
* Copyright(c) 2013 - 2015 Intel Mobile Communications GmbH
|
* Copyright(c) 2013 - 2015 Intel Mobile Communications GmbH
|
||||||
* Copyright(c) 2016 - 2017 Intel Deutschland GmbH
|
* Copyright(c) 2016 - 2017 Intel Deutschland GmbH
|
||||||
|
* Copyright (C) 2018 Intel Corporation
|
||||||
* All rights reserved.
|
* All rights reserved.
|
||||||
*
|
*
|
||||||
* Redistribution and use in source and binary forms, with or without
|
* Redistribution and use in source and binary forms, with or without
|
||||||
|
@ -66,55 +68,117 @@
|
||||||
#include "debugfs.h"
|
#include "debugfs.h"
|
||||||
#include "dbg.h"
|
#include "dbg.h"
|
||||||
|
|
||||||
#define FWRT_DEBUGFS_READ_FILE_OPS(name) \
|
#define FWRT_DEBUGFS_OPEN_WRAPPER(name, buflen, argtype) \
|
||||||
static ssize_t iwl_dbgfs_##name##_read(struct iwl_fw_runtime *fwrt, \
|
struct dbgfs_##name##_data { \
|
||||||
char *buf, size_t count, \
|
argtype *arg; \
|
||||||
loff_t *ppos); \
|
bool read_done; \
|
||||||
static const struct file_operations iwl_dbgfs_##name##_ops = { \
|
ssize_t rlen; \
|
||||||
.read = iwl_dbgfs_##name##_read, \
|
char rbuf[buflen]; \
|
||||||
.open = simple_open, \
|
}; \
|
||||||
.llseek = generic_file_llseek, \
|
static int _iwl_dbgfs_##name##_open(struct inode *inode, \
|
||||||
|
struct file *file) \
|
||||||
|
{ \
|
||||||
|
struct dbgfs_##name##_data *data; \
|
||||||
|
\
|
||||||
|
data = kzalloc(sizeof(*data), GFP_KERNEL); \
|
||||||
|
if (!data) \
|
||||||
|
return -ENOMEM; \
|
||||||
|
\
|
||||||
|
data->read_done = false; \
|
||||||
|
data->arg = inode->i_private; \
|
||||||
|
file->private_data = data; \
|
||||||
|
\
|
||||||
|
return 0; \
|
||||||
}
|
}
|
||||||
|
|
||||||
#define FWRT_DEBUGFS_WRITE_WRAPPER(name, buflen) \
|
#define FWRT_DEBUGFS_READ_WRAPPER(name) \
|
||||||
static ssize_t iwl_dbgfs_##name##_write(struct iwl_fw_runtime *fwrt, \
|
static ssize_t _iwl_dbgfs_##name##_read(struct file *file, \
|
||||||
char *buf, size_t count, \
|
char __user *user_buf, \
|
||||||
loff_t *ppos); \
|
size_t count, loff_t *ppos) \
|
||||||
|
{ \
|
||||||
|
struct dbgfs_##name##_data *data = file->private_data; \
|
||||||
|
\
|
||||||
|
if (!data->read_done) { \
|
||||||
|
data->read_done = true; \
|
||||||
|
data->rlen = iwl_dbgfs_##name##_read(data->arg, \
|
||||||
|
sizeof(data->rbuf),\
|
||||||
|
data->rbuf); \
|
||||||
|
} \
|
||||||
|
\
|
||||||
|
if (data->rlen < 0) \
|
||||||
|
return data->rlen; \
|
||||||
|
return simple_read_from_buffer(user_buf, count, ppos, \
|
||||||
|
data->rbuf, data->rlen); \
|
||||||
|
}
|
||||||
|
|
||||||
|
static int _iwl_dbgfs_release(struct inode *inode, struct file *file)
|
||||||
|
{
|
||||||
|
kfree(file->private_data);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
#define _FWRT_DEBUGFS_READ_FILE_OPS(name, buflen, argtype) \
|
||||||
|
FWRT_DEBUGFS_OPEN_WRAPPER(name, buflen, argtype) \
|
||||||
|
FWRT_DEBUGFS_READ_WRAPPER(name) \
|
||||||
|
static const struct file_operations iwl_dbgfs_##name##_ops = { \
|
||||||
|
.read = _iwl_dbgfs_##name##_read, \
|
||||||
|
.open = _iwl_dbgfs_##name##_open, \
|
||||||
|
.llseek = generic_file_llseek, \
|
||||||
|
.release = _iwl_dbgfs_release, \
|
||||||
|
}
|
||||||
|
|
||||||
|
#define FWRT_DEBUGFS_WRITE_WRAPPER(name, buflen, argtype) \
|
||||||
static ssize_t _iwl_dbgfs_##name##_write(struct file *file, \
|
static ssize_t _iwl_dbgfs_##name##_write(struct file *file, \
|
||||||
const char __user *user_buf, \
|
const char __user *user_buf, \
|
||||||
size_t count, loff_t *ppos) \
|
size_t count, loff_t *ppos) \
|
||||||
{ \
|
{ \
|
||||||
struct iwl_fw_runtime *fwrt = file->private_data; \
|
argtype *arg = \
|
||||||
|
((struct dbgfs_##name##_data *)file->private_data)->arg;\
|
||||||
char buf[buflen] = {}; \
|
char buf[buflen] = {}; \
|
||||||
size_t buf_size = min(count, sizeof(buf) - 1); \
|
size_t buf_size = min(count, sizeof(buf) - 1); \
|
||||||
\
|
\
|
||||||
if (copy_from_user(buf, user_buf, buf_size)) \
|
if (copy_from_user(buf, user_buf, buf_size)) \
|
||||||
return -EFAULT; \
|
return -EFAULT; \
|
||||||
\
|
\
|
||||||
return iwl_dbgfs_##name##_write(fwrt, buf, buf_size, ppos); \
|
return iwl_dbgfs_##name##_write(arg, buf, buf_size); \
|
||||||
}
|
}
|
||||||
|
|
||||||
#define FWRT_DEBUGFS_READ_WRITE_FILE_OPS(name, buflen) \
|
#define _FWRT_DEBUGFS_READ_WRITE_FILE_OPS(name, buflen, argtype) \
|
||||||
FWRT_DEBUGFS_WRITE_WRAPPER(name, buflen) \
|
FWRT_DEBUGFS_OPEN_WRAPPER(name, buflen, argtype) \
|
||||||
|
FWRT_DEBUGFS_WRITE_WRAPPER(name, buflen, argtype) \
|
||||||
|
FWRT_DEBUGFS_READ_WRAPPER(name) \
|
||||||
static const struct file_operations iwl_dbgfs_##name##_ops = { \
|
static const struct file_operations iwl_dbgfs_##name##_ops = { \
|
||||||
.write = _iwl_dbgfs_##name##_write, \
|
.write = _iwl_dbgfs_##name##_write, \
|
||||||
.read = iwl_dbgfs_##name##_read, \
|
.read = _iwl_dbgfs_##name##_read, \
|
||||||
.open = simple_open, \
|
.open = _iwl_dbgfs_##name##_open, \
|
||||||
.llseek = generic_file_llseek, \
|
.llseek = generic_file_llseek, \
|
||||||
|
.release = _iwl_dbgfs_release, \
|
||||||
}
|
}
|
||||||
|
|
||||||
#define FWRT_DEBUGFS_WRITE_FILE_OPS(name, buflen) \
|
#define _FWRT_DEBUGFS_WRITE_FILE_OPS(name, buflen, argtype) \
|
||||||
FWRT_DEBUGFS_WRITE_WRAPPER(name, buflen) \
|
FWRT_DEBUGFS_OPEN_WRAPPER(name, buflen, argtype) \
|
||||||
|
FWRT_DEBUGFS_WRITE_WRAPPER(name, buflen, argtype) \
|
||||||
static const struct file_operations iwl_dbgfs_##name##_ops = { \
|
static const struct file_operations iwl_dbgfs_##name##_ops = { \
|
||||||
.write = _iwl_dbgfs_##name##_write, \
|
.write = _iwl_dbgfs_##name##_write, \
|
||||||
.open = simple_open, \
|
.open = _iwl_dbgfs_##name##_open, \
|
||||||
.llseek = generic_file_llseek, \
|
.llseek = generic_file_llseek, \
|
||||||
|
.release = _iwl_dbgfs_release, \
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#define FWRT_DEBUGFS_READ_FILE_OPS(name, bufsz) \
|
||||||
|
_FWRT_DEBUGFS_READ_FILE_OPS(name, bufsz, struct iwl_fw_runtime)
|
||||||
|
|
||||||
|
#define FWRT_DEBUGFS_WRITE_FILE_OPS(name, bufsz) \
|
||||||
|
_FWRT_DEBUGFS_WRITE_FILE_OPS(name, bufsz, struct iwl_fw_runtime)
|
||||||
|
|
||||||
|
#define FWRT_DEBUGFS_READ_WRITE_FILE_OPS(name, bufsz) \
|
||||||
|
_FWRT_DEBUGFS_READ_WRITE_FILE_OPS(name, bufsz, struct iwl_fw_runtime)
|
||||||
|
|
||||||
#define FWRT_DEBUGFS_ADD_FILE_ALIAS(alias, name, parent, mode) do { \
|
#define FWRT_DEBUGFS_ADD_FILE_ALIAS(alias, name, parent, mode) do { \
|
||||||
if (!debugfs_create_file(alias, mode, parent, fwrt, \
|
if (!debugfs_create_file(alias, mode, parent, fwrt, \
|
||||||
&iwl_dbgfs_##name##_ops)) \
|
&iwl_dbgfs_##name##_ops)) \
|
||||||
goto err; \
|
goto err; \
|
||||||
} while (0)
|
} while (0)
|
||||||
#define FWRT_DEBUGFS_ADD_FILE(name, parent, mode) \
|
#define FWRT_DEBUGFS_ADD_FILE(name, parent, mode) \
|
||||||
FWRT_DEBUGFS_ADD_FILE_ALIAS(#name, name, parent, mode)
|
FWRT_DEBUGFS_ADD_FILE_ALIAS(#name, name, parent, mode)
|
||||||
|
@ -173,8 +237,7 @@ void iwl_fw_trigger_timestamp(struct iwl_fw_runtime *fwrt, u32 delay)
|
||||||
}
|
}
|
||||||
|
|
||||||
static ssize_t iwl_dbgfs_timestamp_marker_write(struct iwl_fw_runtime *fwrt,
|
static ssize_t iwl_dbgfs_timestamp_marker_write(struct iwl_fw_runtime *fwrt,
|
||||||
char *buf, size_t count,
|
char *buf, size_t count)
|
||||||
loff_t *ppos)
|
|
||||||
{
|
{
|
||||||
int ret;
|
int ret;
|
||||||
u32 delay;
|
u32 delay;
|
||||||
|
|
Loading…
Reference in New Issue