import hyperv-daemons-0-0.29.20180415git.el8
This commit is contained in:
		
							parent
							
								
									d76d9977fa
								
							
						
					
					
						commit
						a074fdddb1
					
				
							
								
								
									
										414
									
								
								SOURCES/hvd-Add-vmbus_testing-tool-build-files.patch
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										414
									
								
								SOURCES/hvd-Add-vmbus_testing-tool-build-files.patch
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,414 @@ | |||||||
|  | From d8ca5e0a429b8f7395e136e713980db6a7ac8dc2 Mon Sep 17 00:00:00 2001 | ||||||
|  | From: Mohammed Gamal <mgamal@redhat.com> | ||||||
|  | Date: Wed, 15 Apr 2020 12:00:30 +0200 | ||||||
|  | Subject: [PATCH 2/2] Add vmbus_testing tool build files | ||||||
|  | 
 | ||||||
|  | RH-Author: Mohammed Gamal <mgamal@redhat.com> | ||||||
|  | Message-id: <20200414183955.194006-3-mgamal@redhat.com> | ||||||
|  | Patchwork-id: 94690 | ||||||
|  | O-Subject: [RHEL8.3 virt hyperv-daemons PATCH v5 2/2] Add vmbus_testing tool build files | ||||||
|  | Bugzilla: 1816750 | ||||||
|  | RH-Acked-by: Vitaly Kuznetsov <vkuznets@redhat.com> | ||||||
|  | RH-Acked-by: Cathy Avery <cavery@redhat.com> | ||||||
|  | 
 | ||||||
|  | Add the vmbus_testing tool to redhat build dirs | ||||||
|  | 
 | ||||||
|  | Signed-off-by: Mohammed Gamal <mgamal@redhat.com> | ||||||
|  | ---
 | ||||||
|  |  redhat/hyperv-daemons.spec.template |   2 + | ||||||
|  |  vmbus_testing                       | 376 ++++++++++++++++++++++++++++ | ||||||
|  |  2 files changed, 378 insertions(+) | ||||||
|  |  create mode 100755 vmbus_testing | ||||||
|  | 
 | ||||||
|  | Signed-off-by: Miroslav Rezanina <mrezanin@redhat.com> | ||||||
|  | ---
 | ||||||
|  |  redhat/hyperv-daemons.spec.template |   2 + | ||||||
|  |  vmbus_testing                       | 376 ++++++++++++++++++++++++++++++++++++ | ||||||
|  |  2 files changed, 378 insertions(+) | ||||||
|  |  create mode 100755 vmbus_testing | ||||||
|  | 
 | ||||||
|  | diff --git a/vmbus_testing b/vmbus_testing
 | ||||||
|  | new file mode 100755 | ||||||
|  | index 0000000..e721290
 | ||||||
|  | --- /dev/null
 | ||||||
|  | +++ b/vmbus_testing
 | ||||||
|  | @@ -0,0 +1,376 @@
 | ||||||
|  | +#!/usr/bin/env python3
 | ||||||
|  | +# SPDX-License-Identifier: GPL-2.0
 | ||||||
|  | +#
 | ||||||
|  | +# Program to allow users to fuzz test Hyper-V drivers
 | ||||||
|  | +# by interfacing with Hyper-V debugfs attributes.
 | ||||||
|  | +# Current test methods available:
 | ||||||
|  | +#       1. delay testing
 | ||||||
|  | +#
 | ||||||
|  | +# Current file/directory structure of hyper-V debugfs:
 | ||||||
|  | +#       /sys/kernel/debug/hyperv/UUID
 | ||||||
|  | +#       /sys/kernel/debug/hyperv/UUID/<test-state filename>
 | ||||||
|  | +#       /sys/kernel/debug/hyperv/UUID/<test-method sub-directory>
 | ||||||
|  | +#
 | ||||||
|  | +# author: Branden Bonaby <brandonbonaby94@gmail.com>
 | ||||||
|  | +
 | ||||||
|  | +import os
 | ||||||
|  | +import cmd
 | ||||||
|  | +import argparse
 | ||||||
|  | +import glob
 | ||||||
|  | +from argparse import RawDescriptionHelpFormatter
 | ||||||
|  | +from argparse import RawTextHelpFormatter
 | ||||||
|  | +from enum import Enum
 | ||||||
|  | +
 | ||||||
|  | +# Do not change unless, you change the debugfs attributes
 | ||||||
|  | +# in /drivers/hv/debugfs.c. All fuzz testing
 | ||||||
|  | +# attributes will start with "fuzz_test".
 | ||||||
|  | +
 | ||||||
|  | +# debugfs path for hyperv must exist before proceeding
 | ||||||
|  | +debugfs_hyperv_path = "/sys/kernel/debug/hyperv"
 | ||||||
|  | +if not os.path.isdir(debugfs_hyperv_path):
 | ||||||
|  | +        print("{} doesn't exist/check permissions".format(debugfs_hyperv_path))
 | ||||||
|  | +        exit(-1)
 | ||||||
|  | +
 | ||||||
|  | +class dev_state(Enum):
 | ||||||
|  | +        off = 0
 | ||||||
|  | +        on = 1
 | ||||||
|  | +
 | ||||||
|  | +# File names, that correspond to the files created in
 | ||||||
|  | +# /drivers/hv/debugfs.c
 | ||||||
|  | +class f_names(Enum):
 | ||||||
|  | +        state_f = "fuzz_test_state"
 | ||||||
|  | +        buff_f =  "fuzz_test_buffer_interrupt_delay"
 | ||||||
|  | +        mess_f =  "fuzz_test_message_delay"
 | ||||||
|  | +
 | ||||||
|  | +# Both single_actions and all_actions are used
 | ||||||
|  | +# for error checking and to allow for some subparser
 | ||||||
|  | +# names to be abbreviated. Do not abbreviate the
 | ||||||
|  | +# test method names, as it will become less intuitive
 | ||||||
|  | +# as to what the user can do. If you do decide to
 | ||||||
|  | +# abbreviate the test method name, make sure the main
 | ||||||
|  | +# function reflects this change.
 | ||||||
|  | +
 | ||||||
|  | +all_actions = [
 | ||||||
|  | +        "disable_all",
 | ||||||
|  | +        "D",
 | ||||||
|  | +        "enable_all",
 | ||||||
|  | +        "view_all",
 | ||||||
|  | +        "V"
 | ||||||
|  | +]
 | ||||||
|  | +
 | ||||||
|  | +single_actions = [
 | ||||||
|  | +        "disable_single",
 | ||||||
|  | +        "d",
 | ||||||
|  | +        "enable_single",
 | ||||||
|  | +        "view_single",
 | ||||||
|  | +        "v"
 | ||||||
|  | +]
 | ||||||
|  | +
 | ||||||
|  | +def main():
 | ||||||
|  | +
 | ||||||
|  | +        file_map = recursive_file_lookup(debugfs_hyperv_path, dict())
 | ||||||
|  | +        args = parse_args()
 | ||||||
|  | +        if (not args.action):
 | ||||||
|  | +                print ("Error, no options selected...exiting")
 | ||||||
|  | +                exit(-1)
 | ||||||
|  | +        arg_set = { k for (k,v) in vars(args).items() if v and k != "action" }
 | ||||||
|  | +        arg_set.add(args.action)
 | ||||||
|  | +        path = args.path if "path" in arg_set else None
 | ||||||
|  | +        if (path and path[-1] == "/"):
 | ||||||
|  | +                path = path[:-1]
 | ||||||
|  | +        validate_args_path(path, arg_set, file_map)
 | ||||||
|  | +        if (path and "enable_single" in arg_set):
 | ||||||
|  | +            state_path = locate_state(path, file_map)
 | ||||||
|  | +            set_test_state(state_path, dev_state.on.value, args.quiet)
 | ||||||
|  | +
 | ||||||
|  | +        # Use subparsers as the key for different actions
 | ||||||
|  | +        if ("delay" in arg_set):
 | ||||||
|  | +                validate_delay_values(args.delay_time)
 | ||||||
|  | +                if (args.enable_all):
 | ||||||
|  | +                        set_delay_all_devices(file_map, args.delay_time,
 | ||||||
|  | +                                              args.quiet)
 | ||||||
|  | +                else:
 | ||||||
|  | +                        set_delay_values(path, file_map, args.delay_time,
 | ||||||
|  | +                                         args.quiet)
 | ||||||
|  | +        elif ("disable_all" in arg_set or "D" in arg_set):
 | ||||||
|  | +                disable_all_testing(file_map)
 | ||||||
|  | +        elif ("disable_single" in arg_set or "d" in arg_set):
 | ||||||
|  | +                disable_testing_single_device(path, file_map)
 | ||||||
|  | +        elif ("view_all" in arg_set or "V" in arg_set):
 | ||||||
|  | +                get_all_devices_test_status(file_map)
 | ||||||
|  | +        elif ("view_single" in arg_set or  "v" in arg_set):
 | ||||||
|  | +                get_device_test_values(path, file_map)
 | ||||||
|  | +
 | ||||||
|  | +# Get the state location
 | ||||||
|  | +def locate_state(device, file_map):
 | ||||||
|  | +        return file_map[device][f_names.state_f.value]
 | ||||||
|  | +
 | ||||||
|  | +# Validate delay values to make sure they are acceptable to
 | ||||||
|  | +# enable delays on a device
 | ||||||
|  | +def validate_delay_values(delay):
 | ||||||
|  | +
 | ||||||
|  | +        if (delay[0]  == -1 and delay[1] == -1):
 | ||||||
|  | +                print("\nError, At least 1 value must be greater than 0")
 | ||||||
|  | +                exit(-1)
 | ||||||
|  | +        for i in delay:
 | ||||||
|  | +                if (i < -1 or i == 0 or i > 1000):
 | ||||||
|  | +                        print("\nError, Values must be  equal to -1 "
 | ||||||
|  | +                              "or be > 0 and <= 1000")
 | ||||||
|  | +                        exit(-1)
 | ||||||
|  | +
 | ||||||
|  | +# Validate argument path
 | ||||||
|  | +def validate_args_path(path, arg_set, file_map):
 | ||||||
|  | +
 | ||||||
|  | +        if (not path and any(element in arg_set for element in single_actions)):
 | ||||||
|  | +                print("Error, path (-p) REQUIRED for the specified option. "
 | ||||||
|  | +                      "Use (-h) to check usage.")
 | ||||||
|  | +                exit(-1)
 | ||||||
|  | +        elif (path and any(item in arg_set for item in all_actions)):
 | ||||||
|  | +                print("Error, path (-p) NOT REQUIRED for the specified option. "
 | ||||||
|  | +                      "Use (-h) to check usage." )
 | ||||||
|  | +                exit(-1)
 | ||||||
|  | +        elif (path not in file_map and any(item in arg_set
 | ||||||
|  | +                                           for item in single_actions)):
 | ||||||
|  | +                print("Error, path '{}' not a valid vmbus device".format(path))
 | ||||||
|  | +                exit(-1)
 | ||||||
|  | +
 | ||||||
|  | +# display Testing status of single device
 | ||||||
|  | +def get_device_test_values(path, file_map):
 | ||||||
|  | +
 | ||||||
|  | +        for name in file_map[path]:
 | ||||||
|  | +                file_location = file_map[path][name]
 | ||||||
|  | +                print( name + " = " + str(read_test_files(file_location)))
 | ||||||
|  | +
 | ||||||
|  | +# Create a map of the vmbus devices and their associated files
 | ||||||
|  | +# [key=device, value = [key = filename, value = file path]]
 | ||||||
|  | +def recursive_file_lookup(path, file_map):
 | ||||||
|  | +
 | ||||||
|  | +        for f_path in glob.iglob(path + '**/*'):
 | ||||||
|  | +                if (os.path.isfile(f_path)):
 | ||||||
|  | +                        if (f_path.rsplit("/",2)[0] == debugfs_hyperv_path):
 | ||||||
|  | +                                directory = f_path.rsplit("/",1)[0]
 | ||||||
|  | +                        else:
 | ||||||
|  | +                                directory = f_path.rsplit("/",2)[0]
 | ||||||
|  | +                        f_name = f_path.split("/")[-1]
 | ||||||
|  | +                        if (file_map.get(directory)):
 | ||||||
|  | +                                file_map[directory].update({f_name:f_path})
 | ||||||
|  | +                        else:
 | ||||||
|  | +                                file_map[directory] = {f_name:f_path}
 | ||||||
|  | +                elif (os.path.isdir(f_path)):
 | ||||||
|  | +                        recursive_file_lookup(f_path,file_map)
 | ||||||
|  | +        return file_map
 | ||||||
|  | +
 | ||||||
|  | +# display Testing state of devices
 | ||||||
|  | +def get_all_devices_test_status(file_map):
 | ||||||
|  | +
 | ||||||
|  | +        for device in file_map:
 | ||||||
|  | +                if (get_test_state(locate_state(device, file_map)) is 1):
 | ||||||
|  | +                        print("Testing = ON for: {}"
 | ||||||
|  | +                              .format(device.split("/")[5]))
 | ||||||
|  | +                else:
 | ||||||
|  | +                        print("Testing = OFF for: {}"
 | ||||||
|  | +                              .format(device.split("/")[5]))
 | ||||||
|  | +
 | ||||||
|  | +# read the vmbus device files, path must be absolute path before calling
 | ||||||
|  | +def read_test_files(path):
 | ||||||
|  | +        try:
 | ||||||
|  | +                with open(path,"r") as f:
 | ||||||
|  | +                        file_value = f.readline().strip()
 | ||||||
|  | +                return int(file_value)
 | ||||||
|  | +
 | ||||||
|  | +        except IOError as e:
 | ||||||
|  | +                errno, strerror = e.args
 | ||||||
|  | +                print("I/O error({0}): {1} on file {2}"
 | ||||||
|  | +                      .format(errno, strerror, path))
 | ||||||
|  | +                exit(-1)
 | ||||||
|  | +        except ValueError:
 | ||||||
|  | +                print ("Element to int conversion error in: \n{}".format(path))
 | ||||||
|  | +                exit(-1)
 | ||||||
|  | +
 | ||||||
|  | +# writing to vmbus device files, path must be absolute path before calling
 | ||||||
|  | +def write_test_files(path, value):
 | ||||||
|  | +
 | ||||||
|  | +        try:
 | ||||||
|  | +                with open(path,"w") as f:
 | ||||||
|  | +                        f.write("{}".format(value))
 | ||||||
|  | +        except IOError as e:
 | ||||||
|  | +                errno, strerror = e.args
 | ||||||
|  | +                print("I/O error({0}): {1} on file {2}"
 | ||||||
|  | +                      .format(errno, strerror, path))
 | ||||||
|  | +                exit(-1)
 | ||||||
|  | +
 | ||||||
|  | +# set testing state of device
 | ||||||
|  | +def set_test_state(state_path, state_value, quiet):
 | ||||||
|  | +
 | ||||||
|  | +        write_test_files(state_path, state_value)
 | ||||||
|  | +        if (get_test_state(state_path) is 1):
 | ||||||
|  | +                if (not quiet):
 | ||||||
|  | +                        print("Testing = ON for device: {}"
 | ||||||
|  | +                              .format(state_path.split("/")[5]))
 | ||||||
|  | +        else:
 | ||||||
|  | +                if (not quiet):
 | ||||||
|  | +                        print("Testing = OFF for device: {}"
 | ||||||
|  | +                              .format(state_path.split("/")[5]))
 | ||||||
|  | +
 | ||||||
|  | +# get testing state of device
 | ||||||
|  | +def get_test_state(state_path):
 | ||||||
|  | +        #state == 1 - test = ON
 | ||||||
|  | +        #state == 0 - test = OFF
 | ||||||
|  | +        return  read_test_files(state_path)
 | ||||||
|  | +
 | ||||||
|  | +# write 1 - 1000 microseconds, into a single device using the
 | ||||||
|  | +# fuzz_test_buffer_interrupt_delay and fuzz_test_message_delay
 | ||||||
|  | +# debugfs attributes
 | ||||||
|  | +def set_delay_values(device, file_map, delay_length, quiet):
 | ||||||
|  | +
 | ||||||
|  | +        try:
 | ||||||
|  | +                interrupt = file_map[device][f_names.buff_f.value]
 | ||||||
|  | +                message = file_map[device][f_names.mess_f.value]
 | ||||||
|  | +
 | ||||||
|  | +                # delay[0]- buffer interrupt delay, delay[1]- message delay
 | ||||||
|  | +                if (delay_length[0] >= 0 and delay_length[0] <= 1000):
 | ||||||
|  | +                        write_test_files(interrupt, delay_length[0])
 | ||||||
|  | +                if (delay_length[1] >= 0 and delay_length[1] <= 1000):
 | ||||||
|  | +                        write_test_files(message, delay_length[1])
 | ||||||
|  | +                if (not quiet):
 | ||||||
|  | +                        print("Buffer delay testing = {} for: {}"
 | ||||||
|  | +                              .format(read_test_files(interrupt),
 | ||||||
|  | +                                      interrupt.split("/")[5]))
 | ||||||
|  | +                        print("Message delay testing = {} for: {}"
 | ||||||
|  | +                              .format(read_test_files(message),
 | ||||||
|  | +                                      message.split("/")[5]))
 | ||||||
|  | +        except IOError as e:
 | ||||||
|  | +                errno, strerror = e.args
 | ||||||
|  | +                print("I/O error({0}): {1} on files {2}{3}"
 | ||||||
|  | +                      .format(errno, strerror, interrupt, message))
 | ||||||
|  | +                exit(-1)
 | ||||||
|  | +
 | ||||||
|  | +# enabling delay testing on all devices
 | ||||||
|  | +def set_delay_all_devices(file_map, delay, quiet):
 | ||||||
|  | +
 | ||||||
|  | +        for device in (file_map):
 | ||||||
|  | +                set_test_state(locate_state(device, file_map),
 | ||||||
|  | +                               dev_state.on.value,
 | ||||||
|  | +                               quiet)
 | ||||||
|  | +                set_delay_values(device, file_map, delay, quiet)
 | ||||||
|  | +
 | ||||||
|  | +# disable all testing on a SINGLE device.
 | ||||||
|  | +def disable_testing_single_device(device, file_map):
 | ||||||
|  | +
 | ||||||
|  | +        for name in file_map[device]:
 | ||||||
|  | +                file_location = file_map[device][name]
 | ||||||
|  | +                write_test_files(file_location, dev_state.off.value)
 | ||||||
|  | +        print("ALL testing now OFF for {}".format(device.split("/")[-1]))
 | ||||||
|  | +
 | ||||||
|  | +# disable all testing on ALL devices
 | ||||||
|  | +def disable_all_testing(file_map):
 | ||||||
|  | +
 | ||||||
|  | +        for device in file_map:
 | ||||||
|  | +                disable_testing_single_device(device, file_map)
 | ||||||
|  | +
 | ||||||
|  | +def parse_args():
 | ||||||
|  | +        parser = argparse.ArgumentParser(prog = "vmbus_testing",usage ="\n"
 | ||||||
|  | +                "%(prog)s [delay]   [-h] [-e|-E] -t [-p]\n"
 | ||||||
|  | +                "%(prog)s [view_all       | V]      [-h]\n"
 | ||||||
|  | +                "%(prog)s [disable_all    | D]      [-h]\n"
 | ||||||
|  | +                "%(prog)s [disable_single | d]      [-h|-p]\n"
 | ||||||
|  | +                "%(prog)s [view_single    | v]      [-h|-p]\n"
 | ||||||
|  | +                "%(prog)s --version\n",
 | ||||||
|  | +                description = "\nUse lsvmbus to get vmbus device type "
 | ||||||
|  | +                "information.\n" "\nThe debugfs root path is "
 | ||||||
|  | +                "/sys/kernel/debug/hyperv",
 | ||||||
|  | +                formatter_class = RawDescriptionHelpFormatter)
 | ||||||
|  | +        subparsers = parser.add_subparsers(dest = "action")
 | ||||||
|  | +        parser.add_argument("--version", action = "version",
 | ||||||
|  | +                version = '%(prog)s 0.1.0')
 | ||||||
|  | +        parser.add_argument("-q","--quiet", action = "store_true",
 | ||||||
|  | +                help = "silence none important test messages."
 | ||||||
|  | +                       " This will only work when enabling testing"
 | ||||||
|  | +                       " on a device.")
 | ||||||
|  | +        # Use the path parser to hold the --path attribute so it can
 | ||||||
|  | +        # be shared between subparsers. Also do the same for the state
 | ||||||
|  | +        # parser, as all testing methods will use --enable_all and
 | ||||||
|  | +        # enable_single.
 | ||||||
|  | +        path_parser = argparse.ArgumentParser(add_help=False)
 | ||||||
|  | +        path_parser.add_argument("-p","--path", metavar = "",
 | ||||||
|  | +                help = "Debugfs path to a vmbus device. The path "
 | ||||||
|  | +                "must be the absolute path to the device.")
 | ||||||
|  | +        state_parser = argparse.ArgumentParser(add_help=False)
 | ||||||
|  | +        state_group = state_parser.add_mutually_exclusive_group(required = True)
 | ||||||
|  | +        state_group.add_argument("-E", "--enable_all", action = "store_const",
 | ||||||
|  | +                                 const = "enable_all",
 | ||||||
|  | +                                 help = "Enable the specified test type "
 | ||||||
|  | +                                 "on ALL vmbus devices.")
 | ||||||
|  | +        state_group.add_argument("-e", "--enable_single",
 | ||||||
|  | +                                 action = "store_const",
 | ||||||
|  | +                                 const = "enable_single",
 | ||||||
|  | +                                 help = "Enable the specified test type on a "
 | ||||||
|  | +                                 "SINGLE vmbus device.")
 | ||||||
|  | +        parser_delay = subparsers.add_parser("delay",
 | ||||||
|  | +                        parents = [state_parser, path_parser],
 | ||||||
|  | +                        help = "Delay the ring buffer interrupt or the "
 | ||||||
|  | +                        "ring buffer message reads in microseconds.",
 | ||||||
|  | +                        prog = "vmbus_testing",
 | ||||||
|  | +                        usage = "%(prog)s [-h]\n"
 | ||||||
|  | +                        "%(prog)s -E -t [value] [value]\n"
 | ||||||
|  | +                        "%(prog)s -e -t [value] [value] -p",
 | ||||||
|  | +                        description = "Delay the ring buffer interrupt for "
 | ||||||
|  | +                        "vmbus devices, or delay the ring buffer message "
 | ||||||
|  | +                        "reads for vmbus devices (both in microseconds). This "
 | ||||||
|  | +                        "is only on the host to guest channel.")
 | ||||||
|  | +        parser_delay.add_argument("-t", "--delay_time", metavar = "", nargs = 2,
 | ||||||
|  | +                        type = check_range, default =[0,0], required = (True),
 | ||||||
|  | +                        help = "Set [buffer] & [message] delay time. "
 | ||||||
|  | +                        "Value constraints: -1 == value "
 | ||||||
|  | +                        "or 0 < value <= 1000.\n"
 | ||||||
|  | +                        "Use -1 to keep the previous value for that delay "
 | ||||||
|  | +                        "type, or a value > 0 <= 1000 to change the delay "
 | ||||||
|  | +                        "time.")
 | ||||||
|  | +        parser_dis_all = subparsers.add_parser("disable_all",
 | ||||||
|  | +                        aliases = ['D'], prog = "vmbus_testing",
 | ||||||
|  | +                        usage = "%(prog)s [disable_all | D] -h\n"
 | ||||||
|  | +                        "%(prog)s [disable_all | D]\n",
 | ||||||
|  | +                        help = "Disable ALL testing on ALL vmbus devices.",
 | ||||||
|  | +                        description = "Disable ALL testing on ALL vmbus "
 | ||||||
|  | +                        "devices.")
 | ||||||
|  | +        parser_dis_single = subparsers.add_parser("disable_single",
 | ||||||
|  | +                        aliases = ['d'],
 | ||||||
|  | +                        parents = [path_parser], prog = "vmbus_testing",
 | ||||||
|  | +                        usage = "%(prog)s [disable_single | d] -h\n"
 | ||||||
|  | +                        "%(prog)s [disable_single | d] -p\n",
 | ||||||
|  | +                        help = "Disable ALL testing on a SINGLE vmbus device.",
 | ||||||
|  | +                        description = "Disable ALL testing on a SINGLE vmbus "
 | ||||||
|  | +                        "device.")
 | ||||||
|  | +        parser_view_all = subparsers.add_parser("view_all", aliases = ['V'],
 | ||||||
|  | +                        help = "View the test state for ALL vmbus devices.",
 | ||||||
|  | +                        prog = "vmbus_testing",
 | ||||||
|  | +                        usage = "%(prog)s [view_all | V] -h\n"
 | ||||||
|  | +                        "%(prog)s [view_all | V]\n",
 | ||||||
|  | +                        description = "This shows the test state for ALL the "
 | ||||||
|  | +                        "vmbus devices.")
 | ||||||
|  | +        parser_view_single = subparsers.add_parser("view_single",
 | ||||||
|  | +                        aliases = ['v'],parents = [path_parser],
 | ||||||
|  | +                        help = "View the test values for a SINGLE vmbus "
 | ||||||
|  | +                        "device.",
 | ||||||
|  | +                        description = "This shows the test values for a SINGLE "
 | ||||||
|  | +                        "vmbus device.", prog = "vmbus_testing",
 | ||||||
|  | +                        usage = "%(prog)s [view_single | v] -h\n"
 | ||||||
|  | +                        "%(prog)s [view_single | v] -p")
 | ||||||
|  | +
 | ||||||
|  | +        return  parser.parse_args()
 | ||||||
|  | +
 | ||||||
|  | +# value checking for range checking input in parser
 | ||||||
|  | +def check_range(arg1):
 | ||||||
|  | +
 | ||||||
|  | +        try:
 | ||||||
|  | +                val = int(arg1)
 | ||||||
|  | +        except ValueError as err:
 | ||||||
|  | +                raise argparse.ArgumentTypeError(str(err))
 | ||||||
|  | +        if val < -1 or val > 1000:
 | ||||||
|  | +                message = ("\n\nvalue must be -1 or  0 < value <= 1000. "
 | ||||||
|  | +                           "Value program received: {}\n").format(val)
 | ||||||
|  | +                raise argparse.ArgumentTypeError(message)
 | ||||||
|  | +        return val
 | ||||||
|  | +
 | ||||||
|  | +if __name__ == "__main__":
 | ||||||
|  | +        main()
 | ||||||
|  | -- 
 | ||||||
|  | 1.8.3.1 | ||||||
|  | 
 | ||||||
| @ -0,0 +1,685 @@ | |||||||
|  | From b0a20fac0e74b0b3eecc20ffe74006e7877da352 Mon Sep 17 00:00:00 2001 | ||||||
|  | From: Mohammed Gamal <mgamal@redhat.com> | ||||||
|  | Date: Wed, 15 Apr 2020 12:00:14 +0200 | ||||||
|  | Subject: [PATCH 1/2] Update C files and scripts to kernel version 5.7-rc1 | ||||||
|  | 
 | ||||||
|  | RH-Author: Mohammed Gamal <mgamal@redhat.com> | ||||||
|  | Message-id: <20200414183955.194006-2-mgamal@redhat.com> | ||||||
|  | Patchwork-id: 94689 | ||||||
|  | O-Subject: [RHEL8.3 virt hyperv-daemons PATCH v5 1/2] Update C files and scripts to kernel version 5.7-rc1 | ||||||
|  | Bugzilla: 1816750 | ||||||
|  | RH-Acked-by: Vitaly Kuznetsov <vkuznets@redhat.com> | ||||||
|  | RH-Acked-by: Cathy Avery <cavery@redhat.com> | ||||||
|  | 
 | ||||||
|  | Signed-off-by: Mohammed Gamal <mgamal@redhat.com> | ||||||
|  | ---
 | ||||||
|  |  hv_fcopy_daemon.c   |  38 ++++++++++++-- | ||||||
|  |  hv_get_dhcp_info.sh |   2 +- | ||||||
|  |  hv_kvp_daemon.c     |  63 ++++++++++++++--------- | ||||||
|  |  hv_set_ifconfig.sh  |   2 +- | ||||||
|  |  hv_vss_daemon.c     | 118 ++++++++++++++++++++++++++++++++++++++------ | ||||||
|  |  lsvmbus             |  75 +++++++++++++++------------- | ||||||
|  |  6 files changed, 220 insertions(+), 78 deletions(-) | ||||||
|  | 
 | ||||||
|  | Signed-off-by: Miroslav Rezanina <mrezanin@redhat.com> | ||||||
|  | ---
 | ||||||
|  |  hv_fcopy_daemon.c   |  38 ++++++++++++++--- | ||||||
|  |  hv_get_dhcp_info.sh |   2 +- | ||||||
|  |  hv_kvp_daemon.c     |  63 ++++++++++++++++++---------- | ||||||
|  |  hv_set_ifconfig.sh  |   2 +- | ||||||
|  |  hv_vss_daemon.c     | 118 +++++++++++++++++++++++++++++++++++++++++++++------- | ||||||
|  |  lsvmbus             |  75 ++++++++++++++++++--------------- | ||||||
|  |  6 files changed, 220 insertions(+), 78 deletions(-) | ||||||
|  | 
 | ||||||
|  | diff --git a/hv_fcopy_daemon.c b/hv_fcopy_daemon.c
 | ||||||
|  | index d78aed8..f40ddaf 100644
 | ||||||
|  | --- a/hv_fcopy_daemon.c
 | ||||||
|  | +++ b/hv_fcopy_daemon.c
 | ||||||
|  | @@ -89,6 +89,8 @@ static int hv_start_fcopy(struct hv_start_fcopy *smsg)
 | ||||||
|  |   | ||||||
|  |  	error = 0; | ||||||
|  |  done: | ||||||
|  | +	if (error)
 | ||||||
|  | +		target_fname[0] = '\0';
 | ||||||
|  |  	return error; | ||||||
|  |  } | ||||||
|  |   | ||||||
|  | @@ -117,15 +119,29 @@ static int hv_copy_data(struct hv_do_fcopy *cpmsg)
 | ||||||
|  |  	return ret; | ||||||
|  |  } | ||||||
|  |   | ||||||
|  | +/*
 | ||||||
|  | + * Reset target_fname to "" in the two below functions for hibernation: if
 | ||||||
|  | + * the fcopy operation is aborted by hibernation, the daemon should remove the
 | ||||||
|  | + * partially-copied file; to achieve this, the hv_utils driver always fakes a
 | ||||||
|  | + * CANCEL_FCOPY message upon suspend, and later when the VM resumes back,
 | ||||||
|  | + * the daemon calls hv_copy_cancel() to remove the file; if a file is copied
 | ||||||
|  | + * successfully before suspend, hv_copy_finished() must reset target_fname to
 | ||||||
|  | + * avoid that the file can be incorrectly removed upon resume, since the faked
 | ||||||
|  | + * CANCEL_FCOPY message is spurious in this case.
 | ||||||
|  | + */
 | ||||||
|  |  static int hv_copy_finished(void) | ||||||
|  |  { | ||||||
|  |  	close(target_fd); | ||||||
|  | +	target_fname[0] = '\0';
 | ||||||
|  |  	return 0; | ||||||
|  |  } | ||||||
|  |  static int hv_copy_cancel(void) | ||||||
|  |  { | ||||||
|  |  	close(target_fd); | ||||||
|  | -	unlink(target_fname);
 | ||||||
|  | +	if (strlen(target_fname) > 0) {
 | ||||||
|  | +		unlink(target_fname);
 | ||||||
|  | +		target_fname[0] = '\0';
 | ||||||
|  | +	}
 | ||||||
|  |  	return 0; | ||||||
|  |   | ||||||
|  |  } | ||||||
|  | @@ -140,7 +156,7 @@ void print_usage(char *argv[])
 | ||||||
|  |   | ||||||
|  |  int main(int argc, char *argv[]) | ||||||
|  |  { | ||||||
|  | -	int fcopy_fd;
 | ||||||
|  | +	int fcopy_fd = -1;
 | ||||||
|  |  	int error; | ||||||
|  |  	int daemonize = 1, long_index = 0, opt; | ||||||
|  |  	int version = FCOPY_CURRENT_VERSION; | ||||||
|  | @@ -150,7 +166,7 @@ int main(int argc, char *argv[])
 | ||||||
|  |  		struct hv_do_fcopy copy; | ||||||
|  |  		__u32 kernel_modver; | ||||||
|  |  	} buffer = { }; | ||||||
|  | -	int in_handshake = 1;
 | ||||||
|  | +	int in_handshake;
 | ||||||
|  |   | ||||||
|  |  	static struct option long_options[] = { | ||||||
|  |  		{"help",	no_argument,	   0,  'h' }, | ||||||
|  | @@ -179,6 +195,12 @@ int main(int argc, char *argv[])
 | ||||||
|  |  	openlog("HV_FCOPY", 0, LOG_USER); | ||||||
|  |  	syslog(LOG_INFO, "starting; pid is:%d", getpid()); | ||||||
|  |   | ||||||
|  | +reopen_fcopy_fd:
 | ||||||
|  | +	if (fcopy_fd != -1)
 | ||||||
|  | +		close(fcopy_fd);
 | ||||||
|  | +	/* Remove any possible partially-copied file on error */
 | ||||||
|  | +	hv_copy_cancel();
 | ||||||
|  | +	in_handshake = 1;
 | ||||||
|  |  	fcopy_fd = open("/dev/vmbus/hv_fcopy", O_RDWR); | ||||||
|  |   | ||||||
|  |  	if (fcopy_fd < 0) { | ||||||
|  | @@ -205,7 +227,7 @@ int main(int argc, char *argv[])
 | ||||||
|  |  		len = pread(fcopy_fd, &buffer, sizeof(buffer), 0); | ||||||
|  |  		if (len < 0) { | ||||||
|  |  			syslog(LOG_ERR, "pread failed: %s", strerror(errno)); | ||||||
|  | -			exit(EXIT_FAILURE);
 | ||||||
|  | +			goto reopen_fcopy_fd;
 | ||||||
|  |  		} | ||||||
|  |   | ||||||
|  |  		if (in_handshake) { | ||||||
|  | @@ -234,14 +256,20 @@ int main(int argc, char *argv[])
 | ||||||
|  |  			break; | ||||||
|  |   | ||||||
|  |  		default: | ||||||
|  | +			error = HV_E_FAIL;
 | ||||||
|  |  			syslog(LOG_ERR, "Unknown operation: %d", | ||||||
|  |  				buffer.hdr.operation); | ||||||
|  |   | ||||||
|  |  		} | ||||||
|  |   | ||||||
|  | +		/*
 | ||||||
|  | +		 * pwrite() may return an error due to the faked CANCEL_FCOPY
 | ||||||
|  | +		 * message upon hibernation. Ignore the error by resetting the
 | ||||||
|  | +		 * dev file, i.e. closing and re-opening it.
 | ||||||
|  | +		 */
 | ||||||
|  |  		if (pwrite(fcopy_fd, &error, sizeof(int), 0) != sizeof(int)) { | ||||||
|  |  			syslog(LOG_ERR, "pwrite failed: %s", strerror(errno)); | ||||||
|  | -			exit(EXIT_FAILURE);
 | ||||||
|  | +			goto reopen_fcopy_fd;
 | ||||||
|  |  		} | ||||||
|  |  	} | ||||||
|  |  } | ||||||
|  | diff --git a/hv_get_dhcp_info.sh b/hv_get_dhcp_info.sh
 | ||||||
|  | index c38686c..2f2a3c7 100644
 | ||||||
|  | --- a/hv_get_dhcp_info.sh
 | ||||||
|  | +++ b/hv_get_dhcp_info.sh
 | ||||||
|  | @@ -13,7 +13,7 @@
 | ||||||
|  |  #	the script prints the string "Disabled" to stdout. | ||||||
|  |  # | ||||||
|  |  # Each Distro is expected to implement this script in a distro specific | ||||||
|  | -# fashion. For instance on Distros that ship with Network Manager enabled,
 | ||||||
|  | +# fashion. For instance, on Distros that ship with Network Manager enabled,
 | ||||||
|  |  # this script can be based on the Network Manager APIs for retrieving DHCP | ||||||
|  |  # information. | ||||||
|  |   | ||||||
|  | diff --git a/hv_kvp_daemon.c b/hv_kvp_daemon.c
 | ||||||
|  | index dbf6e8b..ee9c1bb 100644
 | ||||||
|  | --- a/hv_kvp_daemon.c
 | ||||||
|  | +++ b/hv_kvp_daemon.c
 | ||||||
|  | @@ -76,7 +76,7 @@ enum {
 | ||||||
|  |  	DNS | ||||||
|  |  }; | ||||||
|  |   | ||||||
|  | -static int in_hand_shake = 1;
 | ||||||
|  | +static int in_hand_shake;
 | ||||||
|  |   | ||||||
|  |  static char *os_name = ""; | ||||||
|  |  static char *os_major = ""; | ||||||
|  | @@ -286,7 +286,7 @@ static int kvp_key_delete(int pool, const __u8 *key, int key_size)
 | ||||||
|  |  		 * Found a match; just move the remaining | ||||||
|  |  		 * entries up. | ||||||
|  |  		 */ | ||||||
|  | -		if (i == num_records) {
 | ||||||
|  | +		if (i == (num_records - 1)) {
 | ||||||
|  |  			kvp_file_info[pool].num_records--; | ||||||
|  |  			kvp_update_file(pool); | ||||||
|  |  			return 0; | ||||||
|  | @@ -700,7 +700,7 @@ static void kvp_get_ipconfig_info(char *if_name,
 | ||||||
|  |   | ||||||
|  |   | ||||||
|  |  	/* | ||||||
|  | -	 * Gather the DNS  state.
 | ||||||
|  | +	 * Gather the DNS state.
 | ||||||
|  |  	 * Since there is no standard way to get this information | ||||||
|  |  	 * across various distributions of interest; we just invoke | ||||||
|  |  	 * an external script that needs to be ported across distros | ||||||
|  | @@ -809,7 +809,7 @@ kvp_get_ip_info(int family, char *if_name, int op,
 | ||||||
|  |  	int sn_offset = 0; | ||||||
|  |  	int error = 0; | ||||||
|  |  	char *buffer; | ||||||
|  | -	struct hv_kvp_ipaddr_value *ip_buffer;
 | ||||||
|  | +	struct hv_kvp_ipaddr_value *ip_buffer = NULL;
 | ||||||
|  |  	char cidr_mask[5]; /* /xyz */ | ||||||
|  |  	int weight; | ||||||
|  |  	int i; | ||||||
|  | @@ -1051,7 +1051,7 @@ static int parse_ip_val_buffer(char *in_buf, int *offset,
 | ||||||
|  |  	char *start; | ||||||
|  |   | ||||||
|  |  	/* | ||||||
|  | -	 * in_buf has sequence of characters that are seperated by
 | ||||||
|  | +	 * in_buf has sequence of characters that are separated by
 | ||||||
|  |  	 * the character ';'. The last sequence does not have the | ||||||
|  |  	 * terminating ";" character. | ||||||
|  |  	 */ | ||||||
|  | @@ -1178,6 +1178,7 @@ static int kvp_set_ip_info(char *if_name, struct hv_kvp_ipaddr_value *new_val)
 | ||||||
|  |  	FILE *file; | ||||||
|  |  	char cmd[PATH_MAX]; | ||||||
|  |  	char *mac_addr; | ||||||
|  | +	int str_len;
 | ||||||
|  |   | ||||||
|  |  	/* | ||||||
|  |  	 * Set the configuration for the specified interface with | ||||||
|  | @@ -1301,8 +1302,18 @@ static int kvp_set_ip_info(char *if_name, struct hv_kvp_ipaddr_value *new_val)
 | ||||||
|  |  	 * invoke the external script to do its magic. | ||||||
|  |  	 */ | ||||||
|  |   | ||||||
|  | -	snprintf(cmd, sizeof(cmd), KVP_SCRIPTS_PATH "%s %s",
 | ||||||
|  | -		 "hv_set_ifconfig", if_file);
 | ||||||
|  | +	str_len = snprintf(cmd, sizeof(cmd), KVP_SCRIPTS_PATH "%s %s",
 | ||||||
|  | +			   "hv_set_ifconfig", if_file);
 | ||||||
|  | +	/*
 | ||||||
|  | +	 * This is a little overcautious, but it's necessary to suppress some
 | ||||||
|  | +	 * false warnings from gcc 8.0.1.
 | ||||||
|  | +	 */
 | ||||||
|  | +	if (str_len <= 0 || (unsigned int)str_len >= sizeof(cmd)) {
 | ||||||
|  | +		syslog(LOG_ERR, "Cmd '%s' (len=%d) may be too long",
 | ||||||
|  | +		       cmd, str_len);
 | ||||||
|  | +		return HV_E_FAIL;
 | ||||||
|  | +	}
 | ||||||
|  | +
 | ||||||
|  |  	if (system(cmd)) { | ||||||
|  |  		syslog(LOG_ERR, "Failed to execute cmd '%s'; error: %d %s", | ||||||
|  |  				cmd, errno, strerror(errno)); | ||||||
|  | @@ -1349,7 +1360,7 @@ void print_usage(char *argv[])
 | ||||||
|  |   | ||||||
|  |  int main(int argc, char *argv[]) | ||||||
|  |  { | ||||||
|  | -	int kvp_fd, len;
 | ||||||
|  | +	int kvp_fd = -1, len;
 | ||||||
|  |  	int error; | ||||||
|  |  	struct pollfd pfd; | ||||||
|  |  	char    *p; | ||||||
|  | @@ -1375,6 +1386,8 @@ int main(int argc, char *argv[])
 | ||||||
|  |  			daemonize = 0; | ||||||
|  |  			break; | ||||||
|  |  		case 'h': | ||||||
|  | +			print_usage(argv);
 | ||||||
|  | +			exit(0);
 | ||||||
|  |  		default: | ||||||
|  |  			print_usage(argv); | ||||||
|  |  			exit(EXIT_FAILURE); | ||||||
|  | @@ -1387,14 +1400,6 @@ int main(int argc, char *argv[])
 | ||||||
|  |  	openlog("KVP", 0, LOG_USER); | ||||||
|  |  	syslog(LOG_INFO, "KVP starting; pid is:%d", getpid()); | ||||||
|  |   | ||||||
|  | -	kvp_fd = open("/dev/vmbus/hv_kvp", O_RDWR | O_CLOEXEC);
 | ||||||
|  | -
 | ||||||
|  | -	if (kvp_fd < 0) {
 | ||||||
|  | -		syslog(LOG_ERR, "open /dev/vmbus/hv_kvp failed; error: %d %s",
 | ||||||
|  | -			errno, strerror(errno));
 | ||||||
|  | -		exit(EXIT_FAILURE);
 | ||||||
|  | -	}
 | ||||||
|  | -
 | ||||||
|  |  	/* | ||||||
|  |  	 * Retrieve OS release information. | ||||||
|  |  	 */ | ||||||
|  | @@ -1410,6 +1415,18 @@ int main(int argc, char *argv[])
 | ||||||
|  |  		exit(EXIT_FAILURE); | ||||||
|  |  	} | ||||||
|  |   | ||||||
|  | +reopen_kvp_fd:
 | ||||||
|  | +	if (kvp_fd != -1)
 | ||||||
|  | +		close(kvp_fd);
 | ||||||
|  | +	in_hand_shake = 1;
 | ||||||
|  | +	kvp_fd = open("/dev/vmbus/hv_kvp", O_RDWR | O_CLOEXEC);
 | ||||||
|  | +
 | ||||||
|  | +	if (kvp_fd < 0) {
 | ||||||
|  | +		syslog(LOG_ERR, "open /dev/vmbus/hv_kvp failed; error: %d %s",
 | ||||||
|  | +		       errno, strerror(errno));
 | ||||||
|  | +		exit(EXIT_FAILURE);
 | ||||||
|  | +	}
 | ||||||
|  | +
 | ||||||
|  |  	/* | ||||||
|  |  	 * Register ourselves with the kernel. | ||||||
|  |  	 */ | ||||||
|  | @@ -1443,9 +1460,7 @@ int main(int argc, char *argv[])
 | ||||||
|  |  		if (len != sizeof(struct hv_kvp_msg)) { | ||||||
|  |  			syslog(LOG_ERR, "read failed; error:%d %s", | ||||||
|  |  			       errno, strerror(errno)); | ||||||
|  | -
 | ||||||
|  | -			close(kvp_fd);
 | ||||||
|  | -			return EXIT_FAILURE;
 | ||||||
|  | +			goto reopen_kvp_fd;
 | ||||||
|  |  		} | ||||||
|  |   | ||||||
|  |  		/* | ||||||
|  | @@ -1479,7 +1494,7 @@ int main(int argc, char *argv[])
 | ||||||
|  |  		case KVP_OP_GET_IP_INFO: | ||||||
|  |  			kvp_ip_val = &hv_msg->body.kvp_ip_val; | ||||||
|  |   | ||||||
|  | -			error =  kvp_mac_to_ip(kvp_ip_val);
 | ||||||
|  | +			error = kvp_mac_to_ip(kvp_ip_val);
 | ||||||
|  |   | ||||||
|  |  			if (error) | ||||||
|  |  				hv_msg->error = error; | ||||||
|  | @@ -1604,13 +1619,17 @@ int main(int argc, char *argv[])
 | ||||||
|  |  			break; | ||||||
|  |  		} | ||||||
|  |   | ||||||
|  | -		/* Send the value back to the kernel. */
 | ||||||
|  | +		/*
 | ||||||
|  | +		 * Send the value back to the kernel. Note: the write() may
 | ||||||
|  | +		 * return an error due to hibernation; we can ignore the error
 | ||||||
|  | +		 * by resetting the dev file, i.e. closing and re-opening it.
 | ||||||
|  | +		 */
 | ||||||
|  |  kvp_done: | ||||||
|  |  		len = write(kvp_fd, hv_msg, sizeof(struct hv_kvp_msg)); | ||||||
|  |  		if (len != sizeof(struct hv_kvp_msg)) { | ||||||
|  |  			syslog(LOG_ERR, "write failed; error: %d %s", errno, | ||||||
|  |  			       strerror(errno)); | ||||||
|  | -			exit(EXIT_FAILURE);
 | ||||||
|  | +			goto reopen_kvp_fd;
 | ||||||
|  |  		} | ||||||
|  |  	} | ||||||
|  |   | ||||||
|  | diff --git a/hv_set_ifconfig.sh b/hv_set_ifconfig.sh
 | ||||||
|  | index 18b27cc..3dd064c 100644
 | ||||||
|  | --- a/hv_set_ifconfig.sh
 | ||||||
|  | +++ b/hv_set_ifconfig.sh
 | ||||||
|  | @@ -12,7 +12,7 @@
 | ||||||
|  |  # be used to configure the interface. | ||||||
|  |  # | ||||||
|  |  # Each Distro is expected to implement this script in a distro specific | ||||||
|  | -# fashion. For instance on Distros that ship with Network Manager enabled,
 | ||||||
|  | +# fashion. For instance, on Distros that ship with Network Manager enabled,
 | ||||||
|  |  # this script can be based on the Network Manager APIs for configuring the | ||||||
|  |  # interface. | ||||||
|  |  # | ||||||
|  | diff --git a/hv_vss_daemon.c b/hv_vss_daemon.c
 | ||||||
|  | index 34031a2..8fe0a5c 100644
 | ||||||
|  | --- a/hv_vss_daemon.c
 | ||||||
|  | +++ b/hv_vss_daemon.c
 | ||||||
|  | @@ -36,6 +36,10 @@
 | ||||||
|  |  #include <linux/hyperv.h> | ||||||
|  |  #include <syslog.h> | ||||||
|  |  #include <getopt.h> | ||||||
|  | +#include <stdbool.h>
 | ||||||
|  | +#include <dirent.h>
 | ||||||
|  | +
 | ||||||
|  | +static bool fs_frozen;
 | ||||||
|  |   | ||||||
|  |  /* Don't use syslog() in the function since that can cause write to disk */ | ||||||
|  |  static int vss_do_freeze(char *dir, unsigned int cmd) | ||||||
|  | @@ -51,7 +55,7 @@ static int vss_do_freeze(char *dir, unsigned int cmd)
 | ||||||
|  |  	 * If a partition is mounted more than once, only the first | ||||||
|  |  	 * FREEZE/THAW can succeed and the later ones will get | ||||||
|  |  	 * EBUSY/EINVAL respectively: there could be 2 cases: | ||||||
|  | -	 * 1) a user may mount the same partition to differnt directories
 | ||||||
|  | +	 * 1) a user may mount the same partition to different directories
 | ||||||
|  |  	 *  by mistake or on purpose; | ||||||
|  |  	 * 2) The subvolume of btrfs appears to have the same partition | ||||||
|  |  	 * mounted more than once. | ||||||
|  | @@ -68,6 +72,55 @@ static int vss_do_freeze(char *dir, unsigned int cmd)
 | ||||||
|  |  	return !!ret; | ||||||
|  |  } | ||||||
|  |   | ||||||
|  | +static bool is_dev_loop(const char *blkname)
 | ||||||
|  | +{
 | ||||||
|  | +	char *buffer;
 | ||||||
|  | +	DIR *dir;
 | ||||||
|  | +	struct dirent *entry;
 | ||||||
|  | +	bool ret = false;
 | ||||||
|  | +
 | ||||||
|  | +	buffer = malloc(PATH_MAX);
 | ||||||
|  | +	if (!buffer) {
 | ||||||
|  | +		syslog(LOG_ERR, "Can't allocate memory!");
 | ||||||
|  | +		exit(1);
 | ||||||
|  | +	}
 | ||||||
|  | +
 | ||||||
|  | +	snprintf(buffer, PATH_MAX, "%s/loop", blkname);
 | ||||||
|  | +	if (!access(buffer, R_OK | X_OK)) {
 | ||||||
|  | +		ret = true;
 | ||||||
|  | +		goto free_buffer;
 | ||||||
|  | +	} else if (errno != ENOENT) {
 | ||||||
|  | +		syslog(LOG_ERR, "Can't access: %s; error:%d %s!",
 | ||||||
|  | +		       buffer, errno, strerror(errno));
 | ||||||
|  | +	}
 | ||||||
|  | +
 | ||||||
|  | +	snprintf(buffer, PATH_MAX, "%s/slaves", blkname);
 | ||||||
|  | +	dir = opendir(buffer);
 | ||||||
|  | +	if (!dir) {
 | ||||||
|  | +		if (errno != ENOENT)
 | ||||||
|  | +			syslog(LOG_ERR, "Can't opendir: %s; error:%d %s!",
 | ||||||
|  | +			       buffer, errno, strerror(errno));
 | ||||||
|  | +		goto free_buffer;
 | ||||||
|  | +	}
 | ||||||
|  | +
 | ||||||
|  | +	while ((entry = readdir(dir)) != NULL) {
 | ||||||
|  | +		if (strcmp(entry->d_name, ".") == 0 ||
 | ||||||
|  | +		    strcmp(entry->d_name, "..") == 0)
 | ||||||
|  | +			continue;
 | ||||||
|  | +
 | ||||||
|  | +		snprintf(buffer, PATH_MAX, "%s/slaves/%s", blkname,
 | ||||||
|  | +			 entry->d_name);
 | ||||||
|  | +		if (is_dev_loop(buffer)) {
 | ||||||
|  | +			ret = true;
 | ||||||
|  | +			break;
 | ||||||
|  | +		}
 | ||||||
|  | +	}
 | ||||||
|  | +	closedir(dir);
 | ||||||
|  | +free_buffer:
 | ||||||
|  | +	free(buffer);
 | ||||||
|  | +	return ret;
 | ||||||
|  | +}
 | ||||||
|  | +
 | ||||||
|  |  static int vss_operate(int operation) | ||||||
|  |  { | ||||||
|  |  	char match[] = "/dev/"; | ||||||
|  | @@ -75,6 +128,7 @@ static int vss_operate(int operation)
 | ||||||
|  |  	struct mntent *ent; | ||||||
|  |  	struct stat sb; | ||||||
|  |  	char errdir[1024] = {0}; | ||||||
|  | +	char blkdir[23]; /* /sys/dev/block/XXX:XXX */
 | ||||||
|  |  	unsigned int cmd; | ||||||
|  |  	int error = 0, root_seen = 0, save_errno = 0; | ||||||
|  |   | ||||||
|  | @@ -96,10 +150,15 @@ static int vss_operate(int operation)
 | ||||||
|  |  	while ((ent = getmntent(mounts))) { | ||||||
|  |  		if (strncmp(ent->mnt_fsname, match, strlen(match))) | ||||||
|  |  			continue; | ||||||
|  | -		if (stat(ent->mnt_fsname, &sb) == -1)
 | ||||||
|  | -			continue;
 | ||||||
|  | -		if (S_ISBLK(sb.st_mode) && major(sb.st_rdev) == LOOP_MAJOR)
 | ||||||
|  | -			continue;
 | ||||||
|  | +		if (stat(ent->mnt_fsname, &sb)) {
 | ||||||
|  | +			syslog(LOG_ERR, "Can't stat: %s; error:%d %s!",
 | ||||||
|  | +			       ent->mnt_fsname, errno, strerror(errno));
 | ||||||
|  | +		} else {
 | ||||||
|  | +			sprintf(blkdir, "/sys/dev/block/%d:%d",
 | ||||||
|  | +				major(sb.st_rdev), minor(sb.st_rdev));
 | ||||||
|  | +			if (is_dev_loop(blkdir))
 | ||||||
|  | +				continue;
 | ||||||
|  | +		}
 | ||||||
|  |  		if (hasmntopt(ent, MNTOPT_RO) != NULL) | ||||||
|  |  			continue; | ||||||
|  |  		if (strcmp(ent->mnt_type, "vfat") == 0) | ||||||
|  | @@ -109,18 +168,27 @@ static int vss_operate(int operation)
 | ||||||
|  |  			continue; | ||||||
|  |  		} | ||||||
|  |  		error |= vss_do_freeze(ent->mnt_dir, cmd); | ||||||
|  | -		if (error && operation == VSS_OP_FREEZE)
 | ||||||
|  | -			goto err;
 | ||||||
|  | +		if (operation == VSS_OP_FREEZE) {
 | ||||||
|  | +			if (error)
 | ||||||
|  | +				goto err;
 | ||||||
|  | +			fs_frozen = true;
 | ||||||
|  | +		}
 | ||||||
|  |  	} | ||||||
|  |   | ||||||
|  |  	endmntent(mounts); | ||||||
|  |   | ||||||
|  |  	if (root_seen) { | ||||||
|  |  		error |= vss_do_freeze("/", cmd); | ||||||
|  | -		if (error && operation == VSS_OP_FREEZE)
 | ||||||
|  | -			goto err;
 | ||||||
|  | +		if (operation == VSS_OP_FREEZE) {
 | ||||||
|  | +			if (error)
 | ||||||
|  | +				goto err;
 | ||||||
|  | +			fs_frozen = true;
 | ||||||
|  | +		}
 | ||||||
|  |  	} | ||||||
|  |   | ||||||
|  | +	if (operation == VSS_OP_THAW && !error)
 | ||||||
|  | +		fs_frozen = false;
 | ||||||
|  | +
 | ||||||
|  |  	goto out; | ||||||
|  |  err: | ||||||
|  |  	save_errno = errno; | ||||||
|  | @@ -129,6 +197,7 @@ err:
 | ||||||
|  |  		endmntent(mounts); | ||||||
|  |  	} | ||||||
|  |  	vss_operate(VSS_OP_THAW); | ||||||
|  | +	fs_frozen = false;
 | ||||||
|  |  	/* Call syslog after we thaw all filesystems */ | ||||||
|  |  	if (ent) | ||||||
|  |  		syslog(LOG_ERR, "FREEZE of %s failed; error:%d %s", | ||||||
|  | @@ -150,13 +219,13 @@ void print_usage(char *argv[])
 | ||||||
|  |   | ||||||
|  |  int main(int argc, char *argv[]) | ||||||
|  |  { | ||||||
|  | -	int vss_fd, len;
 | ||||||
|  | +	int vss_fd = -1, len;
 | ||||||
|  |  	int error; | ||||||
|  |  	struct pollfd pfd; | ||||||
|  |  	int	op; | ||||||
|  |  	struct hv_vss_msg vss_msg[1]; | ||||||
|  |  	int daemonize = 1, long_index = 0, opt; | ||||||
|  | -	int in_handshake = 1;
 | ||||||
|  | +	int in_handshake;
 | ||||||
|  |  	__u32 kernel_modver; | ||||||
|  |   | ||||||
|  |  	static struct option long_options[] = { | ||||||
|  | @@ -172,6 +241,8 @@ int main(int argc, char *argv[])
 | ||||||
|  |  			daemonize = 0; | ||||||
|  |  			break; | ||||||
|  |  		case 'h': | ||||||
|  | +			print_usage(argv);
 | ||||||
|  | +			exit(0);
 | ||||||
|  |  		default: | ||||||
|  |  			print_usage(argv); | ||||||
|  |  			exit(EXIT_FAILURE); | ||||||
|  | @@ -184,6 +255,18 @@ int main(int argc, char *argv[])
 | ||||||
|  |  	openlog("Hyper-V VSS", 0, LOG_USER); | ||||||
|  |  	syslog(LOG_INFO, "VSS starting; pid is:%d", getpid()); | ||||||
|  |   | ||||||
|  | +reopen_vss_fd:
 | ||||||
|  | +	if (vss_fd != -1)
 | ||||||
|  | +		close(vss_fd);
 | ||||||
|  | +	if (fs_frozen) {
 | ||||||
|  | +		if (vss_operate(VSS_OP_THAW) || fs_frozen) {
 | ||||||
|  | +			syslog(LOG_ERR, "failed to thaw file system: err=%d",
 | ||||||
|  | +			       errno);
 | ||||||
|  | +			exit(EXIT_FAILURE);
 | ||||||
|  | +		}
 | ||||||
|  | +	}
 | ||||||
|  | +
 | ||||||
|  | +	in_handshake = 1;
 | ||||||
|  |  	vss_fd = open("/dev/vmbus/hv_vss", O_RDWR); | ||||||
|  |  	if (vss_fd < 0) { | ||||||
|  |  		syslog(LOG_ERR, "open /dev/vmbus/hv_vss failed; error: %d %s", | ||||||
|  | @@ -236,8 +319,7 @@ int main(int argc, char *argv[])
 | ||||||
|  |  		if (len != sizeof(struct hv_vss_msg)) { | ||||||
|  |  			syslog(LOG_ERR, "read failed; error:%d %s", | ||||||
|  |  			       errno, strerror(errno)); | ||||||
|  | -			close(vss_fd);
 | ||||||
|  | -			return EXIT_FAILURE;
 | ||||||
|  | +			goto reopen_vss_fd;
 | ||||||
|  |  		} | ||||||
|  |   | ||||||
|  |  		op = vss_msg->vss_hdr.operation; | ||||||
|  | @@ -264,14 +346,18 @@ int main(int argc, char *argv[])
 | ||||||
|  |  		default: | ||||||
|  |  			syslog(LOG_ERR, "Illegal op:%d\n", op); | ||||||
|  |  		} | ||||||
|  | +
 | ||||||
|  | +		/*
 | ||||||
|  | +		 * The write() may return an error due to the faked VSS_OP_THAW
 | ||||||
|  | +		 * message upon hibernation. Ignore the error by resetting the
 | ||||||
|  | +		 * dev file, i.e. closing and re-opening it.
 | ||||||
|  | +		 */
 | ||||||
|  |  		vss_msg->error = error; | ||||||
|  |  		len = write(vss_fd, vss_msg, sizeof(struct hv_vss_msg)); | ||||||
|  |  		if (len != sizeof(struct hv_vss_msg)) { | ||||||
|  |  			syslog(LOG_ERR, "write failed; error: %d %s", errno, | ||||||
|  |  			       strerror(errno)); | ||||||
|  | -
 | ||||||
|  | -			if (op == VSS_OP_FREEZE)
 | ||||||
|  | -				vss_operate(VSS_OP_THAW);
 | ||||||
|  | +			goto reopen_vss_fd;
 | ||||||
|  |  		} | ||||||
|  |  	} | ||||||
|  |   | ||||||
|  | diff --git a/lsvmbus b/lsvmbus
 | ||||||
|  | index 55e7374..099f2c4 100644
 | ||||||
|  | --- a/lsvmbus
 | ||||||
|  | +++ b/lsvmbus
 | ||||||
|  | @@ -4,10 +4,10 @@
 | ||||||
|  |  import os | ||||||
|  |  from optparse import OptionParser | ||||||
|  |   | ||||||
|  | +help_msg = "print verbose messages. Try -vv, -vvv for  more verbose messages"
 | ||||||
|  |  parser = OptionParser() | ||||||
|  | -parser.add_option("-v", "--verbose", dest="verbose",
 | ||||||
|  | -		   help="print verbose messages. Try -vv, -vvv for \
 | ||||||
|  | -			more verbose messages", action="count")
 | ||||||
|  | +parser.add_option(
 | ||||||
|  | +	"-v", "--verbose", dest="verbose", help=help_msg, action="count")
 | ||||||
|  |   | ||||||
|  |  (options, args) = parser.parse_args() | ||||||
|  |   | ||||||
|  | @@ -21,27 +21,28 @@ if not os.path.isdir(vmbus_sys_path):
 | ||||||
|  |  	exit(-1) | ||||||
|  |   | ||||||
|  |  vmbus_dev_dict = { | ||||||
|  | -	'{0e0b6031-5213-4934-818b-38d90ced39db}' : '[Operating system shutdown]',
 | ||||||
|  | -	'{9527e630-d0ae-497b-adce-e80ab0175caf}' : '[Time Synchronization]',
 | ||||||
|  | -	'{57164f39-9115-4e78-ab55-382f3bd5422d}' : '[Heartbeat]',
 | ||||||
|  | -	'{a9a0f4e7-5a45-4d96-b827-8a841e8c03e6}' : '[Data Exchange]',
 | ||||||
|  | -	'{35fa2e29-ea23-4236-96ae-3a6ebacba440}' : '[Backup (volume checkpoint)]',
 | ||||||
|  | -	'{34d14be3-dee4-41c8-9ae7-6b174977c192}' : '[Guest services]',
 | ||||||
|  | -	'{525074dc-8985-46e2-8057-a307dc18a502}' : '[Dynamic Memory]',
 | ||||||
|  | -	'{cfa8b69e-5b4a-4cc0-b98b-8ba1a1f3f95a}' : 'Synthetic mouse',
 | ||||||
|  | -	'{f912ad6d-2b17-48ea-bd65-f927a61c7684}' : 'Synthetic keyboard',
 | ||||||
|  | -	'{da0a7802-e377-4aac-8e77-0558eb1073f8}' : 'Synthetic framebuffer adapter',
 | ||||||
|  | -	'{f8615163-df3e-46c5-913f-f2d2f965ed0e}' : 'Synthetic network adapter',
 | ||||||
|  | -	'{32412632-86cb-44a2-9b5c-50d1417354f5}' : 'Synthetic IDE Controller',
 | ||||||
|  | -	'{ba6163d9-04a1-4d29-b605-72e2ffb1dc7f}' : 'Synthetic SCSI Controller',
 | ||||||
|  | -	'{2f9bcc4a-0069-4af3-b76b-6fd0be528cda}' : 'Synthetic fiber channel adapter',
 | ||||||
|  | -	'{8c2eaf3d-32a7-4b09-ab99-bd1f1c86b501}' : 'Synthetic RDMA adapter',
 | ||||||
|  | -	'{44c4f61d-4444-4400-9d52-802e27ede19f}' : 'PCI Express pass-through',
 | ||||||
|  | -	'{276aacf4-ac15-426c-98dd-7521ad3f01fe}' : '[Reserved system device]',
 | ||||||
|  | -	'{f8e65716-3cb3-4a06-9a60-1889c5cccab5}' : '[Reserved system device]',
 | ||||||
|  | -	'{3375baf4-9e15-4b30-b765-67acb10d607b}' : '[Reserved system device]',
 | ||||||
|  | +	'{0e0b6031-5213-4934-818b-38d90ced39db}': '[Operating system shutdown]',
 | ||||||
|  | +	'{9527e630-d0ae-497b-adce-e80ab0175caf}': '[Time Synchronization]',
 | ||||||
|  | +	'{57164f39-9115-4e78-ab55-382f3bd5422d}': '[Heartbeat]',
 | ||||||
|  | +	'{a9a0f4e7-5a45-4d96-b827-8a841e8c03e6}': '[Data Exchange]',
 | ||||||
|  | +	'{35fa2e29-ea23-4236-96ae-3a6ebacba440}': '[Backup (volume checkpoint)]',
 | ||||||
|  | +	'{34d14be3-dee4-41c8-9ae7-6b174977c192}': '[Guest services]',
 | ||||||
|  | +	'{525074dc-8985-46e2-8057-a307dc18a502}': '[Dynamic Memory]',
 | ||||||
|  | +	'{cfa8b69e-5b4a-4cc0-b98b-8ba1a1f3f95a}': 'Synthetic mouse',
 | ||||||
|  | +	'{f912ad6d-2b17-48ea-bd65-f927a61c7684}': 'Synthetic keyboard',
 | ||||||
|  | +	'{da0a7802-e377-4aac-8e77-0558eb1073f8}': 'Synthetic framebuffer adapter',
 | ||||||
|  | +	'{f8615163-df3e-46c5-913f-f2d2f965ed0e}': 'Synthetic network adapter',
 | ||||||
|  | +	'{32412632-86cb-44a2-9b5c-50d1417354f5}': 'Synthetic IDE Controller',
 | ||||||
|  | +	'{ba6163d9-04a1-4d29-b605-72e2ffb1dc7f}': 'Synthetic SCSI Controller',
 | ||||||
|  | +	'{2f9bcc4a-0069-4af3-b76b-6fd0be528cda}': 'Synthetic fiber channel adapter',
 | ||||||
|  | +	'{8c2eaf3d-32a7-4b09-ab99-bd1f1c86b501}': 'Synthetic RDMA adapter',
 | ||||||
|  | +	'{44c4f61d-4444-4400-9d52-802e27ede19f}': 'PCI Express pass-through',
 | ||||||
|  | +	'{276aacf4-ac15-426c-98dd-7521ad3f01fe}': '[Reserved system device]',
 | ||||||
|  | +	'{f8e65716-3cb3-4a06-9a60-1889c5cccab5}': '[Reserved system device]',
 | ||||||
|  | +	'{3375baf4-9e15-4b30-b765-67acb10d607b}': '[Reserved system device]',
 | ||||||
|  |  } | ||||||
|  |   | ||||||
|  | +
 | ||||||
|  |  def get_vmbus_dev_attr(dev_name, attr): | ||||||
|  |  	try: | ||||||
|  |  		f = open('%s/%s/%s' % (vmbus_sys_path, dev_name, attr), 'r') | ||||||
|  | @@ -52,6 +53,7 @@ def get_vmbus_dev_attr(dev_name, attr):
 | ||||||
|  |   | ||||||
|  |  	return lines | ||||||
|  |   | ||||||
|  | +
 | ||||||
|  |  class VMBus_Dev: | ||||||
|  |  	pass | ||||||
|  |   | ||||||
|  | @@ -66,12 +68,13 @@ for f in os.listdir(vmbus_sys_path):
 | ||||||
|  |   | ||||||
|  |  	chn_vp_mapping = get_vmbus_dev_attr(f, 'channel_vp_mapping') | ||||||
|  |  	chn_vp_mapping = [c.strip() for c in chn_vp_mapping] | ||||||
|  | -	chn_vp_mapping = sorted(chn_vp_mapping,
 | ||||||
|  | -		key = lambda c : int(c.split(':')[0]))
 | ||||||
|  | +	chn_vp_mapping = sorted(
 | ||||||
|  | +		chn_vp_mapping, key=lambda c: int(c.split(':')[0]))
 | ||||||
|  |   | ||||||
|  | -	chn_vp_mapping = ['\tRel_ID=%s, target_cpu=%s' %
 | ||||||
|  | -				(c.split(':')[0], c.split(':')[1])
 | ||||||
|  | -					for c in chn_vp_mapping]
 | ||||||
|  | +	chn_vp_mapping = [
 | ||||||
|  | +		'\tRel_ID=%s, target_cpu=%s' %
 | ||||||
|  | +		(c.split(':')[0], c.split(':')[1]) for c in chn_vp_mapping
 | ||||||
|  | +	]
 | ||||||
|  |  	d = VMBus_Dev() | ||||||
|  |  	d.sysfs_path = '%s/%s' % (vmbus_sys_path, f) | ||||||
|  |  	d.vmbus_id = vmbus_id | ||||||
|  | @@ -85,7 +88,7 @@ for f in os.listdir(vmbus_sys_path):
 | ||||||
|  |  	vmbus_dev_list.append(d) | ||||||
|  |   | ||||||
|  |   | ||||||
|  | -vmbus_dev_list  = sorted(vmbus_dev_list, key = lambda d : int(d.vmbus_id))
 | ||||||
|  | +vmbus_dev_list = sorted(vmbus_dev_list, key=lambda d: int(d.vmbus_id))
 | ||||||
|  |   | ||||||
|  |  format0 = '%2s: %s' | ||||||
|  |  format1 = '%2s: Class_ID = %s - %s\n%s' | ||||||
|  | @@ -95,9 +98,15 @@ for d in vmbus_dev_list:
 | ||||||
|  |  	if verbose == 0: | ||||||
|  |  		print(('VMBUS ID ' + format0) % (d.vmbus_id, d.dev_desc)) | ||||||
|  |  	elif verbose == 1: | ||||||
|  | -		print (('VMBUS ID ' + format1) %	\
 | ||||||
|  | -			(d.vmbus_id, d.class_id, d.dev_desc, d.chn_vp_mapping))
 | ||||||
|  | +		print(
 | ||||||
|  | +			('VMBUS ID ' + format1) %
 | ||||||
|  | +			(d.vmbus_id, d.class_id, d.dev_desc, d.chn_vp_mapping)
 | ||||||
|  | +		)
 | ||||||
|  |  	else: | ||||||
|  | -		print (('VMBUS ID ' + format2) % \
 | ||||||
|  | -			(d.vmbus_id, d.class_id, d.dev_desc, \
 | ||||||
|  | -			d.device_id, d.sysfs_path, d.chn_vp_mapping))
 | ||||||
|  | +		print(
 | ||||||
|  | +			('VMBUS ID ' + format2) %
 | ||||||
|  | +			(
 | ||||||
|  | +				d.vmbus_id, d.class_id, d.dev_desc,
 | ||||||
|  | +				d.device_id, d.sysfs_path, d.chn_vp_mapping
 | ||||||
|  | +			)
 | ||||||
|  | +		)
 | ||||||
|  | -- 
 | ||||||
|  | 1.8.3.1 | ||||||
|  | 
 | ||||||
| @ -13,7 +13,7 @@ | |||||||
| 
 | 
 | ||||||
| Name:     hyperv-daemons | Name:     hyperv-daemons | ||||||
| Version:  0 | Version:  0 | ||||||
| Release:  0.28%{?snapver}%{?dist} | Release:  0.29%{?snapver}%{?dist} | ||||||
| Summary:  Hyper-V daemons suite | Summary:  Hyper-V daemons suite | ||||||
| 
 | 
 | ||||||
| Group:    System Environment/Daemons | Group:    System Environment/Daemons | ||||||
| @ -52,6 +52,10 @@ Patch0: 0001-tools-hv-update-lsvmbus-to-be-compatible-with-python.patch | |||||||
| Patch1: 0002-hv_set_ifconfig_nm_enable.patch | Patch1: 0002-hv_set_ifconfig_nm_enable.patch | ||||||
| # For bz#1769920 - [Hyper-V][RHEL8] Running 'systemctl isolate' on any target stops hyperv-daemon services | # For bz#1769920 - [Hyper-V][RHEL8] Running 'systemctl isolate' on any target stops hyperv-daemon services | ||||||
| Patch2: hpvd-Set-IgnoreOnIsolate-1-in-systemd-units.patch | Patch2: hpvd-Set-IgnoreOnIsolate-1-in-systemd-units.patch | ||||||
|  | # For bz#1816750 - [Hyper-V][RHEL8.3] Update Hyper-V tools | ||||||
|  | Patch3: hvd-Update-C-files-and-scripts-to-kernel-version-5.7-rc1.patch | ||||||
|  | # For bz#1816750 - [Hyper-V][RHEL8.3] Update Hyper-V tools | ||||||
|  | Patch4: hvd-Add-vmbus_testing-tool-build-files.patch | ||||||
| 
 | 
 | ||||||
| # Hyper-V is available only on x86 architectures | # Hyper-V is available only on x86 architectures | ||||||
| # The base empty (a.k.a. virtual) package can not be noarch | # The base empty (a.k.a. virtual) package can not be noarch | ||||||
| @ -159,6 +163,8 @@ cp -pvL %{SOURCE301} lsvmbus | |||||||
| cp -pvL %{SOURCE4} hv_set_ifconfig.sh | cp -pvL %{SOURCE4} hv_set_ifconfig.sh | ||||||
| %patch1 -p0 -b .hv_set_ifconfig_nm_enable | %patch1 -p0 -b .hv_set_ifconfig_nm_enable | ||||||
| %patch2 -p1 | %patch2 -p1 | ||||||
|  | %patch3 -p1 | ||||||
|  | %patch4 -p1 | ||||||
| 
 | 
 | ||||||
| %build | %build | ||||||
| # HYPERV KVP DAEMON | # HYPERV KVP DAEMON | ||||||
| @ -201,6 +207,7 @@ mkdir -p %{buildroot}%{_sharedstatedir}/hyperv | |||||||
| # Tools | # Tools | ||||||
| install -p -m 0755 lsvmbus %{buildroot}%{_sbindir}/ | install -p -m 0755 lsvmbus %{buildroot}%{_sbindir}/ | ||||||
| sed -i 's,#!/usr/bin/env python,#!%{__python3},' %{buildroot}%{_sbindir}/lsvmbus | sed -i 's,#!/usr/bin/env python,#!%{__python3},' %{buildroot}%{_sbindir}/lsvmbus | ||||||
|  | install -p -m 0755 vmbus_testing %{buildroot}%{_sbindir}/ | ||||||
| 
 | 
 | ||||||
| %post -n hypervkvpd | %post -n hypervkvpd | ||||||
| if [ $1 -gt 1 ] ; then | if [ $1 -gt 1 ] ; then | ||||||
| @ -272,8 +279,15 @@ fi | |||||||
| 
 | 
 | ||||||
| %files -n hyperv-tools | %files -n hyperv-tools | ||||||
| %{_sbindir}/lsvmbus | %{_sbindir}/lsvmbus | ||||||
|  | %{_sbindir}/vmbus_testing | ||||||
| 
 | 
 | ||||||
| %changelog | %changelog | ||||||
|  | * Wed Apr 15 2020 Miroslav Rezanina <mrezanin@redhat.com> - 0-0.29.20180415git.el8 | ||||||
|  | - hvd-Update-C-files-and-scripts-to-kernel-version-5.7-rc1.patch [bz#1816750] | ||||||
|  | - hvd-Add-vmbus_testing-tool-build-files.patch [bz#1816750] | ||||||
|  | - Resolves: bz#1816750 | ||||||
|  |   ([Hyper-V][RHEL8.3] Update Hyper-V tools) | ||||||
|  | 
 | ||||||
| * Tue Nov 19 2019 Miroslav Rezanina <mrezanin@redhat.com> - 0-0.28.20180415git.el8 | * Tue Nov 19 2019 Miroslav Rezanina <mrezanin@redhat.com> - 0-0.28.20180415git.el8 | ||||||
| - hpvd-Set-IgnoreOnIsolate-1-in-systemd-units.patch [bz#1769920] | - hpvd-Set-IgnoreOnIsolate-1-in-systemd-units.patch [bz#1769920] | ||||||
| - Resolves: bz#1769920 | - Resolves: bz#1769920 | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user