DPDK patches and discussions
 help / color / mirror / Atom feed
From: "Juraj Linkeš" <juraj.linkes@pantheon.tech>
To: thomas@monjalon.net, david.marchand@redhat.com,
	Honnappa.Nagarahalli@arm.com, ohilyard@iol.unh.edu,
	lijuan.tu@intel.com
Cc: dev@dpdk.org, "Juraj Linkeš" <juraj.linkes@pantheon.tech>
Subject: [RFC PATCH v1 02/23] dts: merge DTS framework/plotting.py to DPDK
Date: Wed,  6 Apr 2022 15:18:42 +0000	[thread overview]
Message-ID: <20220406151903.2916254-3-juraj.linkes@pantheon.tech> (raw)
In-Reply-To: <20220406151903.2916254-1-juraj.linkes@pantheon.tech>

---
 dts/framework/plotting.py | 246 ++++++++++++++++++++++++++++++++++++++
 1 file changed, 246 insertions(+)
 create mode 100644 dts/framework/plotting.py

diff --git a/dts/framework/plotting.py b/dts/framework/plotting.py
new file mode 100644
index 0000000000..83499061ef
--- /dev/null
+++ b/dts/framework/plotting.py
@@ -0,0 +1,246 @@
+# BSD LICENSE
+#
+# Copyright(c) 2010-2014 Intel Corporation. All rights reserved.
+# All rights reserved.
+#
+# Redistribution and use in source and binary forms, with or without
+# modification, are permitted provided that the following conditions
+# are met:
+#
+#   * Redistributions of source code must retain the above copyright
+#     notice, this list of conditions and the following disclaimer.
+#   * Redistributions in binary form must reproduce the above copyright
+#     notice, this list of conditions and the following disclaimer in
+#     the documentation and/or other materials provided with the
+#     distribution.
+#   * Neither the name of Intel Corporation nor the names of its
+#     contributors may be used to endorse or promote products derived
+#     from this software without specific prior written permission.
+#
+# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+# "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+# LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+# A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+# OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+# SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+# LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+# DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+# THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+# OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+
+import os
+import shutil
+
+from docutils.parsers.rst.directives import path
+
+import framework.plotgraph as plotgraph
+import framework.utils as utils
+
+from .exception import VerifyFailure
+from .plotgraph import Plot2DGraph
+from .rst import path2Result
+
+"""
+Generate Plots for performance test results
+"""
+
+
+class tableData(object):
+    def __init__(self):
+        self.headers = []
+        """
+        Each data array corresponds to a column related to one of the headers above
+        """
+        self.data = []
+
+
+class Plotting(object):
+
+    path_2_result = path2Result
+    plots_subfolder = "images"
+    image_format = "png"
+
+    default_bar_colours = [
+        "#f70202",
+        "#0f0b0b",
+        "#123eed",
+        "#07601b",
+        "#36f760",
+        "#87210d",
+        "#512f28",
+        "#11c6b1",
+        "#45f94e",
+        "#f94566",
+    ]
+
+    default_line_markers = ["o"]
+
+    default_line_styles = ["--"]
+
+    def __init__(self, crb, target, nic):
+
+        # Ensure the folder exist
+        try:
+
+            path = "/".join(
+                [Plotting.path_2_result, crb, target, nic, Plotting.plots_subfolder]
+            )
+
+            if not os.path.exists(path):
+                os.makedirs(path)
+
+            self.plots_path = path
+
+        except Exception as e:
+            raise VerifyFailure("Plot Error: " + str(e))
+
+    def clear_all_plots(self, crb, target):
+        shutil.rmtree(self.plots_path, True)
+
+    def create_bars_plot(
+        self,
+        image_filename,
+        plot_title,
+        xdata,
+        ydata,
+        xlabel="",
+        ylabel="",
+        legend=[],
+        bar_colours=default_bar_colours,
+    ):
+
+        for yseries in ydata:
+            if len(xdata) != len(yseries):
+                print(
+                    utils.RED(
+                        "The number of items in X axis (%s) and Y axis (%s) does not match."
+                        % (xdata, ydata)
+                    )
+                )
+                return ""
+
+        image_path = "%s/%s.%s" % (
+            self.plots_path,
+            image_filename,
+            Plotting.image_format,
+        )
+
+        pgraph = Plot2DGraph()
+        pgraph.resetMe()
+
+        # Set the number of bars, ydata contains a array per set of data
+        pgraph.setNumPlots(len(ydata))
+        pgraph.setNumSubplots(1)
+
+        pgraph.setColors(bar_colours)
+        pgraph.addBarYlabel(0, ylabel)
+        pgraph.setBarLegends(0, legend)
+
+        # For each value in the x axis add corresponding bar (array in ydata)
+        for xvalue in range(len(xdata)):
+            yvalues = [_[xvalue] for _ in ydata]
+            pgraph.addBarData(0, xdata[xvalue], yvalues)
+
+        # Dynamic adjustment of the bar widths for better plot appearance
+        bar_width = 0.30 - 0.005 * ((len(xdata) * len(legend)) - 4)
+        pgraph.setBarWidth(0, bar_width)
+
+        pgraph.generateBar(plotName=image_path, title=plot_title)
+
+        return image_path
+
+    def create_lines_plot(
+        self,
+        image_filename,
+        plot_title,
+        xdata,
+        ydata,
+        xticks=[],
+        yticks=[],
+        xlabel="",
+        ylabel="",
+        legend=[],
+        line_colours=default_bar_colours,
+        line_markers=default_line_markers,
+        line_styles=default_line_styles,
+        addHline=False,
+        hLine={},
+        testing=False,
+    ):
+
+        image_path = "%s/%s.%s" % (
+            self.plots_path,
+            image_filename,
+            Plotting.image_format,
+        )
+
+        pgraph = Plot2DGraph()
+        pgraph.resetMe()
+
+        numPlots = len(ydata) / len(xticks)
+        numticks = len(xticks)
+
+        # Set the number of bars, ydata contains a array per set of data
+        pgraph.setNumPlots(numPlots)
+        # TODO more than one plot per figure needs to be tested
+        pgraph.setNumSubplots(1)
+
+        # workaround
+        if numPlots > len(line_colours):
+            print("WARNING - numPlots > len(line_colours)")
+            r = 0x00
+            g = 0x66
+            b = 0xFF
+            for _ in range(numPlots - len(line_colours)):
+                r = r % 256
+                g = g % 256
+                b = b % 256
+                _ = "#%0.2x%0.2x%0.2x" % (r, g, b)
+                line_colours.append(_)
+                r += 7
+                g -= 10
+                b -= 9
+
+        line_markers = line_markers * numPlots
+        line_styles = line_styles * numPlots
+
+        pgraph.setColors(line_colours)
+        pgraph.setMarkers(line_markers)
+        pgraph.setLineStyle(line_styles)
+
+        pgraph.addBarYlabel(0, ylabel)
+        pgraph.setBarLegends(0, legend)
+
+        # For each value in the x axis add corresponding bar (array in ydata)
+        for i in list(range(numPlots)):
+            yDataStart = i * numticks
+            pgraph.addPlotData(
+                i,
+                "Number of active pipes per output port",
+                ylabel,
+                xticks,
+                [],
+                xdata,
+                ydata[yDataStart : (yDataStart + numticks)],
+                [],
+                [],
+            )
+
+        pgraph.xLen = 0.6
+        pgraph.titleFontSize = 18
+        pgraph.titleYOffset = 0.96
+        pgraph.titleXOffset = 0.35
+
+        if addHline:
+            pgraph.horizontalLine = True
+            pgraph.hLineName = hLine["name"]
+            pgraph.hLine = hLine["value"]
+            pgraph.hLineBoxX = hLine["boxXvalue"]
+            pgraph.hLineBoxY = hLine["boxYvalue"]
+
+        pgraph.generatePlot(
+            plotName=image_path, keys=legend, title=plot_title, firstYvalue=1
+        )
+
+        return image_path
-- 
2.20.1


  parent reply	other threads:[~2022-04-06 15:19 UTC|newest]

Thread overview: 24+ messages / expand[flat|nested]  mbox.gz  Atom feed  top
2022-04-06 15:18 [RFC PATCH v1 00/23] merge DTS test resource files " Juraj Linkeš
2022-04-06 15:18 ` [RFC PATCH v1 01/23] dts: merge DTS framework/plotgraph.py " Juraj Linkeš
2022-04-06 15:18 ` Juraj Linkeš [this message]
2022-04-06 15:18 ` [RFC PATCH v1 03/23] dts: merge DTS framework/pmd_output.py " Juraj Linkeš
2022-04-06 15:18 ` [RFC PATCH v1 04/23] dts: merge DTS framework/qemu_kvm.py " Juraj Linkeš
2022-04-06 15:18 ` [RFC PATCH v1 05/23] dts: merge DTS framework/qemu_libvirt.py " Juraj Linkeš
2022-04-06 15:18 ` [RFC PATCH v1 06/23] dts: merge DTS framework/test_capabilities.py " Juraj Linkeš
2022-04-06 15:18 ` [RFC PATCH v1 07/23] dts: merge DTS framework/test_case.py " Juraj Linkeš
2022-04-06 15:18 ` [RFC PATCH v1 08/23] dts: merge DTS framework/virt_base.py " Juraj Linkeš
2022-04-06 15:18 ` [RFC PATCH v1 09/23] dts: merge DTS framework/virt_common.py " Juraj Linkeš
2022-04-06 15:18 ` [RFC PATCH v1 10/23] dts: merge DTS framework/virt_dut.py " Juraj Linkeš
2022-04-06 15:18 ` [RFC PATCH v1 11/23] dts: merge DTS framework/virt_resource.py " Juraj Linkeš
2022-04-06 15:18 ` [RFC PATCH v1 12/23] dts: merge DTS framework/virt_scene.py " Juraj Linkeš
2022-04-06 15:18 ` [RFC PATCH v1 13/23] dts: merge DTS nics/__init__.py " Juraj Linkeš
2022-04-06 15:18 ` [RFC PATCH v1 14/23] dts: merge DTS nics/system_info.py " Juraj Linkeš
2022-04-06 15:18 ` [RFC PATCH v1 15/23] dts: merge DTS framework/flow/__init__.py " Juraj Linkeš
2022-04-06 15:18 ` [RFC PATCH v1 16/23] dts: merge DTS framework/flow/enums.py " Juraj Linkeš
2022-04-06 15:18 ` [RFC PATCH v1 17/23] dts: merge DTS framework/flow/exceptions.py " Juraj Linkeš
2022-04-06 15:18 ` [RFC PATCH v1 18/23] dts: merge DTS framework/flow/flow.py " Juraj Linkeš
2022-04-06 15:18 ` [RFC PATCH v1 19/23] dts: merge DTS framework/flow/flow_action_items.py " Juraj Linkeš
2022-04-06 15:19 ` [RFC PATCH v1 20/23] dts: merge DTS framework/flow/flow_items.py " Juraj Linkeš
2022-04-06 15:19 ` [RFC PATCH v1 21/23] dts: merge DTS framework/flow/flow_pattern_items.py " Juraj Linkeš
2022-04-06 15:19 ` [RFC PATCH v1 22/23] dts: merge DTS framework/flow/flow_rule.py " Juraj Linkeš
2022-04-06 15:19 ` [RFC PATCH v1 23/23] dts: merge DTS framework/flow/generator.py " Juraj Linkeš

Reply instructions:

You may reply publicly to this message via plain-text email
using any one of the following methods:

* Save the following mbox file, import it into your mail client,
  and reply-to-all from there: mbox

  Avoid top-posting and favor interleaved quoting:
  https://en.wikipedia.org/wiki/Posting_style#Interleaved_style

* Reply using the --to, --cc, and --in-reply-to
  switches of git-send-email(1):

  git send-email \
    --in-reply-to=20220406151903.2916254-3-juraj.linkes@pantheon.tech \
    --to=juraj.linkes@pantheon.tech \
    --cc=Honnappa.Nagarahalli@arm.com \
    --cc=david.marchand@redhat.com \
    --cc=dev@dpdk.org \
    --cc=lijuan.tu@intel.com \
    --cc=ohilyard@iol.unh.edu \
    --cc=thomas@monjalon.net \
    /path/to/YOUR_REPLY

  https://kernel.org/pub/software/scm/git/docs/git-send-email.html

* If your mail client supports setting the In-Reply-To header
  via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).