[qca-nss-clients] adding Hawkeye profiling support
Allowing user program to change status event counter configuration;
Revised counter Tx process from using stop/start to directly use counter request.
Change-Id: I267ac171b1bb35cbc692425153cabaac51750163
Signed-off-by: Guojun Jin <g.jin@codeaurora.org>
diff --git a/profiler/profile.c b/profiler/profile.c
index 50be1fc..e4038de 100644
--- a/profiler/profile.c
+++ b/profiler/profile.c
@@ -34,6 +34,7 @@
#include <asm/uaccess.h>
#include <asm/page.h>
#include <asm/thread_info.h>
+#include <linux/ctype.h>
#include <nss_api_if.h>
#include "profilenode.h"
@@ -87,6 +88,9 @@
* LINUX and Ultra counters must all fit in one packet
*/
#define PROFILE_LINUX_MAX_COUNTERS 40
+#define PROFILE_STS_EVENT_COUNTERS 8
+#define PROFILE_STS_EVENT_THREAD_BITS 5
+
static int profile_num_counters = 0;
static volatile unsigned int *profile_counter[PROFILE_LINUX_MAX_COUNTERS];
static char profile_name[PROFILE_LINUX_MAX_COUNTERS][PROFILE_COUNTER_NAME_LENGTH];
@@ -183,7 +187,7 @@
ph.pph.ddr_freq = pn->pnc.un.ddr_freq;
ph.pph.cpu_id = pn->pnc.un.cpu_id;
ph.pph.seq_num = htonl(pn->profile_sequence_num);
- ph.pph.sample_stack_words = htonl(PROFILE_STACK_WORDS);
+ ph.pph.sample_stack_words = PROFILE_STACK_WORDS;
ns = (blen - sizeof(ph)) / sizeof(struct profile_sample);
profileInfo("%X: blen %d ns = %d psc_hd count %d ssets %d phs %d pss %d\n", pn->profile_sequence_num, blen, ns, psc_hd->count, psc_hd->exh.sample_sets, sizeof(ph), sizeof(struct profile_sample));
@@ -262,7 +266,7 @@
};
/*
- * make a packet full of performance counters
+ * make a packet full of performance counters (software)
*/
static int profile_make_stats_packet(char *buf, int bytes, struct profile_io *pn)
{
@@ -344,16 +348,24 @@
if (!pn->pnc.enabled && nss_get_state(pn->ctx) == NSS_STATE_INITIALIZED) {
nss_tx_status_t ret;
+
+ /*
+ * sw_ksp_ptr is used as event flag. NULL means normal I/O
+ */
+ pn->sw_ksp_ptr = NULL;
pn->pnc.enabled = 1;
pn->profile_first_packet = 1;
pn->pnc.un.hd_magic = UBI32_PROFILE_HD_MAGIC | NSS_PROFILER_START_MSG;
- ret = nss_profiler_if_tx_buf(pn->ctx, &pn->pnc.un, sizeof(pn->pnc.un), profiler_handle_reply);
- profileInfo("%s: %d -- %p: ccl %p sp %p\n", __func__, ret, pn, pn->ccl, pn->pnc.samples);
+ ret = nss_profiler_if_tx_buf(pn->ctx, &pn->pnc.un,
+ sizeof(pn->pnc.un), profiler_handle_reply, pn);
+ profileInfo("%s: %d -- %p: ccl %p sp %p\n", __func__, ret,
+ pn, pn->ccl, pn->pnc.samples);
filp->private_data = pn;
return 0;
}
- profileWarn("profile ena %d nss stat %x\n", pn->pnc.enabled, nss_get_state(pn->ctx));
+ profileWarn("profile ena %d nss stat %x\n", pn->pnc.enabled,
+ nss_get_state(pn->ctx));
return -EBUSY;
}
@@ -373,6 +385,14 @@
if (!pn->pnc.enabled) {
return -EPERM;
}
+ if (pn->sw_ksp_ptr) {
+ struct debug_box *db = (struct debug_box *) pn->sw_ksp_ptr;
+ slen = (PROFILE_STS_EVENT_COUNTERS + 1) * sizeof(db->data[0]);
+ if (copy_to_user(buf, db->data, slen))
+ return -EFAULT;
+ return slen;
+ }
+
if (!pn->pnc.samples) {
return -ENOMEM;
}
@@ -408,8 +428,10 @@
nss_tx_status_t ret;
pn->pnc.enabled = 1;
pn->pnc.un.hd_magic = UBI32_PROFILE_HD_MAGIC | NSS_PROFILER_START_MSG;
- ret = nss_profiler_if_tx_buf(pn->ctx, &pn->pnc.un, sizeof(pn->pnc.un), profiler_handle_reply);
- profileWarn("%s: restart %d -- %p: ccl %p sp %p\n", __func__, ret, pn, pn->ccl, pn->pnc.samples);
+ ret = nss_profiler_if_tx_buf(pn->ctx, &pn->pnc.un, sizeof(pn->pnc.un),
+ profiler_handle_reply, pn);
+ profileWarn("%s: restart %d -- %p: ccl %p sp %p\n", __func__,
+ ret, pn, pn->ccl, pn->pnc.samples);
}
return result + slen;
@@ -427,9 +449,11 @@
if (pn->pnc.enabled) {
nss_tx_status_t ret;
+ pn->sw_ksp_ptr = NULL;
pn->pnc.enabled = 0;
pn->pnc.un.hd_magic = UBI32_PROFILE_HD_MAGIC | NSS_PROFILER_STOP_MSG;
- ret = nss_profiler_if_tx_buf(pn->ctx, &pn->pnc.un, sizeof(pn->pnc.un), profiler_handle_reply);
+ ret = nss_profiler_if_tx_buf(pn->ctx, &pn->pnc.un,
+ sizeof(pn->pnc.un), profiler_handle_reply, pn);
profileInfo("%s: %p %d\n", __func__, pn, ret);
return 0;
}
@@ -438,8 +462,121 @@
return -EBADF;
}
-#define isspace(c) (c==' ' || c=='\t')
+/*
+ * profiler_handle_stat_event_reply()
+ * print current FW stat event counter configurations
+ */
+static void profiler_handle_stat_event_reply(struct nss_ctx_instance *nss_ctx,
+ struct nss_cmn_msg *ncm)
+{
+ struct profile_io *pio = (struct profile_io *) ncm->app_data;
+ struct debug_box *pdb = (struct debug_box *) &pio->pnc;
+ struct debug_box *db = (struct debug_box *) &ncm[1];
+ int i, thrds;
+ for (i = 0; i < db->dlen; i++)
+ printk("stat counter %d: %x\n", i, db->data[i]);
+
+ thrds = db->data[i];
+ i = (1 << PROFILE_STS_EVENT_THREAD_BITS) - 1;
+ profileInfo("%d: event end mark %x, ThrA %d ThrB %d\n",
+ ncm->len, thrds, (thrds & i) + 1,
+ ((thrds >> PROFILE_STS_EVENT_THREAD_BITS) & i) + 1);
+
+ /*
+ * save data for read()
+ */
+ memcpy(pdb->data, db->data, (db->dlen + 1) * sizeof(db->data[0]));
+}
+
+/*
+ * parse_sys_stat_event_req()
+ * process FW stat events request: event#1 index#1 event#2 index#2 ...
+ */
+static int parse_sys_stat_event_req(const char *buf, size_t count,
+ struct debug_box *db, struct profile_io *pio)
+{
+ char *cp;
+ int result;
+
+ printk("%d cmd buf %s\n", count, buf);
+ if (count < 19) /* minimum data for sys_stat_event request */
+ return -EINVAL;
+
+ if (strncmp(buf, "get-sys-stat-events", 19) == 0) {
+ db->hd_magic = UBI32_PROFILE_HD_MAGIC | NSS_PROFILER_GET_SYS_STAT_EVENT;
+ db->dlen = result;
+ result = nss_profiler_if_tx_buf(pio->ctx, &pio->pnc.un,
+ sizeof(pio->pnc.un),
+ profiler_handle_stat_event_reply, pio);
+ profileInfo("get_sys_stat_events: %d\n", result);
+ return result == NSS_TX_SUCCESS ? count : -EFAULT;
+ }
+
+ if (strncmp(buf, "set-sys-stat-events", 19)) {
+ printk("unknow event: %s\n", buf);
+ return -EINVAL;
+ }
+
+ db->dlen = sizeof(pio->pnc.un);
+ memset(db->data, 0, PROFILE_STS_EVENT_COUNTERS * sizeof(db->data[0]));
+
+ cp = strchr(buf, ' ');
+ if (!cp) {
+ printk("no enough paramters %s\n", buf);
+ return -EINVAL;
+ }
+
+ do {
+ int idx, event;
+
+ while (isspace(*cp))
+ cp++;
+ event = kstrtoul(cp, NULL, 0);
+
+ cp = strchr(cp, ' ');
+ if (!cp) {
+ printk("missing index %s\n", buf);
+ return -EINVAL;
+ }
+ while (isspace(*cp))
+ cp++;
+ idx = event >> 16;
+ if (idx) {
+ if ((event & 0x1FF) < 50) {
+ printk("thr ID (%d) ignored for event %d\n",
+ idx, event & 0x1FF);
+ } else if (idx > 12) {
+ if ((idx >>= 5) > 12) {
+ printk("tID %d too big [1..12]\n", idx);
+ return -E2BIG;
+ }
+ }
+ }
+ idx = kstrtoul(cp, NULL, 10);
+ if (idx < 0 || idx > 7) {
+ printk("index %d out of range [0..7]\n", idx);
+ return -ERANGE;
+ }
+ printk("%p: e %d i %d\n", db, event, idx);
+ db->data[idx] = event;
+ cp = strchr(cp, ' ');
+ } while (cp);
+ db->hd_magic = UBI32_PROFILE_HD_MAGIC | NSS_PROFILER_SET_SYS_STAT_EVENT;
+ result = nss_profiler_if_tx_buf(pio->ctx, &pio->pnc.un, sizeof(pio->pnc.un),
+ profiler_handle_stat_event_reply, pio);
+ profileInfo("%p: %d send cmd %x to FW ret %d\n",
+ db, count, db->hd_magic, result);
+ return count;
+}
+
+/*
+ * parseDbgData()
+ * parsing debug requests: base_address [options] cmd length
+ *
+ * cmd is either read or write
+ * option is one of mio, moveio, h [heap security verify], etc.
+ */
static int parseDbgData(const char *buf, size_t count, struct debug_box *db)
{
char *cp;
@@ -523,7 +660,7 @@
{
int result;
struct debug_box *db;
- struct profile_io *pio = node[0];
+ struct profile_io *pio = (struct profile_io *)filp->private_data;
if (!pio) {
return -ENOENT;
@@ -535,6 +672,19 @@
db = (struct debug_box *) &pio->pnc;
db->dlen = db->opts = 0;
+
+ if (!isdigit(buf[0])) {
+ result = parse_sys_stat_event_req(buf, count, db, pio);
+
+ if ((result > 0) && (filp->f_flags & O_RDWR)) {
+ /*
+ * set flag so event-counter can read the data from FW
+ */
+ pio->sw_ksp_ptr = (uint32_t *)db;
+ }
+ return result;
+ }
+
result = parseDbgData(buf, count, db);
if (result < 0) {
return result;
@@ -546,7 +696,8 @@
db->hd_magic = UBI32_PROFILE_HD_MAGIC | NSS_PROFILER_DEBUG_WR_MSG;
db->dlen = result;
}
- result = nss_profiler_if_tx_buf(pio->ctx, &pio->pnc.un, sizeof(pio->pnc.un), profiler_handle_debug_reply);
+ result = nss_profiler_if_tx_buf(pio->ctx, &pio->pnc.un,
+ sizeof(pio->pnc.un), profiler_handle_debug_reply, pio);
printk("dbg res %d dlen = %d opt %x\n", result, db->dlen, db->opts);
return count;
}
@@ -664,7 +815,7 @@
pn->pnc.un.cpu_id = ntohl(pTx->cpu_id);
pn->pnc.un.cpu_freq = ntohl(pTx->cpu_freq);
pn->pnc.un.ddr_freq = ntohl(pTx->ddr_freq);
- pn->pnc.un.num_counters = ntohl(pTx->num_counters);
+ pn->pnc.un.num_counters = pTx->num_counters;
} else {
pn->pnc.un = *pTx;
}
@@ -680,7 +831,9 @@
if (pn->pnc.enabled > 0) {
pn->pnc.enabled = -1;
pn->pnc.un.hd_magic = UBI32_PROFILE_HD_MAGIC | NSS_PROFILER_STOP_MSG;
- ret = nss_profiler_if_tx_buf(pn->ctx, &pn->pnc.un, sizeof(pn->pnc.un), profiler_handle_reply);
+ ret = nss_profiler_if_tx_buf(pn->ctx,
+ &pn->pnc.un, sizeof(pn->pnc.un),
+ profiler_handle_reply, pn);
profileWarn("%d temp stop sampling engine %d\n", swap, ret);
}
if (swap < 3) {
@@ -696,19 +849,16 @@
memcpy(&nsb->psc_header, buf, buf_len); /* pn->pnc.pn2h->psc_header = *psc_hd; maybe faster, but take more memory */
nsb->mh.md_type = PINGPONG_FULL;
- //kxdump((void*)(nsb->samples + 23), sizeof(*nsb->samples) << 1, "1st 2 samples");
+
+ /*
+ * ask for perf_counters (software counters) update every 32 samples
+ */
if (!wr) {
- /*
- * should be UBI32_PROFILE_HD_MAGIC | NSS_PROFILER_COUNTERS_MSG
- * but FW is hard to change due to packge warehouse, so using
- * STOP/START instead till PROFILER_COUNTERS_MSG done in FW
- */
- pn->pnc.un.hd_magic = UBI32_PROFILE_HD_MAGIC | NSS_PROFILER_STOP_MSG;
- ret = nss_profiler_if_tx_buf(pn->ctx, &pn->pnc.un, sizeof(pn->pnc.un), profiler_handle_reply);
+ pn->pnc.un.hd_magic = UBI32_PROFILE_HD_MAGIC | NSS_PROFILER_COUNTERS_MSG;
+ ret = nss_profiler_if_tx_buf(pn->ctx, &pn->pnc.un,
+ sizeof(pn->pnc.un), profiler_handle_reply, pn);
if (ret == NSS_TX_FAILURE)
- printk("STOP Cmd failed %d %d\n", ret, wr);
- pn->pnc.un.hd_magic = UBI32_PROFILE_HD_MAGIC | NSS_PROFILER_START_MSG;
- ret = nss_profiler_if_tx_buf(pn->ctx, &pn->pnc.un, sizeof(pn->pnc.un), profiler_handle_reply);
+ printk("req counters Cmd failed %d %d\n", ret, wr);
}
profileInfo("filled %p %p wr %d\n", nsb, nsb->samples, pn->ccl_write);
}
@@ -748,9 +898,11 @@
}
/*
- * sw_ksp is an array of pointers to struct thread_info, the current task executing for each linux virtual processor
+ * sw_ksp is an array of pointers to struct thread_info,
+ * the current task executing for each linux virtual processor
node->sw_ksp_ptr = sw_ksp;
*/
+ node->sw_ksp_ptr = NULL;
node->task_offset = offsetof(struct thread_info, task);
node->pid_offset = offsetof(struct task_struct, tgid);
}