@@ -11,6 +11,7 @@
// C++
#include<iostream>
+#include <limits>
// KernelShark
#include "KsPlugins.hpp"
@@ -414,3 +415,24 @@ void eventFieldIntervalPlot(KsCppArgV *argvCpp,
<< exc.what() << std::endl;
}
}
+
+/**
+ * @brief Distance between the click and the shape. Used to decide if
+ * the double click action must be executed.
+ *
+ * @param x: X coordinate of the click.
+ * @param y: Y coordinate of the click.
+ *
+ * @returns If the click is inside the box, the distance is zero.
+ * Otherwise infinity.
+ */
+double LatencyBox::distance(int x, int y) const
+{
+ if (x < pointX(0) || x > pointX(2))
+ return std::numeric_limits<double>::max();
+
+ if (y < pointY(0) || y > pointY(1))
+ return std::numeric_limits<double>::max();
+
+ return 0;
+}
@@ -13,6 +13,7 @@
#define _KS_PLUGINS_H
// C++
+#include <vector>
#include <functional>
// KernelShark
@@ -101,4 +102,51 @@ void eventFieldIntervalPlot(KsCppArgV *argvCpp,
KsPlot::Color col,
float size);
+/**
+ * This class represents the graphical element visualizing the latency between
+ * two events.
+ */
+class LatencyBox : public KsPlot::Rectangle
+{
+ /** On double click do. */
+ void _doubleClick() const override {}
+
+public:
+ /** The trace record data that corresponds to this LatencyBox. */
+ std::vector<kshark_data_field_int64 *> _data;
+
+ double distance(int x, int y) const override;
+};
+
+/**
+ * This template function make shape of hollow box for two events and
+ * return the shape to be plotted by KernelShark on top of the existing
+ * Graph generated by the model.
+ */
+template<class T> KsPlot::PlotObject *
+makeLatencyBox(std::vector<const KsPlot::Graph *> graph,
+ std::vector<int> bins,
+ std::vector<kshark_data_field_int64 *> data,
+ KsPlot::Color col, float size)
+{
+ LatencyBox *rec = new T;
+ rec->_data = data;
+
+ KsPlot::Point p0 = graph[0]->bin(bins[0])._base;
+ KsPlot::Point p1 = graph[0]->bin(bins[1])._base;
+ int height = graph[0]->height() * .3;
+
+ rec->setFill(false);
+ rec->setPoint(0, p0.x() - 1, p0.y() - height);
+ rec->setPoint(1, p0.x() - 1, p0.y() - 1);
+
+ rec->setPoint(3, p1.x() - 1, p1.y() - height);
+ rec->setPoint(2, p1.x() - 1, p1.y() - 1);
+
+ rec->_size = size;
+ rec->_color = col;
+
+ return rec;
+}
+
#endif
@@ -11,9 +11,6 @@
* preempted by another task.
*/
-// C++
-#include <vector>
-
// KernelShark
#include "libkshark.h"
#include "libkshark-plugin.h"
@@ -24,7 +21,7 @@
using namespace KsPlot;
-static KsMainWindow *ks_ptr;
+static KsMainWindow *ks4sched_ptr;
/**
* @brief Provide the plugin with a pointer to the KsMainWindow object (the GUI
@@ -32,72 +29,24 @@ static KsMainWindow *ks_ptr;
*/
__hidden void *plugin_set_gui_ptr(void *gui_ptr)
{
- ks_ptr = static_cast<KsMainWindow *>(gui_ptr);
+ ks4sched_ptr = static_cast<KsMainWindow *>(gui_ptr);
return nullptr;
}
/**
- * This class represents the graphical element visualizing the latency between
- * sched_waking and sched_switch events.
+ * This child class represents the graphical element visualizing the latency
+ * between sched_waking and sched_switch events. It is defined to re-implement
+ * the handler for double-click.
*/
-class LatencyBox : public Rectangle
+class SchedLatencyBox : public LatencyBox
{
/** On double click do. */
void _doubleClick() const override
{
- ks_ptr->markEntry(_data[1]->entry, DualMarkerState::B);
- ks_ptr->markEntry(_data[0]->entry, DualMarkerState::A);
+ ks4sched_ptr->markEntry(_data[1]->entry, DualMarkerState::B);
+ ks4sched_ptr->markEntry(_data[0]->entry, DualMarkerState::A);
}
-public:
- /** The trace record data that corresponds to this LatencyBox. */
- std::vector<kshark_data_field_int64 *> _data;
-
- /**
- * @brief Distance between the click and the shape. Used to decide if
- * the double click action must be executed.
- *
- * @param x: X coordinate of the click.
- * @param y: Y coordinate of the click.
- *
- * @returns If the click is inside the box, the distance is zero.
- * Otherwise infinity.
- */
- double distance(int x, int y) const override
- {
- if (x < pointX(0) || x > pointX(2))
- return std::numeric_limits<double>::max();
-
- if (y < pointY(0) || y > pointY(1))
- return std::numeric_limits<double>::max();
-
- return 0;
- }
-};
-
-static PlotObject *makeShape(std::vector<const Graph *> graph,
- std::vector<int> bins,
- std::vector<kshark_data_field_int64 *> data,
- Color col, float size)
-{
- LatencyBox *rec = new LatencyBox;
- rec->_data = data;
-
- Point p0 = graph[0]->bin(bins[0])._base;
- Point p1 = graph[0]->bin(bins[1])._base;
- int height = graph[0]->height() * .3;
-
- rec->setFill(false);
- rec->setPoint(0, p0.x() - 1, p0.y() - height);
- rec->setPoint(1, p0.x() - 1, p0.y() - 1);
-
- rec->setPoint(3, p1.x() - 1, p1.y() - height);
- rec->setPoint(2, p1.x() - 1, p1.y() - 1);
-
- rec->_size = size;
- rec->_color = col;
-
- return rec;
};
/*
@@ -191,14 +140,14 @@ __hidden void plugin_draw(kshark_cpp_argv *argv_c,
eventFieldIntervalPlot(argvCpp,
plugin_ctx->sw_data, checkFieldSW,
plugin_ctx->ss_data, checkEntryPid,
- makeShape,
+ makeLatencyBox<SchedLatencyBox>,
{0, 255, 0}, // Green
-1); // Default size
eventFieldIntervalPlot(argvCpp,
plugin_ctx->ss_data, checkFieldSS,
plugin_ctx->ss_data, checkEntryPid,
- makeShape,
+ makeLatencyBox<SchedLatencyBox>,
{255, 0, 0}, // Red
-1); // Default size
}
new file mode 100644
@@ -0,0 +1,64 @@
+/* SPDX-License-Identifier: LGPL-2.1 */
+
+/*
+ * Copyright (C) 2018 VMware Inc, Yordan Karadzhov (VMware) <y.karadz@gmail.com>
+ * 2021 Intel Inc, Hongzhan Chen <hongzhan.chen@intel.com>
+ */
+
+/**
+ * @file common_sched.h
+ * @brief Common definitions for sched plugins.
+ */
+
+#ifndef _KS_PLUGIN_COMMON_SCHED_H
+#define _KS_PLUGIN_COMMON_SCHED_H
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+/** The type of the numerical data field used by the 'tep' APIs. */
+typedef unsigned long long tep_num_field_t;
+
+/** The type of the data field stored in the kshark_data_container object. */
+typedef int64_t ks_num_field_t;
+
+/** prev_state offset in data field */
+#define PREV_STATE_SHIFT ((int) ((sizeof(ks_num_field_t) - 1) * 8))
+
+/** Bit mask used when converting data to prev_state */
+#define PREV_STATE_MASK (((ks_num_field_t) 1 << 8) - 1)
+
+/** Bit mask used when converting data to PID */
+#define PID_MASK (((ks_num_field_t) 1 << PREV_STATE_SHIFT) - 1)
+
+/**
+ * @brief Set the PID value for the data field stored in the
+ * kshark_data_container object.
+ *
+ * @param field: Input pointer to data field.
+ * @param pid: Input pid to set in data field.
+ */
+static inline void plugin_sched_set_pid(ks_num_field_t *field,
+ tep_num_field_t pid)
+{
+ *field &= ~PID_MASK;
+ *field |= pid & PID_MASK;
+}
+
+/**
+ * @brief Retrieve the PID value from the data field stored in the
+ * kshark_data_container object.
+ *
+ * @param field: Input location for the data field.
+ */
+static inline int plugin_sched_get_pid(ks_num_field_t field)
+{
+ return field & PID_MASK;
+}
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif
@@ -22,36 +22,6 @@
/** Plugin context instance. */
-//! @cond Doxygen_Suppress
-
-typedef unsigned long long tep_num_field_t;
-
-#define PREV_STATE_SHIFT ((int) ((sizeof(ks_num_field_t) - 1) * 8))
-
-#define PREV_STATE_MASK (((ks_num_field_t) 1 << 8) - 1)
-
-#define PID_MASK (((ks_num_field_t) 1 << PREV_STATE_SHIFT) - 1)
-
-//! @endcond
-
-static void plugin_sched_set_pid(ks_num_field_t *field,
- tep_num_field_t pid)
-{
- *field &= ~PID_MASK;
- *field = pid & PID_MASK;
-}
-
-/**
- * @brief Retrieve the PID value from the data field stored in the
- * kshark_data_container object.
- *
- * @param field: Input location for the data field.
- */
-__hidden int plugin_sched_get_pid(ks_num_field_t field)
-{
- return field & PID_MASK;
-}
-
/* Use the most significant byte to store the value of "prev_state". */
static void plugin_sched_set_prev_state(ks_num_field_t *field,
tep_num_field_t prev_state)
@@ -135,7 +105,7 @@ static void plugin_sched_swith_action(struct kshark_data_stream *stream,
struct tep_record *record = (struct tep_record *) rec;
struct plugin_sched_context *plugin_ctx;
unsigned long long next_pid, prev_state;
- ks_num_field_t ks_field;
+ ks_num_field_t ks_field = 0;
int ret;
plugin_ctx = __get_context(stream->stream_id);
@@ -15,6 +15,7 @@
// KernelShark
#include "libkshark.h"
#include "libkshark-plugin.h"
+#include "plugins/common_sched.h"
#ifdef __cplusplus
extern "C" {
@@ -55,10 +56,6 @@ struct plugin_sched_context {
KS_DECLARE_PLUGIN_CONTEXT_METHODS(struct plugin_sched_context)
-/** The type of the data field stored in the kshark_data_container object. */
-typedef int64_t ks_num_field_t;
-
-int plugin_sched_get_pid(ks_num_field_t field);
int plugin_sched_get_prev_state(ks_num_field_t field);
To avoid code duplication, move some common APIs and definitions out from plugin SchedEvent to share with other plugins. Signed-off-by: Hongzhan Chen <hongzhan.chen@intel.com>