|
|
1bdc94 |
From 49ebd26734d492cc4189b97704052780b3641854 Mon Sep 17 00:00:00 2001
|
|
|
4a2fec |
From: "Danilo C. L. de Paula" <ddepaula@redhat.com>
|
|
|
4a2fec |
Date: Mon, 16 Jan 2017 11:52:49 +0100
|
|
|
4a2fec |
Subject: Revert "kvm_stat: Remove"
|
|
|
4a2fec |
|
|
|
4a2fec |
RH-Author: ddepaula <ddepaula@redhat.com>
|
|
|
4a2fec |
Message-id: <1479302806-10135-2-git-send-email-ddepaula@redhat.com>
|
|
|
4a2fec |
Patchwork-id: 72851
|
|
|
4a2fec |
O-Subject: [RHEV-7.4 qemu-kvm-rhev PATCH v3 1/3] Revert "kvm_stat: Remove"
|
|
|
4a2fec |
Bugzilla: 1389238
|
|
|
4a2fec |
RH-Acked-by: John Snow <jsnow@redhat.com>
|
|
|
4a2fec |
RH-Acked-by: David Hildenbrand <david@redhat.com>
|
|
|
4a2fec |
RH-Acked-by: Miroslav Rezanina <mrezanin@redhat.com>
|
|
|
4a2fec |
|
|
|
4a2fec |
kvm_stat script was removed in QEMU 2.7.0 as it become part of kernel
|
|
|
4a2fec |
tree. However kvm_stat is shipped in qemu-kvm-tools package in RHEL.
|
|
|
4a2fec |
|
|
|
4a2fec |
This reverts commit 60b412dd18362bd4ddc44ba7022aacb6af074b5d.
|
|
|
4a2fec |
|
|
|
4a2fec |
Signed-off-by: Danilo Cesar Lemes de Paula <ddepaula@redhat.com>
|
|
|
4a2fec |
Signed-off-by: Miroslav Rezanina <mrezanin@redhat.com>
|
|
|
4a2fec |
|
|
|
1bdc94 |
Merged patches (2.11.0):
|
|
|
1bdc94 |
- beffa30342 tools/kvm_stat: hide cursor
|
|
|
1bdc94 |
- 98cf56caaa tools/kvm_stat: catch curses exceptions only
|
|
|
1bdc94 |
- dada85e20a tools/kvm_stat: handle SIGINT in log and batch modes
|
|
|
1bdc94 |
- 8654f68785 tools/kvm_stat: fix misc glitches
|
|
|
1bdc94 |
- d4525d7460 tools/kvm_stat: fix trace setup glitch on field updates in TracepointProvider
|
|
|
1bdc94 |
- 5f41d9716a tools/kvm_stat: full PEP8 compliance
|
|
|
1bdc94 |
- 5bf653923c tools/kvm_stat: reduce perceived idle time on filter updates
|
|
|
1bdc94 |
- 2ef292f12d tools/kvm_stat: document list of interactive commands
|
|
|
1bdc94 |
- 3e1d2fc34c tools/kvm_stat: display guest name when using pid filter
|
|
|
1bdc94 |
- 66f2e8203e tools/kvm_stat: remove pid filter on empty input
|
|
|
1bdc94 |
- 05cc1c1aab tools/kvm_stat: print error messages on faulty pid filter input
|
|
|
1bdc94 |
- a0c6451760 tools/kvm_stat: display regex when set to non-default
|
|
|
1bdc94 |
- 3c7dae7507 tools/kvm_stat: remove regex filter on empty input
|
|
|
1bdc94 |
- 8b7b31fc24 tools/kvm_stat: add option '--guest'
|
|
|
1bdc94 |
- 38362f9d75 tools/kvm_stat: add interactive command 'c'
|
|
|
1bdc94 |
- 243d90a041 tools/kvm_stat: add interactive command 'r'
|
|
|
1bdc94 |
- 4f80c9081b tools/kvm_stat: add '%Total' column
|
|
|
1bdc94 |
- e21fb4a96e tools/kvm_stat: fix typo
|
|
|
1bdc94 |
- 7bc419ee18 tools/kvm_stat: fix event counts display for interrupted intervals
|
|
|
1bdc94 |
- d54183dd4b tools/kvm_stat: fix undue use of initial sleeptime
|
|
|
1bdc94 |
- 6514729674 tools/kvm_stat: remove unnecessary header redraws
|
|
|
1bdc94 |
- c4f941df75 tools/kvm_stat: simplify line print logic
|
|
|
1bdc94 |
- 35c050e4fd tools/kvm_stat: removed unused function
|
|
|
1bdc94 |
- 92d06e843c tools/kvm_stat: remove extra statement
|
|
|
1bdc94 |
- d0ced131bf tools/kvm_stat: simplify initializers
|
|
|
1bdc94 |
- 1dbc0659ac tools/kvm_stat: move functions to corresponding classes
|
|
|
1bdc94 |
- bf53d77372 tools/kvm_stat: show cursor in selection screens
|
|
|
1bdc94 |
- 6b557dc235 tools/kvm_stat: display message indicating lack of events
|
|
|
1bdc94 |
- 7f50a8be62 tools/kvm_stat: make heading look a bit more like 'top'
|
|
|
1bdc94 |
- 8ba2fd89d1 tools/kvm_stat: rename 'Current' column to 'CurAvg/s'
|
|
|
1bdc94 |
- 95b5c46145 tools/kvm_stat: add new interactive command 'h'
|
|
|
1bdc94 |
- 25c56e3212 tools/kvm_stat: add new interactive command 's'
|
|
|
1bdc94 |
- 55f7ed9b2c tools/kvm_stat: add new interactive command 'o'
|
|
|
1bdc94 |
- 53a1267b00 tools/kvm_stat: display guest list in pid/guest selection screens
|
|
|
1bdc94 |
- 554fa10df5 tools/kvm_stat: display guest list in pid/guest selection screens
|
|
|
1bdc94 |
- 72f660ddfc tools/kvm_stat: add new command line switch '-i'
|
|
|
1bdc94 |
- 17165a0bc8 tools/kvm_stat: add new interactive command 'b'
|
|
|
1bdc94 |
- ac6ba9a14b tools/kvm_stat: use variables instead of hard paths in help output
|
|
|
1bdc94 |
- 592b801e5c tools/kvm_stat: add '-f help' to get the available event list
|
|
|
1bdc94 |
- 283aa25f75 tools/kvm_stat: fix command line option '-g'
|
|
|
1bdc94 |
|
|
|
4a2fec |
Merged patches (2.9.0):
|
|
|
4a2fec |
- 1e69b1b Include kvm_stat in qemu-kvm.spec
|
|
|
4a2fec |
- 7fcfc94 tools: kvm_stat: Powerpc related fixes
|
|
|
4a2fec |
- 7f89136 tools: kvm_stat: Introduce pid monitoring
|
|
|
4a2fec |
- c728a6b tools: kvm_stat: Add comments
|
|
|
4a2fec |
- 27fb856 Package man page of "kvm_stat" tool
|
|
|
4a2fec |
|
|
|
4a2fec |
(cherry picked from commit d4a8e35b84072816c79e23f5d0a69a2145217004)
|
|
|
1bdc94 |
(cherry picked from commit e0425f69f136a05a59ee5cb7022409ed2be94a0b)
|
|
|
1bdc94 |
(cherry picked from commit 57e760aee8b28f3b2c6b9f3aad0d2b916dd13595)
|
|
|
1bdc94 |
(cherry picked from commit d711afa1b8ef8aba1397158626816008936ed0d6)
|
|
|
1bdc94 |
(cherry picked from commit 8ceca7bebce67440898c22d058047daa8ec37031)
|
|
|
4a2fec |
---
|
|
|
4a2fec |
Makefile | 8 +
|
|
|
4a2fec |
redhat/qemu-kvm.spec.template | 7 +-
|
|
|
1bdc94 |
scripts/kvm/kvm_stat | 1585 +++++++++++++++++++++++++++++++++++++++++
|
|
|
1bdc94 |
scripts/kvm/kvm_stat.texi | 104 +++
|
|
|
1bdc94 |
4 files changed, 1703 insertions(+), 1 deletion(-)
|
|
|
4a2fec |
create mode 100755 scripts/kvm/kvm_stat
|
|
|
4a2fec |
create mode 100644 scripts/kvm/kvm_stat.texi
|
|
|
4a2fec |
|
|
|
4a2fec |
diff --git a/Makefile b/Makefile
|
|
|
1bdc94 |
index 89ba4c5..2ffac2b 100644
|
|
|
4a2fec |
--- a/Makefile
|
|
|
4a2fec |
+++ b/Makefile
|
|
|
1bdc94 |
@@ -354,6 +354,9 @@ DOCS=qemu-doc.html qemu-doc.txt qemu.1 qemu-img.1 qemu-nbd.8 qemu-ga.8
|
|
|
4a2fec |
DOCS+=docs/interop/qemu-qmp-ref.html docs/interop/qemu-qmp-ref.txt docs/interop/qemu-qmp-ref.7
|
|
|
4a2fec |
DOCS+=docs/interop/qemu-ga-ref.html docs/interop/qemu-ga-ref.txt docs/interop/qemu-ga-ref.7
|
|
|
1bdc94 |
DOCS+=docs/qemu-block-drivers.7
|
|
|
4a2fec |
+ifdef CONFIG_LINUX
|
|
|
4a2fec |
+DOCS+=kvm_stat.1
|
|
|
4a2fec |
+endif
|
|
|
4a2fec |
ifdef CONFIG_VIRTFS
|
|
|
4a2fec |
DOCS+=fsdev/virtfs-proxy-helper.1
|
|
|
4a2fec |
endif
|
|
|
1bdc94 |
@@ -955,6 +958,11 @@ html: qemu-doc.html docs/interop/qemu-qmp-ref.html docs/interop/qemu-ga-ref.html
|
|
|
4a2fec |
info: qemu-doc.info docs/interop/qemu-qmp-ref.info docs/interop/qemu-ga-ref.info
|
|
|
4a2fec |
pdf: qemu-doc.pdf docs/interop/qemu-qmp-ref.pdf docs/interop/qemu-ga-ref.pdf
|
|
|
4a2fec |
txt: qemu-doc.txt docs/interop/qemu-qmp-ref.txt docs/interop/qemu-ga-ref.txt
|
|
|
4a2fec |
+kvm_stat.1: scripts/kvm/kvm_stat.texi
|
|
|
4a2fec |
+ $(call quiet-command, \
|
|
|
4a2fec |
+ perl -Ww -- $(SRC_PATH)/scripts/texi2pod.pl $< kvm_stat.pod && \
|
|
|
4a2fec |
+ $(POD2MAN) --section=1 --center=" " --release=" " kvm_stat.pod > $@, \
|
|
|
4a2fec |
+ " GEN $@")
|
|
|
4a2fec |
|
|
|
4a2fec |
qemu-doc.html qemu-doc.info qemu-doc.pdf qemu-doc.txt: \
|
|
|
4a2fec |
qemu-img.texi qemu-nbd.texi qemu-options.texi qemu-option-trace.texi \
|
|
|
4a2fec |
diff --git a/scripts/kvm/kvm_stat b/scripts/kvm/kvm_stat
|
|
|
4a2fec |
new file mode 100755
|
|
|
1bdc94 |
index 0000000..c74a9a0
|
|
|
4a2fec |
--- /dev/null
|
|
|
4a2fec |
+++ b/scripts/kvm/kvm_stat
|
|
|
1bdc94 |
@@ -0,0 +1,1585 @@
|
|
|
4a2fec |
+#!/usr/bin/python
|
|
|
4a2fec |
+#
|
|
|
4a2fec |
+# top-like utility for displaying kvm statistics
|
|
|
4a2fec |
+#
|
|
|
4a2fec |
+# Copyright 2006-2008 Qumranet Technologies
|
|
|
4a2fec |
+# Copyright 2008-2011 Red Hat, Inc.
|
|
|
4a2fec |
+#
|
|
|
4a2fec |
+# Authors:
|
|
|
4a2fec |
+# Avi Kivity <avi@redhat.com>
|
|
|
4a2fec |
+#
|
|
|
4a2fec |
+# This work is licensed under the terms of the GNU GPL, version 2. See
|
|
|
4a2fec |
+# the COPYING file in the top-level directory.
|
|
|
4a2fec |
+"""The kvm_stat module outputs statistics about running KVM VMs
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+Three different ways of output formatting are available:
|
|
|
4a2fec |
+- as a top-like text ui
|
|
|
4a2fec |
+- in a key -> value format
|
|
|
4a2fec |
+- in an all keys, all values format
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+The data is sampled from the KVM's debugfs entries and its perf events.
|
|
|
4a2fec |
+"""
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+import curses
|
|
|
4a2fec |
+import sys
|
|
|
4a2fec |
+import os
|
|
|
4a2fec |
+import time
|
|
|
4a2fec |
+import optparse
|
|
|
4a2fec |
+import ctypes
|
|
|
4a2fec |
+import fcntl
|
|
|
4a2fec |
+import resource
|
|
|
4a2fec |
+import struct
|
|
|
4a2fec |
+import re
|
|
|
1bdc94 |
+import subprocess
|
|
|
4a2fec |
+from collections import defaultdict
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+VMX_EXIT_REASONS = {
|
|
|
4a2fec |
+ 'EXCEPTION_NMI': 0,
|
|
|
4a2fec |
+ 'EXTERNAL_INTERRUPT': 1,
|
|
|
4a2fec |
+ 'TRIPLE_FAULT': 2,
|
|
|
4a2fec |
+ 'PENDING_INTERRUPT': 7,
|
|
|
4a2fec |
+ 'NMI_WINDOW': 8,
|
|
|
4a2fec |
+ 'TASK_SWITCH': 9,
|
|
|
4a2fec |
+ 'CPUID': 10,
|
|
|
4a2fec |
+ 'HLT': 12,
|
|
|
4a2fec |
+ 'INVLPG': 14,
|
|
|
4a2fec |
+ 'RDPMC': 15,
|
|
|
4a2fec |
+ 'RDTSC': 16,
|
|
|
4a2fec |
+ 'VMCALL': 18,
|
|
|
4a2fec |
+ 'VMCLEAR': 19,
|
|
|
4a2fec |
+ 'VMLAUNCH': 20,
|
|
|
4a2fec |
+ 'VMPTRLD': 21,
|
|
|
4a2fec |
+ 'VMPTRST': 22,
|
|
|
4a2fec |
+ 'VMREAD': 23,
|
|
|
4a2fec |
+ 'VMRESUME': 24,
|
|
|
4a2fec |
+ 'VMWRITE': 25,
|
|
|
4a2fec |
+ 'VMOFF': 26,
|
|
|
4a2fec |
+ 'VMON': 27,
|
|
|
4a2fec |
+ 'CR_ACCESS': 28,
|
|
|
4a2fec |
+ 'DR_ACCESS': 29,
|
|
|
4a2fec |
+ 'IO_INSTRUCTION': 30,
|
|
|
4a2fec |
+ 'MSR_READ': 31,
|
|
|
4a2fec |
+ 'MSR_WRITE': 32,
|
|
|
4a2fec |
+ 'INVALID_STATE': 33,
|
|
|
4a2fec |
+ 'MWAIT_INSTRUCTION': 36,
|
|
|
4a2fec |
+ 'MONITOR_INSTRUCTION': 39,
|
|
|
4a2fec |
+ 'PAUSE_INSTRUCTION': 40,
|
|
|
4a2fec |
+ 'MCE_DURING_VMENTRY': 41,
|
|
|
4a2fec |
+ 'TPR_BELOW_THRESHOLD': 43,
|
|
|
4a2fec |
+ 'APIC_ACCESS': 44,
|
|
|
4a2fec |
+ 'EPT_VIOLATION': 48,
|
|
|
4a2fec |
+ 'EPT_MISCONFIG': 49,
|
|
|
4a2fec |
+ 'WBINVD': 54,
|
|
|
4a2fec |
+ 'XSETBV': 55,
|
|
|
4a2fec |
+ 'APIC_WRITE': 56,
|
|
|
4a2fec |
+ 'INVPCID': 58,
|
|
|
4a2fec |
+}
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+SVM_EXIT_REASONS = {
|
|
|
4a2fec |
+ 'READ_CR0': 0x000,
|
|
|
4a2fec |
+ 'READ_CR3': 0x003,
|
|
|
4a2fec |
+ 'READ_CR4': 0x004,
|
|
|
4a2fec |
+ 'READ_CR8': 0x008,
|
|
|
4a2fec |
+ 'WRITE_CR0': 0x010,
|
|
|
4a2fec |
+ 'WRITE_CR3': 0x013,
|
|
|
4a2fec |
+ 'WRITE_CR4': 0x014,
|
|
|
4a2fec |
+ 'WRITE_CR8': 0x018,
|
|
|
4a2fec |
+ 'READ_DR0': 0x020,
|
|
|
4a2fec |
+ 'READ_DR1': 0x021,
|
|
|
4a2fec |
+ 'READ_DR2': 0x022,
|
|
|
4a2fec |
+ 'READ_DR3': 0x023,
|
|
|
4a2fec |
+ 'READ_DR4': 0x024,
|
|
|
4a2fec |
+ 'READ_DR5': 0x025,
|
|
|
4a2fec |
+ 'READ_DR6': 0x026,
|
|
|
4a2fec |
+ 'READ_DR7': 0x027,
|
|
|
4a2fec |
+ 'WRITE_DR0': 0x030,
|
|
|
4a2fec |
+ 'WRITE_DR1': 0x031,
|
|
|
4a2fec |
+ 'WRITE_DR2': 0x032,
|
|
|
4a2fec |
+ 'WRITE_DR3': 0x033,
|
|
|
4a2fec |
+ 'WRITE_DR4': 0x034,
|
|
|
4a2fec |
+ 'WRITE_DR5': 0x035,
|
|
|
4a2fec |
+ 'WRITE_DR6': 0x036,
|
|
|
4a2fec |
+ 'WRITE_DR7': 0x037,
|
|
|
4a2fec |
+ 'EXCP_BASE': 0x040,
|
|
|
4a2fec |
+ 'INTR': 0x060,
|
|
|
4a2fec |
+ 'NMI': 0x061,
|
|
|
4a2fec |
+ 'SMI': 0x062,
|
|
|
4a2fec |
+ 'INIT': 0x063,
|
|
|
4a2fec |
+ 'VINTR': 0x064,
|
|
|
4a2fec |
+ 'CR0_SEL_WRITE': 0x065,
|
|
|
4a2fec |
+ 'IDTR_READ': 0x066,
|
|
|
4a2fec |
+ 'GDTR_READ': 0x067,
|
|
|
4a2fec |
+ 'LDTR_READ': 0x068,
|
|
|
4a2fec |
+ 'TR_READ': 0x069,
|
|
|
4a2fec |
+ 'IDTR_WRITE': 0x06a,
|
|
|
4a2fec |
+ 'GDTR_WRITE': 0x06b,
|
|
|
4a2fec |
+ 'LDTR_WRITE': 0x06c,
|
|
|
4a2fec |
+ 'TR_WRITE': 0x06d,
|
|
|
4a2fec |
+ 'RDTSC': 0x06e,
|
|
|
4a2fec |
+ 'RDPMC': 0x06f,
|
|
|
4a2fec |
+ 'PUSHF': 0x070,
|
|
|
4a2fec |
+ 'POPF': 0x071,
|
|
|
4a2fec |
+ 'CPUID': 0x072,
|
|
|
4a2fec |
+ 'RSM': 0x073,
|
|
|
4a2fec |
+ 'IRET': 0x074,
|
|
|
4a2fec |
+ 'SWINT': 0x075,
|
|
|
4a2fec |
+ 'INVD': 0x076,
|
|
|
4a2fec |
+ 'PAUSE': 0x077,
|
|
|
4a2fec |
+ 'HLT': 0x078,
|
|
|
4a2fec |
+ 'INVLPG': 0x079,
|
|
|
4a2fec |
+ 'INVLPGA': 0x07a,
|
|
|
4a2fec |
+ 'IOIO': 0x07b,
|
|
|
4a2fec |
+ 'MSR': 0x07c,
|
|
|
4a2fec |
+ 'TASK_SWITCH': 0x07d,
|
|
|
4a2fec |
+ 'FERR_FREEZE': 0x07e,
|
|
|
4a2fec |
+ 'SHUTDOWN': 0x07f,
|
|
|
4a2fec |
+ 'VMRUN': 0x080,
|
|
|
4a2fec |
+ 'VMMCALL': 0x081,
|
|
|
4a2fec |
+ 'VMLOAD': 0x082,
|
|
|
4a2fec |
+ 'VMSAVE': 0x083,
|
|
|
4a2fec |
+ 'STGI': 0x084,
|
|
|
4a2fec |
+ 'CLGI': 0x085,
|
|
|
4a2fec |
+ 'SKINIT': 0x086,
|
|
|
4a2fec |
+ 'RDTSCP': 0x087,
|
|
|
4a2fec |
+ 'ICEBP': 0x088,
|
|
|
4a2fec |
+ 'WBINVD': 0x089,
|
|
|
4a2fec |
+ 'MONITOR': 0x08a,
|
|
|
4a2fec |
+ 'MWAIT': 0x08b,
|
|
|
4a2fec |
+ 'MWAIT_COND': 0x08c,
|
|
|
4a2fec |
+ 'XSETBV': 0x08d,
|
|
|
4a2fec |
+ 'NPF': 0x400,
|
|
|
4a2fec |
+}
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+# EC definition of HSR (from arch/arm64/include/asm/kvm_arm.h)
|
|
|
4a2fec |
+AARCH64_EXIT_REASONS = {
|
|
|
4a2fec |
+ 'UNKNOWN': 0x00,
|
|
|
4a2fec |
+ 'WFI': 0x01,
|
|
|
4a2fec |
+ 'CP15_32': 0x03,
|
|
|
4a2fec |
+ 'CP15_64': 0x04,
|
|
|
4a2fec |
+ 'CP14_MR': 0x05,
|
|
|
4a2fec |
+ 'CP14_LS': 0x06,
|
|
|
4a2fec |
+ 'FP_ASIMD': 0x07,
|
|
|
4a2fec |
+ 'CP10_ID': 0x08,
|
|
|
4a2fec |
+ 'CP14_64': 0x0C,
|
|
|
4a2fec |
+ 'ILL_ISS': 0x0E,
|
|
|
4a2fec |
+ 'SVC32': 0x11,
|
|
|
4a2fec |
+ 'HVC32': 0x12,
|
|
|
4a2fec |
+ 'SMC32': 0x13,
|
|
|
4a2fec |
+ 'SVC64': 0x15,
|
|
|
4a2fec |
+ 'HVC64': 0x16,
|
|
|
4a2fec |
+ 'SMC64': 0x17,
|
|
|
4a2fec |
+ 'SYS64': 0x18,
|
|
|
4a2fec |
+ 'IABT': 0x20,
|
|
|
4a2fec |
+ 'IABT_HYP': 0x21,
|
|
|
4a2fec |
+ 'PC_ALIGN': 0x22,
|
|
|
4a2fec |
+ 'DABT': 0x24,
|
|
|
4a2fec |
+ 'DABT_HYP': 0x25,
|
|
|
4a2fec |
+ 'SP_ALIGN': 0x26,
|
|
|
4a2fec |
+ 'FP_EXC32': 0x28,
|
|
|
4a2fec |
+ 'FP_EXC64': 0x2C,
|
|
|
4a2fec |
+ 'SERROR': 0x2F,
|
|
|
4a2fec |
+ 'BREAKPT': 0x30,
|
|
|
4a2fec |
+ 'BREAKPT_HYP': 0x31,
|
|
|
4a2fec |
+ 'SOFTSTP': 0x32,
|
|
|
4a2fec |
+ 'SOFTSTP_HYP': 0x33,
|
|
|
4a2fec |
+ 'WATCHPT': 0x34,
|
|
|
4a2fec |
+ 'WATCHPT_HYP': 0x35,
|
|
|
4a2fec |
+ 'BKPT32': 0x38,
|
|
|
4a2fec |
+ 'VECTOR32': 0x3A,
|
|
|
4a2fec |
+ 'BRK64': 0x3C,
|
|
|
4a2fec |
+}
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+# From include/uapi/linux/kvm.h, KVM_EXIT_xxx
|
|
|
4a2fec |
+USERSPACE_EXIT_REASONS = {
|
|
|
4a2fec |
+ 'UNKNOWN': 0,
|
|
|
4a2fec |
+ 'EXCEPTION': 1,
|
|
|
4a2fec |
+ 'IO': 2,
|
|
|
4a2fec |
+ 'HYPERCALL': 3,
|
|
|
4a2fec |
+ 'DEBUG': 4,
|
|
|
4a2fec |
+ 'HLT': 5,
|
|
|
4a2fec |
+ 'MMIO': 6,
|
|
|
4a2fec |
+ 'IRQ_WINDOW_OPEN': 7,
|
|
|
4a2fec |
+ 'SHUTDOWN': 8,
|
|
|
4a2fec |
+ 'FAIL_ENTRY': 9,
|
|
|
4a2fec |
+ 'INTR': 10,
|
|
|
4a2fec |
+ 'SET_TPR': 11,
|
|
|
4a2fec |
+ 'TPR_ACCESS': 12,
|
|
|
4a2fec |
+ 'S390_SIEIC': 13,
|
|
|
4a2fec |
+ 'S390_RESET': 14,
|
|
|
4a2fec |
+ 'DCR': 15,
|
|
|
4a2fec |
+ 'NMI': 16,
|
|
|
4a2fec |
+ 'INTERNAL_ERROR': 17,
|
|
|
4a2fec |
+ 'OSI': 18,
|
|
|
4a2fec |
+ 'PAPR_HCALL': 19,
|
|
|
4a2fec |
+ 'S390_UCONTROL': 20,
|
|
|
4a2fec |
+ 'WATCHDOG': 21,
|
|
|
4a2fec |
+ 'S390_TSCH': 22,
|
|
|
4a2fec |
+ 'EPR': 23,
|
|
|
4a2fec |
+ 'SYSTEM_EVENT': 24,
|
|
|
4a2fec |
+}
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+IOCTL_NUMBERS = {
|
|
|
4a2fec |
+ 'SET_FILTER': 0x40082406,
|
|
|
4a2fec |
+ 'ENABLE': 0x00002400,
|
|
|
4a2fec |
+ 'DISABLE': 0x00002401,
|
|
|
4a2fec |
+ 'RESET': 0x00002403,
|
|
|
4a2fec |
+}
|
|
|
4a2fec |
+
|
|
|
1bdc94 |
+
|
|
|
4a2fec |
+class Arch(object):
|
|
|
4a2fec |
+ """Encapsulates global architecture specific data.
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ Contains the performance event open syscall and ioctl numbers, as
|
|
|
4a2fec |
+ well as the VM exit reasons for the architecture it runs on.
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ """
|
|
|
4a2fec |
+ @staticmethod
|
|
|
4a2fec |
+ def get_arch():
|
|
|
4a2fec |
+ machine = os.uname()[4]
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ if machine.startswith('ppc'):
|
|
|
4a2fec |
+ return ArchPPC()
|
|
|
4a2fec |
+ elif machine.startswith('aarch64'):
|
|
|
4a2fec |
+ return ArchA64()
|
|
|
4a2fec |
+ elif machine.startswith('s390'):
|
|
|
4a2fec |
+ return ArchS390()
|
|
|
4a2fec |
+ else:
|
|
|
4a2fec |
+ # X86_64
|
|
|
4a2fec |
+ for line in open('/proc/cpuinfo'):
|
|
|
4a2fec |
+ if not line.startswith('flags'):
|
|
|
4a2fec |
+ continue
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ flags = line.split()
|
|
|
4a2fec |
+ if 'vmx' in flags:
|
|
|
4a2fec |
+ return ArchX86(VMX_EXIT_REASONS)
|
|
|
4a2fec |
+ if 'svm' in flags:
|
|
|
4a2fec |
+ return ArchX86(SVM_EXIT_REASONS)
|
|
|
4a2fec |
+ return
|
|
|
4a2fec |
+
|
|
|
1bdc94 |
+
|
|
|
4a2fec |
+class ArchX86(Arch):
|
|
|
4a2fec |
+ def __init__(self, exit_reasons):
|
|
|
4a2fec |
+ self.sc_perf_evt_open = 298
|
|
|
4a2fec |
+ self.ioctl_numbers = IOCTL_NUMBERS
|
|
|
4a2fec |
+ self.exit_reasons = exit_reasons
|
|
|
4a2fec |
+
|
|
|
1bdc94 |
+
|
|
|
4a2fec |
+class ArchPPC(Arch):
|
|
|
4a2fec |
+ def __init__(self):
|
|
|
4a2fec |
+ self.sc_perf_evt_open = 319
|
|
|
4a2fec |
+ self.ioctl_numbers = IOCTL_NUMBERS
|
|
|
4a2fec |
+ self.ioctl_numbers['ENABLE'] = 0x20002400
|
|
|
4a2fec |
+ self.ioctl_numbers['DISABLE'] = 0x20002401
|
|
|
4a2fec |
+ self.ioctl_numbers['RESET'] = 0x20002403
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ # PPC comes in 32 and 64 bit and some generated ioctl
|
|
|
4a2fec |
+ # numbers depend on the wordsize.
|
|
|
4a2fec |
+ char_ptr_size = ctypes.sizeof(ctypes.c_char_p)
|
|
|
4a2fec |
+ self.ioctl_numbers['SET_FILTER'] = 0x80002406 | char_ptr_size << 16
|
|
|
4a2fec |
+ self.exit_reasons = {}
|
|
|
4a2fec |
+
|
|
|
1bdc94 |
+
|
|
|
4a2fec |
+class ArchA64(Arch):
|
|
|
4a2fec |
+ def __init__(self):
|
|
|
4a2fec |
+ self.sc_perf_evt_open = 241
|
|
|
4a2fec |
+ self.ioctl_numbers = IOCTL_NUMBERS
|
|
|
4a2fec |
+ self.exit_reasons = AARCH64_EXIT_REASONS
|
|
|
4a2fec |
+
|
|
|
1bdc94 |
+
|
|
|
4a2fec |
+class ArchS390(Arch):
|
|
|
4a2fec |
+ def __init__(self):
|
|
|
4a2fec |
+ self.sc_perf_evt_open = 331
|
|
|
4a2fec |
+ self.ioctl_numbers = IOCTL_NUMBERS
|
|
|
4a2fec |
+ self.exit_reasons = None
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ARCH = Arch.get_arch()
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+class perf_event_attr(ctypes.Structure):
|
|
|
4a2fec |
+ """Struct that holds the necessary data to set up a trace event.
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ For an extensive explanation see perf_event_open(2) and
|
|
|
4a2fec |
+ include/uapi/linux/perf_event.h, struct perf_event_attr
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ All fields that are not initialized in the constructor are 0.
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ """
|
|
|
4a2fec |
+ _fields_ = [('type', ctypes.c_uint32),
|
|
|
4a2fec |
+ ('size', ctypes.c_uint32),
|
|
|
4a2fec |
+ ('config', ctypes.c_uint64),
|
|
|
4a2fec |
+ ('sample_freq', ctypes.c_uint64),
|
|
|
4a2fec |
+ ('sample_type', ctypes.c_uint64),
|
|
|
4a2fec |
+ ('read_format', ctypes.c_uint64),
|
|
|
4a2fec |
+ ('flags', ctypes.c_uint64),
|
|
|
4a2fec |
+ ('wakeup_events', ctypes.c_uint32),
|
|
|
4a2fec |
+ ('bp_type', ctypes.c_uint32),
|
|
|
4a2fec |
+ ('bp_addr', ctypes.c_uint64),
|
|
|
4a2fec |
+ ('bp_len', ctypes.c_uint64),
|
|
|
4a2fec |
+ ]
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ def __init__(self):
|
|
|
4a2fec |
+ super(self.__class__, self).__init__()
|
|
|
4a2fec |
+ self.type = PERF_TYPE_TRACEPOINT
|
|
|
4a2fec |
+ self.size = ctypes.sizeof(self)
|
|
|
4a2fec |
+ self.read_format = PERF_FORMAT_GROUP
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+PERF_TYPE_TRACEPOINT = 2
|
|
|
4a2fec |
+PERF_FORMAT_GROUP = 1 << 3
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+PATH_DEBUGFS_TRACING = '/sys/kernel/debug/tracing'
|
|
|
4a2fec |
+PATH_DEBUGFS_KVM = '/sys/kernel/debug/kvm'
|
|
|
4a2fec |
+
|
|
|
1bdc94 |
+
|
|
|
4a2fec |
+class Group(object):
|
|
|
4a2fec |
+ """Represents a perf event group."""
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ def __init__(self):
|
|
|
4a2fec |
+ self.events = []
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ def add_event(self, event):
|
|
|
4a2fec |
+ self.events.append(event)
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ def read(self):
|
|
|
4a2fec |
+ """Returns a dict with 'event name: value' for all events in the
|
|
|
4a2fec |
+ group.
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ Values are read by reading from the file descriptor of the
|
|
|
4a2fec |
+ event that is the group leader. See perf_event_open(2) for
|
|
|
4a2fec |
+ details.
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ Read format for the used event configuration is:
|
|
|
4a2fec |
+ struct read_format {
|
|
|
4a2fec |
+ u64 nr; /* The number of events */
|
|
|
4a2fec |
+ struct {
|
|
|
4a2fec |
+ u64 value; /* The value of the event */
|
|
|
4a2fec |
+ } values[nr];
|
|
|
4a2fec |
+ };
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ """
|
|
|
4a2fec |
+ length = 8 * (1 + len(self.events))
|
|
|
4a2fec |
+ read_format = 'xxxxxxxx' + 'Q' * len(self.events)
|
|
|
4a2fec |
+ return dict(zip([event.name for event in self.events],
|
|
|
4a2fec |
+ struct.unpack(read_format,
|
|
|
4a2fec |
+ os.read(self.events[0].fd, length))))
|
|
|
4a2fec |
+
|
|
|
1bdc94 |
+
|
|
|
4a2fec |
+class Event(object):
|
|
|
4a2fec |
+ """Represents a performance event and manages its life cycle."""
|
|
|
4a2fec |
+ def __init__(self, name, group, trace_cpu, trace_pid, trace_point,
|
|
|
4a2fec |
+ trace_filter, trace_set='kvm'):
|
|
|
1bdc94 |
+ self.libc = ctypes.CDLL('libc.so.6', use_errno=True)
|
|
|
1bdc94 |
+ self.syscall = self.libc.syscall
|
|
|
4a2fec |
+ self.name = name
|
|
|
4a2fec |
+ self.fd = None
|
|
|
4a2fec |
+ self.setup_event(group, trace_cpu, trace_pid, trace_point,
|
|
|
4a2fec |
+ trace_filter, trace_set)
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ def __del__(self):
|
|
|
4a2fec |
+ """Closes the event's file descriptor.
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ As no python file object was created for the file descriptor,
|
|
|
4a2fec |
+ python will not reference count the descriptor and will not
|
|
|
4a2fec |
+ close it itself automatically, so we do it.
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ """
|
|
|
4a2fec |
+ if self.fd:
|
|
|
4a2fec |
+ os.close(self.fd)
|
|
|
4a2fec |
+
|
|
|
1bdc94 |
+ def perf_event_open(self, attr, pid, cpu, group_fd, flags):
|
|
|
1bdc94 |
+ """Wrapper for the sys_perf_evt_open() syscall.
|
|
|
1bdc94 |
+
|
|
|
1bdc94 |
+ Used to set up performance events, returns a file descriptor or -1
|
|
|
1bdc94 |
+ on error.
|
|
|
1bdc94 |
+
|
|
|
1bdc94 |
+ Attributes are:
|
|
|
1bdc94 |
+ - syscall number
|
|
|
1bdc94 |
+ - struct perf_event_attr *
|
|
|
1bdc94 |
+ - pid or -1 to monitor all pids
|
|
|
1bdc94 |
+ - cpu number or -1 to monitor all cpus
|
|
|
1bdc94 |
+ - The file descriptor of the group leader or -1 to create a group.
|
|
|
1bdc94 |
+ - flags
|
|
|
1bdc94 |
+
|
|
|
1bdc94 |
+ """
|
|
|
1bdc94 |
+ return self.syscall(ARCH.sc_perf_evt_open, ctypes.pointer(attr),
|
|
|
1bdc94 |
+ ctypes.c_int(pid), ctypes.c_int(cpu),
|
|
|
1bdc94 |
+ ctypes.c_int(group_fd), ctypes.c_long(flags))
|
|
|
1bdc94 |
+
|
|
|
4a2fec |
+ def setup_event_attribute(self, trace_set, trace_point):
|
|
|
4a2fec |
+ """Returns an initialized ctype perf_event_attr struct."""
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ id_path = os.path.join(PATH_DEBUGFS_TRACING, 'events', trace_set,
|
|
|
4a2fec |
+ trace_point, 'id')
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ event_attr = perf_event_attr()
|
|
|
4a2fec |
+ event_attr.config = int(open(id_path).read())
|
|
|
4a2fec |
+ return event_attr
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ def setup_event(self, group, trace_cpu, trace_pid, trace_point,
|
|
|
4a2fec |
+ trace_filter, trace_set):
|
|
|
4a2fec |
+ """Sets up the perf event in Linux.
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ Issues the syscall to register the event in the kernel and
|
|
|
4a2fec |
+ then sets the optional filter.
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ """
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ event_attr = self.setup_event_attribute(trace_set, trace_point)
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ # First event will be group leader.
|
|
|
4a2fec |
+ group_leader = -1
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ # All others have to pass the leader's descriptor instead.
|
|
|
4a2fec |
+ if group.events:
|
|
|
4a2fec |
+ group_leader = group.events[0].fd
|
|
|
4a2fec |
+
|
|
|
1bdc94 |
+ fd = self.perf_event_open(event_attr, trace_pid,
|
|
|
1bdc94 |
+ trace_cpu, group_leader, 0)
|
|
|
4a2fec |
+ if fd == -1:
|
|
|
4a2fec |
+ err = ctypes.get_errno()
|
|
|
4a2fec |
+ raise OSError(err, os.strerror(err),
|
|
|
4a2fec |
+ 'while calling sys_perf_event_open().')
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ if trace_filter:
|
|
|
4a2fec |
+ fcntl.ioctl(fd, ARCH.ioctl_numbers['SET_FILTER'],
|
|
|
4a2fec |
+ trace_filter)
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ self.fd = fd
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ def enable(self):
|
|
|
4a2fec |
+ """Enables the trace event in the kernel.
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ Enabling the group leader makes reading counters from it and the
|
|
|
4a2fec |
+ events under it possible.
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ """
|
|
|
4a2fec |
+ fcntl.ioctl(self.fd, ARCH.ioctl_numbers['ENABLE'], 0)
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ def disable(self):
|
|
|
4a2fec |
+ """Disables the trace event in the kernel.
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ Disabling the group leader makes reading all counters under it
|
|
|
4a2fec |
+ impossible.
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ """
|
|
|
4a2fec |
+ fcntl.ioctl(self.fd, ARCH.ioctl_numbers['DISABLE'], 0)
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ def reset(self):
|
|
|
4a2fec |
+ """Resets the count of the trace event in the kernel."""
|
|
|
4a2fec |
+ fcntl.ioctl(self.fd, ARCH.ioctl_numbers['RESET'], 0)
|
|
|
4a2fec |
+
|
|
|
1bdc94 |
+
|
|
|
1bdc94 |
+class Provider(object):
|
|
|
1bdc94 |
+ """Encapsulates functionalities used by all providers."""
|
|
|
1bdc94 |
+ @staticmethod
|
|
|
1bdc94 |
+ def is_field_wanted(fields_filter, field):
|
|
|
1bdc94 |
+ """Indicate whether field is valid according to fields_filter."""
|
|
|
1bdc94 |
+ if not fields_filter or fields_filter == "help":
|
|
|
1bdc94 |
+ return True
|
|
|
1bdc94 |
+ return re.match(fields_filter, field) is not None
|
|
|
1bdc94 |
+
|
|
|
1bdc94 |
+ @staticmethod
|
|
|
1bdc94 |
+ def walkdir(path):
|
|
|
1bdc94 |
+ """Returns os.walk() data for specified directory.
|
|
|
1bdc94 |
+
|
|
|
1bdc94 |
+ As it is only a wrapper it returns the same 3-tuple of (dirpath,
|
|
|
1bdc94 |
+ dirnames, filenames).
|
|
|
1bdc94 |
+ """
|
|
|
1bdc94 |
+ return next(os.walk(path))
|
|
|
1bdc94 |
+
|
|
|
1bdc94 |
+
|
|
|
1bdc94 |
+class TracepointProvider(Provider):
|
|
|
4a2fec |
+ """Data provider for the stats class.
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ Manages the events/groups from which it acquires its data.
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ """
|
|
|
1bdc94 |
+ def __init__(self, pid, fields_filter):
|
|
|
4a2fec |
+ self.group_leaders = []
|
|
|
1bdc94 |
+ self.filters = self.get_filters()
|
|
|
1bdc94 |
+ self.update_fields(fields_filter)
|
|
|
1bdc94 |
+ self.pid = pid
|
|
|
1bdc94 |
+
|
|
|
1bdc94 |
+ @staticmethod
|
|
|
1bdc94 |
+ def get_filters():
|
|
|
1bdc94 |
+ """Returns a dict of trace events, their filter ids and
|
|
|
1bdc94 |
+ the values that can be filtered.
|
|
|
1bdc94 |
+
|
|
|
1bdc94 |
+ Trace events can be filtered for special values by setting a
|
|
|
1bdc94 |
+ filter string via an ioctl. The string normally has the format
|
|
|
1bdc94 |
+ identifier==value. For each filter a new event will be created, to
|
|
|
1bdc94 |
+ be able to distinguish the events.
|
|
|
1bdc94 |
+
|
|
|
1bdc94 |
+ """
|
|
|
1bdc94 |
+ filters = {}
|
|
|
1bdc94 |
+ filters['kvm_userspace_exit'] = ('reason', USERSPACE_EXIT_REASONS)
|
|
|
1bdc94 |
+ if ARCH.exit_reasons:
|
|
|
1bdc94 |
+ filters['kvm_exit'] = ('exit_reason', ARCH.exit_reasons)
|
|
|
1bdc94 |
+ return filters
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ def get_available_fields(self):
|
|
|
4a2fec |
+ """Returns a list of available event's of format 'event name(filter
|
|
|
4a2fec |
+ name)'.
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ All available events have directories under
|
|
|
4a2fec |
+ /sys/kernel/debug/tracing/events/ which export information
|
|
|
4a2fec |
+ about the specific event. Therefore, listing the dirs gives us
|
|
|
4a2fec |
+ a list of all available events.
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ Some events like the vm exit reasons can be filtered for
|
|
|
4a2fec |
+ specific values. To take account for that, the routine below
|
|
|
4a2fec |
+ creates special fields with the following format:
|
|
|
4a2fec |
+ event name(filter name)
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ """
|
|
|
4a2fec |
+ path = os.path.join(PATH_DEBUGFS_TRACING, 'events', 'kvm')
|
|
|
1bdc94 |
+ fields = self.walkdir(path)[1]
|
|
|
4a2fec |
+ extra = []
|
|
|
4a2fec |
+ for field in fields:
|
|
|
4a2fec |
+ if field in self.filters:
|
|
|
4a2fec |
+ filter_name_, filter_dicts = self.filters[field]
|
|
|
4a2fec |
+ for name in filter_dicts:
|
|
|
4a2fec |
+ extra.append(field + '(' + name + ')')
|
|
|
4a2fec |
+ fields += extra
|
|
|
4a2fec |
+ return fields
|
|
|
4a2fec |
+
|
|
|
1bdc94 |
+ def update_fields(self, fields_filter):
|
|
|
1bdc94 |
+ """Refresh fields, applying fields_filter"""
|
|
|
1bdc94 |
+ self._fields = [field for field in self.get_available_fields()
|
|
|
1bdc94 |
+ if self.is_field_wanted(fields_filter, field)]
|
|
|
1bdc94 |
+
|
|
|
1bdc94 |
+ @staticmethod
|
|
|
1bdc94 |
+ def get_online_cpus():
|
|
|
1bdc94 |
+ """Returns a list of cpu id integers."""
|
|
|
1bdc94 |
+ def parse_int_list(list_string):
|
|
|
1bdc94 |
+ """Returns an int list from a string of comma separated integers and
|
|
|
1bdc94 |
+ integer ranges."""
|
|
|
1bdc94 |
+ integers = []
|
|
|
1bdc94 |
+ members = list_string.split(',')
|
|
|
1bdc94 |
+
|
|
|
1bdc94 |
+ for member in members:
|
|
|
1bdc94 |
+ if '-' not in member:
|
|
|
1bdc94 |
+ integers.append(int(member))
|
|
|
1bdc94 |
+ else:
|
|
|
1bdc94 |
+ int_range = member.split('-')
|
|
|
1bdc94 |
+ integers.extend(range(int(int_range[0]),
|
|
|
1bdc94 |
+ int(int_range[1]) + 1))
|
|
|
1bdc94 |
+
|
|
|
1bdc94 |
+ return integers
|
|
|
1bdc94 |
+
|
|
|
1bdc94 |
+ with open('/sys/devices/system/cpu/online') as cpu_list:
|
|
|
1bdc94 |
+ cpu_string = cpu_list.readline()
|
|
|
1bdc94 |
+ return parse_int_list(cpu_string)
|
|
|
1bdc94 |
+
|
|
|
4a2fec |
+ def setup_traces(self):
|
|
|
4a2fec |
+ """Creates all event and group objects needed to be able to retrieve
|
|
|
4a2fec |
+ data."""
|
|
|
1bdc94 |
+ fields = self.get_available_fields()
|
|
|
4a2fec |
+ if self._pid > 0:
|
|
|
4a2fec |
+ # Fetch list of all threads of the monitored pid, as qemu
|
|
|
4a2fec |
+ # starts a thread for each vcpu.
|
|
|
4a2fec |
+ path = os.path.join('/proc', str(self._pid), 'task')
|
|
|
1bdc94 |
+ groupids = self.walkdir(path)[1]
|
|
|
4a2fec |
+ else:
|
|
|
1bdc94 |
+ groupids = self.get_online_cpus()
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ # The constant is needed as a buffer for python libs, std
|
|
|
4a2fec |
+ # streams and other files that the script opens.
|
|
|
1bdc94 |
+ newlim = len(groupids) * len(fields) + 50
|
|
|
4a2fec |
+ try:
|
|
|
4a2fec |
+ softlim_, hardlim = resource.getrlimit(resource.RLIMIT_NOFILE)
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ if hardlim < newlim:
|
|
|
4a2fec |
+ # Now we need CAP_SYS_RESOURCE, to increase the hard limit.
|
|
|
4a2fec |
+ resource.setrlimit(resource.RLIMIT_NOFILE, (newlim, newlim))
|
|
|
4a2fec |
+ else:
|
|
|
4a2fec |
+ # Raising the soft limit is sufficient.
|
|
|
4a2fec |
+ resource.setrlimit(resource.RLIMIT_NOFILE, (newlim, hardlim))
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ except ValueError:
|
|
|
4a2fec |
+ sys.exit("NOFILE rlimit could not be raised to {0}".format(newlim))
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ for groupid in groupids:
|
|
|
4a2fec |
+ group = Group()
|
|
|
1bdc94 |
+ for name in fields:
|
|
|
4a2fec |
+ tracepoint = name
|
|
|
4a2fec |
+ tracefilter = None
|
|
|
4a2fec |
+ match = re.match(r'(.*)\((.*)\)', name)
|
|
|
4a2fec |
+ if match:
|
|
|
4a2fec |
+ tracepoint, sub = match.groups()
|
|
|
4a2fec |
+ tracefilter = ('%s==%d\0' %
|
|
|
4a2fec |
+ (self.filters[tracepoint][0],
|
|
|
4a2fec |
+ self.filters[tracepoint][1][sub]))
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ # From perf_event_open(2):
|
|
|
4a2fec |
+ # pid > 0 and cpu == -1
|
|
|
4a2fec |
+ # This measures the specified process/thread on any CPU.
|
|
|
4a2fec |
+ #
|
|
|
4a2fec |
+ # pid == -1 and cpu >= 0
|
|
|
4a2fec |
+ # This measures all processes/threads on the specified CPU.
|
|
|
4a2fec |
+ trace_cpu = groupid if self._pid == 0 else -1
|
|
|
4a2fec |
+ trace_pid = int(groupid) if self._pid != 0 else -1
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ group.add_event(Event(name=name,
|
|
|
4a2fec |
+ group=group,
|
|
|
4a2fec |
+ trace_cpu=trace_cpu,
|
|
|
4a2fec |
+ trace_pid=trace_pid,
|
|
|
4a2fec |
+ trace_point=tracepoint,
|
|
|
4a2fec |
+ trace_filter=tracefilter))
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ self.group_leaders.append(group)
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ @property
|
|
|
4a2fec |
+ def fields(self):
|
|
|
4a2fec |
+ return self._fields
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ @fields.setter
|
|
|
4a2fec |
+ def fields(self, fields):
|
|
|
4a2fec |
+ """Enables/disables the (un)wanted events"""
|
|
|
4a2fec |
+ self._fields = fields
|
|
|
4a2fec |
+ for group in self.group_leaders:
|
|
|
4a2fec |
+ for index, event in enumerate(group.events):
|
|
|
4a2fec |
+ if event.name in fields:
|
|
|
4a2fec |
+ event.reset()
|
|
|
4a2fec |
+ event.enable()
|
|
|
4a2fec |
+ else:
|
|
|
4a2fec |
+ # Do not disable the group leader.
|
|
|
4a2fec |
+ # It would disable all of its events.
|
|
|
4a2fec |
+ if index != 0:
|
|
|
4a2fec |
+ event.disable()
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ @property
|
|
|
4a2fec |
+ def pid(self):
|
|
|
4a2fec |
+ return self._pid
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ @pid.setter
|
|
|
4a2fec |
+ def pid(self, pid):
|
|
|
4a2fec |
+ """Changes the monitored pid by setting new traces."""
|
|
|
4a2fec |
+ self._pid = pid
|
|
|
4a2fec |
+ # The garbage collector will get rid of all Event/Group
|
|
|
4a2fec |
+ # objects and open files after removing the references.
|
|
|
4a2fec |
+ self.group_leaders = []
|
|
|
4a2fec |
+ self.setup_traces()
|
|
|
4a2fec |
+ self.fields = self._fields
|
|
|
4a2fec |
+
|
|
|
1bdc94 |
+ def read(self, by_guest=0):
|
|
|
4a2fec |
+ """Returns 'event name: current value' for all enabled events."""
|
|
|
4a2fec |
+ ret = defaultdict(int)
|
|
|
4a2fec |
+ for group in self.group_leaders:
|
|
|
4a2fec |
+ for name, val in group.read().iteritems():
|
|
|
4a2fec |
+ if name in self._fields:
|
|
|
4a2fec |
+ ret[name] += val
|
|
|
4a2fec |
+ return ret
|
|
|
4a2fec |
+
|
|
|
1bdc94 |
+ def reset(self):
|
|
|
1bdc94 |
+ """Reset all field counters"""
|
|
|
1bdc94 |
+ for group in self.group_leaders:
|
|
|
1bdc94 |
+ for event in group.events:
|
|
|
1bdc94 |
+ event.reset()
|
|
|
1bdc94 |
+
|
|
|
1bdc94 |
+
|
|
|
1bdc94 |
+class DebugfsProvider(Provider):
|
|
|
4a2fec |
+ """Provides data from the files that KVM creates in the kvm debugfs
|
|
|
4a2fec |
+ folder."""
|
|
|
1bdc94 |
+ def __init__(self, pid, fields_filter, include_past):
|
|
|
1bdc94 |
+ self.update_fields(fields_filter)
|
|
|
1bdc94 |
+ self._baseline = {}
|
|
|
4a2fec |
+ self.do_read = True
|
|
|
1bdc94 |
+ self.paths = []
|
|
|
1bdc94 |
+ self.pid = pid
|
|
|
1bdc94 |
+ if include_past:
|
|
|
1bdc94 |
+ self.restore()
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ def get_available_fields(self):
|
|
|
4a2fec |
+ """"Returns a list of available fields.
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ The fields are all available KVM debugfs files
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ """
|
|
|
1bdc94 |
+ return self.walkdir(PATH_DEBUGFS_KVM)[2]
|
|
|
1bdc94 |
+
|
|
|
1bdc94 |
+ def update_fields(self, fields_filter):
|
|
|
1bdc94 |
+ """Refresh fields, applying fields_filter"""
|
|
|
1bdc94 |
+ self._fields = [field for field in self.get_available_fields()
|
|
|
1bdc94 |
+ if self.is_field_wanted(fields_filter, field)]
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ @property
|
|
|
4a2fec |
+ def fields(self):
|
|
|
4a2fec |
+ return self._fields
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ @fields.setter
|
|
|
4a2fec |
+ def fields(self, fields):
|
|
|
4a2fec |
+ self._fields = fields
|
|
|
1bdc94 |
+ self.reset()
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ @property
|
|
|
4a2fec |
+ def pid(self):
|
|
|
4a2fec |
+ return self._pid
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ @pid.setter
|
|
|
4a2fec |
+ def pid(self, pid):
|
|
|
1bdc94 |
+ self._pid = pid
|
|
|
4a2fec |
+ if pid != 0:
|
|
|
1bdc94 |
+ vms = self.walkdir(PATH_DEBUGFS_KVM)[1]
|
|
|
4a2fec |
+ if len(vms) == 0:
|
|
|
4a2fec |
+ self.do_read = False
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ self.paths = filter(lambda x: "{}-".format(pid) in x, vms)
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ else:
|
|
|
1bdc94 |
+ self.paths = []
|
|
|
4a2fec |
+ self.do_read = True
|
|
|
1bdc94 |
+ self.reset()
|
|
|
4a2fec |
+
|
|
|
1bdc94 |
+ def read(self, reset=0, by_guest=0):
|
|
|
1bdc94 |
+ """Returns a dict with format:'file name / field -> current value'.
|
|
|
1bdc94 |
+
|
|
|
1bdc94 |
+ Parameter 'reset':
|
|
|
1bdc94 |
+ 0 plain read
|
|
|
1bdc94 |
+ 1 reset field counts to 0
|
|
|
1bdc94 |
+ 2 restore the original field counts
|
|
|
1bdc94 |
+
|
|
|
1bdc94 |
+ """
|
|
|
4a2fec |
+ results = {}
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ # If no debugfs filtering support is available, then don't read.
|
|
|
4a2fec |
+ if not self.do_read:
|
|
|
4a2fec |
+ return results
|
|
|
4a2fec |
+
|
|
|
1bdc94 |
+ paths = self.paths
|
|
|
1bdc94 |
+ if self._pid == 0:
|
|
|
1bdc94 |
+ paths = []
|
|
|
1bdc94 |
+ for entry in os.walk(PATH_DEBUGFS_KVM):
|
|
|
1bdc94 |
+ for dir in entry[1]:
|
|
|
1bdc94 |
+ paths.append(dir)
|
|
|
1bdc94 |
+ for path in paths:
|
|
|
4a2fec |
+ for field in self._fields:
|
|
|
1bdc94 |
+ value = self.read_field(field, path)
|
|
|
1bdc94 |
+ key = path + field
|
|
|
1bdc94 |
+ if reset == 1:
|
|
|
1bdc94 |
+ self._baseline[key] = value
|
|
|
1bdc94 |
+ if reset == 2:
|
|
|
1bdc94 |
+ self._baseline[key] = 0
|
|
|
1bdc94 |
+ if self._baseline.get(key, -1) == -1:
|
|
|
1bdc94 |
+ self._baseline[key] = value
|
|
|
1bdc94 |
+ increment = (results.get(field, 0) + value -
|
|
|
1bdc94 |
+ self._baseline.get(key, 0))
|
|
|
1bdc94 |
+ if by_guest:
|
|
|
1bdc94 |
+ pid = key.split('-')[0]
|
|
|
1bdc94 |
+ if pid in results:
|
|
|
1bdc94 |
+ results[pid] += increment
|
|
|
1bdc94 |
+ else:
|
|
|
1bdc94 |
+ results[pid] = increment
|
|
|
1bdc94 |
+ else:
|
|
|
1bdc94 |
+ results[field] = increment
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ return results
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ def read_field(self, field, path):
|
|
|
4a2fec |
+ """Returns the value of a single field from a specific VM."""
|
|
|
4a2fec |
+ try:
|
|
|
4a2fec |
+ return int(open(os.path.join(PATH_DEBUGFS_KVM,
|
|
|
4a2fec |
+ path,
|
|
|
4a2fec |
+ field))
|
|
|
4a2fec |
+ .read())
|
|
|
4a2fec |
+ except IOError:
|
|
|
4a2fec |
+ return 0
|
|
|
4a2fec |
+
|
|
|
1bdc94 |
+ def reset(self):
|
|
|
1bdc94 |
+ """Reset field counters"""
|
|
|
1bdc94 |
+ self._baseline = {}
|
|
|
1bdc94 |
+ self.read(1)
|
|
|
1bdc94 |
+
|
|
|
1bdc94 |
+ def restore(self):
|
|
|
1bdc94 |
+ """Reset field counters"""
|
|
|
1bdc94 |
+ self._baseline = {}
|
|
|
1bdc94 |
+ self.read(2)
|
|
|
1bdc94 |
+
|
|
|
1bdc94 |
+
|
|
|
4a2fec |
+class Stats(object):
|
|
|
4a2fec |
+ """Manages the data providers and the data they provide.
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ It is used to set filters on the provider's data and collect all
|
|
|
4a2fec |
+ provider data.
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ """
|
|
|
1bdc94 |
+ def __init__(self, options):
|
|
|
1bdc94 |
+ self.providers = self.get_providers(options)
|
|
|
1bdc94 |
+ self._pid_filter = options.pid
|
|
|
1bdc94 |
+ self._fields_filter = options.fields
|
|
|
4a2fec |
+ self.values = {}
|
|
|
1bdc94 |
+
|
|
|
1bdc94 |
+ @staticmethod
|
|
|
1bdc94 |
+ def get_providers(options):
|
|
|
1bdc94 |
+ """Returns a list of data providers depending on the passed options."""
|
|
|
1bdc94 |
+ providers = []
|
|
|
1bdc94 |
+
|
|
|
1bdc94 |
+ if options.debugfs:
|
|
|
1bdc94 |
+ providers.append(DebugfsProvider(options.pid, options.fields,
|
|
|
1bdc94 |
+ options.dbgfs_include_past))
|
|
|
1bdc94 |
+ if options.tracepoints or not providers:
|
|
|
1bdc94 |
+ providers.append(TracepointProvider(options.pid, options.fields))
|
|
|
1bdc94 |
+
|
|
|
1bdc94 |
+ return providers
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ def update_provider_filters(self):
|
|
|
4a2fec |
+ """Propagates fields filters to providers."""
|
|
|
4a2fec |
+ # As we reset the counters when updating the fields we can
|
|
|
4a2fec |
+ # also clear the cache of old values.
|
|
|
4a2fec |
+ self.values = {}
|
|
|
4a2fec |
+ for provider in self.providers:
|
|
|
1bdc94 |
+ provider.update_fields(self._fields_filter)
|
|
|
4a2fec |
+
|
|
|
1bdc94 |
+ def reset(self):
|
|
|
1bdc94 |
+ self.values = {}
|
|
|
4a2fec |
+ for provider in self.providers:
|
|
|
1bdc94 |
+ provider.reset()
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ @property
|
|
|
4a2fec |
+ def fields_filter(self):
|
|
|
4a2fec |
+ return self._fields_filter
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ @fields_filter.setter
|
|
|
4a2fec |
+ def fields_filter(self, fields_filter):
|
|
|
1bdc94 |
+ if fields_filter != self._fields_filter:
|
|
|
1bdc94 |
+ self._fields_filter = fields_filter
|
|
|
1bdc94 |
+ self.update_provider_filters()
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ @property
|
|
|
4a2fec |
+ def pid_filter(self):
|
|
|
4a2fec |
+ return self._pid_filter
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ @pid_filter.setter
|
|
|
4a2fec |
+ def pid_filter(self, pid):
|
|
|
1bdc94 |
+ if pid != self._pid_filter:
|
|
|
1bdc94 |
+ self._pid_filter = pid
|
|
|
1bdc94 |
+ self.values = {}
|
|
|
1bdc94 |
+ for provider in self.providers:
|
|
|
1bdc94 |
+ provider.pid = self._pid_filter
|
|
|
4a2fec |
+
|
|
|
1bdc94 |
+ def get(self, by_guest=0):
|
|
|
4a2fec |
+ """Returns a dict with field -> (value, delta to last value) of all
|
|
|
4a2fec |
+ provider data."""
|
|
|
4a2fec |
+ for provider in self.providers:
|
|
|
1bdc94 |
+ new = provider.read(by_guest=by_guest)
|
|
|
1bdc94 |
+ for key in new if by_guest else provider.fields:
|
|
|
1bdc94 |
+ oldval = self.values.get(key, (0, 0))[0]
|
|
|
4a2fec |
+ newval = new.get(key, 0)
|
|
|
1bdc94 |
+ newdelta = newval - oldval
|
|
|
4a2fec |
+ self.values[key] = (newval, newdelta)
|
|
|
4a2fec |
+ return self.values
|
|
|
4a2fec |
+
|
|
|
1bdc94 |
+ def toggle_display_guests(self, to_pid):
|
|
|
1bdc94 |
+ """Toggle between collection of stats by individual event and by
|
|
|
1bdc94 |
+ guest pid
|
|
|
1bdc94 |
+
|
|
|
1bdc94 |
+ Events reported by DebugfsProvider change when switching to/from
|
|
|
1bdc94 |
+ reading by guest values. Hence we have to remove the excess event
|
|
|
1bdc94 |
+ names from self.values.
|
|
|
1bdc94 |
+
|
|
|
1bdc94 |
+ """
|
|
|
1bdc94 |
+ if any(isinstance(ins, TracepointProvider) for ins in self.providers):
|
|
|
1bdc94 |
+ return 1
|
|
|
1bdc94 |
+ if to_pid:
|
|
|
1bdc94 |
+ for provider in self.providers:
|
|
|
1bdc94 |
+ if isinstance(provider, DebugfsProvider):
|
|
|
1bdc94 |
+ for key in provider.fields:
|
|
|
1bdc94 |
+ if key in self.values.keys():
|
|
|
1bdc94 |
+ del self.values[key]
|
|
|
1bdc94 |
+ else:
|
|
|
1bdc94 |
+ oldvals = self.values.copy()
|
|
|
1bdc94 |
+ for key in oldvals:
|
|
|
1bdc94 |
+ if key.isdigit():
|
|
|
1bdc94 |
+ del self.values[key]
|
|
|
1bdc94 |
+ # Update oldval (see get())
|
|
|
1bdc94 |
+ self.get(to_pid)
|
|
|
1bdc94 |
+ return 0
|
|
|
1bdc94 |
+
|
|
|
1bdc94 |
+DELAY_DEFAULT = 3.0
|
|
|
1bdc94 |
+MAX_GUEST_NAME_LEN = 48
|
|
|
1bdc94 |
+MAX_REGEX_LEN = 44
|
|
|
1bdc94 |
+DEFAULT_REGEX = r'^[^\(]*$'
|
|
|
1bdc94 |
+SORT_DEFAULT = 0
|
|
|
1bdc94 |
+
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+class Tui(object):
|
|
|
4a2fec |
+ """Instruments curses to draw a nice text ui."""
|
|
|
4a2fec |
+ def __init__(self, stats):
|
|
|
4a2fec |
+ self.stats = stats
|
|
|
4a2fec |
+ self.screen = None
|
|
|
1bdc94 |
+ self._delay_initial = 0.25
|
|
|
1bdc94 |
+ self._delay_regular = DELAY_DEFAULT
|
|
|
1bdc94 |
+ self._sorting = SORT_DEFAULT
|
|
|
1bdc94 |
+ self._display_guests = 0
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ def __enter__(self):
|
|
|
4a2fec |
+ """Initialises curses for later use. Based on curses.wrapper
|
|
|
4a2fec |
+ implementation from the Python standard library."""
|
|
|
4a2fec |
+ self.screen = curses.initscr()
|
|
|
4a2fec |
+ curses.noecho()
|
|
|
4a2fec |
+ curses.cbreak()
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ # The try/catch works around a minor bit of
|
|
|
4a2fec |
+ # over-conscientiousness in the curses module, the error
|
|
|
4a2fec |
+ # return from C start_color() is ignorable.
|
|
|
4a2fec |
+ try:
|
|
|
4a2fec |
+ curses.start_color()
|
|
|
1bdc94 |
+ except curses.error:
|
|
|
1bdc94 |
+ pass
|
|
|
1bdc94 |
+
|
|
|
1bdc94 |
+ # Hide cursor in extra statement as some monochrome terminals
|
|
|
1bdc94 |
+ # might support hiding but not colors.
|
|
|
1bdc94 |
+ try:
|
|
|
1bdc94 |
+ curses.curs_set(0)
|
|
|
1bdc94 |
+ except curses.error:
|
|
|
4a2fec |
+ pass
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ curses.use_default_colors()
|
|
|
4a2fec |
+ return self
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ def __exit__(self, *exception):
|
|
|
1bdc94 |
+ """Resets the terminal to its normal state. Based on curses.wrapper
|
|
|
4a2fec |
+ implementation from the Python standard library."""
|
|
|
4a2fec |
+ if self.screen:
|
|
|
4a2fec |
+ self.screen.keypad(0)
|
|
|
4a2fec |
+ curses.echo()
|
|
|
4a2fec |
+ curses.nocbreak()
|
|
|
4a2fec |
+ curses.endwin()
|
|
|
4a2fec |
+
|
|
|
1bdc94 |
+ @staticmethod
|
|
|
1bdc94 |
+ def get_all_gnames():
|
|
|
1bdc94 |
+ """Returns a list of (pid, gname) tuples of all running guests"""
|
|
|
1bdc94 |
+ res = []
|
|
|
1bdc94 |
+ try:
|
|
|
1bdc94 |
+ child = subprocess.Popen(['ps', '-A', '--format', 'pid,args'],
|
|
|
1bdc94 |
+ stdout=subprocess.PIPE)
|
|
|
1bdc94 |
+ except:
|
|
|
1bdc94 |
+ raise Exception
|
|
|
1bdc94 |
+ for line in child.stdout:
|
|
|
1bdc94 |
+ line = line.lstrip().split(' ', 1)
|
|
|
1bdc94 |
+ # perform a sanity check before calling the more expensive
|
|
|
1bdc94 |
+ # function to possibly extract the guest name
|
|
|
1bdc94 |
+ if ' -name ' in line[1]:
|
|
|
1bdc94 |
+ res.append((line[0], Tui.get_gname_from_pid(line[0])))
|
|
|
1bdc94 |
+ child.stdout.close()
|
|
|
1bdc94 |
+
|
|
|
1bdc94 |
+ return res
|
|
|
1bdc94 |
+
|
|
|
1bdc94 |
+ def print_all_gnames(self, row):
|
|
|
1bdc94 |
+ """Print a list of all running guests along with their pids."""
|
|
|
1bdc94 |
+ self.screen.addstr(row, 2, '%8s %-60s' %
|
|
|
1bdc94 |
+ ('Pid', 'Guest Name (fuzzy list, might be '
|
|
|
1bdc94 |
+ 'inaccurate!)'),
|
|
|
1bdc94 |
+ curses.A_UNDERLINE)
|
|
|
1bdc94 |
+ row += 1
|
|
|
1bdc94 |
+ try:
|
|
|
1bdc94 |
+ for line in self.get_all_gnames():
|
|
|
1bdc94 |
+ self.screen.addstr(row, 2, '%8s %-60s' % (line[0], line[1]))
|
|
|
1bdc94 |
+ row += 1
|
|
|
1bdc94 |
+ if row >= self.screen.getmaxyx()[0]:
|
|
|
1bdc94 |
+ break
|
|
|
1bdc94 |
+ except Exception:
|
|
|
1bdc94 |
+ self.screen.addstr(row + 1, 2, 'Not available')
|
|
|
1bdc94 |
+
|
|
|
1bdc94 |
+ @staticmethod
|
|
|
1bdc94 |
+ def get_pid_from_gname(gname):
|
|
|
1bdc94 |
+ """Fuzzy function to convert guest name to QEMU process pid.
|
|
|
1bdc94 |
+
|
|
|
1bdc94 |
+ Returns a list of potential pids, can be empty if no match found.
|
|
|
1bdc94 |
+ Throws an exception on processing errors.
|
|
|
1bdc94 |
+
|
|
|
1bdc94 |
+ """
|
|
|
1bdc94 |
+ pids = []
|
|
|
1bdc94 |
+ for line in Tui.get_all_gnames():
|
|
|
1bdc94 |
+ if gname == line[1]:
|
|
|
1bdc94 |
+ pids.append(int(line[0]))
|
|
|
1bdc94 |
+
|
|
|
1bdc94 |
+ return pids
|
|
|
1bdc94 |
+
|
|
|
1bdc94 |
+ @staticmethod
|
|
|
1bdc94 |
+ def get_gname_from_pid(pid):
|
|
|
1bdc94 |
+ """Returns the guest name for a QEMU process pid.
|
|
|
1bdc94 |
+
|
|
|
1bdc94 |
+ Extracts the guest name from the QEMU comma line by processing the
|
|
|
1bdc94 |
+ '-name' option. Will also handle names specified out of sequence.
|
|
|
1bdc94 |
+
|
|
|
1bdc94 |
+ """
|
|
|
1bdc94 |
+ name = ''
|
|
|
1bdc94 |
+ try:
|
|
|
1bdc94 |
+ line = open('/proc/{}/cmdline'
|
|
|
1bdc94 |
+ .format(pid), 'rb').read().split('\0')
|
|
|
1bdc94 |
+ parms = line[line.index('-name') + 1].split(',')
|
|
|
1bdc94 |
+ while '' in parms:
|
|
|
1bdc94 |
+ # commas are escaped (i.e. ',,'), hence e.g. 'foo,bar' results
|
|
|
1bdc94 |
+ # in # ['foo', '', 'bar'], which we revert here
|
|
|
1bdc94 |
+ idx = parms.index('')
|
|
|
1bdc94 |
+ parms[idx - 1] += ',' + parms[idx + 1]
|
|
|
1bdc94 |
+ del parms[idx:idx+2]
|
|
|
1bdc94 |
+ # the '-name' switch allows for two ways to specify the guest name,
|
|
|
1bdc94 |
+ # where the plain name overrides the name specified via 'guest='
|
|
|
1bdc94 |
+ for arg in parms:
|
|
|
1bdc94 |
+ if '=' not in arg:
|
|
|
1bdc94 |
+ name = arg
|
|
|
1bdc94 |
+ break
|
|
|
1bdc94 |
+ if arg[:6] == 'guest=':
|
|
|
1bdc94 |
+ name = arg[6:]
|
|
|
1bdc94 |
+ except (ValueError, IOError, IndexError):
|
|
|
1bdc94 |
+ pass
|
|
|
1bdc94 |
+
|
|
|
1bdc94 |
+ return name
|
|
|
1bdc94 |
+
|
|
|
4a2fec |
+ def update_drilldown(self):
|
|
|
4a2fec |
+ """Sets or removes a filter that only allows fields without braces."""
|
|
|
4a2fec |
+ if not self.stats.fields_filter:
|
|
|
1bdc94 |
+ self.stats.fields_filter = DEFAULT_REGEX
|
|
|
4a2fec |
+
|
|
|
1bdc94 |
+ elif self.stats.fields_filter == DEFAULT_REGEX:
|
|
|
4a2fec |
+ self.stats.fields_filter = None
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ def update_pid(self, pid):
|
|
|
4a2fec |
+ """Propagates pid selection to stats object."""
|
|
|
4a2fec |
+ self.stats.pid_filter = pid
|
|
|
4a2fec |
+
|
|
|
1bdc94 |
+ def refresh_header(self, pid=None):
|
|
|
1bdc94 |
+ """Refreshes the header."""
|
|
|
1bdc94 |
+ if pid is None:
|
|
|
1bdc94 |
+ pid = self.stats.pid_filter
|
|
|
4a2fec |
+ self.screen.erase()
|
|
|
1bdc94 |
+ gname = self.get_gname_from_pid(pid)
|
|
|
1bdc94 |
+ if gname:
|
|
|
1bdc94 |
+ gname = ('({})'.format(gname[:MAX_GUEST_NAME_LEN] + '...'
|
|
|
1bdc94 |
+ if len(gname) > MAX_GUEST_NAME_LEN
|
|
|
1bdc94 |
+ else gname))
|
|
|
1bdc94 |
+ if pid > 0:
|
|
|
1bdc94 |
+ self.screen.addstr(0, 0, 'kvm statistics - pid {0} {1}'
|
|
|
1bdc94 |
+ .format(pid, gname), curses.A_BOLD)
|
|
|
4a2fec |
+ else:
|
|
|
4a2fec |
+ self.screen.addstr(0, 0, 'kvm statistics - summary', curses.A_BOLD)
|
|
|
1bdc94 |
+ if self.stats.fields_filter and self.stats.fields_filter \
|
|
|
1bdc94 |
+ != DEFAULT_REGEX:
|
|
|
1bdc94 |
+ regex = self.stats.fields_filter
|
|
|
1bdc94 |
+ if len(regex) > MAX_REGEX_LEN:
|
|
|
1bdc94 |
+ regex = regex[:MAX_REGEX_LEN] + '...'
|
|
|
1bdc94 |
+ self.screen.addstr(1, 17, 'regex filter: {0}'.format(regex))
|
|
|
1bdc94 |
+ if self._display_guests:
|
|
|
1bdc94 |
+ col_name = 'Guest Name'
|
|
|
1bdc94 |
+ else:
|
|
|
1bdc94 |
+ col_name = 'Event'
|
|
|
1bdc94 |
+ self.screen.addstr(2, 1, '%-40s %10s%7s %8s' %
|
|
|
1bdc94 |
+ (col_name, 'Total', '%Total', 'CurAvg/s'),
|
|
|
1bdc94 |
+ curses.A_STANDOUT)
|
|
|
1bdc94 |
+ self.screen.addstr(4, 1, 'Collecting data...')
|
|
|
1bdc94 |
+ self.screen.refresh()
|
|
|
1bdc94 |
+
|
|
|
1bdc94 |
+ def refresh_body(self, sleeptime):
|
|
|
4a2fec |
+ row = 3
|
|
|
1bdc94 |
+ self.screen.move(row, 0)
|
|
|
1bdc94 |
+ self.screen.clrtobot()
|
|
|
1bdc94 |
+ stats = self.stats.get(self._display_guests)
|
|
|
1bdc94 |
+
|
|
|
1bdc94 |
+ def sortCurAvg(x):
|
|
|
1bdc94 |
+ # sort by current events if available
|
|
|
4a2fec |
+ if stats[x][1]:
|
|
|
4a2fec |
+ return (-stats[x][1], -stats[x][0])
|
|
|
4a2fec |
+ else:
|
|
|
4a2fec |
+ return (0, -stats[x][0])
|
|
|
1bdc94 |
+
|
|
|
1bdc94 |
+ def sortTotal(x):
|
|
|
1bdc94 |
+ # sort by totals
|
|
|
1bdc94 |
+ return (0, -stats[x][0])
|
|
|
1bdc94 |
+ total = 0.
|
|
|
1bdc94 |
+ for val in stats.values():
|
|
|
1bdc94 |
+ total += val[0]
|
|
|
1bdc94 |
+ if self._sorting == SORT_DEFAULT:
|
|
|
1bdc94 |
+ sortkey = sortCurAvg
|
|
|
1bdc94 |
+ else:
|
|
|
1bdc94 |
+ sortkey = sortTotal
|
|
|
4a2fec |
+ for key in sorted(stats.keys(), key=sortkey):
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ if row >= self.screen.getmaxyx()[0]:
|
|
|
4a2fec |
+ break
|
|
|
4a2fec |
+ values = stats[key]
|
|
|
4a2fec |
+ if not values[0] and not values[1]:
|
|
|
4a2fec |
+ break
|
|
|
1bdc94 |
+ if values[0] is not None:
|
|
|
1bdc94 |
+ cur = int(round(values[1] / sleeptime)) if values[1] else ''
|
|
|
1bdc94 |
+ if self._display_guests:
|
|
|
1bdc94 |
+ key = self.get_gname_from_pid(key)
|
|
|
1bdc94 |
+ self.screen.addstr(row, 1, '%-40s %10d%7.1f %8s' %
|
|
|
1bdc94 |
+ (key, values[0], values[0] * 100 / total,
|
|
|
1bdc94 |
+ cur))
|
|
|
4a2fec |
+ row += 1
|
|
|
1bdc94 |
+ if row == 3:
|
|
|
1bdc94 |
+ self.screen.addstr(4, 1, 'No matching events reported yet')
|
|
|
4a2fec |
+ self.screen.refresh()
|
|
|
4a2fec |
+
|
|
|
1bdc94 |
+ def show_msg(self, text):
|
|
|
1bdc94 |
+ """Display message centered text and exit on key press"""
|
|
|
1bdc94 |
+ hint = 'Press any key to continue'
|
|
|
1bdc94 |
+ curses.cbreak()
|
|
|
1bdc94 |
+ self.screen.erase()
|
|
|
1bdc94 |
+ (x, term_width) = self.screen.getmaxyx()
|
|
|
1bdc94 |
+ row = 2
|
|
|
1bdc94 |
+ for line in text:
|
|
|
1bdc94 |
+ start = (term_width - len(line)) / 2
|
|
|
1bdc94 |
+ self.screen.addstr(row, start, line)
|
|
|
1bdc94 |
+ row += 1
|
|
|
1bdc94 |
+ self.screen.addstr(row + 1, (term_width - len(hint)) / 2, hint,
|
|
|
1bdc94 |
+ curses.A_STANDOUT)
|
|
|
1bdc94 |
+ self.screen.getkey()
|
|
|
1bdc94 |
+
|
|
|
1bdc94 |
+ def show_help_interactive(self):
|
|
|
1bdc94 |
+ """Display help with list of interactive commands"""
|
|
|
1bdc94 |
+ msg = (' b toggle events by guests (debugfs only, honors'
|
|
|
1bdc94 |
+ ' filters)',
|
|
|
1bdc94 |
+ ' c clear filter',
|
|
|
1bdc94 |
+ ' f filter by regular expression',
|
|
|
1bdc94 |
+ ' g filter by guest name',
|
|
|
1bdc94 |
+ ' h display interactive commands reference',
|
|
|
1bdc94 |
+ ' o toggle sorting order (Total vs CurAvg/s)',
|
|
|
1bdc94 |
+ ' p filter by PID',
|
|
|
1bdc94 |
+ ' q quit',
|
|
|
1bdc94 |
+ ' r reset stats',
|
|
|
1bdc94 |
+ ' s set update interval',
|
|
|
1bdc94 |
+ ' x toggle reporting of stats for individual child trace'
|
|
|
1bdc94 |
+ ' events',
|
|
|
1bdc94 |
+ 'Any other key refreshes statistics immediately')
|
|
|
1bdc94 |
+ curses.cbreak()
|
|
|
1bdc94 |
+ self.screen.erase()
|
|
|
1bdc94 |
+ self.screen.addstr(0, 0, "Interactive commands reference",
|
|
|
1bdc94 |
+ curses.A_BOLD)
|
|
|
1bdc94 |
+ self.screen.addstr(2, 0, "Press any key to exit", curses.A_STANDOUT)
|
|
|
1bdc94 |
+ row = 4
|
|
|
1bdc94 |
+ for line in msg:
|
|
|
1bdc94 |
+ self.screen.addstr(row, 0, line)
|
|
|
1bdc94 |
+ row += 1
|
|
|
1bdc94 |
+ self.screen.getkey()
|
|
|
1bdc94 |
+ self.refresh_header()
|
|
|
1bdc94 |
+
|
|
|
4a2fec |
+ def show_filter_selection(self):
|
|
|
4a2fec |
+ """Draws filter selection mask.
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ Asks for a valid regex and sets the fields filter accordingly.
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ """
|
|
|
4a2fec |
+ while True:
|
|
|
4a2fec |
+ self.screen.erase()
|
|
|
4a2fec |
+ self.screen.addstr(0, 0,
|
|
|
4a2fec |
+ "Show statistics for events matching a regex.",
|
|
|
4a2fec |
+ curses.A_BOLD)
|
|
|
4a2fec |
+ self.screen.addstr(2, 0,
|
|
|
4a2fec |
+ "Current regex: {0}"
|
|
|
4a2fec |
+ .format(self.stats.fields_filter))
|
|
|
4a2fec |
+ self.screen.addstr(3, 0, "New regex: ")
|
|
|
4a2fec |
+ curses.echo()
|
|
|
4a2fec |
+ regex = self.screen.getstr()
|
|
|
4a2fec |
+ curses.noecho()
|
|
|
4a2fec |
+ if len(regex) == 0:
|
|
|
1bdc94 |
+ self.stats.fields_filter = DEFAULT_REGEX
|
|
|
1bdc94 |
+ self.refresh_header()
|
|
|
4a2fec |
+ return
|
|
|
4a2fec |
+ try:
|
|
|
4a2fec |
+ re.compile(regex)
|
|
|
4a2fec |
+ self.stats.fields_filter = regex
|
|
|
1bdc94 |
+ self.refresh_header()
|
|
|
4a2fec |
+ return
|
|
|
4a2fec |
+ except re.error:
|
|
|
4a2fec |
+ continue
|
|
|
4a2fec |
+
|
|
|
1bdc94 |
+ def show_vm_selection_by_pid(self):
|
|
|
4a2fec |
+ """Draws PID selection mask.
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ Asks for a pid until a valid pid or 0 has been entered.
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ """
|
|
|
1bdc94 |
+ msg = ''
|
|
|
4a2fec |
+ while True:
|
|
|
4a2fec |
+ self.screen.erase()
|
|
|
4a2fec |
+ self.screen.addstr(0, 0,
|
|
|
4a2fec |
+ 'Show statistics for specific pid.',
|
|
|
4a2fec |
+ curses.A_BOLD)
|
|
|
4a2fec |
+ self.screen.addstr(1, 0,
|
|
|
4a2fec |
+ 'This might limit the shown data to the trace '
|
|
|
4a2fec |
+ 'statistics.')
|
|
|
1bdc94 |
+ self.screen.addstr(5, 0, msg)
|
|
|
1bdc94 |
+ self.print_all_gnames(7)
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ curses.echo()
|
|
|
4a2fec |
+ self.screen.addstr(3, 0, "Pid [0 or pid]: ")
|
|
|
4a2fec |
+ pid = self.screen.getstr()
|
|
|
4a2fec |
+ curses.noecho()
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ try:
|
|
|
1bdc94 |
+ if len(pid) > 0:
|
|
|
1bdc94 |
+ pid = int(pid)
|
|
|
1bdc94 |
+ if pid != 0 and not os.path.isdir(os.path.join('/proc/',
|
|
|
1bdc94 |
+ str(pid))):
|
|
|
1bdc94 |
+ msg = '"' + str(pid) + '": Not a running process'
|
|
|
1bdc94 |
+ continue
|
|
|
4a2fec |
+ else:
|
|
|
1bdc94 |
+ pid = 0
|
|
|
1bdc94 |
+ self.refresh_header(pid)
|
|
|
1bdc94 |
+ self.update_pid(pid)
|
|
|
1bdc94 |
+ break
|
|
|
1bdc94 |
+ except ValueError:
|
|
|
1bdc94 |
+ msg = '"' + str(pid) + '": Not a valid pid'
|
|
|
1bdc94 |
+
|
|
|
1bdc94 |
+ def show_set_update_interval(self):
|
|
|
1bdc94 |
+ """Draws update interval selection mask."""
|
|
|
1bdc94 |
+ msg = ''
|
|
|
1bdc94 |
+ while True:
|
|
|
1bdc94 |
+ self.screen.erase()
|
|
|
1bdc94 |
+ self.screen.addstr(0, 0, 'Set update interval (defaults to %fs).' %
|
|
|
1bdc94 |
+ DELAY_DEFAULT, curses.A_BOLD)
|
|
|
1bdc94 |
+ self.screen.addstr(4, 0, msg)
|
|
|
1bdc94 |
+ self.screen.addstr(2, 0, 'Change delay from %.1fs to ' %
|
|
|
1bdc94 |
+ self._delay_regular)
|
|
|
1bdc94 |
+ curses.echo()
|
|
|
1bdc94 |
+ val = self.screen.getstr()
|
|
|
1bdc94 |
+ curses.noecho()
|
|
|
1bdc94 |
+
|
|
|
1bdc94 |
+ try:
|
|
|
1bdc94 |
+ if len(val) > 0:
|
|
|
1bdc94 |
+ delay = float(val)
|
|
|
1bdc94 |
+ if delay < 0.1:
|
|
|
1bdc94 |
+ msg = '"' + str(val) + '": Value must be >=0.1'
|
|
|
4a2fec |
+ continue
|
|
|
1bdc94 |
+ if delay > 25.5:
|
|
|
1bdc94 |
+ msg = '"' + str(val) + '": Value must be <=25.5'
|
|
|
1bdc94 |
+ continue
|
|
|
1bdc94 |
+ else:
|
|
|
1bdc94 |
+ delay = DELAY_DEFAULT
|
|
|
1bdc94 |
+ self._delay_regular = delay
|
|
|
1bdc94 |
+ break
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ except ValueError:
|
|
|
1bdc94 |
+ msg = '"' + str(val) + '": Invalid value'
|
|
|
1bdc94 |
+ self.refresh_header()
|
|
|
1bdc94 |
+
|
|
|
1bdc94 |
+ def show_vm_selection_by_guest_name(self):
|
|
|
1bdc94 |
+ """Draws guest selection mask.
|
|
|
1bdc94 |
+
|
|
|
1bdc94 |
+ Asks for a guest name until a valid guest name or '' is entered.
|
|
|
1bdc94 |
+
|
|
|
1bdc94 |
+ """
|
|
|
1bdc94 |
+ msg = ''
|
|
|
1bdc94 |
+ while True:
|
|
|
1bdc94 |
+ self.screen.erase()
|
|
|
1bdc94 |
+ self.screen.addstr(0, 0,
|
|
|
1bdc94 |
+ 'Show statistics for specific guest.',
|
|
|
1bdc94 |
+ curses.A_BOLD)
|
|
|
1bdc94 |
+ self.screen.addstr(1, 0,
|
|
|
1bdc94 |
+ 'This might limit the shown data to the trace '
|
|
|
1bdc94 |
+ 'statistics.')
|
|
|
1bdc94 |
+ self.screen.addstr(5, 0, msg)
|
|
|
1bdc94 |
+ self.print_all_gnames(7)
|
|
|
1bdc94 |
+ curses.echo()
|
|
|
1bdc94 |
+ self.screen.addstr(3, 0, "Guest [ENTER or guest]: ")
|
|
|
1bdc94 |
+ gname = self.screen.getstr()
|
|
|
1bdc94 |
+ curses.noecho()
|
|
|
1bdc94 |
+
|
|
|
1bdc94 |
+ if not gname:
|
|
|
1bdc94 |
+ self.refresh_header(0)
|
|
|
1bdc94 |
+ self.update_pid(0)
|
|
|
1bdc94 |
+ break
|
|
|
1bdc94 |
+ else:
|
|
|
1bdc94 |
+ pids = []
|
|
|
1bdc94 |
+ try:
|
|
|
1bdc94 |
+ pids = self.get_pid_from_gname(gname)
|
|
|
1bdc94 |
+ except:
|
|
|
1bdc94 |
+ msg = '"' + gname + '": Internal error while searching, ' \
|
|
|
1bdc94 |
+ 'use pid filter instead'
|
|
|
1bdc94 |
+ continue
|
|
|
1bdc94 |
+ if len(pids) == 0:
|
|
|
1bdc94 |
+ msg = '"' + gname + '": Not an active guest'
|
|
|
1bdc94 |
+ continue
|
|
|
1bdc94 |
+ if len(pids) > 1:
|
|
|
1bdc94 |
+ msg = '"' + gname + '": Multiple matches found, use pid ' \
|
|
|
1bdc94 |
+ 'filter instead'
|
|
|
1bdc94 |
+ continue
|
|
|
1bdc94 |
+ self.refresh_header(pids[0])
|
|
|
1bdc94 |
+ self.update_pid(pids[0])
|
|
|
1bdc94 |
+ break
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ def show_stats(self):
|
|
|
4a2fec |
+ """Refreshes the screen and processes user input."""
|
|
|
1bdc94 |
+ sleeptime = self._delay_initial
|
|
|
1bdc94 |
+ self.refresh_header()
|
|
|
1bdc94 |
+ start = 0.0 # result based on init value never appears on screen
|
|
|
4a2fec |
+ while True:
|
|
|
1bdc94 |
+ self.refresh_body(time.time() - start)
|
|
|
4a2fec |
+ curses.halfdelay(int(sleeptime * 10))
|
|
|
1bdc94 |
+ start = time.time()
|
|
|
1bdc94 |
+ sleeptime = self._delay_regular
|
|
|
4a2fec |
+ try:
|
|
|
4a2fec |
+ char = self.screen.getkey()
|
|
|
1bdc94 |
+ if char == 'b':
|
|
|
1bdc94 |
+ self._display_guests = not self._display_guests
|
|
|
1bdc94 |
+ if self.stats.toggle_display_guests(self._display_guests):
|
|
|
1bdc94 |
+ self.show_msg(['Command not available with tracepoints'
|
|
|
1bdc94 |
+ ' enabled', 'Restart with debugfs only '
|
|
|
1bdc94 |
+ '(see option \'-d\') and try again!'])
|
|
|
1bdc94 |
+ self._display_guests = not self._display_guests
|
|
|
1bdc94 |
+ self.refresh_header()
|
|
|
1bdc94 |
+ if char == 'c':
|
|
|
1bdc94 |
+ self.stats.fields_filter = DEFAULT_REGEX
|
|
|
1bdc94 |
+ self.refresh_header(0)
|
|
|
1bdc94 |
+ self.update_pid(0)
|
|
|
4a2fec |
+ if char == 'f':
|
|
|
1bdc94 |
+ curses.curs_set(1)
|
|
|
4a2fec |
+ self.show_filter_selection()
|
|
|
1bdc94 |
+ curses.curs_set(0)
|
|
|
1bdc94 |
+ sleeptime = self._delay_initial
|
|
|
1bdc94 |
+ if char == 'g':
|
|
|
1bdc94 |
+ curses.curs_set(1)
|
|
|
1bdc94 |
+ self.show_vm_selection_by_guest_name()
|
|
|
1bdc94 |
+ curses.curs_set(0)
|
|
|
1bdc94 |
+ sleeptime = self._delay_initial
|
|
|
1bdc94 |
+ if char == 'h':
|
|
|
1bdc94 |
+ self.show_help_interactive()
|
|
|
1bdc94 |
+ if char == 'o':
|
|
|
1bdc94 |
+ self._sorting = not self._sorting
|
|
|
4a2fec |
+ if char == 'p':
|
|
|
1bdc94 |
+ curses.curs_set(1)
|
|
|
1bdc94 |
+ self.show_vm_selection_by_pid()
|
|
|
1bdc94 |
+ curses.curs_set(0)
|
|
|
1bdc94 |
+ sleeptime = self._delay_initial
|
|
|
1bdc94 |
+ if char == 'q':
|
|
|
1bdc94 |
+ break
|
|
|
1bdc94 |
+ if char == 'r':
|
|
|
1bdc94 |
+ self.stats.reset()
|
|
|
1bdc94 |
+ if char == 's':
|
|
|
1bdc94 |
+ curses.curs_set(1)
|
|
|
1bdc94 |
+ self.show_set_update_interval()
|
|
|
1bdc94 |
+ curses.curs_set(0)
|
|
|
1bdc94 |
+ sleeptime = self._delay_initial
|
|
|
1bdc94 |
+ if char == 'x':
|
|
|
1bdc94 |
+ self.update_drilldown()
|
|
|
1bdc94 |
+ # prevents display of current values on next refresh
|
|
|
1bdc94 |
+ self.stats.get()
|
|
|
4a2fec |
+ except KeyboardInterrupt:
|
|
|
4a2fec |
+ break
|
|
|
4a2fec |
+ except curses.error:
|
|
|
4a2fec |
+ continue
|
|
|
4a2fec |
+
|
|
|
1bdc94 |
+
|
|
|
4a2fec |
+def batch(stats):
|
|
|
4a2fec |
+ """Prints statistics in a key, value format."""
|
|
|
1bdc94 |
+ try:
|
|
|
1bdc94 |
+ s = stats.get()
|
|
|
1bdc94 |
+ time.sleep(1)
|
|
|
1bdc94 |
+ s = stats.get()
|
|
|
1bdc94 |
+ for key in sorted(s.keys()):
|
|
|
1bdc94 |
+ values = s[key]
|
|
|
1bdc94 |
+ print '%-42s%10d%10d' % (key, values[0], values[1])
|
|
|
1bdc94 |
+ except KeyboardInterrupt:
|
|
|
1bdc94 |
+ pass
|
|
|
1bdc94 |
+
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+def log(stats):
|
|
|
4a2fec |
+ """Prints statistics as reiterating key block, multiple value blocks."""
|
|
|
4a2fec |
+ keys = sorted(stats.get().iterkeys())
|
|
|
1bdc94 |
+
|
|
|
4a2fec |
+ def banner():
|
|
|
4a2fec |
+ for k in keys:
|
|
|
4a2fec |
+ print '%s' % k,
|
|
|
4a2fec |
+ print
|
|
|
1bdc94 |
+
|
|
|
4a2fec |
+ def statline():
|
|
|
4a2fec |
+ s = stats.get()
|
|
|
4a2fec |
+ for k in keys:
|
|
|
4a2fec |
+ print ' %9d' % s[k][1],
|
|
|
4a2fec |
+ print
|
|
|
4a2fec |
+ line = 0
|
|
|
4a2fec |
+ banner_repeat = 20
|
|
|
4a2fec |
+ while True:
|
|
|
1bdc94 |
+ try:
|
|
|
1bdc94 |
+ time.sleep(1)
|
|
|
1bdc94 |
+ if line % banner_repeat == 0:
|
|
|
1bdc94 |
+ banner()
|
|
|
1bdc94 |
+ statline()
|
|
|
1bdc94 |
+ line += 1
|
|
|
1bdc94 |
+ except KeyboardInterrupt:
|
|
|
1bdc94 |
+ break
|
|
|
1bdc94 |
+
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+def get_options():
|
|
|
4a2fec |
+ """Returns processed program arguments."""
|
|
|
4a2fec |
+ description_text = """
|
|
|
4a2fec |
+This script displays various statistics about VMs running under KVM.
|
|
|
4a2fec |
+The statistics are gathered from the KVM debugfs entries and / or the
|
|
|
4a2fec |
+currently available perf traces.
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+The monitoring takes additional cpu cycles and might affect the VM's
|
|
|
4a2fec |
+performance.
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+Requirements:
|
|
|
4a2fec |
+- Access to:
|
|
|
1bdc94 |
+ %s
|
|
|
1bdc94 |
+ %s/events/*
|
|
|
4a2fec |
+ /proc/pid/task
|
|
|
4a2fec |
+- /proc/sys/kernel/perf_event_paranoid < 1 if user has no
|
|
|
4a2fec |
+ CAP_SYS_ADMIN and perf events are used.
|
|
|
4a2fec |
+- CAP_SYS_RESOURCE if the hard limit is not high enough to allow
|
|
|
4a2fec |
+ the large number of files that are possibly opened.
|
|
|
1bdc94 |
+
|
|
|
1bdc94 |
+Interactive Commands:
|
|
|
1bdc94 |
+ b toggle events by guests (debugfs only, honors filters)
|
|
|
1bdc94 |
+ c clear filter
|
|
|
1bdc94 |
+ f filter by regular expression
|
|
|
1bdc94 |
+ g filter by guest name
|
|
|
1bdc94 |
+ h display interactive commands reference
|
|
|
1bdc94 |
+ o toggle sorting order (Total vs CurAvg/s)
|
|
|
1bdc94 |
+ p filter by PID
|
|
|
1bdc94 |
+ q quit
|
|
|
1bdc94 |
+ r reset stats
|
|
|
1bdc94 |
+ s set update interval
|
|
|
1bdc94 |
+ x toggle reporting of stats for individual child trace events
|
|
|
1bdc94 |
+Press any other key to refresh statistics immediately.
|
|
|
1bdc94 |
+""" % (PATH_DEBUGFS_KVM, PATH_DEBUGFS_TRACING)
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ class PlainHelpFormatter(optparse.IndentedHelpFormatter):
|
|
|
4a2fec |
+ def format_description(self, description):
|
|
|
4a2fec |
+ if description:
|
|
|
4a2fec |
+ return description + "\n"
|
|
|
4a2fec |
+ else:
|
|
|
4a2fec |
+ return ""
|
|
|
4a2fec |
+
|
|
|
1bdc94 |
+ def cb_guest_to_pid(option, opt, val, parser):
|
|
|
1bdc94 |
+ try:
|
|
|
1bdc94 |
+ pids = Tui.get_pid_from_gname(val)
|
|
|
1bdc94 |
+ except:
|
|
|
1bdc94 |
+ raise optparse.OptionValueError('Error while searching for guest '
|
|
|
1bdc94 |
+ '"{}", use "-p" to specify a pid '
|
|
|
1bdc94 |
+ 'instead'.format(val))
|
|
|
1bdc94 |
+ if len(pids) == 0:
|
|
|
1bdc94 |
+ raise optparse.OptionValueError('No guest by the name "{}" '
|
|
|
1bdc94 |
+ 'found'.format(val))
|
|
|
1bdc94 |
+ if len(pids) > 1:
|
|
|
1bdc94 |
+ raise optparse.OptionValueError('Multiple processes found (pids: '
|
|
|
1bdc94 |
+ '{}) - use "-p" to specify a pid '
|
|
|
1bdc94 |
+ 'instead'.format(" ".join(pids)))
|
|
|
1bdc94 |
+ parser.values.pid = pids[0]
|
|
|
1bdc94 |
+
|
|
|
4a2fec |
+ optparser = optparse.OptionParser(description=description_text,
|
|
|
4a2fec |
+ formatter=PlainHelpFormatter())
|
|
|
4a2fec |
+ optparser.add_option('-1', '--once', '--batch',
|
|
|
4a2fec |
+ action='store_true',
|
|
|
4a2fec |
+ default=False,
|
|
|
4a2fec |
+ dest='once',
|
|
|
4a2fec |
+ help='run in batch mode for one second',
|
|
|
4a2fec |
+ )
|
|
|
1bdc94 |
+ optparser.add_option('-i', '--debugfs-include-past',
|
|
|
1bdc94 |
+ action='store_true',
|
|
|
1bdc94 |
+ default=False,
|
|
|
1bdc94 |
+ dest='dbgfs_include_past',
|
|
|
1bdc94 |
+ help='include all available data on past events for '
|
|
|
1bdc94 |
+ 'debugfs',
|
|
|
1bdc94 |
+ )
|
|
|
4a2fec |
+ optparser.add_option('-l', '--log',
|
|
|
4a2fec |
+ action='store_true',
|
|
|
4a2fec |
+ default=False,
|
|
|
4a2fec |
+ dest='log',
|
|
|
4a2fec |
+ help='run in logging mode (like vmstat)',
|
|
|
4a2fec |
+ )
|
|
|
4a2fec |
+ optparser.add_option('-t', '--tracepoints',
|
|
|
4a2fec |
+ action='store_true',
|
|
|
4a2fec |
+ default=False,
|
|
|
4a2fec |
+ dest='tracepoints',
|
|
|
4a2fec |
+ help='retrieve statistics from tracepoints',
|
|
|
4a2fec |
+ )
|
|
|
4a2fec |
+ optparser.add_option('-d', '--debugfs',
|
|
|
4a2fec |
+ action='store_true',
|
|
|
4a2fec |
+ default=False,
|
|
|
4a2fec |
+ dest='debugfs',
|
|
|
4a2fec |
+ help='retrieve statistics from debugfs',
|
|
|
4a2fec |
+ )
|
|
|
4a2fec |
+ optparser.add_option('-f', '--fields',
|
|
|
4a2fec |
+ action='store',
|
|
|
1bdc94 |
+ default=DEFAULT_REGEX,
|
|
|
4a2fec |
+ dest='fields',
|
|
|
1bdc94 |
+ help='''fields to display (regex)
|
|
|
1bdc94 |
+ "-f help" for a list of available events''',
|
|
|
4a2fec |
+ )
|
|
|
4a2fec |
+ optparser.add_option('-p', '--pid',
|
|
|
1bdc94 |
+ action='store',
|
|
|
1bdc94 |
+ default=0,
|
|
|
1bdc94 |
+ type='int',
|
|
|
1bdc94 |
+ dest='pid',
|
|
|
1bdc94 |
+ help='restrict statistics to pid',
|
|
|
1bdc94 |
+ )
|
|
|
1bdc94 |
+ optparser.add_option('-g', '--guest',
|
|
|
1bdc94 |
+ action='callback',
|
|
|
1bdc94 |
+ type='string',
|
|
|
1bdc94 |
+ dest='pid',
|
|
|
1bdc94 |
+ metavar='GUEST',
|
|
|
1bdc94 |
+ help='restrict statistics to guest by name',
|
|
|
1bdc94 |
+ callback=cb_guest_to_pid,
|
|
|
1bdc94 |
+ )
|
|
|
4a2fec |
+ (options, _) = optparser.parse_args(sys.argv)
|
|
|
4a2fec |
+ return options
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+def check_access(options):
|
|
|
4a2fec |
+ """Exits if the current user can't access all needed directories."""
|
|
|
4a2fec |
+ if not os.path.exists('/sys/kernel/debug'):
|
|
|
4a2fec |
+ sys.stderr.write('Please enable CONFIG_DEBUG_FS in your kernel.')
|
|
|
4a2fec |
+ sys.exit(1)
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ if not os.path.exists(PATH_DEBUGFS_KVM):
|
|
|
4a2fec |
+ sys.stderr.write("Please make sure, that debugfs is mounted and "
|
|
|
4a2fec |
+ "readable by the current user:\n"
|
|
|
4a2fec |
+ "('mount -t debugfs debugfs /sys/kernel/debug')\n"
|
|
|
4a2fec |
+ "Also ensure, that the kvm modules are loaded.\n")
|
|
|
4a2fec |
+ sys.exit(1)
|
|
|
4a2fec |
+
|
|
|
1bdc94 |
+ if not os.path.exists(PATH_DEBUGFS_TRACING) and (options.tracepoints or
|
|
|
1bdc94 |
+ not options.debugfs):
|
|
|
4a2fec |
+ sys.stderr.write("Please enable CONFIG_TRACING in your kernel "
|
|
|
4a2fec |
+ "when using the option -t (default).\n"
|
|
|
4a2fec |
+ "If it is enabled, make {0} readable by the "
|
|
|
4a2fec |
+ "current user.\n"
|
|
|
4a2fec |
+ .format(PATH_DEBUGFS_TRACING))
|
|
|
4a2fec |
+ if options.tracepoints:
|
|
|
4a2fec |
+ sys.exit(1)
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ sys.stderr.write("Falling back to debugfs statistics!\n")
|
|
|
4a2fec |
+ options.debugfs = True
|
|
|
1bdc94 |
+ time.sleep(5)
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ return options
|
|
|
4a2fec |
+
|
|
|
1bdc94 |
+
|
|
|
4a2fec |
+def main():
|
|
|
4a2fec |
+ options = get_options()
|
|
|
4a2fec |
+ options = check_access(options)
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ if (options.pid > 0 and
|
|
|
4a2fec |
+ not os.path.isdir(os.path.join('/proc/',
|
|
|
4a2fec |
+ str(options.pid)))):
|
|
|
4a2fec |
+ sys.stderr.write('Did you use a (unsupported) tid instead of a pid?\n')
|
|
|
4a2fec |
+ sys.exit('Specified pid does not exist.')
|
|
|
4a2fec |
+
|
|
|
1bdc94 |
+ stats = Stats(options)
|
|
|
1bdc94 |
+
|
|
|
1bdc94 |
+ if options.fields == "help":
|
|
|
1bdc94 |
+ event_list = "\n"
|
|
|
1bdc94 |
+ s = stats.get()
|
|
|
1bdc94 |
+ for key in s.keys():
|
|
|
1bdc94 |
+ if key.find('(') != -1:
|
|
|
1bdc94 |
+ key = key[0:key.find('(')]
|
|
|
1bdc94 |
+ if event_list.find('\n' + key + '\n') == -1:
|
|
|
1bdc94 |
+ event_list += key + '\n'
|
|
|
1bdc94 |
+ sys.stdout.write(event_list)
|
|
|
1bdc94 |
+ return ""
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+ if options.log:
|
|
|
4a2fec |
+ log(stats)
|
|
|
4a2fec |
+ elif not options.once:
|
|
|
4a2fec |
+ with Tui(stats) as tui:
|
|
|
4a2fec |
+ tui.show_stats()
|
|
|
4a2fec |
+ else:
|
|
|
4a2fec |
+ batch(stats)
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+if __name__ == "__main__":
|
|
|
4a2fec |
+ main()
|
|
|
4a2fec |
diff --git a/scripts/kvm/kvm_stat.texi b/scripts/kvm/kvm_stat.texi
|
|
|
4a2fec |
new file mode 100644
|
|
|
1bdc94 |
index 0000000..5d964f6
|
|
|
4a2fec |
--- /dev/null
|
|
|
4a2fec |
+++ b/scripts/kvm/kvm_stat.texi
|
|
|
1bdc94 |
@@ -0,0 +1,104 @@
|
|
|
4a2fec |
+@example
|
|
|
4a2fec |
+@c man begin SYNOPSIS
|
|
|
4a2fec |
+usage: kvm_stat [OPTION]...
|
|
|
4a2fec |
+@c man end
|
|
|
4a2fec |
+@end example
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+@c man begin DESCRIPTION
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+kvm_stat prints counts of KVM kernel module trace events. These events signify
|
|
|
4a2fec |
+state transitions such as guest mode entry and exit.
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+This tool is useful for observing guest behavior from the host perspective.
|
|
|
4a2fec |
+Often conclusions about performance or buggy behavior can be drawn from the
|
|
|
4a2fec |
+output.
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+The set of KVM kernel module trace events may be specific to the kernel version
|
|
|
4a2fec |
+or architecture. It is best to check the KVM kernel module source code for the
|
|
|
4a2fec |
+meaning of events.
|
|
|
4a2fec |
+
|
|
|
1bdc94 |
+Use batch and logging modes for scripting purposes.
|
|
|
1bdc94 |
+
|
|
|
1bdc94 |
+@section Interactive Commands
|
|
|
1bdc94 |
+
|
|
|
1bdc94 |
+While running in regular (interactive) mode, use any of the following keys:
|
|
|
1bdc94 |
+
|
|
|
1bdc94 |
+@table @key
|
|
|
1bdc94 |
+@item b
|
|
|
1bdc94 |
+@kindex b
|
|
|
1bdc94 |
+toggle events by guests (debugfs only, honors filters)
|
|
|
1bdc94 |
+@item c
|
|
|
1bdc94 |
+@kindex c
|
|
|
1bdc94 |
+clear filter
|
|
|
1bdc94 |
+@item f
|
|
|
1bdc94 |
+@kindex f
|
|
|
1bdc94 |
+filter by regular expression
|
|
|
1bdc94 |
+@item g
|
|
|
1bdc94 |
+@kindex g
|
|
|
1bdc94 |
+filter by guest name
|
|
|
1bdc94 |
+@item h
|
|
|
1bdc94 |
+@kindex h
|
|
|
1bdc94 |
+display interactive commands reference
|
|
|
1bdc94 |
+@item o
|
|
|
1bdc94 |
+@kindex o
|
|
|
1bdc94 |
+toggle sorting order (Total vs CurAvg/s)
|
|
|
1bdc94 |
+@item p
|
|
|
1bdc94 |
+@kindex p
|
|
|
1bdc94 |
+filter by PID
|
|
|
1bdc94 |
+@item q
|
|
|
1bdc94 |
+@kindex q
|
|
|
1bdc94 |
+quit
|
|
|
1bdc94 |
+@item r
|
|
|
1bdc94 |
+@kindex r
|
|
|
1bdc94 |
+reset stats
|
|
|
1bdc94 |
+@item s
|
|
|
1bdc94 |
+@kindex s
|
|
|
1bdc94 |
+set update interval
|
|
|
1bdc94 |
+@item x
|
|
|
1bdc94 |
+@kindex x
|
|
|
1bdc94 |
+toggle reporting of stats for child trace events
|
|
|
1bdc94 |
+@end table
|
|
|
1bdc94 |
+
|
|
|
1bdc94 |
+Press any other key to refresh statistics immediately.
|
|
|
1bdc94 |
+
|
|
|
4a2fec |
+@c man end
|
|
|
4a2fec |
+
|
|
|
1bdc94 |
+
|
|
|
4a2fec |
+@c man begin OPTIONS
|
|
|
4a2fec |
+@table @option
|
|
|
4a2fec |
+@item -1, --once, --batch
|
|
|
4a2fec |
+ run in batch mode for one second
|
|
|
4a2fec |
+@item -l, --log
|
|
|
4a2fec |
+ run in logging mode (like vmstat)
|
|
|
4a2fec |
+@item -t, --tracepoints
|
|
|
4a2fec |
+ retrieve statistics from tracepoints
|
|
|
4a2fec |
+@item -d, --debugfs
|
|
|
4a2fec |
+ retrieve statistics from debugfs
|
|
|
4a2fec |
+@item -p, --pid=@var{pid}
|
|
|
4a2fec |
+ limit statistics to one virtual machine (pid)
|
|
|
1bdc94 |
+@item -i, --debugfs-include-past
|
|
|
1bdc94 |
+ include all available data on past events for debugfs
|
|
|
1bdc94 |
+@item -g, --guest=@var{guest_name}
|
|
|
1bdc94 |
+ limit statistics to one virtual machine (guest name)
|
|
|
4a2fec |
+@item -f, --fields=@var{fields}
|
|
|
4a2fec |
+ fields to display (regex)
|
|
|
4a2fec |
+@item -h, --help
|
|
|
4a2fec |
+ show help message
|
|
|
4a2fec |
+@end table
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+@c man end
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+@ignore
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+@setfilename kvm_stat
|
|
|
4a2fec |
+@settitle Report KVM kernel module event counters.
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+@c man begin AUTHOR
|
|
|
4a2fec |
+Stefan Hajnoczi <stefanha@redhat.com>
|
|
|
4a2fec |
+@c man end
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+@c man begin SEEALSO
|
|
|
4a2fec |
+perf(1), trace-cmd(1)
|
|
|
4a2fec |
+@c man end
|
|
|
4a2fec |
+
|
|
|
4a2fec |
+@end ignore
|
|
|
4a2fec |
--
|
|
|
4a2fec |
1.8.3.1
|
|
|
4a2fec |
|