|
@@ -47,6 +47,10 @@
|
|
|
#define MAX_USER_COMMAND_SIZE_WOWL_PATTERN 512
|
|
|
#define MAX_USER_COMMAND_SIZE_FRAME 4096
|
|
|
|
|
|
+#ifdef WLAN_POWER_DEBUGFS
|
|
|
+#define POWER_DEBUGFS_BUFFER_MAX_LEN 4096
|
|
|
+#endif
|
|
|
+
|
|
|
/**
|
|
|
* __wcnss_wowenable_write() - wow_enable debugfs handler
|
|
|
* @file: debugfs file handle
|
|
@@ -506,6 +510,241 @@ static ssize_t wcnss_patterngen_write(struct file *file,
|
|
|
return ret;
|
|
|
}
|
|
|
|
|
|
+#ifdef WLAN_POWER_DEBUGFS
|
|
|
+/**
|
|
|
+ * hdd_power_debugstats_cb() - callback routine for Power stats debugs
|
|
|
+ * @response: Pointer to Power stats response
|
|
|
+ * @context: Pointer to statsContext
|
|
|
+ *
|
|
|
+ * Return: None
|
|
|
+ */
|
|
|
+static void hdd_power_debugstats_cb(struct power_stats_response *response,
|
|
|
+ void *context)
|
|
|
+{
|
|
|
+ struct statsContext *stats_context;
|
|
|
+ struct power_stats_response *power_stats;
|
|
|
+ hdd_adapter_t *adapter;
|
|
|
+ uint32_t power_stats_len;
|
|
|
+ uint32_t stats_registers_len;
|
|
|
+
|
|
|
+ ENTER();
|
|
|
+ if (!context) {
|
|
|
+ hdd_err("context is NULL");
|
|
|
+ return;
|
|
|
+ }
|
|
|
+
|
|
|
+ stats_context = (struct statsContext *)context;
|
|
|
+
|
|
|
+ spin_lock(&hdd_context_lock);
|
|
|
+ adapter = stats_context->pAdapter;
|
|
|
+ if ((POWER_STATS_MAGIC != stats_context->magic) ||
|
|
|
+ (!adapter) || (WLAN_HDD_ADAPTER_MAGIC != adapter->magic)) {
|
|
|
+ spin_unlock(&hdd_context_lock);
|
|
|
+ hdd_err("Invalid context, adapter [%p] magic [%08x]",
|
|
|
+ adapter, stats_context->magic);
|
|
|
+ return;
|
|
|
+ }
|
|
|
+
|
|
|
+ stats_context->magic = 0;
|
|
|
+ stats_registers_len = (sizeof(response->debug_registers[0]) *
|
|
|
+ response->num_debug_register);
|
|
|
+ power_stats_len = stats_registers_len + sizeof(*power_stats);
|
|
|
+ adapter->chip_power_stats = qdf_mem_malloc(power_stats_len);
|
|
|
+ if (!adapter->chip_power_stats) {
|
|
|
+ hdd_err("Power stats memory alloc fails!");
|
|
|
+ goto exit_stats_cb;
|
|
|
+ }
|
|
|
+
|
|
|
+ power_stats = adapter->chip_power_stats;
|
|
|
+ power_stats->cumulative_sleep_time_ms
|
|
|
+ = response->cumulative_sleep_time_ms;
|
|
|
+ power_stats->cumulative_total_on_time_ms
|
|
|
+ = response->cumulative_total_on_time_ms;
|
|
|
+ power_stats->deep_sleep_enter_counter
|
|
|
+ = response->deep_sleep_enter_counter;
|
|
|
+ power_stats->last_deep_sleep_enter_tstamp_ms
|
|
|
+ = response->last_deep_sleep_enter_tstamp_ms;
|
|
|
+ power_stats->debug_register_fmt
|
|
|
+ = response->debug_register_fmt;
|
|
|
+ power_stats->num_debug_register
|
|
|
+ = response->num_debug_register;
|
|
|
+
|
|
|
+ power_stats->debug_registers = (uint32_t *)(power_stats + 1);
|
|
|
+
|
|
|
+ qdf_mem_copy(power_stats->debug_registers,
|
|
|
+ response->debug_registers, stats_registers_len);
|
|
|
+
|
|
|
+exit_stats_cb:
|
|
|
+ complete(&stats_context->completion);
|
|
|
+ spin_unlock(&hdd_context_lock);
|
|
|
+ EXIT();
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+ * __wlan_hdd_read_power_debugfs() - API to collect Chip power stats from FW
|
|
|
+ * @file: file pointer
|
|
|
+ * @buf: buffer
|
|
|
+ * @count: count
|
|
|
+ * @pos: position pointer
|
|
|
+ *
|
|
|
+ * Return: Number of bytes read on success, error number otherwise
|
|
|
+ */
|
|
|
+static ssize_t __wlan_hdd_read_power_debugfs(struct file *file,
|
|
|
+ char __user *buf,
|
|
|
+ size_t count, loff_t *pos)
|
|
|
+{
|
|
|
+ hdd_adapter_t *adapter;
|
|
|
+ hdd_context_t *hdd_ctx;
|
|
|
+ static struct statsContext context;
|
|
|
+ struct power_stats_response *chip_power_stats;
|
|
|
+ ssize_t ret_cnt = 0;
|
|
|
+ int rc = 0, j;
|
|
|
+ unsigned int len = 0;
|
|
|
+ char *power_debugfs_buf;
|
|
|
+
|
|
|
+ ENTER();
|
|
|
+ adapter = (hdd_adapter_t *)file->private_data;
|
|
|
+ if ((!adapter) || (WLAN_HDD_ADAPTER_MAGIC != adapter->magic)) {
|
|
|
+ hdd_err("Invalid adapter or adapter has invalid magic");
|
|
|
+ return -EINVAL;
|
|
|
+ }
|
|
|
+
|
|
|
+ hdd_ctx = WLAN_HDD_GET_CTX(adapter);
|
|
|
+ ret_cnt = wlan_hdd_validate_context(hdd_ctx);
|
|
|
+ if (0 != ret_cnt)
|
|
|
+ return ret_cnt;
|
|
|
+
|
|
|
+ if (adapter->chip_power_stats)
|
|
|
+ qdf_mem_free(adapter->chip_power_stats);
|
|
|
+
|
|
|
+ adapter->chip_power_stats = NULL;
|
|
|
+ context.pAdapter = adapter;
|
|
|
+ context.magic = POWER_STATS_MAGIC;
|
|
|
+
|
|
|
+ init_completion(&context.completion);
|
|
|
+
|
|
|
+ if (QDF_STATUS_SUCCESS !=
|
|
|
+ sme_power_debug_stats_req(hdd_ctx->hHal,
|
|
|
+ hdd_power_debugstats_cb,
|
|
|
+ &context)) {
|
|
|
+ hdd_err("chip power stats request failed");
|
|
|
+ return -EINVAL;
|
|
|
+ }
|
|
|
+
|
|
|
+ rc = wait_for_completion_timeout(&context.completion,
|
|
|
+ msecs_to_jiffies(WLAN_WAIT_TIME_POWER_STATS));
|
|
|
+ if (!rc) {
|
|
|
+ hdd_err("Target response timed out Power stats");
|
|
|
+ /* Invalidate the Stats context magic */
|
|
|
+ spin_lock(&hdd_context_lock);
|
|
|
+ context.magic = 0;
|
|
|
+ spin_unlock(&hdd_context_lock);
|
|
|
+ return -ETIMEDOUT;
|
|
|
+ }
|
|
|
+
|
|
|
+ chip_power_stats = adapter->chip_power_stats;
|
|
|
+ if (!chip_power_stats) {
|
|
|
+ hdd_err("Power stats retrieval fails!");
|
|
|
+ return -EINVAL;
|
|
|
+ }
|
|
|
+
|
|
|
+ power_debugfs_buf = qdf_mem_malloc(POWER_DEBUGFS_BUFFER_MAX_LEN);
|
|
|
+ if (!power_debugfs_buf) {
|
|
|
+ hdd_err("Power stats buffer alloc fails!");
|
|
|
+ qdf_mem_free(chip_power_stats);
|
|
|
+ adapter->chip_power_stats = NULL;
|
|
|
+ return -EINVAL;
|
|
|
+ }
|
|
|
+
|
|
|
+ len += scnprintf(power_debugfs_buf, POWER_DEBUGFS_BUFFER_MAX_LEN,
|
|
|
+ "POWER DEBUG STATS\n=================\n"
|
|
|
+ "cumulative_sleep_time_ms: %d\n"
|
|
|
+ "cumulative_total_on_time_ms: %d\n"
|
|
|
+ "deep_sleep_enter_counter: %d\n"
|
|
|
+ "last_deep_sleep_enter_tstamp_ms: %d\n"
|
|
|
+ "debug_register_fmt: %d\n"
|
|
|
+ "num_debug_register: %d\n",
|
|
|
+ chip_power_stats->cumulative_sleep_time_ms,
|
|
|
+ chip_power_stats->cumulative_total_on_time_ms,
|
|
|
+ chip_power_stats->deep_sleep_enter_counter,
|
|
|
+ chip_power_stats->last_deep_sleep_enter_tstamp_ms,
|
|
|
+ chip_power_stats->debug_register_fmt,
|
|
|
+ chip_power_stats->num_debug_register);
|
|
|
+
|
|
|
+ for (j = 0; j < chip_power_stats->num_debug_register; j++) {
|
|
|
+ if ((POWER_DEBUGFS_BUFFER_MAX_LEN - len) > 0)
|
|
|
+ len += scnprintf(power_debugfs_buf + len,
|
|
|
+ POWER_DEBUGFS_BUFFER_MAX_LEN - len,
|
|
|
+ "debug_registers[%d]: 0x%x\n", j,
|
|
|
+ chip_power_stats->debug_registers[j]);
|
|
|
+ else
|
|
|
+ j = chip_power_stats->num_debug_register;
|
|
|
+ }
|
|
|
+
|
|
|
+ qdf_mem_free(chip_power_stats);
|
|
|
+ adapter->chip_power_stats = NULL;
|
|
|
+
|
|
|
+ ret_cnt = simple_read_from_buffer(buf, count, pos,
|
|
|
+ power_debugfs_buf, len);
|
|
|
+ qdf_mem_free(power_debugfs_buf);
|
|
|
+ return ret_cnt;
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+ * wlan_hdd_read_power_debugfs() - SSR wrapper function to read power debugfs
|
|
|
+ * @file: file pointer
|
|
|
+ * @buf: buffer
|
|
|
+ * @count: count
|
|
|
+ * @pos: position pointer
|
|
|
+ *
|
|
|
+ * Return: Number of bytes read on success, error number otherwise
|
|
|
+ */
|
|
|
+static ssize_t wlan_hdd_read_power_debugfs(struct file *file,
|
|
|
+ char __user *buf,
|
|
|
+ size_t count, loff_t *pos)
|
|
|
+{
|
|
|
+ int ret;
|
|
|
+
|
|
|
+ cds_ssr_protect(__func__);
|
|
|
+ ret = __wlan_hdd_read_power_debugfs(file, buf, count, pos);
|
|
|
+ cds_ssr_unprotect(__func__);
|
|
|
+
|
|
|
+ return ret;
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+ * __wlan_hdd_open_power_debugfs() - Function to save private on open
|
|
|
+ * @inode: Pointer to inode structure
|
|
|
+ * @file: file pointer
|
|
|
+ *
|
|
|
+ * Return: zero
|
|
|
+ */
|
|
|
+static int __wlan_hdd_open_power_debugfs(struct inode *inode, struct file *file)
|
|
|
+{
|
|
|
+ file->private_data = inode->i_private;
|
|
|
+ return 0;
|
|
|
+}
|
|
|
+
|
|
|
+
|
|
|
+/**
|
|
|
+ * wlan_hdd_open_power_debugfs() - SSR wrapper function to save private on open
|
|
|
+ * @inode: Pointer to inode structure
|
|
|
+ * @file: file pointer
|
|
|
+ *
|
|
|
+ * Return: zero
|
|
|
+ */
|
|
|
+static int wlan_hdd_open_power_debugfs(struct inode *inode, struct file *file)
|
|
|
+{
|
|
|
+ int ret;
|
|
|
+
|
|
|
+ cds_ssr_protect(__func__);
|
|
|
+ ret = __wlan_hdd_open_power_debugfs(inode, file);
|
|
|
+ cds_ssr_unprotect(__func__);
|
|
|
+
|
|
|
+ return ret;
|
|
|
+}
|
|
|
+#endif
|
|
|
+
|
|
|
/**
|
|
|
* __wcnss_debugfs_open() - Generic debugfs open() handler
|
|
|
* @inode: inode of the debugfs file
|
|
@@ -577,6 +816,37 @@ static const struct file_operations fops_patterngen = {
|
|
|
.llseek = default_llseek,
|
|
|
};
|
|
|
|
|
|
+#ifdef WLAN_POWER_DEBUGFS
|
|
|
+static const struct file_operations fops_powerdebugs = {
|
|
|
+ .read = wlan_hdd_read_power_debugfs,
|
|
|
+ .open = wlan_hdd_open_power_debugfs,
|
|
|
+ .owner = THIS_MODULE,
|
|
|
+ .llseek = default_llseek,
|
|
|
+};
|
|
|
+
|
|
|
+/**
|
|
|
+ * wlan_hdd_create_power_stats_file() - API to create power stats file
|
|
|
+ * @adapter: interface adapter pointer
|
|
|
+ *
|
|
|
+ * Return: QDF_STATUS
|
|
|
+ */
|
|
|
+static QDF_STATUS wlan_hdd_create_power_stats_file(hdd_adapter_t *adapter)
|
|
|
+{
|
|
|
+ if (!debugfs_create_file("power_stats", S_IRUSR | S_IRGRP | S_IROTH,
|
|
|
+ adapter->debugfs_phy, adapter,
|
|
|
+ &fops_powerdebugs))
|
|
|
+ return QDF_STATUS_E_FAILURE;
|
|
|
+
|
|
|
+ return QDF_STATUS_SUCCESS;
|
|
|
+}
|
|
|
+
|
|
|
+#else
|
|
|
+static QDF_STATUS wlan_hdd_create_power_stats_file(hdd_adapter_t *adapter)
|
|
|
+{
|
|
|
+ return QDF_STATUS_SUCCESS;
|
|
|
+}
|
|
|
+#endif
|
|
|
+
|
|
|
/**
|
|
|
* hdd_debugfs_init() - Initialize debugfs interface
|
|
|
* @adapter: interface adapter pointer
|
|
@@ -612,6 +882,9 @@ QDF_STATUS hdd_debugfs_init(hdd_adapter_t *adapter)
|
|
|
&fops_patterngen))
|
|
|
return QDF_STATUS_E_FAILURE;
|
|
|
|
|
|
+ if (QDF_STATUS_SUCCESS != wlan_hdd_create_power_stats_file(adapter))
|
|
|
+ return QDF_STATUS_E_FAILURE;
|
|
|
+
|
|
|
return QDF_STATUS_SUCCESS;
|
|
|
}
|
|
|
|