summaryrefslogtreecommitdiffstats
path: root/qemu/scripts/kvm
diff options
context:
space:
mode:
Diffstat (limited to 'qemu/scripts/kvm')
-rwxr-xr-xqemu/scripts/kvm/kvm_flightrecorder126
-rwxr-xr-xqemu/scripts/kvm/kvm_stat825
-rw-r--r--qemu/scripts/kvm/kvm_stat.texi55
-rwxr-xr-xqemu/scripts/kvm/vmxcap260
4 files changed, 0 insertions, 1266 deletions
diff --git a/qemu/scripts/kvm/kvm_flightrecorder b/qemu/scripts/kvm/kvm_flightrecorder
deleted file mode 100755
index 7fb1c2d1a..000000000
--- a/qemu/scripts/kvm/kvm_flightrecorder
+++ /dev/null
@@ -1,126 +0,0 @@
-#!/usr/bin/env python
-#
-# KVM Flight Recorder - ring buffer tracing script
-#
-# Copyright (C) 2012 IBM Corp
-#
-# Author: Stefan Hajnoczi <stefanha@linux.vnet.ibm.com>
-#
-# This script provides a command-line interface to kvm ftrace and is designed
-# to be used as a flight recorder that is always running. To start in-memory
-# recording:
-#
-# sudo kvm_flightrecorder start 8192 # 8 MB per-cpu ring buffers
-#
-# The per-cpu ring buffer size can be given in KB as an optional argument to
-# the 'start' subcommand.
-#
-# To stop the flight recorder:
-#
-# sudo kvm_flightrecorder stop
-#
-# To dump the contents of the flight recorder (this can be done when the
-# recorder is stopped or while it is running):
-#
-# sudo kvm_flightrecorder dump >/path/to/dump.txt
-#
-# To observe the trace while it is running, use the 'tail' subcommand:
-#
-# sudo kvm_flightrecorder tail
-#
-# Note that the flight recorder may impact overall system performance by
-# consuming CPU cycles. No disk I/O is performed since the ring buffer holds a
-# fixed-size in-memory trace.
-
-import sys
-import os
-
-tracing_dir = '/sys/kernel/debug/tracing'
-
-def trace_path(*args):
- return os.path.join(tracing_dir, *args)
-
-def write_file(path, data):
- open(path, 'wb').write(data)
-
-def enable_event(subsystem, event, enable):
- write_file(trace_path('events', subsystem, event, 'enable'), '1' if enable else '0')
-
-def enable_subsystem(subsystem, enable):
- write_file(trace_path('events', subsystem, 'enable'), '1' if enable else '0')
-
-def start_tracing():
- enable_subsystem('kvm', True)
- write_file(trace_path('tracing_on'), '1')
-
-def stop_tracing():
- write_file(trace_path('tracing_on'), '0')
- enable_subsystem('kvm', False)
- write_file(trace_path('events', 'enable'), '0')
- write_file(trace_path('current_tracer'), 'nop')
-
-def dump_trace():
- tracefile = open(trace_path('trace'), 'r')
- try:
- lines = True
- while lines:
- lines = tracefile.readlines(64 * 1024)
- sys.stdout.writelines(lines)
- except KeyboardInterrupt:
- pass
-
-def tail_trace():
- try:
- for line in open(trace_path('trace_pipe'), 'r'):
- sys.stdout.write(line)
- except KeyboardInterrupt:
- pass
-
-def usage():
- print 'Usage: %s start [buffer_size_kb] | stop | dump | tail' % sys.argv[0]
- print 'Control the KVM flight recorder tracing.'
- sys.exit(0)
-
-def main():
- if len(sys.argv) < 2:
- usage()
-
- cmd = sys.argv[1]
- if cmd == '--version':
- print 'kvm_flightrecorder version 1.0'
- sys.exit(0)
-
- if not os.path.isdir(tracing_dir):
- print 'Unable to tracing debugfs directory, try:'
- print 'mount -t debugfs none /sys/kernel/debug'
- sys.exit(1)
- if not os.access(tracing_dir, os.W_OK):
- print 'Unable to write to tracing debugfs directory, please run as root'
- sys.exit(1)
-
- if cmd == 'start':
- stop_tracing() # clean up first
-
- if len(sys.argv) == 3:
- try:
- buffer_size_kb = int(sys.argv[2])
- except ValueError:
- print 'Invalid per-cpu trace buffer size in KB'
- sys.exit(1)
- write_file(trace_path('buffer_size_kb'), str(buffer_size_kb))
- print 'Per-CPU ring buffer size set to %d KB' % buffer_size_kb
-
- start_tracing()
- print 'KVM flight recorder enabled'
- elif cmd == 'stop':
- stop_tracing()
- print 'KVM flight recorder disabled'
- elif cmd == 'dump':
- dump_trace()
- elif cmd == 'tail':
- tail_trace()
- else:
- usage()
-
-if __name__ == '__main__':
- sys.exit(main())
diff --git a/qemu/scripts/kvm/kvm_stat b/qemu/scripts/kvm/kvm_stat
deleted file mode 100755
index 769d884b6..000000000
--- a/qemu/scripts/kvm/kvm_stat
+++ /dev/null
@@ -1,825 +0,0 @@
-#!/usr/bin/python
-#
-# top-like utility for displaying kvm statistics
-#
-# Copyright 2006-2008 Qumranet Technologies
-# Copyright 2008-2011 Red Hat, Inc.
-#
-# Authors:
-# Avi Kivity <avi@redhat.com>
-#
-# This work is licensed under the terms of the GNU GPL, version 2. See
-# the COPYING file in the top-level directory.
-
-import curses
-import sys
-import os
-import time
-import optparse
-import ctypes
-import fcntl
-import resource
-import struct
-import re
-from collections import defaultdict
-from time import sleep
-
-VMX_EXIT_REASONS = {
- 'EXCEPTION_NMI': 0,
- 'EXTERNAL_INTERRUPT': 1,
- 'TRIPLE_FAULT': 2,
- 'PENDING_INTERRUPT': 7,
- 'NMI_WINDOW': 8,
- 'TASK_SWITCH': 9,
- 'CPUID': 10,
- 'HLT': 12,
- 'INVLPG': 14,
- 'RDPMC': 15,
- 'RDTSC': 16,
- 'VMCALL': 18,
- 'VMCLEAR': 19,
- 'VMLAUNCH': 20,
- 'VMPTRLD': 21,
- 'VMPTRST': 22,
- 'VMREAD': 23,
- 'VMRESUME': 24,
- 'VMWRITE': 25,
- 'VMOFF': 26,
- 'VMON': 27,
- 'CR_ACCESS': 28,
- 'DR_ACCESS': 29,
- 'IO_INSTRUCTION': 30,
- 'MSR_READ': 31,
- 'MSR_WRITE': 32,
- 'INVALID_STATE': 33,
- 'MWAIT_INSTRUCTION': 36,
- 'MONITOR_INSTRUCTION': 39,
- 'PAUSE_INSTRUCTION': 40,
- 'MCE_DURING_VMENTRY': 41,
- 'TPR_BELOW_THRESHOLD': 43,
- 'APIC_ACCESS': 44,
- 'EPT_VIOLATION': 48,
- 'EPT_MISCONFIG': 49,
- 'WBINVD': 54,
- 'XSETBV': 55,
- 'APIC_WRITE': 56,
- 'INVPCID': 58,
-}
-
-SVM_EXIT_REASONS = {
- 'READ_CR0': 0x000,
- 'READ_CR3': 0x003,
- 'READ_CR4': 0x004,
- 'READ_CR8': 0x008,
- 'WRITE_CR0': 0x010,
- 'WRITE_CR3': 0x013,
- 'WRITE_CR4': 0x014,
- 'WRITE_CR8': 0x018,
- 'READ_DR0': 0x020,
- 'READ_DR1': 0x021,
- 'READ_DR2': 0x022,
- 'READ_DR3': 0x023,
- 'READ_DR4': 0x024,
- 'READ_DR5': 0x025,
- 'READ_DR6': 0x026,
- 'READ_DR7': 0x027,
- 'WRITE_DR0': 0x030,
- 'WRITE_DR1': 0x031,
- 'WRITE_DR2': 0x032,
- 'WRITE_DR3': 0x033,
- 'WRITE_DR4': 0x034,
- 'WRITE_DR5': 0x035,
- 'WRITE_DR6': 0x036,
- 'WRITE_DR7': 0x037,
- 'EXCP_BASE': 0x040,
- 'INTR': 0x060,
- 'NMI': 0x061,
- 'SMI': 0x062,
- 'INIT': 0x063,
- 'VINTR': 0x064,
- 'CR0_SEL_WRITE': 0x065,
- 'IDTR_READ': 0x066,
- 'GDTR_READ': 0x067,
- 'LDTR_READ': 0x068,
- 'TR_READ': 0x069,
- 'IDTR_WRITE': 0x06a,
- 'GDTR_WRITE': 0x06b,
- 'LDTR_WRITE': 0x06c,
- 'TR_WRITE': 0x06d,
- 'RDTSC': 0x06e,
- 'RDPMC': 0x06f,
- 'PUSHF': 0x070,
- 'POPF': 0x071,
- 'CPUID': 0x072,
- 'RSM': 0x073,
- 'IRET': 0x074,
- 'SWINT': 0x075,
- 'INVD': 0x076,
- 'PAUSE': 0x077,
- 'HLT': 0x078,
- 'INVLPG': 0x079,
- 'INVLPGA': 0x07a,
- 'IOIO': 0x07b,
- 'MSR': 0x07c,
- 'TASK_SWITCH': 0x07d,
- 'FERR_FREEZE': 0x07e,
- 'SHUTDOWN': 0x07f,
- 'VMRUN': 0x080,
- 'VMMCALL': 0x081,
- 'VMLOAD': 0x082,
- 'VMSAVE': 0x083,
- 'STGI': 0x084,
- 'CLGI': 0x085,
- 'SKINIT': 0x086,
- 'RDTSCP': 0x087,
- 'ICEBP': 0x088,
- 'WBINVD': 0x089,
- 'MONITOR': 0x08a,
- 'MWAIT': 0x08b,
- 'MWAIT_COND': 0x08c,
- 'XSETBV': 0x08d,
- 'NPF': 0x400,
-}
-
-# EC definition of HSR (from arch/arm64/include/asm/kvm_arm.h)
-AARCH64_EXIT_REASONS = {
- 'UNKNOWN': 0x00,
- 'WFI': 0x01,
- 'CP15_32': 0x03,
- 'CP15_64': 0x04,
- 'CP14_MR': 0x05,
- 'CP14_LS': 0x06,
- 'FP_ASIMD': 0x07,
- 'CP10_ID': 0x08,
- 'CP14_64': 0x0C,
- 'ILL_ISS': 0x0E,
- 'SVC32': 0x11,
- 'HVC32': 0x12,
- 'SMC32': 0x13,
- 'SVC64': 0x15,
- 'HVC64': 0x16,
- 'SMC64': 0x17,
- 'SYS64': 0x18,
- 'IABT': 0x20,
- 'IABT_HYP': 0x21,
- 'PC_ALIGN': 0x22,
- 'DABT': 0x24,
- 'DABT_HYP': 0x25,
- 'SP_ALIGN': 0x26,
- 'FP_EXC32': 0x28,
- 'FP_EXC64': 0x2C,
- 'SERROR': 0x2F,
- 'BREAKPT': 0x30,
- 'BREAKPT_HYP': 0x31,
- 'SOFTSTP': 0x32,
- 'SOFTSTP_HYP': 0x33,
- 'WATCHPT': 0x34,
- 'WATCHPT_HYP': 0x35,
- 'BKPT32': 0x38,
- 'VECTOR32': 0x3A,
- 'BRK64': 0x3C,
-}
-
-# From include/uapi/linux/kvm.h, KVM_EXIT_xxx
-USERSPACE_EXIT_REASONS = {
- 'UNKNOWN': 0,
- 'EXCEPTION': 1,
- 'IO': 2,
- 'HYPERCALL': 3,
- 'DEBUG': 4,
- 'HLT': 5,
- 'MMIO': 6,
- 'IRQ_WINDOW_OPEN': 7,
- 'SHUTDOWN': 8,
- 'FAIL_ENTRY': 9,
- 'INTR': 10,
- 'SET_TPR': 11,
- 'TPR_ACCESS': 12,
- 'S390_SIEIC': 13,
- 'S390_RESET': 14,
- 'DCR': 15,
- 'NMI': 16,
- 'INTERNAL_ERROR': 17,
- 'OSI': 18,
- 'PAPR_HCALL': 19,
- 'S390_UCONTROL': 20,
- 'WATCHDOG': 21,
- 'S390_TSCH': 22,
- 'EPR': 23,
- 'SYSTEM_EVENT': 24,
-}
-
-IOCTL_NUMBERS = {
- 'SET_FILTER': 0x40082406,
- 'ENABLE': 0x00002400,
- 'DISABLE': 0x00002401,
- 'RESET': 0x00002403,
-}
-
-class Arch(object):
- """Class that encapsulates global architecture specific data like
- syscall and ioctl numbers.
-
- """
- @staticmethod
- def get_arch():
- machine = os.uname()[4]
-
- if machine.startswith('ppc'):
- return ArchPPC()
- elif machine.startswith('aarch64'):
- return ArchA64()
- elif machine.startswith('s390'):
- return ArchS390()
- else:
- # X86_64
- for line in open('/proc/cpuinfo'):
- if not line.startswith('flags'):
- continue
-
- flags = line.split()
- if 'vmx' in flags:
- return ArchX86(VMX_EXIT_REASONS)
- if 'svm' in flags:
- return ArchX86(SVM_EXIT_REASONS)
- return
-
-class ArchX86(Arch):
- def __init__(self, exit_reasons):
- self.sc_perf_evt_open = 298
- self.ioctl_numbers = IOCTL_NUMBERS
- self.exit_reasons = exit_reasons
-
-class ArchPPC(Arch):
- def __init__(self):
- self.sc_perf_evt_open = 319
- self.ioctl_numbers = IOCTL_NUMBERS
- self.ioctl_numbers['ENABLE'] = 0x20002400
- self.ioctl_numbers['DISABLE'] = 0x20002401
-
- # PPC comes in 32 and 64 bit and some generated ioctl
- # numbers depend on the wordsize.
- char_ptr_size = ctypes.sizeof(ctypes.c_char_p)
- self.ioctl_numbers['SET_FILTER'] = 0x80002406 | char_ptr_size << 16
-
-class ArchA64(Arch):
- def __init__(self):
- self.sc_perf_evt_open = 241
- self.ioctl_numbers = IOCTL_NUMBERS
- self.exit_reasons = AARCH64_EXIT_REASONS
-
-class ArchS390(Arch):
- def __init__(self):
- self.sc_perf_evt_open = 331
- self.ioctl_numbers = IOCTL_NUMBERS
- self.exit_reasons = None
-
-ARCH = Arch.get_arch()
-
-
-def walkdir(path):
- """Returns os.walk() data for specified directory.
-
- As it is only a wrapper it returns the same 3-tuple of (dirpath,
- dirnames, filenames).
- """
- return next(os.walk(path))
-
-
-def parse_int_list(list_string):
- """Returns an int list from a string of comma separated integers and
- integer ranges."""
- integers = []
- members = list_string.split(',')
-
- for member in members:
- if '-' not in member:
- integers.append(int(member))
- else:
- int_range = member.split('-')
- integers.extend(range(int(int_range[0]),
- int(int_range[1]) + 1))
-
- return integers
-
-
-def get_online_cpus():
- with open('/sys/devices/system/cpu/online') as cpu_list:
- cpu_string = cpu_list.readline()
- return parse_int_list(cpu_string)
-
-
-def get_filters():
- filters = {}
- filters['kvm_userspace_exit'] = ('reason', USERSPACE_EXIT_REASONS)
- if ARCH.exit_reasons:
- filters['kvm_exit'] = ('exit_reason', ARCH.exit_reasons)
- return filters
-
-libc = ctypes.CDLL('libc.so.6', use_errno=True)
-syscall = libc.syscall
-
-class perf_event_attr(ctypes.Structure):
- _fields_ = [('type', ctypes.c_uint32),
- ('size', ctypes.c_uint32),
- ('config', ctypes.c_uint64),
- ('sample_freq', ctypes.c_uint64),
- ('sample_type', ctypes.c_uint64),
- ('read_format', ctypes.c_uint64),
- ('flags', ctypes.c_uint64),
- ('wakeup_events', ctypes.c_uint32),
- ('bp_type', ctypes.c_uint32),
- ('bp_addr', ctypes.c_uint64),
- ('bp_len', ctypes.c_uint64),
- ]
-
- def __init__(self):
- super(self.__class__, self).__init__()
- self.type = PERF_TYPE_TRACEPOINT
- self.size = ctypes.sizeof(self)
- self.read_format = PERF_FORMAT_GROUP
-
-def perf_event_open(attr, pid, cpu, group_fd, flags):
- return syscall(ARCH.sc_perf_evt_open, ctypes.pointer(attr),
- ctypes.c_int(pid), ctypes.c_int(cpu),
- ctypes.c_int(group_fd), ctypes.c_long(flags))
-
-PERF_TYPE_TRACEPOINT = 2
-PERF_FORMAT_GROUP = 1 << 3
-
-PATH_DEBUGFS_TRACING = '/sys/kernel/debug/tracing'
-PATH_DEBUGFS_KVM = '/sys/kernel/debug/kvm'
-
-class Group(object):
- def __init__(self):
- self.events = []
-
- def add_event(self, event):
- self.events.append(event)
-
- def read(self):
- length = 8 * (1 + len(self.events))
- read_format = 'xxxxxxxx' + 'Q' * len(self.events)
- return dict(zip([event.name for event in self.events],
- struct.unpack(read_format,
- os.read(self.events[0].fd, length))))
-
-class Event(object):
- def __init__(self, name, group, trace_cpu, trace_point, trace_filter,
- trace_set='kvm'):
- self.name = name
- self.fd = None
- self.setup_event(group, trace_cpu, trace_point, trace_filter,
- trace_set)
-
- def setup_event_attribute(self, trace_set, trace_point):
- id_path = os.path.join(PATH_DEBUGFS_TRACING, 'events', trace_set,
- trace_point, 'id')
-
- event_attr = perf_event_attr()
- event_attr.config = int(open(id_path).read())
- return event_attr
-
- def setup_event(self, group, trace_cpu, trace_point, trace_filter,
- trace_set):
- event_attr = self.setup_event_attribute(trace_set, trace_point)
-
- group_leader = -1
- if group.events:
- group_leader = group.events[0].fd
-
- fd = perf_event_open(event_attr, -1, trace_cpu,
- group_leader, 0)
- if fd == -1:
- err = ctypes.get_errno()
- raise OSError(err, os.strerror(err),
- 'while calling sys_perf_event_open().')
-
- if trace_filter:
- fcntl.ioctl(fd, ARCH.ioctl_numbers['SET_FILTER'],
- trace_filter)
-
- self.fd = fd
-
- def enable(self):
- fcntl.ioctl(self.fd, ARCH.ioctl_numbers['ENABLE'], 0)
-
- def disable(self):
- fcntl.ioctl(self.fd, ARCH.ioctl_numbers['DISABLE'], 0)
-
- def reset(self):
- fcntl.ioctl(self.fd, ARCH.ioctl_numbers['RESET'], 0)
-
-class TracepointProvider(object):
- def __init__(self):
- self.group_leaders = []
- self.filters = get_filters()
- self._fields = self.get_available_fields()
- self.setup_traces()
- self.fields = self._fields
-
- def get_available_fields(self):
- path = os.path.join(PATH_DEBUGFS_TRACING, 'events', 'kvm')
- fields = walkdir(path)[1]
- extra = []
- for field in fields:
- if field in self.filters:
- filter_name_, filter_dicts = self.filters[field]
- for name in filter_dicts:
- extra.append(field + '(' + name + ')')
- fields += extra
- return fields
-
- def setup_traces(self):
- cpus = get_online_cpus()
-
- # The constant is needed as a buffer for python libs, std
- # streams and other files that the script opens.
- newlim = len(cpus) * len(self._fields) + 50
- try:
- softlim_, hardlim = resource.getrlimit(resource.RLIMIT_NOFILE)
-
- if hardlim < newlim:
- # Now we need CAP_SYS_RESOURCE, to increase the hard limit.
- resource.setrlimit(resource.RLIMIT_NOFILE, (newlim, newlim))
- else:
- # Raising the soft limit is sufficient.
- resource.setrlimit(resource.RLIMIT_NOFILE, (newlim, hardlim))
-
- except ValueError:
- sys.exit("NOFILE rlimit could not be raised to {0}".format(newlim))
-
- for cpu in cpus:
- group = Group()
- for name in self._fields:
- tracepoint = name
- tracefilter = None
- match = re.match(r'(.*)\((.*)\)', name)
- if match:
- tracepoint, sub = match.groups()
- tracefilter = ('%s==%d\0' %
- (self.filters[tracepoint][0],
- self.filters[tracepoint][1][sub]))
-
- group.add_event(Event(name=name,
- group=group,
- trace_cpu=cpu,
- trace_point=tracepoint,
- trace_filter=tracefilter))
- self.group_leaders.append(group)
-
- def available_fields(self):
- return self.get_available_fields()
-
- @property
- def fields(self):
- return self._fields
-
- @fields.setter
- def fields(self, fields):
- self._fields = fields
- for group in self.group_leaders:
- for index, event in enumerate(group.events):
- if event.name in fields:
- event.reset()
- event.enable()
- else:
- # Do not disable the group leader.
- # It would disable all of its events.
- if index != 0:
- event.disable()
-
- def read(self):
- ret = defaultdict(int)
- for group in self.group_leaders:
- for name, val in group.read().iteritems():
- if name in self._fields:
- ret[name] += val
- return ret
-
-class DebugfsProvider(object):
- def __init__(self):
- self._fields = self.get_available_fields()
-
- def get_available_fields(self):
- return walkdir(PATH_DEBUGFS_KVM)[2]
-
- @property
- def fields(self):
- return self._fields
-
- @fields.setter
- def fields(self, fields):
- self._fields = fields
-
- def read(self):
- def val(key):
- return int(file(PATH_DEBUGFS_KVM + '/' + key).read())
- return dict([(key, val(key)) for key in self._fields])
-
-class Stats(object):
- def __init__(self, providers, fields=None):
- self.providers = providers
- self._fields_filter = fields
- self.values = {}
- self.update_provider_filters()
-
- def update_provider_filters(self):
- def wanted(key):
- if not self._fields_filter:
- return True
- return re.match(self._fields_filter, key) is not None
-
- # As we reset the counters when updating the fields we can
- # also clear the cache of old values.
- self.values = {}
- for provider in self.providers:
- provider_fields = [key for key in provider.get_available_fields()
- if wanted(key)]
- provider.fields = provider_fields
-
- @property
- def fields_filter(self):
- return self._fields_filter
-
- @fields_filter.setter
- def fields_filter(self, fields_filter):
- self._fields_filter = fields_filter
- self.update_provider_filters()
-
- def get(self):
- for provider in self.providers:
- new = provider.read()
- for key in provider.fields:
- oldval = self.values.get(key, (0, 0))
- newval = new.get(key, 0)
- newdelta = None
- if oldval is not None:
- newdelta = newval - oldval[0]
- self.values[key] = (newval, newdelta)
- return self.values
-
-LABEL_WIDTH = 40
-NUMBER_WIDTH = 10
-
-class Tui(object):
- def __init__(self, stats):
- self.stats = stats
- self.screen = None
- self.drilldown = False
- self.update_drilldown()
-
- def __enter__(self):
- """Initialises curses for later use. Based on curses.wrapper
- implementation from the Python standard library."""
- self.screen = curses.initscr()
- curses.noecho()
- curses.cbreak()
-
- # The try/catch works around a minor bit of
- # over-conscientiousness in the curses module, the error
- # return from C start_color() is ignorable.
- try:
- curses.start_color()
- except:
- pass
-
- curses.use_default_colors()
- return self
-
- def __exit__(self, *exception):
- """Resets the terminal to its normal state. Based on curses.wrappre
- implementation from the Python standard library."""
- if self.screen:
- self.screen.keypad(0)
- curses.echo()
- curses.nocbreak()
- curses.endwin()
-
- def update_drilldown(self):
- if not self.stats.fields_filter:
- self.stats.fields_filter = r'^[^\(]*$'
-
- elif self.stats.fields_filter == r'^[^\(]*$':
- self.stats.fields_filter = None
-
- def refresh(self, sleeptime):
- self.screen.erase()
- self.screen.addstr(0, 0, 'kvm statistics - summary', curses.A_BOLD)
- self.screen.addstr(2, 1, 'Event')
- self.screen.addstr(2, 1 + LABEL_WIDTH + NUMBER_WIDTH -
- len('Total'), 'Total')
- self.screen.addstr(2, 1 + LABEL_WIDTH + NUMBER_WIDTH + 8 -
- len('Current'), 'Current')
- row = 3
- stats = self.stats.get()
- def sortkey(x):
- if stats[x][1]:
- return (-stats[x][1], -stats[x][0])
- else:
- return (0, -stats[x][0])
- for key in sorted(stats.keys(), key=sortkey):
-
- if row >= self.screen.getmaxyx()[0]:
- break
- values = stats[key]
- if not values[0] and not values[1]:
- break
- col = 1
- self.screen.addstr(row, col, key)
- col += LABEL_WIDTH
- self.screen.addstr(row, col, '%10d' % (values[0],))
- col += NUMBER_WIDTH
- if values[1] is not None:
- self.screen.addstr(row, col, '%8d' % (values[1] / sleeptime,))
- row += 1
- self.screen.refresh()
-
- def show_filter_selection(self):
- while True:
- self.screen.erase()
- self.screen.addstr(0, 0,
- "Show statistics for events matching a regex.",
- curses.A_BOLD)
- self.screen.addstr(2, 0,
- "Current regex: {0}"
- .format(self.stats.fields_filter))
- self.screen.addstr(3, 0, "New regex: ")
- curses.echo()
- regex = self.screen.getstr()
- curses.noecho()
- if len(regex) == 0:
- return
- try:
- re.compile(regex)
- self.stats.fields_filter = regex
- return
- except re.error:
- continue
-
- def show_stats(self):
- sleeptime = 0.25
- while True:
- self.refresh(sleeptime)
- curses.halfdelay(int(sleeptime * 10))
- sleeptime = 3
- try:
- char = self.screen.getkey()
- if char == 'x':
- self.drilldown = not self.drilldown
- self.update_drilldown()
- if char == 'q':
- break
- if char == 'f':
- self.show_filter_selection()
- except KeyboardInterrupt:
- break
- except curses.error:
- continue
-
-def batch(stats):
- s = stats.get()
- time.sleep(1)
- s = stats.get()
- for key in sorted(s.keys()):
- values = s[key]
- print '%-42s%10d%10d' % (key, values[0], values[1])
-
-def log(stats):
- keys = sorted(stats.get().iterkeys())
- def banner():
- for k in keys:
- print '%s' % k,
- print
- def statline():
- s = stats.get()
- for k in keys:
- print ' %9d' % s[k][1],
- print
- line = 0
- banner_repeat = 20
- while True:
- time.sleep(1)
- if line % banner_repeat == 0:
- banner()
- statline()
- line += 1
-
-def get_options():
- description_text = """
-This script displays various statistics about VMs running under KVM.
-The statistics are gathered from the KVM debugfs entries and / or the
-currently available perf traces.
-
-The monitoring takes additional cpu cycles and might affect the VM's
-performance.
-
-Requirements:
-- Access to:
- /sys/kernel/debug/kvm
- /sys/kernel/debug/trace/events/*
- /proc/pid/task
-- /proc/sys/kernel/perf_event_paranoid < 1 if user has no
- CAP_SYS_ADMIN and perf events are used.
-- CAP_SYS_RESOURCE if the hard limit is not high enough to allow
- the large number of files that are possibly opened.
-"""
-
- class PlainHelpFormatter(optparse.IndentedHelpFormatter):
- def format_description(self, description):
- if description:
- return description + "\n"
- else:
- return ""
-
- optparser = optparse.OptionParser(description=description_text,
- formatter=PlainHelpFormatter())
- optparser.add_option('-1', '--once', '--batch',
- action='store_true',
- default=False,
- dest='once',
- help='run in batch mode for one second',
- )
- optparser.add_option('-l', '--log',
- action='store_true',
- default=False,
- dest='log',
- help='run in logging mode (like vmstat)',
- )
- optparser.add_option('-t', '--tracepoints',
- action='store_true',
- default=False,
- dest='tracepoints',
- help='retrieve statistics from tracepoints',
- )
- optparser.add_option('-d', '--debugfs',
- action='store_true',
- default=False,
- dest='debugfs',
- help='retrieve statistics from debugfs',
- )
- optparser.add_option('-f', '--fields',
- action='store',
- default=None,
- dest='fields',
- help='fields to display (regex)',
- )
- (options, _) = optparser.parse_args(sys.argv)
- return options
-
-def get_providers(options):
- providers = []
-
- if options.tracepoints:
- providers.append(TracepointProvider())
- if options.debugfs:
- providers.append(DebugfsProvider())
- if len(providers) == 0:
- providers.append(TracepointProvider())
-
- return providers
-
-def check_access(options):
- if not os.path.exists('/sys/kernel/debug'):
- sys.stderr.write('Please enable CONFIG_DEBUG_FS in your kernel.')
- sys.exit(1)
-
- if not os.path.exists(PATH_DEBUGFS_KVM):
- sys.stderr.write("Please make sure, that debugfs is mounted and "
- "readable by the current user:\n"
- "('mount -t debugfs debugfs /sys/kernel/debug')\n"
- "Also ensure, that the kvm modules are loaded.\n")
- sys.exit(1)
-
- if not os.path.exists(PATH_DEBUGFS_TRACING) and (options.tracepoints
- or not options.debugfs):
- sys.stderr.write("Please enable CONFIG_TRACING in your kernel "
- "when using the option -t (default).\n"
- "If it is enabled, make {0} readable by the "
- "current user.\n"
- .format(PATH_DEBUGFS_TRACING))
- if options.tracepoints:
- sys.exit(1)
-
- sys.stderr.write("Falling back to debugfs statistics!\n")
- options.debugfs = True
- sleep(5)
-
- return options
-
-def main():
- options = get_options()
- options = check_access(options)
- providers = get_providers(options)
- stats = Stats(providers, fields=options.fields)
-
- if options.log:
- log(stats)
- elif not options.once:
- with Tui(stats) as tui:
- tui.show_stats()
- else:
- batch(stats)
-
-if __name__ == "__main__":
- main()
diff --git a/qemu/scripts/kvm/kvm_stat.texi b/qemu/scripts/kvm/kvm_stat.texi
deleted file mode 100644
index 6ce00d80e..000000000
--- a/qemu/scripts/kvm/kvm_stat.texi
+++ /dev/null
@@ -1,55 +0,0 @@
-@example
-@c man begin SYNOPSIS
-usage: kvm_stat [OPTION]...
-@c man end
-@end example
-
-@c man begin DESCRIPTION
-
-kvm_stat prints counts of KVM kernel module trace events. These events signify
-state transitions such as guest mode entry and exit.
-
-This tool is useful for observing guest behavior from the host perspective.
-Often conclusions about performance or buggy behavior can be drawn from the
-output.
-
-The set of KVM kernel module trace events may be specific to the kernel version
-or architecture. It is best to check the KVM kernel module source code for the
-meaning of events.
-
-Note that trace events are counted globally across all running guests.
-
-@c man end
-
-@c man begin OPTIONS
-@table @option
-@item -1, --once, --batch
- run in batch mode for one second
-@item -l, --log
- run in logging mode (like vmstat)
-@item -t, --tracepoints
- retrieve statistics from tracepoints
-@item -d, --debugfs
- retrieve statistics from debugfs
-@item -f, --fields=@var{fields}
- fields to display (regex)
-@item -h, --help
- show help message
-@end table
-
-@c man end
-
-@ignore
-
-@setfilename kvm_stat
-@settitle Report KVM kernel module event counters.
-
-@c man begin AUTHOR
-Stefan Hajnoczi <stefanha@redhat.com>
-@c man end
-
-@c man begin SEEALSO
-perf(1), trace-cmd(1)
-@c man end
-
-@end ignore
diff --git a/qemu/scripts/kvm/vmxcap b/qemu/scripts/kvm/vmxcap
deleted file mode 100755
index 8f0371f49..000000000
--- a/qemu/scripts/kvm/vmxcap
+++ /dev/null
@@ -1,260 +0,0 @@
-#!/usr/bin/python
-#
-# tool for querying VMX capabilities
-#
-# Copyright 2009-2010 Red Hat, Inc.
-#
-# Authors:
-# Avi Kivity <avi@redhat.com>
-#
-# This work is licensed under the terms of the GNU GPL, version 2. See
-# the COPYING file in the top-level directory.
-
-MSR_IA32_VMX_BASIC = 0x480
-MSR_IA32_VMX_PINBASED_CTLS = 0x481
-MSR_IA32_VMX_PROCBASED_CTLS = 0x482
-MSR_IA32_VMX_EXIT_CTLS = 0x483
-MSR_IA32_VMX_ENTRY_CTLS = 0x484
-MSR_IA32_VMX_MISC_CTLS = 0x485
-MSR_IA32_VMX_PROCBASED_CTLS2 = 0x48B
-MSR_IA32_VMX_EPT_VPID_CAP = 0x48C
-MSR_IA32_VMX_TRUE_PINBASED_CTLS = 0x48D
-MSR_IA32_VMX_TRUE_PROCBASED_CTLS = 0x48E
-MSR_IA32_VMX_TRUE_EXIT_CTLS = 0x48F
-MSR_IA32_VMX_TRUE_ENTRY_CTLS = 0x490
-MSR_IA32_VMX_VMFUNC = 0x491
-
-class msr(object):
- def __init__(self):
- try:
- self.f = open('/dev/cpu/0/msr', 'r', 0)
- except:
- self.f = open('/dev/msr0', 'r', 0)
- def read(self, index, default = None):
- import struct
- self.f.seek(index)
- try:
- return struct.unpack('Q', self.f.read(8))[0]
- except:
- return default
-
-class Control(object):
- def __init__(self, name, bits, cap_msr, true_cap_msr = None):
- self.name = name
- self.bits = bits
- self.cap_msr = cap_msr
- self.true_cap_msr = true_cap_msr
- def read2(self, nr):
- m = msr()
- val = m.read(nr, 0)
- return (val & 0xffffffff, val >> 32)
- def show(self):
- print self.name
- mbz, mb1 = self.read2(self.cap_msr)
- tmbz, tmb1 = 0, 0
- if self.true_cap_msr:
- tmbz, tmb1 = self.read2(self.true_cap_msr)
- for bit in sorted(self.bits.keys()):
- zero = not (mbz & (1 << bit))
- one = mb1 & (1 << bit)
- true_zero = not (tmbz & (1 << bit))
- true_one = tmb1 & (1 << bit)
- s= '?'
- if (self.true_cap_msr and true_zero and true_one
- and one and not zero):
- s = 'default'
- elif zero and not one:
- s = 'no'
- elif one and not zero:
- s = 'forced'
- elif one and zero:
- s = 'yes'
- print ' %-40s %s' % (self.bits[bit], s)
-
-class Misc(object):
- def __init__(self, name, bits, msr):
- self.name = name
- self.bits = bits
- self.msr = msr
- def show(self):
- print self.name
- value = msr().read(self.msr, 0)
- def first_bit(key):
- if type(key) is tuple:
- return key[0]
- else:
- return key
- for bits in sorted(self.bits.keys(), key = first_bit):
- if type(bits) is tuple:
- lo, hi = bits
- fmt = int
- else:
- lo = hi = bits
- def fmt(x):
- return { True: 'yes', False: 'no' }[x]
- v = (value >> lo) & ((1 << (hi - lo + 1)) - 1)
- print ' %-40s %s' % (self.bits[bits], fmt(v))
-
-controls = [
- Misc(
- name = 'Basic VMX Information',
- bits = {
- (0, 30): 'Revision',
- (32,44): 'VMCS size',
- 48: 'VMCS restricted to 32 bit addresses',
- 49: 'Dual-monitor support',
- (50, 53): 'VMCS memory type',
- 54: 'INS/OUTS instruction information',
- 55: 'IA32_VMX_TRUE_*_CTLS support',
- },
- msr = MSR_IA32_VMX_BASIC,
- ),
- Control(
- name = 'pin-based controls',
- bits = {
- 0: 'External interrupt exiting',
- 3: 'NMI exiting',
- 5: 'Virtual NMIs',
- 6: 'Activate VMX-preemption timer',
- 7: 'Process posted interrupts',
- },
- cap_msr = MSR_IA32_VMX_PINBASED_CTLS,
- true_cap_msr = MSR_IA32_VMX_TRUE_PINBASED_CTLS,
- ),
-
- Control(
- name = 'primary processor-based controls',
- bits = {
- 2: 'Interrupt window exiting',
- 3: 'Use TSC offsetting',
- 7: 'HLT exiting',
- 9: 'INVLPG exiting',
- 10: 'MWAIT exiting',
- 11: 'RDPMC exiting',
- 12: 'RDTSC exiting',
- 15: 'CR3-load exiting',
- 16: 'CR3-store exiting',
- 19: 'CR8-load exiting',
- 20: 'CR8-store exiting',
- 21: 'Use TPR shadow',
- 22: 'NMI-window exiting',
- 23: 'MOV-DR exiting',
- 24: 'Unconditional I/O exiting',
- 25: 'Use I/O bitmaps',
- 27: 'Monitor trap flag',
- 28: 'Use MSR bitmaps',
- 29: 'MONITOR exiting',
- 30: 'PAUSE exiting',
- 31: 'Activate secondary control',
- },
- cap_msr = MSR_IA32_VMX_PROCBASED_CTLS,
- true_cap_msr = MSR_IA32_VMX_TRUE_PROCBASED_CTLS,
- ),
-
- Control(
- name = 'secondary processor-based controls',
- bits = {
- 0: 'Virtualize APIC accesses',
- 1: 'Enable EPT',
- 2: 'Descriptor-table exiting',
- 3: 'Enable RDTSCP',
- 4: 'Virtualize x2APIC mode',
- 5: 'Enable VPID',
- 6: 'WBINVD exiting',
- 7: 'Unrestricted guest',
- 8: 'APIC register emulation',
- 9: 'Virtual interrupt delivery',
- 10: 'PAUSE-loop exiting',
- 11: 'RDRAND exiting',
- 12: 'Enable INVPCID',
- 13: 'Enable VM functions',
- 14: 'VMCS shadowing',
- 16: 'RDSEED exiting',
- 18: 'EPT-violation #VE',
- 20: 'Enable XSAVES/XRSTORS',
- },
- cap_msr = MSR_IA32_VMX_PROCBASED_CTLS2,
- ),
-
- Control(
- name = 'VM-Exit controls',
- bits = {
- 2: 'Save debug controls',
- 9: 'Host address-space size',
- 12: 'Load IA32_PERF_GLOBAL_CTRL',
- 15: 'Acknowledge interrupt on exit',
- 18: 'Save IA32_PAT',
- 19: 'Load IA32_PAT',
- 20: 'Save IA32_EFER',
- 21: 'Load IA32_EFER',
- 22: 'Save VMX-preemption timer value',
- },
- cap_msr = MSR_IA32_VMX_EXIT_CTLS,
- true_cap_msr = MSR_IA32_VMX_TRUE_EXIT_CTLS,
- ),
-
- Control(
- name = 'VM-Entry controls',
- bits = {
- 2: 'Load debug controls',
- 9: 'IA-32e mode guest',
- 10: 'Entry to SMM',
- 11: 'Deactivate dual-monitor treatment',
- 13: 'Load IA32_PERF_GLOBAL_CTRL',
- 14: 'Load IA32_PAT',
- 15: 'Load IA32_EFER',
- },
- cap_msr = MSR_IA32_VMX_ENTRY_CTLS,
- true_cap_msr = MSR_IA32_VMX_TRUE_ENTRY_CTLS,
- ),
-
- Misc(
- name = 'Miscellaneous data',
- bits = {
- (0,4): 'VMX-preemption timer scale (log2)',
- 5: 'Store EFER.LMA into IA-32e mode guest control',
- 6: 'HLT activity state',
- 7: 'Shutdown activity state',
- 8: 'Wait-for-SIPI activity state',
- 15: 'IA32_SMBASE support',
- (16,24): 'Number of CR3-target values',
- (25,27): 'MSR-load/store count recommendation',
- 28: 'IA32_SMM_MONITOR_CTL[2] can be set to 1',
- 29: 'VMWRITE to VM-exit information fields',
- (32,63): 'MSEG revision identifier',
- },
- msr = MSR_IA32_VMX_MISC_CTLS,
- ),
-
- Misc(
- name = 'VPID and EPT capabilities',
- bits = {
- 0: 'Execute-only EPT translations',
- 6: 'Page-walk length 4',
- 8: 'Paging-structure memory type UC',
- 14: 'Paging-structure memory type WB',
- 16: '2MB EPT pages',
- 17: '1GB EPT pages',
- 20: 'INVEPT supported',
- 21: 'EPT accessed and dirty flags',
- 25: 'Single-context INVEPT',
- 26: 'All-context INVEPT',
- 32: 'INVVPID supported',
- 40: 'Individual-address INVVPID',
- 41: 'Single-context INVVPID',
- 42: 'All-context INVVPID',
- 43: 'Single-context-retaining-globals INVVPID',
- },
- msr = MSR_IA32_VMX_EPT_VPID_CAP,
- ),
- Misc(
- name = 'VM Functions',
- bits = {
- 0: 'EPTP Switching',
- },
- msr = MSR_IA32_VMX_VMFUNC,
- ),
- ]
-
-for c in controls:
- c.show()