2019-05-29 23:57:35 +00:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-only
|
2009-07-13 23:02:34 +00:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2009, Microsoft Corporation.
|
|
|
|
*
|
|
|
|
* Authors:
|
|
|
|
* Haiyang Zhang <haiyangz@microsoft.com>
|
|
|
|
* Hank Janssen <hjanssen@microsoft.com>
|
2011-04-29 20:45:15 +00:00
|
|
|
* K. Y. Srinivasan <kys@microsoft.com>
|
2009-07-13 23:02:34 +00:00
|
|
|
*/
|
2011-03-29 20:58:47 +00:00
|
|
|
#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
|
|
|
|
|
2009-07-13 23:02:34 +00:00
|
|
|
#include <linux/init.h>
|
|
|
|
#include <linux/module.h>
|
|
|
|
#include <linux/device.h>
|
|
|
|
#include <linux/interrupt.h>
|
|
|
|
#include <linux/sysctl.h>
|
include cleanup: Update gfp.h and slab.h includes to prepare for breaking implicit slab.h inclusion from percpu.h
percpu.h is included by sched.h and module.h and thus ends up being
included when building most .c files. percpu.h includes slab.h which
in turn includes gfp.h making everything defined by the two files
universally available and complicating inclusion dependencies.
percpu.h -> slab.h dependency is about to be removed. Prepare for
this change by updating users of gfp and slab facilities include those
headers directly instead of assuming availability. As this conversion
needs to touch large number of source files, the following script is
used as the basis of conversion.
http://userweb.kernel.org/~tj/misc/slabh-sweep.py
The script does the followings.
* Scan files for gfp and slab usages and update includes such that
only the necessary includes are there. ie. if only gfp is used,
gfp.h, if slab is used, slab.h.
* When the script inserts a new include, it looks at the include
blocks and try to put the new include such that its order conforms
to its surrounding. It's put in the include block which contains
core kernel includes, in the same order that the rest are ordered -
alphabetical, Christmas tree, rev-Xmas-tree or at the end if there
doesn't seem to be any matching order.
* If the script can't find a place to put a new include (mostly
because the file doesn't have fitting include block), it prints out
an error message indicating which .h file needs to be added to the
file.
The conversion was done in the following steps.
1. The initial automatic conversion of all .c files updated slightly
over 4000 files, deleting around 700 includes and adding ~480 gfp.h
and ~3000 slab.h inclusions. The script emitted errors for ~400
files.
2. Each error was manually checked. Some didn't need the inclusion,
some needed manual addition while adding it to implementation .h or
embedding .c file was more appropriate for others. This step added
inclusions to around 150 files.
3. The script was run again and the output was compared to the edits
from #2 to make sure no file was left behind.
4. Several build tests were done and a couple of problems were fixed.
e.g. lib/decompress_*.c used malloc/free() wrappers around slab
APIs requiring slab.h to be added manually.
5. The script was run on all .h files but without automatically
editing them as sprinkling gfp.h and slab.h inclusions around .h
files could easily lead to inclusion dependency hell. Most gfp.h
inclusion directives were ignored as stuff from gfp.h was usually
wildly available and often used in preprocessor macros. Each
slab.h inclusion directive was examined and added manually as
necessary.
6. percpu.h was updated not to include slab.h.
7. Build test were done on the following configurations and failures
were fixed. CONFIG_GCOV_KERNEL was turned off for all tests (as my
distributed build env didn't work with gcov compiles) and a few
more options had to be turned off depending on archs to make things
build (like ipr on powerpc/64 which failed due to missing writeq).
* x86 and x86_64 UP and SMP allmodconfig and a custom test config.
* powerpc and powerpc64 SMP allmodconfig
* sparc and sparc64 SMP allmodconfig
* ia64 SMP allmodconfig
* s390 SMP allmodconfig
* alpha SMP allmodconfig
* um on x86_64 SMP allmodconfig
8. percpu.h modifications were reverted so that it could be applied as
a separate patch and serve as bisection point.
Given the fact that I had only a couple of failures from tests on step
6, I'm fairly confident about the coverage of this conversion patch.
If there is a breakage, it's likely to be something in one of the arch
headers which should be easily discoverable easily on most builds of
the specific arch.
Signed-off-by: Tejun Heo <tj@kernel.org>
Guess-its-ok-by: Christoph Lameter <cl@linux-foundation.org>
Cc: Ingo Molnar <mingo@redhat.com>
Cc: Lee Schermerhorn <Lee.Schermerhorn@hp.com>
2010-03-24 08:04:11 +00:00
|
|
|
#include <linux/slab.h>
|
2011-04-29 20:45:15 +00:00
|
|
|
#include <linux/acpi.h>
|
2010-05-28 23:22:44 +00:00
|
|
|
#include <linux/completion.h>
|
2011-10-04 19:29:52 +00:00
|
|
|
#include <linux/hyperv.h>
|
2012-12-01 14:46:54 +00:00
|
|
|
#include <linux/kernel_stat.h>
|
2015-01-10 07:54:32 +00:00
|
|
|
#include <linux/clockchips.h>
|
2015-02-27 19:25:51 +00:00
|
|
|
#include <linux/cpu.h>
|
2017-02-08 17:51:37 +00:00
|
|
|
#include <linux/sched/task_stack.h>
|
|
|
|
|
2013-02-17 19:30:44 +00:00
|
|
|
#include <asm/mshyperv.h>
|
2019-09-05 23:01:20 +00:00
|
|
|
#include <linux/delay.h>
|
2015-02-28 19:39:01 +00:00
|
|
|
#include <linux/notifier.h>
|
|
|
|
#include <linux/ptrace.h>
|
2015-08-05 07:52:37 +00:00
|
|
|
#include <linux/screen_info.h>
|
2015-08-01 23:08:10 +00:00
|
|
|
#include <linux/kdebug.h>
|
2016-04-05 17:22:55 +00:00
|
|
|
#include <linux/efi.h>
|
2016-05-02 06:14:34 +00:00
|
|
|
#include <linux/random.h>
|
2020-04-06 15:53:31 +00:00
|
|
|
#include <linux/kernel.h>
|
2019-09-05 23:01:16 +00:00
|
|
|
#include <linux/syscore_ops.h>
|
2019-07-01 04:25:56 +00:00
|
|
|
#include <clocksource/hyperv_timer.h>
|
2011-05-13 02:34:28 +00:00
|
|
|
#include "hyperv_vmbus.h"
|
2009-07-13 23:02:34 +00:00
|
|
|
|
2016-12-03 20:34:39 +00:00
|
|
|
struct vmbus_dynid {
|
|
|
|
struct list_head node;
|
|
|
|
struct hv_vmbus_device_id id;
|
|
|
|
};
|
|
|
|
|
2011-06-06 22:49:39 +00:00
|
|
|
static struct acpi_device *hv_acpi_dev;
|
2011-03-15 22:03:32 +00:00
|
|
|
|
2011-04-29 20:45:04 +00:00
|
|
|
static struct completion probe_event;
|
2011-03-15 22:03:44 +00:00
|
|
|
|
2016-12-07 22:53:11 +00:00
|
|
|
static int hyperv_cpuhp_online;
|
2015-02-28 19:39:01 +00:00
|
|
|
|
2018-07-08 02:56:51 +00:00
|
|
|
static void *hv_panic_page;
|
|
|
|
|
2020-04-06 15:53:30 +00:00
|
|
|
/*
|
|
|
|
* Boolean to control whether to report panic messages over Hyper-V.
|
|
|
|
*
|
|
|
|
* It can be set via /proc/sys/kernel/hyperv/record_panic_msg
|
|
|
|
*/
|
|
|
|
static int sysctl_record_panic_msg = 1;
|
|
|
|
|
|
|
|
static int hyperv_report_reg(void)
|
|
|
|
{
|
|
|
|
return !sysctl_record_panic_msg || !hv_panic_page;
|
|
|
|
}
|
|
|
|
|
2015-08-01 23:08:10 +00:00
|
|
|
static int hyperv_panic_event(struct notifier_block *nb, unsigned long val,
|
|
|
|
void *args)
|
|
|
|
{
|
|
|
|
struct pt_regs *regs;
|
|
|
|
|
2020-04-06 15:53:26 +00:00
|
|
|
vmbus_initiate_unload(true);
|
2015-08-01 23:08:10 +00:00
|
|
|
|
2020-04-06 15:53:28 +00:00
|
|
|
/*
|
|
|
|
* Hyper-V should be notified only once about a panic. If we will be
|
|
|
|
* doing hyperv_report_panic_msg() later with kmsg data, don't do
|
|
|
|
* the notification here.
|
|
|
|
*/
|
|
|
|
if (ms_hyperv.misc_features & HV_FEATURE_GUEST_CRASH_MSR_AVAILABLE
|
2020-04-06 15:53:30 +00:00
|
|
|
&& hyperv_report_reg()) {
|
2020-04-06 15:53:26 +00:00
|
|
|
regs = current_pt_regs();
|
2020-04-06 15:53:31 +00:00
|
|
|
hyperv_report_panic(regs, val, false);
|
2020-04-06 15:53:26 +00:00
|
|
|
}
|
2015-02-28 19:39:01 +00:00
|
|
|
return NOTIFY_DONE;
|
|
|
|
}
|
|
|
|
|
2015-08-01 23:08:10 +00:00
|
|
|
static int hyperv_die_event(struct notifier_block *nb, unsigned long val,
|
|
|
|
void *args)
|
|
|
|
{
|
|
|
|
struct die_args *die = (struct die_args *)args;
|
|
|
|
struct pt_regs *regs = die->regs;
|
|
|
|
|
2020-04-06 15:53:28 +00:00
|
|
|
/*
|
|
|
|
* Hyper-V should be notified only once about a panic. If we will be
|
|
|
|
* doing hyperv_report_panic_msg() later with kmsg data, don't do
|
|
|
|
* the notification here.
|
|
|
|
*/
|
2020-04-06 15:53:30 +00:00
|
|
|
if (hyperv_report_reg())
|
2020-04-06 15:53:31 +00:00
|
|
|
hyperv_report_panic(regs, val, true);
|
2015-08-01 23:08:10 +00:00
|
|
|
return NOTIFY_DONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct notifier_block hyperv_die_block = {
|
|
|
|
.notifier_call = hyperv_die_event,
|
|
|
|
};
|
2015-02-28 19:39:01 +00:00
|
|
|
static struct notifier_block hyperv_panic_block = {
|
|
|
|
.notifier_call = hyperv_panic_event,
|
|
|
|
};
|
|
|
|
|
2016-04-05 17:22:55 +00:00
|
|
|
static const char *fb_mmio_name = "fb_range";
|
|
|
|
static struct resource *fb_mmio;
|
2016-09-07 12:39:33 +00:00
|
|
|
static struct resource *hyperv_mmio;
|
2019-11-01 20:00:04 +00:00
|
|
|
static DEFINE_MUTEX(hyperv_mmio_lock);
|
2011-03-15 22:03:44 +00:00
|
|
|
|
2011-12-01 17:59:34 +00:00
|
|
|
static int vmbus_exists(void)
|
|
|
|
{
|
|
|
|
if (hv_acpi_dev == NULL)
|
|
|
|
return -ENODEV;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-09-02 16:25:56 +00:00
|
|
|
#define VMBUS_ALIAS_LEN ((sizeof((struct hv_vmbus_device_id *)0)->guid) * 2)
|
|
|
|
static void print_alias_name(struct hv_device *hv_dev, char *alias_name)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
for (i = 0; i < VMBUS_ALIAS_LEN; i += 2)
|
|
|
|
sprintf(&alias_name[i], "%02x", hv_dev->dev_type.b[i/2]);
|
|
|
|
}
|
|
|
|
|
2017-09-22 03:58:49 +00:00
|
|
|
static u8 channel_monitor_group(const struct vmbus_channel *channel)
|
2013-09-13 18:32:56 +00:00
|
|
|
{
|
|
|
|
return (u8)channel->offermsg.monitorid / 32;
|
|
|
|
}
|
|
|
|
|
2017-09-22 03:58:49 +00:00
|
|
|
static u8 channel_monitor_offset(const struct vmbus_channel *channel)
|
2013-09-13 18:32:56 +00:00
|
|
|
{
|
|
|
|
return (u8)channel->offermsg.monitorid % 32;
|
|
|
|
}
|
|
|
|
|
2017-09-22 03:58:49 +00:00
|
|
|
static u32 channel_pending(const struct vmbus_channel *channel,
|
|
|
|
const struct hv_monitor_page *monitor_page)
|
2013-09-13 18:32:56 +00:00
|
|
|
{
|
|
|
|
u8 monitor_group = channel_monitor_group(channel);
|
2017-09-22 03:58:49 +00:00
|
|
|
|
2013-09-13 18:32:56 +00:00
|
|
|
return monitor_page->trigger_group[monitor_group].pending;
|
|
|
|
}
|
|
|
|
|
2017-09-22 03:58:49 +00:00
|
|
|
static u32 channel_latency(const struct vmbus_channel *channel,
|
|
|
|
const struct hv_monitor_page *monitor_page)
|
2013-09-13 18:32:57 +00:00
|
|
|
{
|
|
|
|
u8 monitor_group = channel_monitor_group(channel);
|
|
|
|
u8 monitor_offset = channel_monitor_offset(channel);
|
2017-09-22 03:58:49 +00:00
|
|
|
|
2013-09-13 18:32:57 +00:00
|
|
|
return monitor_page->latency[monitor_group][monitor_offset];
|
|
|
|
}
|
|
|
|
|
2013-09-13 18:32:58 +00:00
|
|
|
static u32 channel_conn_id(struct vmbus_channel *channel,
|
|
|
|
struct hv_monitor_page *monitor_page)
|
|
|
|
{
|
|
|
|
u8 monitor_group = channel_monitor_group(channel);
|
|
|
|
u8 monitor_offset = channel_monitor_offset(channel);
|
|
|
|
return monitor_page->parameter[monitor_group][monitor_offset].connectionid.u.id;
|
|
|
|
}
|
|
|
|
|
2013-09-13 18:32:49 +00:00
|
|
|
static ssize_t id_show(struct device *dev, struct device_attribute *dev_attr,
|
|
|
|
char *buf)
|
|
|
|
{
|
|
|
|
struct hv_device *hv_dev = device_to_hv_device(dev);
|
|
|
|
|
|
|
|
if (!hv_dev->channel)
|
|
|
|
return -ENODEV;
|
|
|
|
return sprintf(buf, "%d\n", hv_dev->channel->offermsg.child_relid);
|
|
|
|
}
|
|
|
|
static DEVICE_ATTR_RO(id);
|
|
|
|
|
2013-09-13 18:32:50 +00:00
|
|
|
static ssize_t state_show(struct device *dev, struct device_attribute *dev_attr,
|
|
|
|
char *buf)
|
|
|
|
{
|
|
|
|
struct hv_device *hv_dev = device_to_hv_device(dev);
|
|
|
|
|
|
|
|
if (!hv_dev->channel)
|
|
|
|
return -ENODEV;
|
|
|
|
return sprintf(buf, "%d\n", hv_dev->channel->state);
|
|
|
|
}
|
|
|
|
static DEVICE_ATTR_RO(state);
|
|
|
|
|
2013-09-13 18:32:51 +00:00
|
|
|
static ssize_t monitor_id_show(struct device *dev,
|
|
|
|
struct device_attribute *dev_attr, char *buf)
|
|
|
|
{
|
|
|
|
struct hv_device *hv_dev = device_to_hv_device(dev);
|
|
|
|
|
|
|
|
if (!hv_dev->channel)
|
|
|
|
return -ENODEV;
|
|
|
|
return sprintf(buf, "%d\n", hv_dev->channel->offermsg.monitorid);
|
|
|
|
}
|
|
|
|
static DEVICE_ATTR_RO(monitor_id);
|
|
|
|
|
2013-09-13 18:32:53 +00:00
|
|
|
static ssize_t class_id_show(struct device *dev,
|
|
|
|
struct device_attribute *dev_attr, char *buf)
|
|
|
|
{
|
|
|
|
struct hv_device *hv_dev = device_to_hv_device(dev);
|
|
|
|
|
|
|
|
if (!hv_dev->channel)
|
|
|
|
return -ENODEV;
|
|
|
|
return sprintf(buf, "{%pUl}\n",
|
|
|
|
hv_dev->channel->offermsg.offer.if_type.b);
|
|
|
|
}
|
|
|
|
static DEVICE_ATTR_RO(class_id);
|
|
|
|
|
2013-09-13 18:32:54 +00:00
|
|
|
static ssize_t device_id_show(struct device *dev,
|
|
|
|
struct device_attribute *dev_attr, char *buf)
|
|
|
|
{
|
|
|
|
struct hv_device *hv_dev = device_to_hv_device(dev);
|
|
|
|
|
|
|
|
if (!hv_dev->channel)
|
|
|
|
return -ENODEV;
|
|
|
|
return sprintf(buf, "{%pUl}\n",
|
|
|
|
hv_dev->channel->offermsg.offer.if_instance.b);
|
|
|
|
}
|
|
|
|
static DEVICE_ATTR_RO(device_id);
|
|
|
|
|
2013-09-13 18:32:52 +00:00
|
|
|
static ssize_t modalias_show(struct device *dev,
|
|
|
|
struct device_attribute *dev_attr, char *buf)
|
|
|
|
{
|
|
|
|
struct hv_device *hv_dev = device_to_hv_device(dev);
|
|
|
|
char alias_name[VMBUS_ALIAS_LEN + 1];
|
|
|
|
|
|
|
|
print_alias_name(hv_dev, alias_name);
|
|
|
|
return sprintf(buf, "vmbus:%s\n", alias_name);
|
|
|
|
}
|
|
|
|
static DEVICE_ATTR_RO(modalias);
|
|
|
|
|
2018-07-28 21:58:48 +00:00
|
|
|
#ifdef CONFIG_NUMA
|
|
|
|
static ssize_t numa_node_show(struct device *dev,
|
|
|
|
struct device_attribute *attr, char *buf)
|
|
|
|
{
|
|
|
|
struct hv_device *hv_dev = device_to_hv_device(dev);
|
|
|
|
|
|
|
|
if (!hv_dev->channel)
|
|
|
|
return -ENODEV;
|
|
|
|
|
|
|
|
return sprintf(buf, "%d\n", hv_dev->channel->numa_node);
|
|
|
|
}
|
|
|
|
static DEVICE_ATTR_RO(numa_node);
|
|
|
|
#endif
|
|
|
|
|
2013-09-13 18:32:56 +00:00
|
|
|
static ssize_t server_monitor_pending_show(struct device *dev,
|
|
|
|
struct device_attribute *dev_attr,
|
|
|
|
char *buf)
|
|
|
|
{
|
|
|
|
struct hv_device *hv_dev = device_to_hv_device(dev);
|
|
|
|
|
|
|
|
if (!hv_dev->channel)
|
|
|
|
return -ENODEV;
|
|
|
|
return sprintf(buf, "%d\n",
|
|
|
|
channel_pending(hv_dev->channel,
|
2019-02-19 05:38:06 +00:00
|
|
|
vmbus_connection.monitor_pages[0]));
|
2013-09-13 18:32:56 +00:00
|
|
|
}
|
|
|
|
static DEVICE_ATTR_RO(server_monitor_pending);
|
|
|
|
|
|
|
|
static ssize_t client_monitor_pending_show(struct device *dev,
|
|
|
|
struct device_attribute *dev_attr,
|
|
|
|
char *buf)
|
|
|
|
{
|
|
|
|
struct hv_device *hv_dev = device_to_hv_device(dev);
|
|
|
|
|
|
|
|
if (!hv_dev->channel)
|
|
|
|
return -ENODEV;
|
|
|
|
return sprintf(buf, "%d\n",
|
|
|
|
channel_pending(hv_dev->channel,
|
|
|
|
vmbus_connection.monitor_pages[1]));
|
|
|
|
}
|
|
|
|
static DEVICE_ATTR_RO(client_monitor_pending);
|
2013-09-13 18:32:53 +00:00
|
|
|
|
2013-09-13 18:32:57 +00:00
|
|
|
static ssize_t server_monitor_latency_show(struct device *dev,
|
|
|
|
struct device_attribute *dev_attr,
|
|
|
|
char *buf)
|
|
|
|
{
|
|
|
|
struct hv_device *hv_dev = device_to_hv_device(dev);
|
|
|
|
|
|
|
|
if (!hv_dev->channel)
|
|
|
|
return -ENODEV;
|
|
|
|
return sprintf(buf, "%d\n",
|
|
|
|
channel_latency(hv_dev->channel,
|
|
|
|
vmbus_connection.monitor_pages[0]));
|
|
|
|
}
|
|
|
|
static DEVICE_ATTR_RO(server_monitor_latency);
|
|
|
|
|
|
|
|
static ssize_t client_monitor_latency_show(struct device *dev,
|
|
|
|
struct device_attribute *dev_attr,
|
|
|
|
char *buf)
|
|
|
|
{
|
|
|
|
struct hv_device *hv_dev = device_to_hv_device(dev);
|
|
|
|
|
|
|
|
if (!hv_dev->channel)
|
|
|
|
return -ENODEV;
|
|
|
|
return sprintf(buf, "%d\n",
|
|
|
|
channel_latency(hv_dev->channel,
|
|
|
|
vmbus_connection.monitor_pages[1]));
|
|
|
|
}
|
|
|
|
static DEVICE_ATTR_RO(client_monitor_latency);
|
|
|
|
|
2013-09-13 18:32:58 +00:00
|
|
|
static ssize_t server_monitor_conn_id_show(struct device *dev,
|
|
|
|
struct device_attribute *dev_attr,
|
|
|
|
char *buf)
|
|
|
|
{
|
|
|
|
struct hv_device *hv_dev = device_to_hv_device(dev);
|
|
|
|
|
|
|
|
if (!hv_dev->channel)
|
|
|
|
return -ENODEV;
|
|
|
|
return sprintf(buf, "%d\n",
|
|
|
|
channel_conn_id(hv_dev->channel,
|
|
|
|
vmbus_connection.monitor_pages[0]));
|
|
|
|
}
|
|
|
|
static DEVICE_ATTR_RO(server_monitor_conn_id);
|
|
|
|
|
|
|
|
static ssize_t client_monitor_conn_id_show(struct device *dev,
|
|
|
|
struct device_attribute *dev_attr,
|
|
|
|
char *buf)
|
|
|
|
{
|
|
|
|
struct hv_device *hv_dev = device_to_hv_device(dev);
|
|
|
|
|
|
|
|
if (!hv_dev->channel)
|
|
|
|
return -ENODEV;
|
|
|
|
return sprintf(buf, "%d\n",
|
|
|
|
channel_conn_id(hv_dev->channel,
|
|
|
|
vmbus_connection.monitor_pages[1]));
|
|
|
|
}
|
|
|
|
static DEVICE_ATTR_RO(client_monitor_conn_id);
|
|
|
|
|
2013-09-13 18:33:01 +00:00
|
|
|
static ssize_t out_intr_mask_show(struct device *dev,
|
|
|
|
struct device_attribute *dev_attr, char *buf)
|
|
|
|
{
|
|
|
|
struct hv_device *hv_dev = device_to_hv_device(dev);
|
|
|
|
struct hv_ring_buffer_debug_info outbound;
|
2018-12-17 20:16:09 +00:00
|
|
|
int ret;
|
2013-09-13 18:33:01 +00:00
|
|
|
|
|
|
|
if (!hv_dev->channel)
|
|
|
|
return -ENODEV;
|
2018-12-17 20:16:09 +00:00
|
|
|
|
|
|
|
ret = hv_ringbuffer_get_debuginfo(&hv_dev->channel->outbound,
|
|
|
|
&outbound);
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
|
2013-09-13 18:33:01 +00:00
|
|
|
return sprintf(buf, "%d\n", outbound.current_interrupt_mask);
|
|
|
|
}
|
|
|
|
static DEVICE_ATTR_RO(out_intr_mask);
|
|
|
|
|
|
|
|
static ssize_t out_read_index_show(struct device *dev,
|
|
|
|
struct device_attribute *dev_attr, char *buf)
|
|
|
|
{
|
|
|
|
struct hv_device *hv_dev = device_to_hv_device(dev);
|
|
|
|
struct hv_ring_buffer_debug_info outbound;
|
2018-12-17 20:16:09 +00:00
|
|
|
int ret;
|
2013-09-13 18:33:01 +00:00
|
|
|
|
|
|
|
if (!hv_dev->channel)
|
|
|
|
return -ENODEV;
|
2018-12-17 20:16:09 +00:00
|
|
|
|
|
|
|
ret = hv_ringbuffer_get_debuginfo(&hv_dev->channel->outbound,
|
|
|
|
&outbound);
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
2013-09-13 18:33:01 +00:00
|
|
|
return sprintf(buf, "%d\n", outbound.current_read_index);
|
|
|
|
}
|
|
|
|
static DEVICE_ATTR_RO(out_read_index);
|
|
|
|
|
|
|
|
static ssize_t out_write_index_show(struct device *dev,
|
|
|
|
struct device_attribute *dev_attr,
|
|
|
|
char *buf)
|
|
|
|
{
|
|
|
|
struct hv_device *hv_dev = device_to_hv_device(dev);
|
|
|
|
struct hv_ring_buffer_debug_info outbound;
|
2018-12-17 20:16:09 +00:00
|
|
|
int ret;
|
2013-09-13 18:33:01 +00:00
|
|
|
|
|
|
|
if (!hv_dev->channel)
|
|
|
|
return -ENODEV;
|
2018-12-17 20:16:09 +00:00
|
|
|
|
|
|
|
ret = hv_ringbuffer_get_debuginfo(&hv_dev->channel->outbound,
|
|
|
|
&outbound);
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
2013-09-13 18:33:01 +00:00
|
|
|
return sprintf(buf, "%d\n", outbound.current_write_index);
|
|
|
|
}
|
|
|
|
static DEVICE_ATTR_RO(out_write_index);
|
|
|
|
|
|
|
|
static ssize_t out_read_bytes_avail_show(struct device *dev,
|
|
|
|
struct device_attribute *dev_attr,
|
|
|
|
char *buf)
|
|
|
|
{
|
|
|
|
struct hv_device *hv_dev = device_to_hv_device(dev);
|
|
|
|
struct hv_ring_buffer_debug_info outbound;
|
2018-12-17 20:16:09 +00:00
|
|
|
int ret;
|
2013-09-13 18:33:01 +00:00
|
|
|
|
|
|
|
if (!hv_dev->channel)
|
|
|
|
return -ENODEV;
|
2018-12-17 20:16:09 +00:00
|
|
|
|
|
|
|
ret = hv_ringbuffer_get_debuginfo(&hv_dev->channel->outbound,
|
|
|
|
&outbound);
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
2013-09-13 18:33:01 +00:00
|
|
|
return sprintf(buf, "%d\n", outbound.bytes_avail_toread);
|
|
|
|
}
|
|
|
|
static DEVICE_ATTR_RO(out_read_bytes_avail);
|
|
|
|
|
|
|
|
static ssize_t out_write_bytes_avail_show(struct device *dev,
|
|
|
|
struct device_attribute *dev_attr,
|
|
|
|
char *buf)
|
|
|
|
{
|
|
|
|
struct hv_device *hv_dev = device_to_hv_device(dev);
|
|
|
|
struct hv_ring_buffer_debug_info outbound;
|
2018-12-17 20:16:09 +00:00
|
|
|
int ret;
|
2013-09-13 18:33:01 +00:00
|
|
|
|
|
|
|
if (!hv_dev->channel)
|
|
|
|
return -ENODEV;
|
2018-12-17 20:16:09 +00:00
|
|
|
|
|
|
|
ret = hv_ringbuffer_get_debuginfo(&hv_dev->channel->outbound,
|
|
|
|
&outbound);
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
2013-09-13 18:33:01 +00:00
|
|
|
return sprintf(buf, "%d\n", outbound.bytes_avail_towrite);
|
|
|
|
}
|
|
|
|
static DEVICE_ATTR_RO(out_write_bytes_avail);
|
|
|
|
|
|
|
|
static ssize_t in_intr_mask_show(struct device *dev,
|
|
|
|
struct device_attribute *dev_attr, char *buf)
|
|
|
|
{
|
|
|
|
struct hv_device *hv_dev = device_to_hv_device(dev);
|
|
|
|
struct hv_ring_buffer_debug_info inbound;
|
2018-12-17 20:16:09 +00:00
|
|
|
int ret;
|
2013-09-13 18:33:01 +00:00
|
|
|
|
|
|
|
if (!hv_dev->channel)
|
|
|
|
return -ENODEV;
|
2018-12-17 20:16:09 +00:00
|
|
|
|
|
|
|
ret = hv_ringbuffer_get_debuginfo(&hv_dev->channel->inbound, &inbound);
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
|
2013-09-13 18:33:01 +00:00
|
|
|
return sprintf(buf, "%d\n", inbound.current_interrupt_mask);
|
|
|
|
}
|
|
|
|
static DEVICE_ATTR_RO(in_intr_mask);
|
|
|
|
|
|
|
|
static ssize_t in_read_index_show(struct device *dev,
|
|
|
|
struct device_attribute *dev_attr, char *buf)
|
|
|
|
{
|
|
|
|
struct hv_device *hv_dev = device_to_hv_device(dev);
|
|
|
|
struct hv_ring_buffer_debug_info inbound;
|
2018-12-17 20:16:09 +00:00
|
|
|
int ret;
|
2013-09-13 18:33:01 +00:00
|
|
|
|
|
|
|
if (!hv_dev->channel)
|
|
|
|
return -ENODEV;
|
2018-12-17 20:16:09 +00:00
|
|
|
|
|
|
|
ret = hv_ringbuffer_get_debuginfo(&hv_dev->channel->inbound, &inbound);
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
|
2013-09-13 18:33:01 +00:00
|
|
|
return sprintf(buf, "%d\n", inbound.current_read_index);
|
|
|
|
}
|
|
|
|
static DEVICE_ATTR_RO(in_read_index);
|
|
|
|
|
|
|
|
static ssize_t in_write_index_show(struct device *dev,
|
|
|
|
struct device_attribute *dev_attr, char *buf)
|
|
|
|
{
|
|
|
|
struct hv_device *hv_dev = device_to_hv_device(dev);
|
|
|
|
struct hv_ring_buffer_debug_info inbound;
|
2018-12-17 20:16:09 +00:00
|
|
|
int ret;
|
2013-09-13 18:33:01 +00:00
|
|
|
|
|
|
|
if (!hv_dev->channel)
|
|
|
|
return -ENODEV;
|
2018-12-17 20:16:09 +00:00
|
|
|
|
|
|
|
ret = hv_ringbuffer_get_debuginfo(&hv_dev->channel->inbound, &inbound);
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
|
2013-09-13 18:33:01 +00:00
|
|
|
return sprintf(buf, "%d\n", inbound.current_write_index);
|
|
|
|
}
|
|
|
|
static DEVICE_ATTR_RO(in_write_index);
|
|
|
|
|
|
|
|
static ssize_t in_read_bytes_avail_show(struct device *dev,
|
|
|
|
struct device_attribute *dev_attr,
|
|
|
|
char *buf)
|
|
|
|
{
|
|
|
|
struct hv_device *hv_dev = device_to_hv_device(dev);
|
|
|
|
struct hv_ring_buffer_debug_info inbound;
|
2018-12-17 20:16:09 +00:00
|
|
|
int ret;
|
2013-09-13 18:33:01 +00:00
|
|
|
|
|
|
|
if (!hv_dev->channel)
|
|
|
|
return -ENODEV;
|
2018-12-17 20:16:09 +00:00
|
|
|
|
|
|
|
ret = hv_ringbuffer_get_debuginfo(&hv_dev->channel->inbound, &inbound);
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
|
2013-09-13 18:33:01 +00:00
|
|
|
return sprintf(buf, "%d\n", inbound.bytes_avail_toread);
|
|
|
|
}
|
|
|
|
static DEVICE_ATTR_RO(in_read_bytes_avail);
|
|
|
|
|
|
|
|
static ssize_t in_write_bytes_avail_show(struct device *dev,
|
|
|
|
struct device_attribute *dev_attr,
|
|
|
|
char *buf)
|
|
|
|
{
|
|
|
|
struct hv_device *hv_dev = device_to_hv_device(dev);
|
|
|
|
struct hv_ring_buffer_debug_info inbound;
|
2018-12-17 20:16:09 +00:00
|
|
|
int ret;
|
2013-09-13 18:33:01 +00:00
|
|
|
|
|
|
|
if (!hv_dev->channel)
|
|
|
|
return -ENODEV;
|
2018-12-17 20:16:09 +00:00
|
|
|
|
|
|
|
ret = hv_ringbuffer_get_debuginfo(&hv_dev->channel->inbound, &inbound);
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
|
2013-09-13 18:33:01 +00:00
|
|
|
return sprintf(buf, "%d\n", inbound.bytes_avail_towrite);
|
|
|
|
}
|
|
|
|
static DEVICE_ATTR_RO(in_write_bytes_avail);
|
|
|
|
|
2015-08-05 07:52:43 +00:00
|
|
|
static ssize_t channel_vp_mapping_show(struct device *dev,
|
|
|
|
struct device_attribute *dev_attr,
|
|
|
|
char *buf)
|
|
|
|
{
|
|
|
|
struct hv_device *hv_dev = device_to_hv_device(dev);
|
|
|
|
struct vmbus_channel *channel = hv_dev->channel, *cur_sc;
|
|
|
|
unsigned long flags;
|
|
|
|
int buf_size = PAGE_SIZE, n_written, tot_written;
|
|
|
|
struct list_head *cur;
|
|
|
|
|
|
|
|
if (!channel)
|
|
|
|
return -ENODEV;
|
|
|
|
|
|
|
|
tot_written = snprintf(buf, buf_size, "%u:%u\n",
|
|
|
|
channel->offermsg.child_relid, channel->target_cpu);
|
|
|
|
|
|
|
|
spin_lock_irqsave(&channel->lock, flags);
|
|
|
|
|
|
|
|
list_for_each(cur, &channel->sc_list) {
|
|
|
|
if (tot_written >= buf_size - 1)
|
|
|
|
break;
|
|
|
|
|
|
|
|
cur_sc = list_entry(cur, struct vmbus_channel, sc_list);
|
|
|
|
n_written = scnprintf(buf + tot_written,
|
|
|
|
buf_size - tot_written,
|
|
|
|
"%u:%u\n",
|
|
|
|
cur_sc->offermsg.child_relid,
|
|
|
|
cur_sc->target_cpu);
|
|
|
|
tot_written += n_written;
|
|
|
|
}
|
|
|
|
|
|
|
|
spin_unlock_irqrestore(&channel->lock, flags);
|
|
|
|
|
|
|
|
return tot_written;
|
|
|
|
}
|
|
|
|
static DEVICE_ATTR_RO(channel_vp_mapping);
|
|
|
|
|
2015-12-26 04:00:30 +00:00
|
|
|
static ssize_t vendor_show(struct device *dev,
|
|
|
|
struct device_attribute *dev_attr,
|
|
|
|
char *buf)
|
|
|
|
{
|
|
|
|
struct hv_device *hv_dev = device_to_hv_device(dev);
|
|
|
|
return sprintf(buf, "0x%x\n", hv_dev->vendor_id);
|
|
|
|
}
|
|
|
|
static DEVICE_ATTR_RO(vendor);
|
|
|
|
|
|
|
|
static ssize_t device_show(struct device *dev,
|
|
|
|
struct device_attribute *dev_attr,
|
|
|
|
char *buf)
|
|
|
|
{
|
|
|
|
struct hv_device *hv_dev = device_to_hv_device(dev);
|
|
|
|
return sprintf(buf, "0x%x\n", hv_dev->device_id);
|
|
|
|
}
|
|
|
|
static DEVICE_ATTR_RO(device);
|
|
|
|
|
2018-08-10 23:06:08 +00:00
|
|
|
static ssize_t driver_override_store(struct device *dev,
|
|
|
|
struct device_attribute *attr,
|
|
|
|
const char *buf, size_t count)
|
|
|
|
{
|
|
|
|
struct hv_device *hv_dev = device_to_hv_device(dev);
|
|
|
|
char *driver_override, *old, *cp;
|
|
|
|
|
|
|
|
/* We need to keep extra room for a newline */
|
|
|
|
if (count >= (PAGE_SIZE - 1))
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
driver_override = kstrndup(buf, count, GFP_KERNEL);
|
|
|
|
if (!driver_override)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
cp = strchr(driver_override, '\n');
|
|
|
|
if (cp)
|
|
|
|
*cp = '\0';
|
|
|
|
|
|
|
|
device_lock(dev);
|
|
|
|
old = hv_dev->driver_override;
|
|
|
|
if (strlen(driver_override)) {
|
|
|
|
hv_dev->driver_override = driver_override;
|
|
|
|
} else {
|
|
|
|
kfree(driver_override);
|
|
|
|
hv_dev->driver_override = NULL;
|
|
|
|
}
|
|
|
|
device_unlock(dev);
|
|
|
|
|
|
|
|
kfree(old);
|
|
|
|
|
|
|
|
return count;
|
|
|
|
}
|
|
|
|
|
|
|
|
static ssize_t driver_override_show(struct device *dev,
|
|
|
|
struct device_attribute *attr, char *buf)
|
|
|
|
{
|
|
|
|
struct hv_device *hv_dev = device_to_hv_device(dev);
|
|
|
|
ssize_t len;
|
|
|
|
|
|
|
|
device_lock(dev);
|
|
|
|
len = snprintf(buf, PAGE_SIZE, "%s\n", hv_dev->driver_override);
|
|
|
|
device_unlock(dev);
|
|
|
|
|
|
|
|
return len;
|
|
|
|
}
|
|
|
|
static DEVICE_ATTR_RW(driver_override);
|
|
|
|
|
2013-09-13 18:33:01 +00:00
|
|
|
/* Set up per device attributes in /sys/bus/vmbus/devices/<bus device> */
|
2016-12-03 20:34:39 +00:00
|
|
|
static struct attribute *vmbus_dev_attrs[] = {
|
2013-09-13 18:32:49 +00:00
|
|
|
&dev_attr_id.attr,
|
2013-09-13 18:32:50 +00:00
|
|
|
&dev_attr_state.attr,
|
2013-09-13 18:32:51 +00:00
|
|
|
&dev_attr_monitor_id.attr,
|
2013-09-13 18:32:53 +00:00
|
|
|
&dev_attr_class_id.attr,
|
2013-09-13 18:32:54 +00:00
|
|
|
&dev_attr_device_id.attr,
|
2013-09-13 18:32:52 +00:00
|
|
|
&dev_attr_modalias.attr,
|
2018-07-28 21:58:48 +00:00
|
|
|
#ifdef CONFIG_NUMA
|
|
|
|
&dev_attr_numa_node.attr,
|
|
|
|
#endif
|
2013-09-13 18:32:56 +00:00
|
|
|
&dev_attr_server_monitor_pending.attr,
|
|
|
|
&dev_attr_client_monitor_pending.attr,
|
2013-09-13 18:32:57 +00:00
|
|
|
&dev_attr_server_monitor_latency.attr,
|
|
|
|
&dev_attr_client_monitor_latency.attr,
|
2013-09-13 18:32:58 +00:00
|
|
|
&dev_attr_server_monitor_conn_id.attr,
|
|
|
|
&dev_attr_client_monitor_conn_id.attr,
|
2013-09-13 18:33:01 +00:00
|
|
|
&dev_attr_out_intr_mask.attr,
|
|
|
|
&dev_attr_out_read_index.attr,
|
|
|
|
&dev_attr_out_write_index.attr,
|
|
|
|
&dev_attr_out_read_bytes_avail.attr,
|
|
|
|
&dev_attr_out_write_bytes_avail.attr,
|
|
|
|
&dev_attr_in_intr_mask.attr,
|
|
|
|
&dev_attr_in_read_index.attr,
|
|
|
|
&dev_attr_in_write_index.attr,
|
|
|
|
&dev_attr_in_read_bytes_avail.attr,
|
|
|
|
&dev_attr_in_write_bytes_avail.attr,
|
2015-08-05 07:52:43 +00:00
|
|
|
&dev_attr_channel_vp_mapping.attr,
|
2015-12-26 04:00:30 +00:00
|
|
|
&dev_attr_vendor.attr,
|
|
|
|
&dev_attr_device.attr,
|
2018-08-10 23:06:08 +00:00
|
|
|
&dev_attr_driver_override.attr,
|
2013-09-13 18:32:49 +00:00
|
|
|
NULL,
|
|
|
|
};
|
2019-03-19 04:04:01 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Device-level attribute_group callback function. Returns the permission for
|
|
|
|
* each attribute, and returns 0 if an attribute is not visible.
|
|
|
|
*/
|
|
|
|
static umode_t vmbus_dev_attr_is_visible(struct kobject *kobj,
|
|
|
|
struct attribute *attr, int idx)
|
|
|
|
{
|
|
|
|
struct device *dev = kobj_to_dev(kobj);
|
|
|
|
const struct hv_device *hv_dev = device_to_hv_device(dev);
|
|
|
|
|
|
|
|
/* Hide the monitor attributes if the monitor mechanism is not used. */
|
|
|
|
if (!hv_dev->channel->offermsg.monitor_allocated &&
|
|
|
|
(attr == &dev_attr_monitor_id.attr ||
|
|
|
|
attr == &dev_attr_server_monitor_pending.attr ||
|
|
|
|
attr == &dev_attr_client_monitor_pending.attr ||
|
|
|
|
attr == &dev_attr_server_monitor_latency.attr ||
|
|
|
|
attr == &dev_attr_client_monitor_latency.attr ||
|
|
|
|
attr == &dev_attr_server_monitor_conn_id.attr ||
|
|
|
|
attr == &dev_attr_client_monitor_conn_id.attr))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
return attr->mode;
|
|
|
|
}
|
|
|
|
|
|
|
|
static const struct attribute_group vmbus_dev_group = {
|
|
|
|
.attrs = vmbus_dev_attrs,
|
|
|
|
.is_visible = vmbus_dev_attr_is_visible
|
|
|
|
};
|
|
|
|
__ATTRIBUTE_GROUPS(vmbus_dev);
|
2013-09-13 18:32:49 +00:00
|
|
|
|
2011-03-15 22:03:37 +00:00
|
|
|
/*
|
|
|
|
* vmbus_uevent - add uevent for our device
|
|
|
|
*
|
|
|
|
* This routine is invoked when a device is added or removed on the vmbus to
|
|
|
|
* generate a uevent to udev in the userspace. The udev will then look at its
|
|
|
|
* rule and the uevent generated here to load the appropriate driver
|
2011-08-25 16:48:38 +00:00
|
|
|
*
|
|
|
|
* The alias string will be of the form vmbus:guid where guid is the string
|
|
|
|
* representation of the device guid (each byte of the guid will be
|
|
|
|
* represented with two hex characters.
|
2011-03-15 22:03:37 +00:00
|
|
|
*/
|
|
|
|
static int vmbus_uevent(struct device *device, struct kobj_uevent_env *env)
|
|
|
|
{
|
|
|
|
struct hv_device *dev = device_to_hv_device(device);
|
2011-09-02 16:25:56 +00:00
|
|
|
int ret;
|
|
|
|
char alias_name[VMBUS_ALIAS_LEN + 1];
|
2011-08-25 16:48:38 +00:00
|
|
|
|
2011-09-02 16:25:56 +00:00
|
|
|
print_alias_name(dev, alias_name);
|
2011-08-25 16:48:38 +00:00
|
|
|
ret = add_uevent_var(env, "MODALIAS=vmbus:%s", alias_name);
|
|
|
|
return ret;
|
2011-03-15 22:03:37 +00:00
|
|
|
}
|
|
|
|
|
2018-08-10 23:06:08 +00:00
|
|
|
static const struct hv_vmbus_device_id *
|
2019-01-10 14:25:32 +00:00
|
|
|
hv_vmbus_dev_match(const struct hv_vmbus_device_id *id, const guid_t *guid)
|
2018-08-10 23:06:08 +00:00
|
|
|
{
|
|
|
|
if (id == NULL)
|
|
|
|
return NULL; /* empty device table */
|
|
|
|
|
2019-01-10 14:25:32 +00:00
|
|
|
for (; !guid_is_null(&id->guid); id++)
|
|
|
|
if (guid_equal(&id->guid, guid))
|
2018-08-10 23:06:08 +00:00
|
|
|
return id;
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static const struct hv_vmbus_device_id *
|
2019-01-10 14:25:32 +00:00
|
|
|
hv_vmbus_dynid_match(struct hv_driver *drv, const guid_t *guid)
|
2011-09-13 17:59:37 +00:00
|
|
|
{
|
2016-12-03 20:34:39 +00:00
|
|
|
const struct hv_vmbus_device_id *id = NULL;
|
|
|
|
struct vmbus_dynid *dynid;
|
|
|
|
|
|
|
|
spin_lock(&drv->dynids.lock);
|
|
|
|
list_for_each_entry(dynid, &drv->dynids.list, node) {
|
2019-01-10 14:25:32 +00:00
|
|
|
if (guid_equal(&dynid->id.guid, guid)) {
|
2016-12-03 20:34:39 +00:00
|
|
|
id = &dynid->id;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
spin_unlock(&drv->dynids.lock);
|
|
|
|
|
2018-08-10 23:06:08 +00:00
|
|
|
return id;
|
|
|
|
}
|
2016-12-03 20:34:39 +00:00
|
|
|
|
2019-01-10 14:25:32 +00:00
|
|
|
static const struct hv_vmbus_device_id vmbus_device_null;
|
2016-12-03 20:34:39 +00:00
|
|
|
|
2018-08-10 23:06:08 +00:00
|
|
|
/*
|
|
|
|
* Return a matching hv_vmbus_device_id pointer.
|
|
|
|
* If there is no match, return NULL.
|
|
|
|
*/
|
|
|
|
static const struct hv_vmbus_device_id *hv_vmbus_get_id(struct hv_driver *drv,
|
|
|
|
struct hv_device *dev)
|
|
|
|
{
|
2019-01-10 14:25:32 +00:00
|
|
|
const guid_t *guid = &dev->dev_type;
|
2018-08-10 23:06:08 +00:00
|
|
|
const struct hv_vmbus_device_id *id;
|
2011-09-13 17:59:37 +00:00
|
|
|
|
2018-08-10 23:06:08 +00:00
|
|
|
/* When driver_override is set, only bind to the matching driver */
|
|
|
|
if (dev->driver_override && strcmp(dev->driver_override, drv->name))
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
/* Look at the dynamic ids first, before the static ones */
|
|
|
|
id = hv_vmbus_dynid_match(drv, guid);
|
|
|
|
if (!id)
|
|
|
|
id = hv_vmbus_dev_match(drv->id_table, guid);
|
|
|
|
|
|
|
|
/* driver_override will always match, send a dummy id */
|
|
|
|
if (!id && dev->driver_override)
|
|
|
|
id = &vmbus_device_null;
|
|
|
|
|
|
|
|
return id;
|
2011-09-13 17:59:37 +00:00
|
|
|
}
|
|
|
|
|
2016-12-03 20:34:39 +00:00
|
|
|
/* vmbus_add_dynid - add a new device ID to this driver and re-probe devices */
|
2019-01-10 14:25:32 +00:00
|
|
|
static int vmbus_add_dynid(struct hv_driver *drv, guid_t *guid)
|
2016-12-03 20:34:39 +00:00
|
|
|
{
|
|
|
|
struct vmbus_dynid *dynid;
|
|
|
|
|
|
|
|
dynid = kzalloc(sizeof(*dynid), GFP_KERNEL);
|
|
|
|
if (!dynid)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
dynid->id.guid = *guid;
|
|
|
|
|
|
|
|
spin_lock(&drv->dynids.lock);
|
|
|
|
list_add_tail(&dynid->node, &drv->dynids.list);
|
|
|
|
spin_unlock(&drv->dynids.lock);
|
|
|
|
|
|
|
|
return driver_attach(&drv->driver);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void vmbus_free_dynids(struct hv_driver *drv)
|
|
|
|
{
|
|
|
|
struct vmbus_dynid *dynid, *n;
|
|
|
|
|
|
|
|
spin_lock(&drv->dynids.lock);
|
|
|
|
list_for_each_entry_safe(dynid, n, &drv->dynids.list, node) {
|
|
|
|
list_del(&dynid->node);
|
|
|
|
kfree(dynid);
|
|
|
|
}
|
|
|
|
spin_unlock(&drv->dynids.lock);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* store_new_id - sysfs frontend to vmbus_add_dynid()
|
|
|
|
*
|
|
|
|
* Allow GUIDs to be added to an existing driver via sysfs.
|
|
|
|
*/
|
|
|
|
static ssize_t new_id_store(struct device_driver *driver, const char *buf,
|
|
|
|
size_t count)
|
|
|
|
{
|
|
|
|
struct hv_driver *drv = drv_to_hv_drv(driver);
|
2019-01-10 14:25:32 +00:00
|
|
|
guid_t guid;
|
2016-12-03 20:34:39 +00:00
|
|
|
ssize_t retval;
|
|
|
|
|
2019-01-10 14:25:32 +00:00
|
|
|
retval = guid_parse(buf, &guid);
|
2017-05-18 17:46:06 +00:00
|
|
|
if (retval)
|
|
|
|
return retval;
|
2016-12-03 20:34:39 +00:00
|
|
|
|
2018-08-10 23:06:08 +00:00
|
|
|
if (hv_vmbus_dynid_match(drv, &guid))
|
2016-12-03 20:34:39 +00:00
|
|
|
return -EEXIST;
|
|
|
|
|
|
|
|
retval = vmbus_add_dynid(drv, &guid);
|
|
|
|
if (retval)
|
|
|
|
return retval;
|
|
|
|
return count;
|
|
|
|
}
|
|
|
|
static DRIVER_ATTR_WO(new_id);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* store_remove_id - remove a PCI device ID from this driver
|
|
|
|
*
|
|
|
|
* Removes a dynamic pci device ID to this driver.
|
|
|
|
*/
|
|
|
|
static ssize_t remove_id_store(struct device_driver *driver, const char *buf,
|
|
|
|
size_t count)
|
|
|
|
{
|
|
|
|
struct hv_driver *drv = drv_to_hv_drv(driver);
|
|
|
|
struct vmbus_dynid *dynid, *n;
|
2019-01-10 14:25:32 +00:00
|
|
|
guid_t guid;
|
2017-05-18 17:46:06 +00:00
|
|
|
ssize_t retval;
|
2016-12-03 20:34:39 +00:00
|
|
|
|
2019-01-10 14:25:32 +00:00
|
|
|
retval = guid_parse(buf, &guid);
|
2017-05-18 17:46:06 +00:00
|
|
|
if (retval)
|
|
|
|
return retval;
|
2016-12-03 20:34:39 +00:00
|
|
|
|
2017-05-18 17:46:06 +00:00
|
|
|
retval = -ENODEV;
|
2016-12-03 20:34:39 +00:00
|
|
|
spin_lock(&drv->dynids.lock);
|
|
|
|
list_for_each_entry_safe(dynid, n, &drv->dynids.list, node) {
|
|
|
|
struct hv_vmbus_device_id *id = &dynid->id;
|
|
|
|
|
2019-01-10 14:25:32 +00:00
|
|
|
if (guid_equal(&id->guid, &guid)) {
|
2016-12-03 20:34:39 +00:00
|
|
|
list_del(&dynid->node);
|
|
|
|
kfree(dynid);
|
|
|
|
retval = count;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
spin_unlock(&drv->dynids.lock);
|
|
|
|
|
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
static DRIVER_ATTR_WO(remove_id);
|
|
|
|
|
|
|
|
static struct attribute *vmbus_drv_attrs[] = {
|
|
|
|
&driver_attr_new_id.attr,
|
|
|
|
&driver_attr_remove_id.attr,
|
|
|
|
NULL,
|
|
|
|
};
|
|
|
|
ATTRIBUTE_GROUPS(vmbus_drv);
|
2011-09-13 17:59:37 +00:00
|
|
|
|
2011-03-15 22:03:38 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* vmbus_match - Attempt to match the specified device to the specified driver
|
|
|
|
*/
|
|
|
|
static int vmbus_match(struct device *device, struct device_driver *driver)
|
|
|
|
{
|
|
|
|
struct hv_driver *drv = drv_to_hv_drv(driver);
|
2011-06-06 22:50:04 +00:00
|
|
|
struct hv_device *hv_dev = device_to_hv_device(device);
|
2011-03-15 22:03:38 +00:00
|
|
|
|
2016-01-28 06:29:41 +00:00
|
|
|
/* The hv_sock driver handles all hv_sock offers. */
|
|
|
|
if (is_hvsock_channel(hv_dev->channel))
|
|
|
|
return drv->hvsock;
|
|
|
|
|
2018-08-10 23:06:08 +00:00
|
|
|
if (hv_vmbus_get_id(drv, hv_dev))
|
2011-09-13 17:59:37 +00:00
|
|
|
return 1;
|
2011-04-26 16:20:24 +00:00
|
|
|
|
2011-08-25 16:48:39 +00:00
|
|
|
return 0;
|
2011-03-15 22:03:38 +00:00
|
|
|
}
|
|
|
|
|
2011-03-15 22:03:39 +00:00
|
|
|
/*
|
|
|
|
* vmbus_probe - Add the new vmbus's child device
|
|
|
|
*/
|
|
|
|
static int vmbus_probe(struct device *child_device)
|
|
|
|
{
|
|
|
|
int ret = 0;
|
|
|
|
struct hv_driver *drv =
|
|
|
|
drv_to_hv_drv(child_device->driver);
|
2011-04-29 20:45:10 +00:00
|
|
|
struct hv_device *dev = device_to_hv_device(child_device);
|
2011-09-13 17:59:38 +00:00
|
|
|
const struct hv_vmbus_device_id *dev_id;
|
2011-03-15 22:03:39 +00:00
|
|
|
|
2018-08-10 23:06:08 +00:00
|
|
|
dev_id = hv_vmbus_get_id(drv, dev);
|
2011-04-29 20:45:10 +00:00
|
|
|
if (drv->probe) {
|
2011-09-13 17:59:38 +00:00
|
|
|
ret = drv->probe(dev, dev_id);
|
2011-04-29 20:45:03 +00:00
|
|
|
if (ret != 0)
|
2011-03-29 20:58:47 +00:00
|
|
|
pr_err("probe failed for device %s (%d)\n",
|
|
|
|
dev_name(child_device), ret);
|
2011-03-15 22:03:39 +00:00
|
|
|
|
|
|
|
} else {
|
2011-03-29 20:58:47 +00:00
|
|
|
pr_err("probe not set for driver %s\n",
|
|
|
|
dev_name(child_device));
|
2011-06-06 22:50:07 +00:00
|
|
|
ret = -ENODEV;
|
2011-03-15 22:03:39 +00:00
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2011-03-15 22:03:40 +00:00
|
|
|
/*
|
|
|
|
* vmbus_remove - Remove a vmbus device
|
|
|
|
*/
|
|
|
|
static int vmbus_remove(struct device *child_device)
|
|
|
|
{
|
2015-02-28 19:18:16 +00:00
|
|
|
struct hv_driver *drv;
|
2011-04-29 20:45:12 +00:00
|
|
|
struct hv_device *dev = device_to_hv_device(child_device);
|
2011-03-15 22:03:40 +00:00
|
|
|
|
2015-02-28 19:18:16 +00:00
|
|
|
if (child_device->driver) {
|
|
|
|
drv = drv_to_hv_drv(child_device->driver);
|
|
|
|
if (drv->remove)
|
|
|
|
drv->remove(dev);
|
|
|
|
}
|
2011-03-15 22:03:40 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-03-15 22:03:41 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* vmbus_shutdown - Shutdown a vmbus device
|
|
|
|
*/
|
|
|
|
static void vmbus_shutdown(struct device *child_device)
|
|
|
|
{
|
|
|
|
struct hv_driver *drv;
|
2011-04-29 20:45:14 +00:00
|
|
|
struct hv_device *dev = device_to_hv_device(child_device);
|
2011-03-15 22:03:41 +00:00
|
|
|
|
|
|
|
|
|
|
|
/* The device may not be attached yet */
|
|
|
|
if (!child_device->driver)
|
|
|
|
return;
|
|
|
|
|
|
|
|
drv = drv_to_hv_drv(child_device->driver);
|
|
|
|
|
2011-04-29 20:45:14 +00:00
|
|
|
if (drv->shutdown)
|
|
|
|
drv->shutdown(dev);
|
2011-03-15 22:03:41 +00:00
|
|
|
}
|
|
|
|
|
2019-09-19 21:46:12 +00:00
|
|
|
#ifdef CONFIG_PM_SLEEP
|
2019-09-05 23:01:17 +00:00
|
|
|
/*
|
|
|
|
* vmbus_suspend - Suspend a vmbus device
|
|
|
|
*/
|
|
|
|
static int vmbus_suspend(struct device *child_device)
|
|
|
|
{
|
|
|
|
struct hv_driver *drv;
|
|
|
|
struct hv_device *dev = device_to_hv_device(child_device);
|
|
|
|
|
|
|
|
/* The device may not be attached yet */
|
|
|
|
if (!child_device->driver)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
drv = drv_to_hv_drv(child_device->driver);
|
|
|
|
if (!drv->suspend)
|
|
|
|
return -EOPNOTSUPP;
|
|
|
|
|
|
|
|
return drv->suspend(dev);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* vmbus_resume - Resume a vmbus device
|
|
|
|
*/
|
|
|
|
static int vmbus_resume(struct device *child_device)
|
|
|
|
{
|
|
|
|
struct hv_driver *drv;
|
|
|
|
struct hv_device *dev = device_to_hv_device(child_device);
|
|
|
|
|
|
|
|
/* The device may not be attached yet */
|
|
|
|
if (!child_device->driver)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
drv = drv_to_hv_drv(child_device->driver);
|
|
|
|
if (!drv->resume)
|
|
|
|
return -EOPNOTSUPP;
|
|
|
|
|
|
|
|
return drv->resume(dev);
|
|
|
|
}
|
2019-09-19 21:46:12 +00:00
|
|
|
#endif /* CONFIG_PM_SLEEP */
|
2011-03-15 22:03:42 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* vmbus_device_release - Final callback release of the vmbus child device
|
|
|
|
*/
|
|
|
|
static void vmbus_device_release(struct device *device)
|
|
|
|
{
|
2011-06-06 22:50:04 +00:00
|
|
|
struct hv_device *hv_dev = device_to_hv_device(device);
|
Drivers: hv: vmbus: fix rescind-offer handling for device without a driver
In the path vmbus_onoffer_rescind() -> vmbus_device_unregister() ->
device_unregister() -> ... -> __device_release_driver(), we can see for a
device without a driver loaded: dev->driver is NULL, so
dev->bus->remove(dev), namely vmbus_remove(), isn't invoked.
As a result, vmbus_remove() -> hv_process_channel_removal() isn't invoked
and some cleanups(like sending a CHANNELMSG_RELID_RELEASED message to the
host) aren't done.
We can demo the issue this way:
1. rmmod hv_utils;
2. disable the Heartbeat Integration Service in Hyper-V Manager and lsvmbus
shows the device disappears.
3. re-enable the Heartbeat in Hyper-V Manager and modprobe hv_utils, but
lsvmbus shows the device can't appear again.
This is because, the host thinks the VM hasn't released the relid, so can't
re-offer the device to the VM.
We can fix the issue by moving hv_process_channel_removal()
from vmbus_close_internal() to vmbus_device_release(), since the latter is
always invoked on device_unregister(), whether or not the dev has a driver
loaded.
Signed-off-by: Dexuan Cui <decui@microsoft.com>
Signed-off-by: K. Y. Srinivasan <kys@microsoft.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
2015-12-15 00:01:49 +00:00
|
|
|
struct vmbus_channel *channel = hv_dev->channel;
|
2011-03-15 22:03:42 +00:00
|
|
|
|
2019-10-03 21:01:49 +00:00
|
|
|
hv_debug_rm_dev_dir(hv_dev);
|
|
|
|
|
2017-04-30 23:21:18 +00:00
|
|
|
mutex_lock(&vmbus_connection.channel_mutex);
|
2018-09-14 16:10:15 +00:00
|
|
|
hv_process_channel_removal(channel);
|
2017-04-30 23:21:18 +00:00
|
|
|
mutex_unlock(&vmbus_connection.channel_mutex);
|
2011-06-06 22:50:04 +00:00
|
|
|
kfree(hv_dev);
|
2011-03-15 22:03:42 +00:00
|
|
|
}
|
|
|
|
|
2019-09-05 23:01:17 +00:00
|
|
|
/*
|
|
|
|
* Note: we must use SET_NOIRQ_SYSTEM_SLEEP_PM_OPS rather than
|
|
|
|
* SET_SYSTEM_SLEEP_PM_OPS: see the comment before vmbus_bus_pm.
|
|
|
|
*/
|
|
|
|
static const struct dev_pm_ops vmbus_pm = {
|
|
|
|
SET_NOIRQ_SYSTEM_SLEEP_PM_OPS(vmbus_suspend, vmbus_resume)
|
|
|
|
};
|
|
|
|
|
2009-07-27 20:47:24 +00:00
|
|
|
/* The one and only one */
|
2011-04-29 20:45:08 +00:00
|
|
|
static struct bus_type hv_bus = {
|
|
|
|
.name = "vmbus",
|
|
|
|
.match = vmbus_match,
|
|
|
|
.shutdown = vmbus_shutdown,
|
|
|
|
.remove = vmbus_remove,
|
|
|
|
.probe = vmbus_probe,
|
|
|
|
.uevent = vmbus_uevent,
|
2016-12-03 20:34:39 +00:00
|
|
|
.dev_groups = vmbus_dev_groups,
|
|
|
|
.drv_groups = vmbus_drv_groups,
|
2019-09-05 23:01:17 +00:00
|
|
|
.pm = &vmbus_pm,
|
2009-07-13 23:02:34 +00:00
|
|
|
};
|
|
|
|
|
2010-12-15 18:48:08 +00:00
|
|
|
struct onmessage_work_context {
|
|
|
|
struct work_struct work;
|
2020-04-06 10:41:51 +00:00
|
|
|
struct {
|
|
|
|
struct hv_message_header header;
|
|
|
|
u8 payload[];
|
|
|
|
} msg;
|
2010-12-15 18:48:08 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
static void vmbus_onmessage_work(struct work_struct *work)
|
|
|
|
{
|
|
|
|
struct onmessage_work_context *ctx;
|
|
|
|
|
2015-02-27 19:25:54 +00:00
|
|
|
/* Do not process messages if we're in DISCONNECTED state */
|
|
|
|
if (vmbus_connection.conn_state == DISCONNECTED)
|
|
|
|
return;
|
|
|
|
|
2010-12-15 18:48:08 +00:00
|
|
|
ctx = container_of(work, struct onmessage_work_context,
|
|
|
|
work);
|
2020-04-06 10:41:52 +00:00
|
|
|
vmbus_onmessage((struct vmbus_channel_message_header *)
|
|
|
|
&ctx->msg.payload);
|
2010-12-15 18:48:08 +00:00
|
|
|
kfree(ctx);
|
|
|
|
}
|
|
|
|
|
2016-02-26 23:13:21 +00:00
|
|
|
void vmbus_on_msg_dpc(unsigned long data)
|
2010-12-02 19:59:22 +00:00
|
|
|
{
|
2017-02-12 06:02:19 +00:00
|
|
|
struct hv_per_cpu_context *hv_cpu = (void *)data;
|
|
|
|
void *page_addr = hv_cpu->synic_message_page;
|
2010-12-02 19:59:22 +00:00
|
|
|
struct hv_message *msg = (struct hv_message *)page_addr +
|
|
|
|
VMBUS_MESSAGE_SINT;
|
2015-03-27 16:10:08 +00:00
|
|
|
struct vmbus_channel_message_header *hdr;
|
2017-03-05 01:27:16 +00:00
|
|
|
const struct vmbus_channel_message_table_entry *entry;
|
2010-12-15 18:48:08 +00:00
|
|
|
struct onmessage_work_context *ctx;
|
2016-05-01 02:21:34 +00:00
|
|
|
u32 message_type = msg->header.message_type;
|
2010-12-02 19:59:22 +00:00
|
|
|
|
2020-04-06 10:43:15 +00:00
|
|
|
/*
|
|
|
|
* 'enum vmbus_channel_message_type' is supposed to always be 'u32' as
|
|
|
|
* it is being used in 'struct vmbus_channel_message_header' definition
|
|
|
|
* which is supposed to match hypervisor ABI.
|
|
|
|
*/
|
|
|
|
BUILD_BUG_ON(sizeof(enum vmbus_channel_message_type) != sizeof(u32));
|
|
|
|
|
2016-05-01 02:21:34 +00:00
|
|
|
if (message_type == HVMSG_NONE)
|
2016-02-26 23:13:15 +00:00
|
|
|
/* no msg */
|
|
|
|
return;
|
2015-03-27 16:10:08 +00:00
|
|
|
|
2016-02-26 23:13:15 +00:00
|
|
|
hdr = (struct vmbus_channel_message_header *)msg->u.payload;
|
2015-03-27 16:10:08 +00:00
|
|
|
|
2017-10-29 19:21:00 +00:00
|
|
|
trace_vmbus_on_msg_dpc(hdr);
|
|
|
|
|
2016-02-26 23:13:15 +00:00
|
|
|
if (hdr->msgtype >= CHANNELMSG_COUNT) {
|
|
|
|
WARN_ONCE(1, "unknown msgtype=%d\n", hdr->msgtype);
|
|
|
|
goto msg_handled;
|
|
|
|
}
|
2015-03-27 16:10:08 +00:00
|
|
|
|
2020-04-06 10:41:50 +00:00
|
|
|
if (msg->header.payload_size > HV_MESSAGE_PAYLOAD_BYTE_COUNT) {
|
|
|
|
WARN_ONCE(1, "payload size is too large (%d)\n",
|
|
|
|
msg->header.payload_size);
|
|
|
|
goto msg_handled;
|
|
|
|
}
|
|
|
|
|
2016-02-26 23:13:15 +00:00
|
|
|
entry = &channel_message_table[hdr->msgtype];
|
2020-01-19 23:29:22 +00:00
|
|
|
|
|
|
|
if (!entry->message_handler)
|
|
|
|
goto msg_handled;
|
|
|
|
|
2020-04-06 10:43:26 +00:00
|
|
|
if (msg->header.payload_size < entry->min_payload_len) {
|
|
|
|
WARN_ONCE(1, "message too short: msgtype=%d len=%d\n",
|
|
|
|
hdr->msgtype, msg->header.payload_size);
|
|
|
|
goto msg_handled;
|
|
|
|
}
|
|
|
|
|
2016-02-26 23:13:15 +00:00
|
|
|
if (entry->handler_type == VMHT_BLOCKING) {
|
2020-04-06 10:41:51 +00:00
|
|
|
ctx = kmalloc(sizeof(*ctx) + msg->header.payload_size,
|
|
|
|
GFP_ATOMIC);
|
2016-02-26 23:13:15 +00:00
|
|
|
if (ctx == NULL)
|
|
|
|
return;
|
2015-03-27 16:10:08 +00:00
|
|
|
|
2016-02-26 23:13:15 +00:00
|
|
|
INIT_WORK(&ctx->work, vmbus_onmessage_work);
|
2020-04-06 10:41:50 +00:00
|
|
|
memcpy(&ctx->msg, msg, sizeof(msg->header) +
|
|
|
|
msg->header.payload_size);
|
2015-03-27 16:10:08 +00:00
|
|
|
|
2017-04-30 23:21:18 +00:00
|
|
|
/*
|
|
|
|
* The host can generate a rescind message while we
|
|
|
|
* may still be handling the original offer. We deal with
|
2020-04-06 00:15:05 +00:00
|
|
|
* this condition by relying on the synchronization provided
|
|
|
|
* by offer_in_progress and by channel_mutex. See also the
|
|
|
|
* inline comments in vmbus_onoffer_rescind().
|
2017-04-30 23:21:18 +00:00
|
|
|
*/
|
|
|
|
switch (hdr->msgtype) {
|
|
|
|
case CHANNELMSG_RESCIND_CHANNELOFFER:
|
|
|
|
/*
|
|
|
|
* If we are handling the rescind message;
|
|
|
|
* schedule the work on the global work queue.
|
2020-04-06 00:15:04 +00:00
|
|
|
*
|
|
|
|
* The OFFER message and the RESCIND message should
|
|
|
|
* not be handled by the same serialized work queue,
|
|
|
|
* because the OFFER handler may call vmbus_open(),
|
|
|
|
* which tries to open the channel by sending an
|
|
|
|
* OPEN_CHANNEL message to the host and waits for
|
|
|
|
* the host's response; however, if the host has
|
|
|
|
* rescinded the channel before it receives the
|
|
|
|
* OPEN_CHANNEL message, the host just silently
|
|
|
|
* ignores the OPEN_CHANNEL message; as a result,
|
|
|
|
* the guest's OFFER handler hangs for ever, if we
|
|
|
|
* handle the RESCIND message in the same serialized
|
|
|
|
* work queue: the RESCIND handler can not start to
|
|
|
|
* run before the OFFER handler finishes.
|
2017-04-30 23:21:18 +00:00
|
|
|
*/
|
2020-04-06 00:15:05 +00:00
|
|
|
schedule_work(&ctx->work);
|
2017-04-30 23:21:18 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case CHANNELMSG_OFFERCHANNEL:
|
2020-04-06 00:15:05 +00:00
|
|
|
/*
|
|
|
|
* The host sends the offer message of a given channel
|
|
|
|
* before sending the rescind message of the same
|
|
|
|
* channel. These messages are sent to the guest's
|
|
|
|
* connect CPU; the guest then starts processing them
|
|
|
|
* in the tasklet handler on this CPU:
|
|
|
|
*
|
|
|
|
* VMBUS_CONNECT_CPU
|
|
|
|
*
|
|
|
|
* [vmbus_on_msg_dpc()]
|
|
|
|
* atomic_inc() // CHANNELMSG_OFFERCHANNEL
|
|
|
|
* queue_work()
|
|
|
|
* ...
|
|
|
|
* [vmbus_on_msg_dpc()]
|
|
|
|
* schedule_work() // CHANNELMSG_RESCIND_CHANNELOFFER
|
|
|
|
*
|
|
|
|
* We rely on the memory-ordering properties of the
|
|
|
|
* queue_work() and schedule_work() primitives, which
|
|
|
|
* guarantee that the atomic increment will be visible
|
|
|
|
* to the CPUs which will execute the offer & rescind
|
|
|
|
* works by the time these works will start execution.
|
|
|
|
*/
|
2017-04-30 23:21:18 +00:00
|
|
|
atomic_inc(&vmbus_connection.offer_in_progress);
|
2020-04-06 00:15:05 +00:00
|
|
|
fallthrough;
|
2017-04-30 23:21:18 +00:00
|
|
|
|
|
|
|
default:
|
|
|
|
queue_work(vmbus_connection.work_queue, &ctx->work);
|
|
|
|
}
|
2016-02-26 23:13:15 +00:00
|
|
|
} else
|
|
|
|
entry->message_handler(hdr);
|
2010-12-02 19:59:22 +00:00
|
|
|
|
2015-03-27 16:10:08 +00:00
|
|
|
msg_handled:
|
2016-05-01 02:21:34 +00:00
|
|
|
vmbus_signal_eom(msg, message_type);
|
2010-12-02 19:59:22 +00:00
|
|
|
}
|
|
|
|
|
2019-09-19 21:46:12 +00:00
|
|
|
#ifdef CONFIG_PM_SLEEP
|
2019-09-05 23:01:20 +00:00
|
|
|
/*
|
|
|
|
* Fake RESCIND_CHANNEL messages to clean up hv_sock channels by force for
|
|
|
|
* hibernation, because hv_sock connections can not persist across hibernation.
|
|
|
|
*/
|
|
|
|
static void vmbus_force_channel_rescinded(struct vmbus_channel *channel)
|
|
|
|
{
|
|
|
|
struct onmessage_work_context *ctx;
|
|
|
|
struct vmbus_channel_rescind_offer *rescind;
|
|
|
|
|
|
|
|
WARN_ON(!is_hvsock_channel(channel));
|
|
|
|
|
|
|
|
/*
|
2020-04-06 10:41:51 +00:00
|
|
|
* Allocation size is small and the allocation should really not fail,
|
2019-09-05 23:01:20 +00:00
|
|
|
* otherwise the state of the hv_sock connections ends up in limbo.
|
|
|
|
*/
|
2020-04-06 10:41:51 +00:00
|
|
|
ctx = kzalloc(sizeof(*ctx) + sizeof(*rescind),
|
|
|
|
GFP_KERNEL | __GFP_NOFAIL);
|
2019-09-05 23:01:20 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* So far, these are not really used by Linux. Just set them to the
|
|
|
|
* reasonable values conforming to the definitions of the fields.
|
|
|
|
*/
|
|
|
|
ctx->msg.header.message_type = 1;
|
|
|
|
ctx->msg.header.payload_size = sizeof(*rescind);
|
|
|
|
|
|
|
|
/* These values are actually used by Linux. */
|
2020-04-06 10:41:51 +00:00
|
|
|
rescind = (struct vmbus_channel_rescind_offer *)ctx->msg.payload;
|
2019-09-05 23:01:20 +00:00
|
|
|
rescind->header.msgtype = CHANNELMSG_RESCIND_CHANNELOFFER;
|
|
|
|
rescind->child_relid = channel->offermsg.child_relid;
|
|
|
|
|
|
|
|
INIT_WORK(&ctx->work, vmbus_onmessage_work);
|
|
|
|
|
2020-04-06 00:15:05 +00:00
|
|
|
queue_work(vmbus_connection.work_queue, &ctx->work);
|
2019-09-05 23:01:20 +00:00
|
|
|
}
|
2019-09-19 21:46:12 +00:00
|
|
|
#endif /* CONFIG_PM_SLEEP */
|
2017-02-12 06:02:20 +00:00
|
|
|
|
2017-02-12 06:02:21 +00:00
|
|
|
/*
|
|
|
|
* Direct callback for channels using other deferred processing
|
|
|
|
*/
|
|
|
|
static void vmbus_channel_isr(struct vmbus_channel *channel)
|
|
|
|
{
|
|
|
|
void (*callback_fn)(void *);
|
|
|
|
|
|
|
|
callback_fn = READ_ONCE(channel->onchannel_callback);
|
|
|
|
if (likely(callback_fn != NULL))
|
|
|
|
(*callback_fn)(channel->channel_callback_context);
|
|
|
|
}
|
|
|
|
|
2017-02-12 06:02:20 +00:00
|
|
|
/*
|
|
|
|
* Schedule all channels with events pending
|
|
|
|
*/
|
|
|
|
static void vmbus_chan_sched(struct hv_per_cpu_context *hv_cpu)
|
|
|
|
{
|
|
|
|
unsigned long *recv_int_page;
|
|
|
|
u32 maxbits, relid;
|
|
|
|
|
|
|
|
if (vmbus_proto_version < VERSION_WIN8) {
|
|
|
|
maxbits = MAX_NUM_CHANNELS_SUPPORTED;
|
|
|
|
recv_int_page = vmbus_connection.recv_int_page;
|
|
|
|
} else {
|
|
|
|
/*
|
|
|
|
* When the host is win8 and beyond, the event page
|
|
|
|
* can be directly checked to get the id of the channel
|
|
|
|
* that has the interrupt pending.
|
|
|
|
*/
|
|
|
|
void *page_addr = hv_cpu->synic_event_page;
|
|
|
|
union hv_synic_event_flags *event
|
|
|
|
= (union hv_synic_event_flags *)page_addr +
|
|
|
|
VMBUS_MESSAGE_SINT;
|
|
|
|
|
|
|
|
maxbits = HV_EVENT_FLAGS_COUNT;
|
|
|
|
recv_int_page = event->flags;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (unlikely(!recv_int_page))
|
|
|
|
return;
|
|
|
|
|
|
|
|
for_each_set_bit(relid, recv_int_page, maxbits) {
|
|
|
|
struct vmbus_channel *channel;
|
|
|
|
|
|
|
|
if (!sync_test_and_clear_bit(relid, recv_int_page))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
/* Special case - vmbus channel protocol msg */
|
|
|
|
if (relid == 0)
|
|
|
|
continue;
|
|
|
|
|
2020-04-06 00:15:06 +00:00
|
|
|
/*
|
|
|
|
* Pairs with the kfree_rcu() in vmbus_chan_release().
|
|
|
|
* Guarantees that the channel data structure doesn't
|
|
|
|
* get freed while the channel pointer below is being
|
|
|
|
* dereferenced.
|
|
|
|
*/
|
2017-03-05 01:13:57 +00:00
|
|
|
rcu_read_lock();
|
|
|
|
|
2017-02-12 06:02:20 +00:00
|
|
|
/* Find channel based on relid */
|
2020-04-06 00:15:06 +00:00
|
|
|
channel = relid2channel(relid);
|
|
|
|
if (channel == NULL)
|
|
|
|
goto sched_unlock_rcu;
|
2017-02-12 06:02:21 +00:00
|
|
|
|
2020-04-06 00:15:06 +00:00
|
|
|
if (channel->rescind)
|
|
|
|
goto sched_unlock_rcu;
|
2017-08-11 17:03:59 +00:00
|
|
|
|
2020-04-06 00:15:06 +00:00
|
|
|
trace_vmbus_chan_sched(channel);
|
2017-10-29 19:21:16 +00:00
|
|
|
|
2020-04-06 00:15:06 +00:00
|
|
|
++channel->interrupts;
|
2017-10-29 18:33:40 +00:00
|
|
|
|
2020-04-06 00:15:06 +00:00
|
|
|
switch (channel->callback_mode) {
|
|
|
|
case HV_CALL_ISR:
|
|
|
|
vmbus_channel_isr(channel);
|
|
|
|
break;
|
2017-02-12 06:02:21 +00:00
|
|
|
|
2020-04-06 00:15:06 +00:00
|
|
|
case HV_CALL_BATCHED:
|
|
|
|
hv_begin_read(&channel->inbound);
|
|
|
|
fallthrough;
|
|
|
|
case HV_CALL_DIRECT:
|
|
|
|
tasklet_schedule(&channel->callback_event);
|
2017-02-12 06:02:20 +00:00
|
|
|
}
|
2017-03-05 01:13:57 +00:00
|
|
|
|
2020-04-06 00:15:06 +00:00
|
|
|
sched_unlock_rcu:
|
2017-03-05 01:13:57 +00:00
|
|
|
rcu_read_unlock();
|
2017-02-12 06:02:20 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-03-05 12:42:14 +00:00
|
|
|
static void vmbus_isr(void)
|
2010-12-02 19:59:22 +00:00
|
|
|
{
|
2017-02-12 06:02:19 +00:00
|
|
|
struct hv_per_cpu_context *hv_cpu
|
|
|
|
= this_cpu_ptr(hv_context.cpu_context);
|
|
|
|
void *page_addr = hv_cpu->synic_event_page;
|
2010-12-02 19:59:22 +00:00
|
|
|
struct hv_message *msg;
|
|
|
|
union hv_synic_event_flags *event;
|
2011-08-27 18:31:35 +00:00
|
|
|
bool handled = false;
|
2010-12-02 19:59:22 +00:00
|
|
|
|
2017-02-12 06:02:19 +00:00
|
|
|
if (unlikely(page_addr == NULL))
|
2014-03-05 12:42:14 +00:00
|
|
|
return;
|
2012-12-01 14:46:55 +00:00
|
|
|
|
|
|
|
event = (union hv_synic_event_flags *)page_addr +
|
|
|
|
VMBUS_MESSAGE_SINT;
|
2011-08-31 21:35:56 +00:00
|
|
|
/*
|
|
|
|
* Check for events before checking for messages. This is the order
|
|
|
|
* in which events and messages are checked in Windows guests on
|
|
|
|
* Hyper-V, and the Windows team suggested we do the same.
|
|
|
|
*/
|
2010-12-02 19:59:22 +00:00
|
|
|
|
2012-12-01 14:46:49 +00:00
|
|
|
if ((vmbus_proto_version == VERSION_WS2008) ||
|
|
|
|
(vmbus_proto_version == VERSION_WIN7)) {
|
2010-12-02 19:59:22 +00:00
|
|
|
|
2012-12-01 14:46:49 +00:00
|
|
|
/* Since we are a child, we only need to check bit 0 */
|
2017-02-06 00:20:31 +00:00
|
|
|
if (sync_test_and_clear_bit(0, event->flags))
|
2012-12-01 14:46:49 +00:00
|
|
|
handled = true;
|
|
|
|
} else {
|
|
|
|
/*
|
|
|
|
* Our host is win8 or above. The signaling mechanism
|
|
|
|
* has changed and we can directly look at the event page.
|
|
|
|
* If bit n is set then we have an interrup on the channel
|
|
|
|
* whose id is n.
|
|
|
|
*/
|
2011-08-27 18:31:35 +00:00
|
|
|
handled = true;
|
|
|
|
}
|
2011-03-15 22:03:43 +00:00
|
|
|
|
2012-12-01 14:46:49 +00:00
|
|
|
if (handled)
|
2017-02-12 06:02:20 +00:00
|
|
|
vmbus_chan_sched(hv_cpu);
|
2012-12-01 14:46:49 +00:00
|
|
|
|
2017-02-12 06:02:19 +00:00
|
|
|
page_addr = hv_cpu->synic_message_page;
|
2011-08-31 21:35:56 +00:00
|
|
|
msg = (struct hv_message *)page_addr + VMBUS_MESSAGE_SINT;
|
|
|
|
|
|
|
|
/* Check if there are actual msgs to be processed */
|
2015-01-10 07:54:32 +00:00
|
|
|
if (msg->header.message_type != HVMSG_NONE) {
|
2019-07-01 04:25:56 +00:00
|
|
|
if (msg->header.message_type == HVMSG_TIMER_EXPIRED) {
|
|
|
|
hv_stimer0_isr();
|
|
|
|
vmbus_signal_eom(msg, HVMSG_TIMER_EXPIRED);
|
|
|
|
} else
|
2017-02-12 06:02:19 +00:00
|
|
|
tasklet_schedule(&hv_cpu->msg_dpc);
|
2015-01-10 07:54:32 +00:00
|
|
|
}
|
2016-05-02 06:14:34 +00:00
|
|
|
|
|
|
|
add_interrupt_randomness(HYPERVISOR_CALLBACK_VECTOR, 0);
|
2011-03-15 22:03:43 +00:00
|
|
|
}
|
|
|
|
|
2018-07-08 02:56:51 +00:00
|
|
|
/*
|
|
|
|
* Callback from kmsg_dump. Grab as much as possible from the end of the kmsg
|
|
|
|
* buffer and call into Hyper-V to transfer the data.
|
|
|
|
*/
|
|
|
|
static void hv_kmsg_dump(struct kmsg_dumper *dumper,
|
|
|
|
enum kmsg_dump_reason reason)
|
|
|
|
{
|
|
|
|
size_t bytes_written;
|
|
|
|
phys_addr_t panic_pa;
|
|
|
|
|
|
|
|
/* We are only interested in panics. */
|
|
|
|
if ((reason != KMSG_DUMP_PANIC) || (!sysctl_record_panic_msg))
|
|
|
|
return;
|
|
|
|
|
|
|
|
panic_pa = virt_to_phys(hv_panic_page);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Write dump contents to the page. No need to synchronize; panic should
|
|
|
|
* be single-threaded.
|
|
|
|
*/
|
2019-07-30 09:49:44 +00:00
|
|
|
kmsg_dump_get_buffer(dumper, true, hv_panic_page, HV_HYP_PAGE_SIZE,
|
2018-07-28 21:58:45 +00:00
|
|
|
&bytes_written);
|
|
|
|
if (bytes_written)
|
|
|
|
hyperv_report_panic_msg(panic_pa, bytes_written);
|
2018-07-08 02:56:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static struct kmsg_dumper hv_kmsg_dumper = {
|
|
|
|
.dump = hv_kmsg_dump,
|
|
|
|
};
|
|
|
|
|
|
|
|
static struct ctl_table_header *hv_ctl_table_hdr;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* sysctl option to allow the user to control whether kmsg data should be
|
|
|
|
* reported to Hyper-V on panic.
|
|
|
|
*/
|
|
|
|
static struct ctl_table hv_ctl_table[] = {
|
|
|
|
{
|
|
|
|
.procname = "hyperv_record_panic_msg",
|
|
|
|
.data = &sysctl_record_panic_msg,
|
|
|
|
.maxlen = sizeof(int),
|
|
|
|
.mode = 0644,
|
|
|
|
.proc_handler = proc_dointvec_minmax,
|
proc/sysctl: add shared variables for range check
In the sysctl code the proc_dointvec_minmax() function is often used to
validate the user supplied value between an allowed range. This
function uses the extra1 and extra2 members from struct ctl_table as
minimum and maximum allowed value.
On sysctl handler declaration, in every source file there are some
readonly variables containing just an integer which address is assigned
to the extra1 and extra2 members, so the sysctl range is enforced.
The special values 0, 1 and INT_MAX are very often used as range
boundary, leading duplication of variables like zero=0, one=1,
int_max=INT_MAX in different source files:
$ git grep -E '\.extra[12].*&(zero|one|int_max)' |wc -l
248
Add a const int array containing the most commonly used values, some
macros to refer more easily to the correct array member, and use them
instead of creating a local one for every object file.
This is the bloat-o-meter output comparing the old and new binary
compiled with the default Fedora config:
# scripts/bloat-o-meter -d vmlinux.o.old vmlinux.o
add/remove: 2/2 grow/shrink: 0/2 up/down: 24/-188 (-164)
Data old new delta
sysctl_vals - 12 +12
__kstrtab_sysctl_vals - 12 +12
max 14 10 -4
int_max 16 - -16
one 68 - -68
zero 128 28 -100
Total: Before=20583249, After=20583085, chg -0.00%
[mcroce@redhat.com: tipc: remove two unused variables]
Link: http://lkml.kernel.org/r/20190530091952.4108-1-mcroce@redhat.com
[akpm@linux-foundation.org: fix net/ipv6/sysctl_net_ipv6.c]
[arnd@arndb.de: proc/sysctl: make firmware loader table conditional]
Link: http://lkml.kernel.org/r/20190617130014.1713870-1-arnd@arndb.de
[akpm@linux-foundation.org: fix fs/eventpoll.c]
Link: http://lkml.kernel.org/r/20190430180111.10688-1-mcroce@redhat.com
Signed-off-by: Matteo Croce <mcroce@redhat.com>
Signed-off-by: Arnd Bergmann <arnd@arndb.de>
Acked-by: Kees Cook <keescook@chromium.org>
Reviewed-by: Aaron Tomlin <atomlin@redhat.com>
Cc: Matthew Wilcox <willy@infradead.org>
Cc: Stephen Rothwell <sfr@canb.auug.org.au>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2019-07-18 22:58:50 +00:00
|
|
|
.extra1 = SYSCTL_ZERO,
|
|
|
|
.extra2 = SYSCTL_ONE
|
2018-07-08 02:56:51 +00:00
|
|
|
},
|
|
|
|
{}
|
|
|
|
};
|
|
|
|
|
|
|
|
static struct ctl_table hv_root_table[] = {
|
|
|
|
{
|
|
|
|
.procname = "kernel",
|
|
|
|
.mode = 0555,
|
|
|
|
.child = hv_ctl_table
|
|
|
|
},
|
|
|
|
{}
|
|
|
|
};
|
2015-02-27 19:25:51 +00:00
|
|
|
|
2010-03-04 22:11:00 +00:00
|
|
|
/*
|
2009-09-02 14:11:14 +00:00
|
|
|
* vmbus_bus_init -Main vmbus driver initialization routine.
|
|
|
|
*
|
|
|
|
* Here, we
|
2010-03-11 22:51:23 +00:00
|
|
|
* - initialize the vmbus driver context
|
|
|
|
* - invoke the vmbus hv main init routine
|
|
|
|
* - retrieve the channel offers
|
2009-09-02 14:11:14 +00:00
|
|
|
*/
|
2015-12-15 00:01:46 +00:00
|
|
|
static int vmbus_bus_init(void)
|
2009-07-13 23:02:34 +00:00
|
|
|
{
|
2009-09-02 14:11:14 +00:00
|
|
|
int ret;
|
2009-07-13 23:02:34 +00:00
|
|
|
|
2010-12-02 20:08:08 +00:00
|
|
|
/* Hypervisor initialization...setup hypercall page..etc */
|
|
|
|
ret = hv_init();
|
2009-09-02 14:11:14 +00:00
|
|
|
if (ret != 0) {
|
2011-03-29 20:58:47 +00:00
|
|
|
pr_err("Unable to initialize the hypervisor - 0x%x\n", ret);
|
2011-06-06 22:50:08 +00:00
|
|
|
return ret;
|
2009-07-13 23:02:34 +00:00
|
|
|
}
|
|
|
|
|
2011-04-29 20:45:08 +00:00
|
|
|
ret = bus_register(&hv_bus);
|
2011-06-06 22:50:08 +00:00
|
|
|
if (ret)
|
2017-01-28 19:37:14 +00:00
|
|
|
return ret;
|
2009-07-13 23:02:34 +00:00
|
|
|
|
2014-03-05 12:42:14 +00:00
|
|
|
hv_setup_vmbus_irq(vmbus_isr);
|
2009-07-13 23:02:34 +00:00
|
|
|
|
2013-06-19 03:28:10 +00:00
|
|
|
ret = hv_synic_alloc();
|
|
|
|
if (ret)
|
|
|
|
goto err_alloc;
|
2019-07-01 04:25:56 +00:00
|
|
|
|
2011-03-15 22:03:33 +00:00
|
|
|
/*
|
2019-07-01 04:25:56 +00:00
|
|
|
* Initialize the per-cpu interrupt state and stimer state.
|
|
|
|
* Then connect to the host.
|
2011-03-15 22:03:33 +00:00
|
|
|
*/
|
2017-12-22 18:19:02 +00:00
|
|
|
ret = cpuhp_setup_state(CPUHP_AP_ONLINE_DYN, "hyperv/vmbus:online",
|
2016-12-07 22:53:11 +00:00
|
|
|
hv_synic_init, hv_synic_cleanup);
|
|
|
|
if (ret < 0)
|
2019-07-01 04:25:56 +00:00
|
|
|
goto err_cpuhp;
|
2016-12-07 22:53:11 +00:00
|
|
|
hyperv_cpuhp_online = ret;
|
|
|
|
|
2011-03-15 22:03:33 +00:00
|
|
|
ret = vmbus_connect();
|
2011-08-31 21:35:55 +00:00
|
|
|
if (ret)
|
2015-12-15 00:01:38 +00:00
|
|
|
goto err_connect;
|
2011-03-15 22:03:33 +00:00
|
|
|
|
2015-02-28 19:39:01 +00:00
|
|
|
/*
|
|
|
|
* Only register if the crash MSRs are available
|
|
|
|
*/
|
2015-08-01 23:08:20 +00:00
|
|
|
if (ms_hyperv.misc_features & HV_FEATURE_GUEST_CRASH_MSR_AVAILABLE) {
|
2018-07-08 02:56:51 +00:00
|
|
|
u64 hyperv_crash_ctl;
|
|
|
|
/*
|
|
|
|
* Sysctl registration is not fatal, since by default
|
|
|
|
* reporting is enabled.
|
|
|
|
*/
|
|
|
|
hv_ctl_table_hdr = register_sysctl_table(hv_root_table);
|
|
|
|
if (!hv_ctl_table_hdr)
|
|
|
|
pr_err("Hyper-V: sysctl table register error");
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Register for panic kmsg callback only if the right
|
|
|
|
* capability is supported by the hypervisor.
|
|
|
|
*/
|
2018-07-28 21:58:47 +00:00
|
|
|
hv_get_crash_ctl(hyperv_crash_ctl);
|
2018-07-08 02:56:51 +00:00
|
|
|
if (hyperv_crash_ctl & HV_CRASH_CTL_CRASH_NOTIFY_MSG) {
|
2019-07-30 09:49:44 +00:00
|
|
|
hv_panic_page = (void *)hv_alloc_hyperv_zeroed_page();
|
2018-07-08 02:56:51 +00:00
|
|
|
if (hv_panic_page) {
|
|
|
|
ret = kmsg_dump_register(&hv_kmsg_dumper);
|
2020-04-06 15:53:27 +00:00
|
|
|
if (ret) {
|
2018-07-08 02:56:51 +00:00
|
|
|
pr_err("Hyper-V: kmsg dump register "
|
|
|
|
"error 0x%x\n", ret);
|
2020-04-06 15:53:27 +00:00
|
|
|
hv_free_hyperv_page(
|
|
|
|
(unsigned long)hv_panic_page);
|
|
|
|
hv_panic_page = NULL;
|
|
|
|
}
|
2018-07-08 02:56:51 +00:00
|
|
|
} else
|
|
|
|
pr_err("Hyper-V: panic message page memory "
|
|
|
|
"allocation failed");
|
|
|
|
}
|
|
|
|
|
2015-08-01 23:08:10 +00:00
|
|
|
register_die_notifier(&hyperv_die_block);
|
2015-02-28 19:39:01 +00:00
|
|
|
}
|
|
|
|
|
2020-04-06 15:53:26 +00:00
|
|
|
/*
|
|
|
|
* Always register the panic notifier because we need to unload
|
|
|
|
* the VMbus channel connection to prevent any VMbus
|
|
|
|
* activity after the VM panics.
|
|
|
|
*/
|
|
|
|
atomic_notifier_chain_register(&panic_notifier_list,
|
|
|
|
&hyperv_panic_block);
|
|
|
|
|
2010-12-02 16:50:58 +00:00
|
|
|
vmbus_request_offers();
|
2010-05-28 23:22:44 +00:00
|
|
|
|
2011-06-06 22:50:08 +00:00
|
|
|
return 0;
|
2011-08-31 21:35:55 +00:00
|
|
|
|
2015-12-15 00:01:38 +00:00
|
|
|
err_connect:
|
2016-12-07 22:53:11 +00:00
|
|
|
cpuhp_remove_state(hyperv_cpuhp_online);
|
2019-07-01 04:25:56 +00:00
|
|
|
err_cpuhp:
|
2013-06-19 03:28:10 +00:00
|
|
|
hv_synic_free();
|
x86/hyperv: Initialize clockevents earlier in CPU onlining
Hyper-V has historically initialized stimer-based clockevents late in the
process of onlining a CPU because clockevents depend on stimer
interrupts. In the original Hyper-V design, stimer interrupts generate a
VMbus message, so the VMbus machinery must be running first, and VMbus
can't be initialized until relatively late. On x86/64, LAPIC timer based
clockevents are used during early initialization before VMbus and
stimer-based clockevents are ready, and again during CPU offlining after
the stimer clockevents have been shut down.
Unfortunately, this design creates problems when offlining CPUs for
hibernation or other purposes. stimer-based clockevents are shut down
relatively early in the offlining process, so clockevents_unbind_device()
must be used to fallback to the LAPIC-based clockevents for the remainder
of the offlining process. Furthermore, the late initialization and early
shutdown of stimer-based clockevents doesn't work well on ARM64 since there
is no other timer like the LAPIC to fallback to. So CPU onlining and
offlining doesn't work properly.
Fix this by recognizing that stimer Direct Mode is the normal path for
newer versions of Hyper-V on x86/64, and the only path on other
architectures. With stimer Direct Mode, stimer interrupts don't require any
VMbus machinery. stimer clockevents can be initialized and shut down
consistent with how it is done for other clockevent devices. While the old
VMbus-based stimer interrupts must still be supported for backward
compatibility on x86, that mode of operation can be treated as legacy.
So add a new Hyper-V stimer entry in the CPU hotplug state list, and use
that new state when in Direct Mode. Update the Hyper-V clocksource driver
to allocate and initialize stimer clockevents earlier during boot. Update
Hyper-V initialization and the VMbus driver to use this new design. As a
result, the LAPIC timer is no longer used during boot or CPU
onlining/offlining and clockevents_unbind_device() is not called. But
retain the old design as a legacy implementation for older versions of
Hyper-V that don't support Direct Mode.
Signed-off-by: Michael Kelley <mikelley@microsoft.com>
Signed-off-by: Thomas Gleixner <tglx@linutronix.de>
Tested-by: Dexuan Cui <decui@microsoft.com>
Reviewed-by: Dexuan Cui <decui@microsoft.com>
Link: https://lkml.kernel.org/r/1573607467-9456-1-git-send-email-mikelley@microsoft.com
2019-11-13 01:11:49 +00:00
|
|
|
err_alloc:
|
2014-03-05 12:42:14 +00:00
|
|
|
hv_remove_vmbus_irq();
|
2011-08-31 21:35:55 +00:00
|
|
|
|
|
|
|
bus_unregister(&hv_bus);
|
2018-07-28 21:58:46 +00:00
|
|
|
unregister_sysctl_table(hv_ctl_table_hdr);
|
|
|
|
hv_ctl_table_hdr = NULL;
|
2011-08-31 21:35:55 +00:00
|
|
|
return ret;
|
2009-07-13 23:02:34 +00:00
|
|
|
}
|
|
|
|
|
2009-09-02 14:11:14 +00:00
|
|
|
/**
|
2015-08-05 07:52:37 +00:00
|
|
|
* __vmbus_child_driver_register() - Register a vmbus's driver
|
|
|
|
* @hv_driver: Pointer to driver structure you want to register
|
2011-08-25 22:07:32 +00:00
|
|
|
* @owner: owner module of the drv
|
|
|
|
* @mod_name: module name string
|
2010-03-04 22:11:00 +00:00
|
|
|
*
|
|
|
|
* Registers the given driver with Linux through the 'driver_register()' call
|
2011-08-25 22:07:32 +00:00
|
|
|
* and sets up the hyper-v vmbus handling for this driver.
|
2010-03-04 22:11:00 +00:00
|
|
|
* It will return the state of the 'driver_register()' call.
|
|
|
|
*
|
2009-09-02 14:11:14 +00:00
|
|
|
*/
|
2011-08-25 22:07:32 +00:00
|
|
|
int __vmbus_driver_register(struct hv_driver *hv_driver, struct module *owner, const char *mod_name)
|
2009-07-13 23:02:34 +00:00
|
|
|
{
|
2009-07-27 20:47:36 +00:00
|
|
|
int ret;
|
2009-07-13 23:02:34 +00:00
|
|
|
|
2011-08-25 22:07:32 +00:00
|
|
|
pr_info("registering driver %s\n", hv_driver->name);
|
2009-07-13 23:02:34 +00:00
|
|
|
|
2011-12-01 17:59:34 +00:00
|
|
|
ret = vmbus_exists();
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
|
2011-08-25 22:07:32 +00:00
|
|
|
hv_driver->driver.name = hv_driver->name;
|
|
|
|
hv_driver->driver.owner = owner;
|
|
|
|
hv_driver->driver.mod_name = mod_name;
|
|
|
|
hv_driver->driver.bus = &hv_bus;
|
2009-07-13 23:02:34 +00:00
|
|
|
|
2016-12-03 20:34:39 +00:00
|
|
|
spin_lock_init(&hv_driver->dynids.lock);
|
|
|
|
INIT_LIST_HEAD(&hv_driver->dynids.list);
|
|
|
|
|
2011-08-25 22:07:32 +00:00
|
|
|
ret = driver_register(&hv_driver->driver);
|
2009-07-13 23:02:34 +00:00
|
|
|
|
2009-07-27 20:47:36 +00:00
|
|
|
return ret;
|
2009-07-13 23:02:34 +00:00
|
|
|
}
|
2011-08-25 22:07:32 +00:00
|
|
|
EXPORT_SYMBOL_GPL(__vmbus_driver_register);
|
2009-07-13 23:02:34 +00:00
|
|
|
|
2009-09-02 14:11:14 +00:00
|
|
|
/**
|
2011-08-25 22:07:32 +00:00
|
|
|
* vmbus_driver_unregister() - Unregister a vmbus's driver
|
2015-08-05 07:52:37 +00:00
|
|
|
* @hv_driver: Pointer to driver structure you want to
|
|
|
|
* un-register
|
2010-03-04 22:11:00 +00:00
|
|
|
*
|
2011-08-25 22:07:32 +00:00
|
|
|
* Un-register the given driver that was previous registered with a call to
|
|
|
|
* vmbus_driver_register()
|
2009-09-02 14:11:14 +00:00
|
|
|
*/
|
2011-08-25 22:07:32 +00:00
|
|
|
void vmbus_driver_unregister(struct hv_driver *hv_driver)
|
2009-07-13 23:02:34 +00:00
|
|
|
{
|
2011-08-25 22:07:32 +00:00
|
|
|
pr_info("unregistering driver %s\n", hv_driver->name);
|
2009-07-13 23:02:34 +00:00
|
|
|
|
2016-12-03 20:34:39 +00:00
|
|
|
if (!vmbus_exists()) {
|
2011-12-27 21:49:37 +00:00
|
|
|
driver_unregister(&hv_driver->driver);
|
2016-12-03 20:34:39 +00:00
|
|
|
vmbus_free_dynids(hv_driver);
|
|
|
|
}
|
2009-07-13 23:02:34 +00:00
|
|
|
}
|
2011-08-25 22:07:32 +00:00
|
|
|
EXPORT_SYMBOL_GPL(vmbus_driver_unregister);
|
2009-07-13 23:02:34 +00:00
|
|
|
|
2017-09-22 03:58:49 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Called when last reference to channel is gone.
|
|
|
|
*/
|
|
|
|
static void vmbus_chan_release(struct kobject *kobj)
|
|
|
|
{
|
|
|
|
struct vmbus_channel *channel
|
|
|
|
= container_of(kobj, struct vmbus_channel, kobj);
|
|
|
|
|
|
|
|
kfree_rcu(channel, rcu);
|
|
|
|
}
|
|
|
|
|
|
|
|
struct vmbus_chan_attribute {
|
|
|
|
struct attribute attr;
|
2019-03-14 20:05:15 +00:00
|
|
|
ssize_t (*show)(struct vmbus_channel *chan, char *buf);
|
2017-09-22 03:58:49 +00:00
|
|
|
ssize_t (*store)(struct vmbus_channel *chan,
|
|
|
|
const char *buf, size_t count);
|
|
|
|
};
|
|
|
|
#define VMBUS_CHAN_ATTR(_name, _mode, _show, _store) \
|
|
|
|
struct vmbus_chan_attribute chan_attr_##_name \
|
|
|
|
= __ATTR(_name, _mode, _show, _store)
|
|
|
|
#define VMBUS_CHAN_ATTR_RW(_name) \
|
|
|
|
struct vmbus_chan_attribute chan_attr_##_name = __ATTR_RW(_name)
|
|
|
|
#define VMBUS_CHAN_ATTR_RO(_name) \
|
|
|
|
struct vmbus_chan_attribute chan_attr_##_name = __ATTR_RO(_name)
|
|
|
|
#define VMBUS_CHAN_ATTR_WO(_name) \
|
|
|
|
struct vmbus_chan_attribute chan_attr_##_name = __ATTR_WO(_name)
|
|
|
|
|
|
|
|
static ssize_t vmbus_chan_attr_show(struct kobject *kobj,
|
|
|
|
struct attribute *attr, char *buf)
|
|
|
|
{
|
|
|
|
const struct vmbus_chan_attribute *attribute
|
|
|
|
= container_of(attr, struct vmbus_chan_attribute, attr);
|
2019-03-14 20:05:15 +00:00
|
|
|
struct vmbus_channel *chan
|
2017-09-22 03:58:49 +00:00
|
|
|
= container_of(kobj, struct vmbus_channel, kobj);
|
|
|
|
|
|
|
|
if (!attribute->show)
|
|
|
|
return -EIO;
|
|
|
|
|
|
|
|
return attribute->show(chan, buf);
|
|
|
|
}
|
|
|
|
|
|
|
|
static const struct sysfs_ops vmbus_chan_sysfs_ops = {
|
|
|
|
.show = vmbus_chan_attr_show,
|
|
|
|
};
|
|
|
|
|
2019-03-14 20:05:15 +00:00
|
|
|
static ssize_t out_mask_show(struct vmbus_channel *channel, char *buf)
|
2017-09-22 03:58:49 +00:00
|
|
|
{
|
2019-03-14 20:05:15 +00:00
|
|
|
struct hv_ring_buffer_info *rbi = &channel->outbound;
|
|
|
|
ssize_t ret;
|
2017-09-22 03:58:49 +00:00
|
|
|
|
2019-03-14 20:05:15 +00:00
|
|
|
mutex_lock(&rbi->ring_buffer_mutex);
|
|
|
|
if (!rbi->ring_buffer) {
|
|
|
|
mutex_unlock(&rbi->ring_buffer_mutex);
|
2019-03-14 20:05:00 +00:00
|
|
|
return -EINVAL;
|
2019-03-14 20:05:15 +00:00
|
|
|
}
|
2019-03-14 20:05:00 +00:00
|
|
|
|
2019-03-14 20:05:15 +00:00
|
|
|
ret = sprintf(buf, "%u\n", rbi->ring_buffer->interrupt_mask);
|
|
|
|
mutex_unlock(&rbi->ring_buffer_mutex);
|
|
|
|
return ret;
|
2017-09-22 03:58:49 +00:00
|
|
|
}
|
2018-01-04 22:13:25 +00:00
|
|
|
static VMBUS_CHAN_ATTR_RO(out_mask);
|
2017-09-22 03:58:49 +00:00
|
|
|
|
2019-03-14 20:05:15 +00:00
|
|
|
static ssize_t in_mask_show(struct vmbus_channel *channel, char *buf)
|
2017-09-22 03:58:49 +00:00
|
|
|
{
|
2019-03-14 20:05:15 +00:00
|
|
|
struct hv_ring_buffer_info *rbi = &channel->inbound;
|
|
|
|
ssize_t ret;
|
2017-09-22 03:58:49 +00:00
|
|
|
|
2019-03-14 20:05:15 +00:00
|
|
|
mutex_lock(&rbi->ring_buffer_mutex);
|
|
|
|
if (!rbi->ring_buffer) {
|
|
|
|
mutex_unlock(&rbi->ring_buffer_mutex);
|
2019-03-14 20:05:00 +00:00
|
|
|
return -EINVAL;
|
2019-03-14 20:05:15 +00:00
|
|
|
}
|
2019-03-14 20:05:00 +00:00
|
|
|
|
2019-03-14 20:05:15 +00:00
|
|
|
ret = sprintf(buf, "%u\n", rbi->ring_buffer->interrupt_mask);
|
|
|
|
mutex_unlock(&rbi->ring_buffer_mutex);
|
|
|
|
return ret;
|
2017-09-22 03:58:49 +00:00
|
|
|
}
|
2018-01-04 22:13:25 +00:00
|
|
|
static VMBUS_CHAN_ATTR_RO(in_mask);
|
2017-09-22 03:58:49 +00:00
|
|
|
|
2019-03-14 20:05:15 +00:00
|
|
|
static ssize_t read_avail_show(struct vmbus_channel *channel, char *buf)
|
2017-09-22 03:58:49 +00:00
|
|
|
{
|
2019-03-14 20:05:15 +00:00
|
|
|
struct hv_ring_buffer_info *rbi = &channel->inbound;
|
|
|
|
ssize_t ret;
|
2017-09-22 03:58:49 +00:00
|
|
|
|
2019-03-14 20:05:15 +00:00
|
|
|
mutex_lock(&rbi->ring_buffer_mutex);
|
|
|
|
if (!rbi->ring_buffer) {
|
|
|
|
mutex_unlock(&rbi->ring_buffer_mutex);
|
2019-03-14 20:05:00 +00:00
|
|
|
return -EINVAL;
|
2019-03-14 20:05:15 +00:00
|
|
|
}
|
2019-03-14 20:05:00 +00:00
|
|
|
|
2019-03-14 20:05:15 +00:00
|
|
|
ret = sprintf(buf, "%u\n", hv_get_bytes_to_read(rbi));
|
|
|
|
mutex_unlock(&rbi->ring_buffer_mutex);
|
|
|
|
return ret;
|
2017-09-22 03:58:49 +00:00
|
|
|
}
|
2018-01-04 22:13:25 +00:00
|
|
|
static VMBUS_CHAN_ATTR_RO(read_avail);
|
2017-09-22 03:58:49 +00:00
|
|
|
|
2019-03-14 20:05:15 +00:00
|
|
|
static ssize_t write_avail_show(struct vmbus_channel *channel, char *buf)
|
2017-09-22 03:58:49 +00:00
|
|
|
{
|
2019-03-14 20:05:15 +00:00
|
|
|
struct hv_ring_buffer_info *rbi = &channel->outbound;
|
|
|
|
ssize_t ret;
|
2017-09-22 03:58:49 +00:00
|
|
|
|
2019-03-14 20:05:15 +00:00
|
|
|
mutex_lock(&rbi->ring_buffer_mutex);
|
|
|
|
if (!rbi->ring_buffer) {
|
|
|
|
mutex_unlock(&rbi->ring_buffer_mutex);
|
2019-03-14 20:05:00 +00:00
|
|
|
return -EINVAL;
|
2019-03-14 20:05:15 +00:00
|
|
|
}
|
2019-03-14 20:05:00 +00:00
|
|
|
|
2019-03-14 20:05:15 +00:00
|
|
|
ret = sprintf(buf, "%u\n", hv_get_bytes_to_write(rbi));
|
|
|
|
mutex_unlock(&rbi->ring_buffer_mutex);
|
|
|
|
return ret;
|
2017-09-22 03:58:49 +00:00
|
|
|
}
|
2018-01-04 22:13:25 +00:00
|
|
|
static VMBUS_CHAN_ATTR_RO(write_avail);
|
2017-09-22 03:58:49 +00:00
|
|
|
|
2019-03-14 20:05:15 +00:00
|
|
|
static ssize_t show_target_cpu(struct vmbus_channel *channel, char *buf)
|
2017-09-22 03:58:49 +00:00
|
|
|
{
|
|
|
|
return sprintf(buf, "%u\n", channel->target_cpu);
|
|
|
|
}
|
2018-01-04 22:13:25 +00:00
|
|
|
static VMBUS_CHAN_ATTR(cpu, S_IRUGO, show_target_cpu, NULL);
|
2017-09-22 03:58:49 +00:00
|
|
|
|
2019-03-14 20:05:15 +00:00
|
|
|
static ssize_t channel_pending_show(struct vmbus_channel *channel,
|
2017-09-22 03:58:49 +00:00
|
|
|
char *buf)
|
|
|
|
{
|
|
|
|
return sprintf(buf, "%d\n",
|
|
|
|
channel_pending(channel,
|
|
|
|
vmbus_connection.monitor_pages[1]));
|
|
|
|
}
|
2018-01-04 22:13:25 +00:00
|
|
|
static VMBUS_CHAN_ATTR(pending, S_IRUGO, channel_pending_show, NULL);
|
2017-09-22 03:58:49 +00:00
|
|
|
|
2019-03-14 20:05:15 +00:00
|
|
|
static ssize_t channel_latency_show(struct vmbus_channel *channel,
|
2017-09-22 03:58:49 +00:00
|
|
|
char *buf)
|
|
|
|
{
|
|
|
|
return sprintf(buf, "%d\n",
|
|
|
|
channel_latency(channel,
|
|
|
|
vmbus_connection.monitor_pages[1]));
|
|
|
|
}
|
2018-01-04 22:13:25 +00:00
|
|
|
static VMBUS_CHAN_ATTR(latency, S_IRUGO, channel_latency_show, NULL);
|
2017-09-22 03:58:49 +00:00
|
|
|
|
2019-03-14 20:05:15 +00:00
|
|
|
static ssize_t channel_interrupts_show(struct vmbus_channel *channel, char *buf)
|
2017-10-29 18:33:40 +00:00
|
|
|
{
|
|
|
|
return sprintf(buf, "%llu\n", channel->interrupts);
|
|
|
|
}
|
2018-01-04 22:13:25 +00:00
|
|
|
static VMBUS_CHAN_ATTR(interrupts, S_IRUGO, channel_interrupts_show, NULL);
|
2017-10-29 18:33:40 +00:00
|
|
|
|
2019-03-14 20:05:15 +00:00
|
|
|
static ssize_t channel_events_show(struct vmbus_channel *channel, char *buf)
|
2017-10-29 18:33:40 +00:00
|
|
|
{
|
|
|
|
return sprintf(buf, "%llu\n", channel->sig_events);
|
|
|
|
}
|
2018-01-04 22:13:25 +00:00
|
|
|
static VMBUS_CHAN_ATTR(events, S_IRUGO, channel_events_show, NULL);
|
2017-10-29 18:33:40 +00:00
|
|
|
|
2019-03-14 20:05:15 +00:00
|
|
|
static ssize_t channel_intr_in_full_show(struct vmbus_channel *channel,
|
2019-02-04 07:13:09 +00:00
|
|
|
char *buf)
|
|
|
|
{
|
|
|
|
return sprintf(buf, "%llu\n",
|
|
|
|
(unsigned long long)channel->intr_in_full);
|
|
|
|
}
|
|
|
|
static VMBUS_CHAN_ATTR(intr_in_full, 0444, channel_intr_in_full_show, NULL);
|
|
|
|
|
2019-03-14 20:05:15 +00:00
|
|
|
static ssize_t channel_intr_out_empty_show(struct vmbus_channel *channel,
|
2019-02-04 07:13:09 +00:00
|
|
|
char *buf)
|
|
|
|
{
|
|
|
|
return sprintf(buf, "%llu\n",
|
|
|
|
(unsigned long long)channel->intr_out_empty);
|
|
|
|
}
|
|
|
|
static VMBUS_CHAN_ATTR(intr_out_empty, 0444, channel_intr_out_empty_show, NULL);
|
|
|
|
|
2019-03-14 20:05:15 +00:00
|
|
|
static ssize_t channel_out_full_first_show(struct vmbus_channel *channel,
|
2019-02-04 07:13:09 +00:00
|
|
|
char *buf)
|
|
|
|
{
|
|
|
|
return sprintf(buf, "%llu\n",
|
|
|
|
(unsigned long long)channel->out_full_first);
|
|
|
|
}
|
|
|
|
static VMBUS_CHAN_ATTR(out_full_first, 0444, channel_out_full_first_show, NULL);
|
|
|
|
|
2019-03-14 20:05:15 +00:00
|
|
|
static ssize_t channel_out_full_total_show(struct vmbus_channel *channel,
|
2019-02-04 07:13:09 +00:00
|
|
|
char *buf)
|
|
|
|
{
|
|
|
|
return sprintf(buf, "%llu\n",
|
|
|
|
(unsigned long long)channel->out_full_total);
|
|
|
|
}
|
|
|
|
static VMBUS_CHAN_ATTR(out_full_total, 0444, channel_out_full_total_show, NULL);
|
|
|
|
|
2019-03-14 20:05:15 +00:00
|
|
|
static ssize_t subchannel_monitor_id_show(struct vmbus_channel *channel,
|
2018-01-09 18:29:06 +00:00
|
|
|
char *buf)
|
|
|
|
{
|
|
|
|
return sprintf(buf, "%u\n", channel->offermsg.monitorid);
|
|
|
|
}
|
|
|
|
static VMBUS_CHAN_ATTR(monitor_id, S_IRUGO, subchannel_monitor_id_show, NULL);
|
|
|
|
|
2019-03-14 20:05:15 +00:00
|
|
|
static ssize_t subchannel_id_show(struct vmbus_channel *channel,
|
2018-01-09 18:29:06 +00:00
|
|
|
char *buf)
|
|
|
|
{
|
|
|
|
return sprintf(buf, "%u\n",
|
|
|
|
channel->offermsg.offer.sub_channel_index);
|
|
|
|
}
|
|
|
|
static VMBUS_CHAN_ATTR_RO(subchannel_id);
|
|
|
|
|
2017-09-22 03:58:49 +00:00
|
|
|
static struct attribute *vmbus_chan_attrs[] = {
|
|
|
|
&chan_attr_out_mask.attr,
|
|
|
|
&chan_attr_in_mask.attr,
|
|
|
|
&chan_attr_read_avail.attr,
|
|
|
|
&chan_attr_write_avail.attr,
|
|
|
|
&chan_attr_cpu.attr,
|
|
|
|
&chan_attr_pending.attr,
|
|
|
|
&chan_attr_latency.attr,
|
2017-10-29 18:33:40 +00:00
|
|
|
&chan_attr_interrupts.attr,
|
|
|
|
&chan_attr_events.attr,
|
2019-02-04 07:13:09 +00:00
|
|
|
&chan_attr_intr_in_full.attr,
|
|
|
|
&chan_attr_intr_out_empty.attr,
|
|
|
|
&chan_attr_out_full_first.attr,
|
|
|
|
&chan_attr_out_full_total.attr,
|
2018-01-09 18:29:06 +00:00
|
|
|
&chan_attr_monitor_id.attr,
|
|
|
|
&chan_attr_subchannel_id.attr,
|
2017-09-22 03:58:49 +00:00
|
|
|
NULL
|
|
|
|
};
|
|
|
|
|
2019-03-19 04:04:01 +00:00
|
|
|
/*
|
|
|
|
* Channel-level attribute_group callback function. Returns the permission for
|
|
|
|
* each attribute, and returns 0 if an attribute is not visible.
|
|
|
|
*/
|
|
|
|
static umode_t vmbus_chan_attr_is_visible(struct kobject *kobj,
|
|
|
|
struct attribute *attr, int idx)
|
|
|
|
{
|
|
|
|
const struct vmbus_channel *channel =
|
|
|
|
container_of(kobj, struct vmbus_channel, kobj);
|
|
|
|
|
|
|
|
/* Hide the monitor attributes if the monitor mechanism is not used. */
|
|
|
|
if (!channel->offermsg.monitor_allocated &&
|
|
|
|
(attr == &chan_attr_pending.attr ||
|
|
|
|
attr == &chan_attr_latency.attr ||
|
|
|
|
attr == &chan_attr_monitor_id.attr))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
return attr->mode;
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct attribute_group vmbus_chan_group = {
|
|
|
|
.attrs = vmbus_chan_attrs,
|
|
|
|
.is_visible = vmbus_chan_attr_is_visible
|
|
|
|
};
|
|
|
|
|
2017-09-22 03:58:49 +00:00
|
|
|
static struct kobj_type vmbus_chan_ktype = {
|
|
|
|
.sysfs_ops = &vmbus_chan_sysfs_ops,
|
|
|
|
.release = vmbus_chan_release,
|
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
|
|
* vmbus_add_channel_kobj - setup a sub-directory under device/channels
|
|
|
|
*/
|
|
|
|
int vmbus_add_channel_kobj(struct hv_device *dev, struct vmbus_channel *channel)
|
|
|
|
{
|
2019-03-19 04:04:01 +00:00
|
|
|
const struct device *device = &dev->device;
|
2017-09-22 03:58:49 +00:00
|
|
|
struct kobject *kobj = &channel->kobj;
|
|
|
|
u32 relid = channel->offermsg.child_relid;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
kobj->kset = dev->channels_kset;
|
|
|
|
ret = kobject_init_and_add(kobj, &vmbus_chan_ktype, NULL,
|
|
|
|
"%u", relid);
|
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
|
2019-03-19 04:04:01 +00:00
|
|
|
ret = sysfs_create_group(kobj, &vmbus_chan_group);
|
|
|
|
|
|
|
|
if (ret) {
|
|
|
|
/*
|
|
|
|
* The calling functions' error handling paths will cleanup the
|
|
|
|
* empty channel directory.
|
|
|
|
*/
|
|
|
|
dev_err(device, "Unable to set up channel sysfs files\n");
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2017-09-22 03:58:49 +00:00
|
|
|
kobject_uevent(kobj, KOBJ_ADD);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2019-03-19 04:04:01 +00:00
|
|
|
/*
|
|
|
|
* vmbus_remove_channel_attr_group - remove the channel's attribute group
|
|
|
|
*/
|
|
|
|
void vmbus_remove_channel_attr_group(struct vmbus_channel *channel)
|
|
|
|
{
|
|
|
|
sysfs_remove_group(&channel->kobj, &vmbus_chan_group);
|
|
|
|
}
|
|
|
|
|
2010-03-04 22:11:00 +00:00
|
|
|
/*
|
2011-09-08 14:24:12 +00:00
|
|
|
* vmbus_device_create - Creates and registers a new child device
|
2010-03-04 22:11:00 +00:00
|
|
|
* on the vmbus.
|
2009-09-02 14:11:14 +00:00
|
|
|
*/
|
2019-01-10 14:25:32 +00:00
|
|
|
struct hv_device *vmbus_device_create(const guid_t *type,
|
|
|
|
const guid_t *instance,
|
2014-06-03 15:38:15 +00:00
|
|
|
struct vmbus_channel *channel)
|
2009-07-13 23:02:34 +00:00
|
|
|
{
|
2009-07-28 15:32:53 +00:00
|
|
|
struct hv_device *child_device_obj;
|
2009-07-13 23:02:34 +00:00
|
|
|
|
2011-03-07 21:35:48 +00:00
|
|
|
child_device_obj = kzalloc(sizeof(struct hv_device), GFP_KERNEL);
|
|
|
|
if (!child_device_obj) {
|
2011-03-29 20:58:47 +00:00
|
|
|
pr_err("Unable to allocate device object for child device\n");
|
2009-07-13 23:02:34 +00:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2010-10-21 16:05:27 +00:00
|
|
|
child_device_obj->channel = channel;
|
2019-01-10 14:25:32 +00:00
|
|
|
guid_copy(&child_device_obj->dev_type, type);
|
|
|
|
guid_copy(&child_device_obj->dev_instance, instance);
|
2015-12-26 04:00:30 +00:00
|
|
|
child_device_obj->vendor_id = 0x1414; /* MSFT vendor ID */
|
2009-07-13 23:02:34 +00:00
|
|
|
|
|
|
|
return child_device_obj;
|
|
|
|
}
|
|
|
|
|
2010-03-04 22:11:00 +00:00
|
|
|
/*
|
2011-09-08 14:24:13 +00:00
|
|
|
* vmbus_device_register - Register the child device
|
2009-09-02 14:11:14 +00:00
|
|
|
*/
|
2011-09-08 14:24:13 +00:00
|
|
|
int vmbus_device_register(struct hv_device *child_device_obj)
|
2009-07-13 23:02:34 +00:00
|
|
|
{
|
2017-09-22 03:58:49 +00:00
|
|
|
struct kobject *kobj = &child_device_obj->device.kobj;
|
|
|
|
int ret;
|
2011-03-07 21:35:48 +00:00
|
|
|
|
2016-11-01 07:01:59 +00:00
|
|
|
dev_set_name(&child_device_obj->device, "%pUl",
|
2016-09-16 16:01:17 +00:00
|
|
|
child_device_obj->channel->offermsg.offer.if_instance.b);
|
2009-07-13 23:02:34 +00:00
|
|
|
|
2011-08-27 18:31:39 +00:00
|
|
|
child_device_obj->device.bus = &hv_bus;
|
2011-06-06 22:49:39 +00:00
|
|
|
child_device_obj->device.parent = &hv_acpi_dev->dev;
|
2011-03-07 21:35:48 +00:00
|
|
|
child_device_obj->device.release = vmbus_device_release;
|
2009-07-13 23:02:34 +00:00
|
|
|
|
2009-09-02 14:11:14 +00:00
|
|
|
/*
|
|
|
|
* Register with the LDM. This will kick off the driver/device
|
|
|
|
* binding...which will eventually call vmbus_match() and vmbus_probe()
|
|
|
|
*/
|
2011-03-07 21:35:48 +00:00
|
|
|
ret = device_register(&child_device_obj->device);
|
2017-09-22 03:58:49 +00:00
|
|
|
if (ret) {
|
2011-03-29 20:58:47 +00:00
|
|
|
pr_err("Unable to register child device\n");
|
2017-09-22 03:58:49 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
child_device_obj->channels_kset = kset_create_and_add("channels",
|
|
|
|
NULL, kobj);
|
|
|
|
if (!child_device_obj->channels_kset) {
|
|
|
|
ret = -ENOMEM;
|
|
|
|
goto err_dev_unregister;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = vmbus_add_channel_kobj(child_device_obj,
|
|
|
|
child_device_obj->channel);
|
|
|
|
if (ret) {
|
|
|
|
pr_err("Unable to register primary channeln");
|
|
|
|
goto err_kset_unregister;
|
|
|
|
}
|
2019-10-03 21:01:49 +00:00
|
|
|
hv_debug_add_dev_dir(child_device_obj);
|
2017-09-22 03:58:49 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
err_kset_unregister:
|
|
|
|
kset_unregister(child_device_obj->channels_kset);
|
2009-07-13 23:02:34 +00:00
|
|
|
|
2017-09-22 03:58:49 +00:00
|
|
|
err_dev_unregister:
|
|
|
|
device_unregister(&child_device_obj->device);
|
2009-07-13 23:02:34 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2010-03-04 22:11:00 +00:00
|
|
|
/*
|
2011-09-08 14:24:14 +00:00
|
|
|
* vmbus_device_unregister - Remove the specified child device
|
2010-03-04 22:11:00 +00:00
|
|
|
* from the vmbus.
|
2009-09-02 14:11:14 +00:00
|
|
|
*/
|
2011-09-08 14:24:14 +00:00
|
|
|
void vmbus_device_unregister(struct hv_device *device_obj)
|
2009-07-13 23:02:34 +00:00
|
|
|
{
|
2013-06-14 23:13:35 +00:00
|
|
|
pr_debug("child device %s unregistered\n",
|
|
|
|
dev_name(&device_obj->device));
|
|
|
|
|
2017-11-14 13:53:32 +00:00
|
|
|
kset_unregister(device_obj->channels_kset);
|
|
|
|
|
2009-09-02 14:11:14 +00:00
|
|
|
/*
|
|
|
|
* Kick off the process of unregistering the device.
|
|
|
|
* This will call vmbus_remove() and eventually vmbus_device_release()
|
|
|
|
*/
|
2011-03-07 21:35:48 +00:00
|
|
|
device_unregister(&device_obj->device);
|
2009-07-13 23:02:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-04-29 20:45:15 +00:00
|
|
|
/*
|
2015-08-05 07:52:36 +00:00
|
|
|
* VMBUS is an acpi enumerated device. Get the information we
|
2014-01-30 02:14:39 +00:00
|
|
|
* need from DSDT.
|
2011-04-29 20:45:15 +00:00
|
|
|
*/
|
2015-08-05 07:52:36 +00:00
|
|
|
#define VTPM_BASE_ADDRESS 0xfed40000
|
2014-01-30 02:14:39 +00:00
|
|
|
static acpi_status vmbus_walk_resources(struct acpi_resource *res, void *ctx)
|
2011-04-29 20:45:15 +00:00
|
|
|
{
|
2015-08-05 07:52:36 +00:00
|
|
|
resource_size_t start = 0;
|
|
|
|
resource_size_t end = 0;
|
|
|
|
struct resource *new_res;
|
|
|
|
struct resource **old_res = &hyperv_mmio;
|
|
|
|
struct resource **prev_res = NULL;
|
|
|
|
|
2014-01-30 02:14:39 +00:00
|
|
|
switch (res->type) {
|
2015-08-05 07:52:36 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* "Address" descriptors are for bus windows. Ignore
|
|
|
|
* "memory" descriptors, which are for registers on
|
|
|
|
* devices.
|
|
|
|
*/
|
|
|
|
case ACPI_RESOURCE_TYPE_ADDRESS32:
|
|
|
|
start = res->data.address32.address.minimum;
|
|
|
|
end = res->data.address32.address.maximum;
|
2014-02-24 13:17:08 +00:00
|
|
|
break;
|
2011-04-29 20:45:15 +00:00
|
|
|
|
2014-01-30 02:14:39 +00:00
|
|
|
case ACPI_RESOURCE_TYPE_ADDRESS64:
|
2015-08-05 07:52:36 +00:00
|
|
|
start = res->data.address64.address.minimum;
|
|
|
|
end = res->data.address64.address.maximum;
|
2014-02-24 13:17:08 +00:00
|
|
|
break;
|
2015-08-05 07:52:36 +00:00
|
|
|
|
|
|
|
default:
|
|
|
|
/* Unused resource type */
|
|
|
|
return AE_OK;
|
|
|
|
|
2011-04-29 20:45:15 +00:00
|
|
|
}
|
2015-08-05 07:52:36 +00:00
|
|
|
/*
|
|
|
|
* Ignore ranges that are below 1MB, as they're not
|
|
|
|
* necessary or useful here.
|
|
|
|
*/
|
|
|
|
if (end < 0x100000)
|
|
|
|
return AE_OK;
|
|
|
|
|
|
|
|
new_res = kzalloc(sizeof(*new_res), GFP_ATOMIC);
|
|
|
|
if (!new_res)
|
|
|
|
return AE_NO_MEMORY;
|
|
|
|
|
|
|
|
/* If this range overlaps the virtual TPM, truncate it. */
|
|
|
|
if (end > VTPM_BASE_ADDRESS && start < VTPM_BASE_ADDRESS)
|
|
|
|
end = VTPM_BASE_ADDRESS;
|
|
|
|
|
|
|
|
new_res->name = "hyperv mmio";
|
|
|
|
new_res->flags = IORESOURCE_MEM;
|
|
|
|
new_res->start = start;
|
|
|
|
new_res->end = end;
|
|
|
|
|
2015-12-15 00:01:52 +00:00
|
|
|
/*
|
|
|
|
* If two ranges are adjacent, merge them.
|
|
|
|
*/
|
2015-08-05 07:52:36 +00:00
|
|
|
do {
|
|
|
|
if (!*old_res) {
|
|
|
|
*old_res = new_res;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2015-12-15 00:01:52 +00:00
|
|
|
if (((*old_res)->end + 1) == new_res->start) {
|
|
|
|
(*old_res)->end = new_res->end;
|
|
|
|
kfree(new_res);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((*old_res)->start == new_res->end + 1) {
|
|
|
|
(*old_res)->start = new_res->start;
|
|
|
|
kfree(new_res);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2016-04-05 17:22:53 +00:00
|
|
|
if ((*old_res)->start > new_res->end) {
|
2015-08-05 07:52:36 +00:00
|
|
|
new_res->sibling = *old_res;
|
|
|
|
if (prev_res)
|
|
|
|
(*prev_res)->sibling = new_res;
|
|
|
|
*old_res = new_res;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
prev_res = old_res;
|
|
|
|
old_res = &(*old_res)->sibling;
|
|
|
|
|
|
|
|
} while (1);
|
2011-04-29 20:45:15 +00:00
|
|
|
|
|
|
|
return AE_OK;
|
|
|
|
}
|
|
|
|
|
2015-08-05 07:52:36 +00:00
|
|
|
static int vmbus_acpi_remove(struct acpi_device *device)
|
|
|
|
{
|
|
|
|
struct resource *cur_res;
|
|
|
|
struct resource *next_res;
|
|
|
|
|
|
|
|
if (hyperv_mmio) {
|
2016-04-05 17:22:55 +00:00
|
|
|
if (fb_mmio) {
|
|
|
|
__release_region(hyperv_mmio, fb_mmio->start,
|
|
|
|
resource_size(fb_mmio));
|
|
|
|
fb_mmio = NULL;
|
|
|
|
}
|
|
|
|
|
2015-08-05 07:52:36 +00:00
|
|
|
for (cur_res = hyperv_mmio; cur_res; cur_res = next_res) {
|
|
|
|
next_res = cur_res->sibling;
|
|
|
|
kfree(cur_res);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2016-04-05 17:22:55 +00:00
|
|
|
static void vmbus_reserve_fb(void)
|
|
|
|
{
|
|
|
|
int size;
|
|
|
|
/*
|
|
|
|
* Make a claim for the frame buffer in the resource tree under the
|
|
|
|
* first node, which will be the one below 4GB. The length seems to
|
|
|
|
* be underreported, particularly in a Generation 1 VM. So start out
|
|
|
|
* reserving a larger area and make it smaller until it succeeds.
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (screen_info.lfb_base) {
|
|
|
|
if (efi_enabled(EFI_BOOT))
|
|
|
|
size = max_t(__u32, screen_info.lfb_size, 0x800000);
|
|
|
|
else
|
|
|
|
size = max_t(__u32, screen_info.lfb_size, 0x4000000);
|
|
|
|
|
|
|
|
for (; !fb_mmio && (size >= 0x100000); size >>= 1) {
|
|
|
|
fb_mmio = __request_region(hyperv_mmio,
|
|
|
|
screen_info.lfb_base, size,
|
|
|
|
fb_mmio_name, 0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-08-05 07:52:37 +00:00
|
|
|
/**
|
|
|
|
* vmbus_allocate_mmio() - Pick a memory-mapped I/O range.
|
|
|
|
* @new: If successful, supplied a pointer to the
|
|
|
|
* allocated MMIO space.
|
|
|
|
* @device_obj: Identifies the caller
|
|
|
|
* @min: Minimum guest physical address of the
|
|
|
|
* allocation
|
|
|
|
* @max: Maximum guest physical address
|
|
|
|
* @size: Size of the range to be allocated
|
|
|
|
* @align: Alignment of the range to be allocated
|
|
|
|
* @fb_overlap_ok: Whether this allocation can be allowed
|
|
|
|
* to overlap the video frame buffer.
|
|
|
|
*
|
|
|
|
* This function walks the resources granted to VMBus by the
|
|
|
|
* _CRS object in the ACPI namespace underneath the parent
|
|
|
|
* "bridge" whether that's a root PCI bus in the Generation 1
|
|
|
|
* case or a Module Device in the Generation 2 case. It then
|
|
|
|
* attempts to allocate from the global MMIO pool in a way that
|
|
|
|
* matches the constraints supplied in these parameters and by
|
|
|
|
* that _CRS.
|
|
|
|
*
|
|
|
|
* Return: 0 on success, -errno on failure
|
|
|
|
*/
|
|
|
|
int vmbus_allocate_mmio(struct resource **new, struct hv_device *device_obj,
|
|
|
|
resource_size_t min, resource_size_t max,
|
|
|
|
resource_size_t size, resource_size_t align,
|
|
|
|
bool fb_overlap_ok)
|
|
|
|
{
|
2016-04-05 17:22:54 +00:00
|
|
|
struct resource *iter, *shadow;
|
2016-04-05 17:22:56 +00:00
|
|
|
resource_size_t range_min, range_max, start;
|
2015-08-05 07:52:37 +00:00
|
|
|
const char *dev_n = dev_name(&device_obj->device);
|
2016-04-05 17:22:56 +00:00
|
|
|
int retval;
|
2016-04-05 17:22:50 +00:00
|
|
|
|
|
|
|
retval = -ENXIO;
|
2019-11-01 20:00:04 +00:00
|
|
|
mutex_lock(&hyperv_mmio_lock);
|
2015-08-05 07:52:37 +00:00
|
|
|
|
2016-04-05 17:22:56 +00:00
|
|
|
/*
|
|
|
|
* If overlaps with frame buffers are allowed, then first attempt to
|
|
|
|
* make the allocation from within the reserved region. Because it
|
|
|
|
* is already reserved, no shadow allocation is necessary.
|
|
|
|
*/
|
|
|
|
if (fb_overlap_ok && fb_mmio && !(min > fb_mmio->end) &&
|
|
|
|
!(max < fb_mmio->start)) {
|
|
|
|
|
|
|
|
range_min = fb_mmio->start;
|
|
|
|
range_max = fb_mmio->end;
|
|
|
|
start = (range_min + align - 1) & ~(align - 1);
|
|
|
|
for (; start + size - 1 <= range_max; start += align) {
|
|
|
|
*new = request_mem_region_exclusive(start, size, dev_n);
|
|
|
|
if (*new) {
|
|
|
|
retval = 0;
|
|
|
|
goto exit;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-08-05 07:52:37 +00:00
|
|
|
for (iter = hyperv_mmio; iter; iter = iter->sibling) {
|
|
|
|
if ((iter->start >= max) || (iter->end <= min))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
range_min = iter->start;
|
|
|
|
range_max = iter->end;
|
2016-04-05 17:22:56 +00:00
|
|
|
start = (range_min + align - 1) & ~(align - 1);
|
|
|
|
for (; start + size - 1 <= range_max; start += align) {
|
|
|
|
shadow = __request_region(iter, start, size, NULL,
|
|
|
|
IORESOURCE_BUSY);
|
|
|
|
if (!shadow)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
*new = request_mem_region_exclusive(start, size, dev_n);
|
|
|
|
if (*new) {
|
|
|
|
shadow->name = (char *)*new;
|
|
|
|
retval = 0;
|
|
|
|
goto exit;
|
2015-08-05 07:52:37 +00:00
|
|
|
}
|
|
|
|
|
2016-04-05 17:22:56 +00:00
|
|
|
__release_region(iter, start, size);
|
2015-08-05 07:52:37 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-04-05 17:22:50 +00:00
|
|
|
exit:
|
2019-11-01 20:00:04 +00:00
|
|
|
mutex_unlock(&hyperv_mmio_lock);
|
2016-04-05 17:22:50 +00:00
|
|
|
return retval;
|
2015-08-05 07:52:37 +00:00
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(vmbus_allocate_mmio);
|
|
|
|
|
2016-04-05 17:22:51 +00:00
|
|
|
/**
|
|
|
|
* vmbus_free_mmio() - Free a memory-mapped I/O range.
|
|
|
|
* @start: Base address of region to release.
|
|
|
|
* @size: Size of the range to be allocated
|
|
|
|
*
|
|
|
|
* This function releases anything requested by
|
|
|
|
* vmbus_mmio_allocate().
|
|
|
|
*/
|
|
|
|
void vmbus_free_mmio(resource_size_t start, resource_size_t size)
|
|
|
|
{
|
2016-04-05 17:22:54 +00:00
|
|
|
struct resource *iter;
|
|
|
|
|
2019-11-01 20:00:04 +00:00
|
|
|
mutex_lock(&hyperv_mmio_lock);
|
2016-04-05 17:22:54 +00:00
|
|
|
for (iter = hyperv_mmio; iter; iter = iter->sibling) {
|
|
|
|
if ((iter->start >= start + size) || (iter->end <= start))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
__release_region(iter, start, size);
|
|
|
|
}
|
2016-04-05 17:22:51 +00:00
|
|
|
release_mem_region(start, size);
|
2019-11-01 20:00:04 +00:00
|
|
|
mutex_unlock(&hyperv_mmio_lock);
|
2016-04-05 17:22:51 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(vmbus_free_mmio);
|
|
|
|
|
2011-04-29 20:45:15 +00:00
|
|
|
static int vmbus_acpi_add(struct acpi_device *device)
|
|
|
|
{
|
|
|
|
acpi_status result;
|
2014-01-30 02:14:39 +00:00
|
|
|
int ret_val = -ENODEV;
|
2015-08-05 07:52:36 +00:00
|
|
|
struct acpi_device *ancestor;
|
2011-04-29 20:45:15 +00:00
|
|
|
|
2011-06-06 22:49:39 +00:00
|
|
|
hv_acpi_dev = device;
|
|
|
|
|
2011-08-27 18:31:38 +00:00
|
|
|
result = acpi_walk_resources(device->handle, METHOD_NAME__CRS,
|
2014-01-30 02:14:39 +00:00
|
|
|
vmbus_walk_resources, NULL);
|
2011-04-29 20:45:15 +00:00
|
|
|
|
2014-01-30 02:14:39 +00:00
|
|
|
if (ACPI_FAILURE(result))
|
|
|
|
goto acpi_walk_err;
|
|
|
|
/*
|
2015-08-05 07:52:36 +00:00
|
|
|
* Some ancestor of the vmbus acpi device (Gen1 or Gen2
|
|
|
|
* firmware) is the VMOD that has the mmio ranges. Get that.
|
2014-01-30 02:14:39 +00:00
|
|
|
*/
|
2015-08-05 07:52:36 +00:00
|
|
|
for (ancestor = device->parent; ancestor; ancestor = ancestor->parent) {
|
|
|
|
result = acpi_walk_resources(ancestor->handle, METHOD_NAME__CRS,
|
|
|
|
vmbus_walk_resources, NULL);
|
2014-01-30 02:14:39 +00:00
|
|
|
|
|
|
|
if (ACPI_FAILURE(result))
|
2015-08-05 07:52:36 +00:00
|
|
|
continue;
|
2016-04-05 17:22:55 +00:00
|
|
|
if (hyperv_mmio) {
|
|
|
|
vmbus_reserve_fb();
|
2015-08-05 07:52:36 +00:00
|
|
|
break;
|
2016-04-05 17:22:55 +00:00
|
|
|
}
|
2011-04-29 20:45:15 +00:00
|
|
|
}
|
2014-01-30 02:14:39 +00:00
|
|
|
ret_val = 0;
|
|
|
|
|
|
|
|
acpi_walk_err:
|
2011-04-29 20:45:15 +00:00
|
|
|
complete(&probe_event);
|
2015-08-05 07:52:36 +00:00
|
|
|
if (ret_val)
|
|
|
|
vmbus_acpi_remove(device);
|
2014-01-30 02:14:39 +00:00
|
|
|
return ret_val;
|
2011-04-29 20:45:15 +00:00
|
|
|
}
|
|
|
|
|
2019-09-19 21:46:12 +00:00
|
|
|
#ifdef CONFIG_PM_SLEEP
|
2019-09-05 23:01:19 +00:00
|
|
|
static int vmbus_bus_suspend(struct device *dev)
|
|
|
|
{
|
2019-09-05 23:01:21 +00:00
|
|
|
struct vmbus_channel *channel, *sc;
|
|
|
|
unsigned long flags;
|
2019-09-05 23:01:20 +00:00
|
|
|
|
|
|
|
while (atomic_read(&vmbus_connection.offer_in_progress) != 0) {
|
|
|
|
/*
|
|
|
|
* We wait here until the completion of any channel
|
|
|
|
* offers that are currently in progress.
|
|
|
|
*/
|
|
|
|
msleep(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
mutex_lock(&vmbus_connection.channel_mutex);
|
|
|
|
list_for_each_entry(channel, &vmbus_connection.chn_list, listentry) {
|
|
|
|
if (!is_hvsock_channel(channel))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
vmbus_force_channel_rescinded(channel);
|
|
|
|
}
|
|
|
|
mutex_unlock(&vmbus_connection.channel_mutex);
|
|
|
|
|
2019-09-05 23:01:21 +00:00
|
|
|
/*
|
|
|
|
* Wait until all the sub-channels and hv_sock channels have been
|
|
|
|
* cleaned up. Sub-channels should be destroyed upon suspend, otherwise
|
|
|
|
* they would conflict with the new sub-channels that will be created
|
|
|
|
* in the resume path. hv_sock channels should also be destroyed, but
|
|
|
|
* a hv_sock channel of an established hv_sock connection can not be
|
|
|
|
* really destroyed since it may still be referenced by the userspace
|
|
|
|
* application, so we just force the hv_sock channel to be rescinded
|
|
|
|
* by vmbus_force_channel_rescinded(), and the userspace application
|
|
|
|
* will thoroughly destroy the channel after hibernation.
|
|
|
|
*
|
|
|
|
* Note: the counter nr_chan_close_on_suspend may never go above 0 if
|
|
|
|
* the VM has no sub-channel and hv_sock channel, e.g. a 1-vCPU VM.
|
|
|
|
*/
|
|
|
|
if (atomic_read(&vmbus_connection.nr_chan_close_on_suspend) > 0)
|
|
|
|
wait_for_completion(&vmbus_connection.ready_for_suspend_event);
|
|
|
|
|
2019-09-05 23:01:22 +00:00
|
|
|
WARN_ON(atomic_read(&vmbus_connection.nr_chan_fixup_on_resume) != 0);
|
|
|
|
|
2019-09-05 23:01:21 +00:00
|
|
|
mutex_lock(&vmbus_connection.channel_mutex);
|
|
|
|
|
|
|
|
list_for_each_entry(channel, &vmbus_connection.chn_list, listentry) {
|
2019-09-05 23:01:22 +00:00
|
|
|
/*
|
2020-04-06 00:15:06 +00:00
|
|
|
* Remove the channel from the array of channels and invalidate
|
|
|
|
* the channel's relid. Upon resume, vmbus_onoffer() will fix
|
|
|
|
* up the relid (and other fields, if necessary) and add the
|
|
|
|
* channel back to the array.
|
2019-09-05 23:01:22 +00:00
|
|
|
*/
|
2020-04-06 00:15:06 +00:00
|
|
|
vmbus_channel_unmap_relid(channel);
|
2019-09-05 23:01:22 +00:00
|
|
|
channel->offermsg.child_relid = INVALID_RELID;
|
|
|
|
|
2019-09-05 23:01:21 +00:00
|
|
|
if (is_hvsock_channel(channel)) {
|
|
|
|
if (!channel->rescind) {
|
|
|
|
pr_err("hv_sock channel not rescinded!\n");
|
|
|
|
WARN_ON_ONCE(1);
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
spin_lock_irqsave(&channel->lock, flags);
|
|
|
|
list_for_each_entry(sc, &channel->sc_list, sc_list) {
|
|
|
|
pr_err("Sub-channel not deleted!\n");
|
|
|
|
WARN_ON_ONCE(1);
|
|
|
|
}
|
|
|
|
spin_unlock_irqrestore(&channel->lock, flags);
|
2019-09-05 23:01:22 +00:00
|
|
|
|
|
|
|
atomic_inc(&vmbus_connection.nr_chan_fixup_on_resume);
|
2019-09-05 23:01:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
mutex_unlock(&vmbus_connection.channel_mutex);
|
|
|
|
|
2019-09-05 23:01:19 +00:00
|
|
|
vmbus_initiate_unload(false);
|
|
|
|
|
2019-09-05 23:01:22 +00:00
|
|
|
/* Reset the event for the next resume. */
|
|
|
|
reinit_completion(&vmbus_connection.ready_for_resume_event);
|
|
|
|
|
2019-09-05 23:01:19 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int vmbus_bus_resume(struct device *dev)
|
|
|
|
{
|
|
|
|
struct vmbus_channel_msginfo *msginfo;
|
|
|
|
size_t msgsize;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* We only use the 'vmbus_proto_version', which was in use before
|
|
|
|
* hibernation, to re-negotiate with the host.
|
|
|
|
*/
|
2019-10-15 11:46:44 +00:00
|
|
|
if (!vmbus_proto_version) {
|
2019-09-05 23:01:19 +00:00
|
|
|
pr_err("Invalid proto version = 0x%x\n", vmbus_proto_version);
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
msgsize = sizeof(*msginfo) +
|
|
|
|
sizeof(struct vmbus_channel_initiate_contact);
|
|
|
|
|
|
|
|
msginfo = kzalloc(msgsize, GFP_KERNEL);
|
|
|
|
|
|
|
|
if (msginfo == NULL)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
ret = vmbus_negotiate_version(msginfo, vmbus_proto_version);
|
|
|
|
|
|
|
|
kfree(msginfo);
|
|
|
|
|
|
|
|
if (ret != 0)
|
|
|
|
return ret;
|
|
|
|
|
2019-09-05 23:01:22 +00:00
|
|
|
WARN_ON(atomic_read(&vmbus_connection.nr_chan_fixup_on_resume) == 0);
|
|
|
|
|
2019-09-05 23:01:19 +00:00
|
|
|
vmbus_request_offers();
|
|
|
|
|
2019-09-05 23:01:22 +00:00
|
|
|
wait_for_completion(&vmbus_connection.ready_for_resume_event);
|
|
|
|
|
2019-09-05 23:01:21 +00:00
|
|
|
/* Reset the event for the next suspend. */
|
|
|
|
reinit_completion(&vmbus_connection.ready_for_suspend_event);
|
|
|
|
|
2019-09-05 23:01:19 +00:00
|
|
|
return 0;
|
|
|
|
}
|
2019-09-19 21:46:12 +00:00
|
|
|
#endif /* CONFIG_PM_SLEEP */
|
2019-09-05 23:01:19 +00:00
|
|
|
|
2011-04-29 20:45:15 +00:00
|
|
|
static const struct acpi_device_id vmbus_acpi_device_ids[] = {
|
|
|
|
{"VMBUS", 0},
|
2011-06-06 22:49:42 +00:00
|
|
|
{"VMBus", 0},
|
2011-04-29 20:45:15 +00:00
|
|
|
{"", 0},
|
|
|
|
};
|
|
|
|
MODULE_DEVICE_TABLE(acpi, vmbus_acpi_device_ids);
|
|
|
|
|
2019-09-05 23:01:19 +00:00
|
|
|
/*
|
|
|
|
* Note: we must use SET_NOIRQ_SYSTEM_SLEEP_PM_OPS rather than
|
|
|
|
* SET_SYSTEM_SLEEP_PM_OPS, otherwise NIC SR-IOV can not work, because the
|
|
|
|
* "pci_dev_pm_ops" uses the "noirq" callbacks: in the resume path, the
|
|
|
|
* pci "noirq" restore callback runs before "non-noirq" callbacks (see
|
|
|
|
* resume_target_kernel() -> dpm_resume_start(), and hibernation_restore() ->
|
|
|
|
* dpm_resume_end()). This means vmbus_bus_resume() and the pci-hyperv's
|
|
|
|
* resume callback must also run via the "noirq" callbacks.
|
|
|
|
*/
|
|
|
|
static const struct dev_pm_ops vmbus_bus_pm = {
|
|
|
|
SET_NOIRQ_SYSTEM_SLEEP_PM_OPS(vmbus_bus_suspend, vmbus_bus_resume)
|
|
|
|
};
|
|
|
|
|
2011-04-29 20:45:15 +00:00
|
|
|
static struct acpi_driver vmbus_acpi_driver = {
|
|
|
|
.name = "vmbus",
|
|
|
|
.ids = vmbus_acpi_device_ids,
|
|
|
|
.ops = {
|
|
|
|
.add = vmbus_acpi_add,
|
2015-04-23 04:31:28 +00:00
|
|
|
.remove = vmbus_acpi_remove,
|
2011-04-29 20:45:15 +00:00
|
|
|
},
|
2019-09-05 23:01:19 +00:00
|
|
|
.drv.pm = &vmbus_bus_pm,
|
2011-04-29 20:45:15 +00:00
|
|
|
};
|
|
|
|
|
2015-08-01 23:08:07 +00:00
|
|
|
static void hv_kexec_handler(void)
|
|
|
|
{
|
2019-07-01 04:25:56 +00:00
|
|
|
hv_stimer_global_cleanup();
|
2016-02-26 23:13:16 +00:00
|
|
|
vmbus_initiate_unload(false);
|
2016-12-07 22:53:12 +00:00
|
|
|
/* Make sure conn_state is set as hv_synic_cleanup checks for it */
|
|
|
|
mb();
|
2016-12-07 22:53:11 +00:00
|
|
|
cpuhp_remove_state(hyperv_cpuhp_online);
|
2017-01-28 19:37:14 +00:00
|
|
|
hyperv_cleanup();
|
2015-08-01 23:08:07 +00:00
|
|
|
};
|
|
|
|
|
2015-08-01 23:08:09 +00:00
|
|
|
static void hv_crash_handler(struct pt_regs *regs)
|
|
|
|
{
|
2019-07-01 04:25:56 +00:00
|
|
|
int cpu;
|
|
|
|
|
2016-02-26 23:13:16 +00:00
|
|
|
vmbus_initiate_unload(true);
|
2015-08-01 23:08:09 +00:00
|
|
|
/*
|
|
|
|
* In crash handler we can't schedule synic cleanup for all CPUs,
|
|
|
|
* doing the cleanup for current CPU only. This should be sufficient
|
|
|
|
* for kdump.
|
|
|
|
*/
|
2019-07-01 04:25:56 +00:00
|
|
|
cpu = smp_processor_id();
|
|
|
|
hv_stimer_cleanup(cpu);
|
2019-11-14 06:32:01 +00:00
|
|
|
hv_synic_disable_regs(cpu);
|
2017-01-28 19:37:14 +00:00
|
|
|
hyperv_cleanup();
|
2015-08-01 23:08:09 +00:00
|
|
|
};
|
|
|
|
|
2019-09-05 23:01:16 +00:00
|
|
|
static int hv_synic_suspend(void)
|
|
|
|
{
|
|
|
|
/*
|
x86/hyperv: Initialize clockevents earlier in CPU onlining
Hyper-V has historically initialized stimer-based clockevents late in the
process of onlining a CPU because clockevents depend on stimer
interrupts. In the original Hyper-V design, stimer interrupts generate a
VMbus message, so the VMbus machinery must be running first, and VMbus
can't be initialized until relatively late. On x86/64, LAPIC timer based
clockevents are used during early initialization before VMbus and
stimer-based clockevents are ready, and again during CPU offlining after
the stimer clockevents have been shut down.
Unfortunately, this design creates problems when offlining CPUs for
hibernation or other purposes. stimer-based clockevents are shut down
relatively early in the offlining process, so clockevents_unbind_device()
must be used to fallback to the LAPIC-based clockevents for the remainder
of the offlining process. Furthermore, the late initialization and early
shutdown of stimer-based clockevents doesn't work well on ARM64 since there
is no other timer like the LAPIC to fallback to. So CPU onlining and
offlining doesn't work properly.
Fix this by recognizing that stimer Direct Mode is the normal path for
newer versions of Hyper-V on x86/64, and the only path on other
architectures. With stimer Direct Mode, stimer interrupts don't require any
VMbus machinery. stimer clockevents can be initialized and shut down
consistent with how it is done for other clockevent devices. While the old
VMbus-based stimer interrupts must still be supported for backward
compatibility on x86, that mode of operation can be treated as legacy.
So add a new Hyper-V stimer entry in the CPU hotplug state list, and use
that new state when in Direct Mode. Update the Hyper-V clocksource driver
to allocate and initialize stimer clockevents earlier during boot. Update
Hyper-V initialization and the VMbus driver to use this new design. As a
result, the LAPIC timer is no longer used during boot or CPU
onlining/offlining and clockevents_unbind_device() is not called. But
retain the old design as a legacy implementation for older versions of
Hyper-V that don't support Direct Mode.
Signed-off-by: Michael Kelley <mikelley@microsoft.com>
Signed-off-by: Thomas Gleixner <tglx@linutronix.de>
Tested-by: Dexuan Cui <decui@microsoft.com>
Reviewed-by: Dexuan Cui <decui@microsoft.com>
Link: https://lkml.kernel.org/r/1573607467-9456-1-git-send-email-mikelley@microsoft.com
2019-11-13 01:11:49 +00:00
|
|
|
* When we reach here, all the non-boot CPUs have been offlined.
|
|
|
|
* If we're in a legacy configuration where stimer Direct Mode is
|
|
|
|
* not enabled, the stimers on the non-boot CPUs have been unbound
|
|
|
|
* in hv_synic_cleanup() -> hv_stimer_legacy_cleanup() ->
|
2019-09-05 23:01:16 +00:00
|
|
|
* hv_stimer_cleanup() -> clockevents_unbind_device().
|
|
|
|
*
|
x86/hyperv: Initialize clockevents earlier in CPU onlining
Hyper-V has historically initialized stimer-based clockevents late in the
process of onlining a CPU because clockevents depend on stimer
interrupts. In the original Hyper-V design, stimer interrupts generate a
VMbus message, so the VMbus machinery must be running first, and VMbus
can't be initialized until relatively late. On x86/64, LAPIC timer based
clockevents are used during early initialization before VMbus and
stimer-based clockevents are ready, and again during CPU offlining after
the stimer clockevents have been shut down.
Unfortunately, this design creates problems when offlining CPUs for
hibernation or other purposes. stimer-based clockevents are shut down
relatively early in the offlining process, so clockevents_unbind_device()
must be used to fallback to the LAPIC-based clockevents for the remainder
of the offlining process. Furthermore, the late initialization and early
shutdown of stimer-based clockevents doesn't work well on ARM64 since there
is no other timer like the LAPIC to fallback to. So CPU onlining and
offlining doesn't work properly.
Fix this by recognizing that stimer Direct Mode is the normal path for
newer versions of Hyper-V on x86/64, and the only path on other
architectures. With stimer Direct Mode, stimer interrupts don't require any
VMbus machinery. stimer clockevents can be initialized and shut down
consistent with how it is done for other clockevent devices. While the old
VMbus-based stimer interrupts must still be supported for backward
compatibility on x86, that mode of operation can be treated as legacy.
So add a new Hyper-V stimer entry in the CPU hotplug state list, and use
that new state when in Direct Mode. Update the Hyper-V clocksource driver
to allocate and initialize stimer clockevents earlier during boot. Update
Hyper-V initialization and the VMbus driver to use this new design. As a
result, the LAPIC timer is no longer used during boot or CPU
onlining/offlining and clockevents_unbind_device() is not called. But
retain the old design as a legacy implementation for older versions of
Hyper-V that don't support Direct Mode.
Signed-off-by: Michael Kelley <mikelley@microsoft.com>
Signed-off-by: Thomas Gleixner <tglx@linutronix.de>
Tested-by: Dexuan Cui <decui@microsoft.com>
Reviewed-by: Dexuan Cui <decui@microsoft.com>
Link: https://lkml.kernel.org/r/1573607467-9456-1-git-send-email-mikelley@microsoft.com
2019-11-13 01:11:49 +00:00
|
|
|
* hv_synic_suspend() only runs on CPU0 with interrupts disabled.
|
|
|
|
* Here we do not call hv_stimer_legacy_cleanup() on CPU0 because:
|
|
|
|
* 1) it's unnecessary as interrupts remain disabled between
|
|
|
|
* syscore_suspend() and syscore_resume(): see create_image() and
|
|
|
|
* resume_target_kernel()
|
2019-09-05 23:01:16 +00:00
|
|
|
* 2) the stimer on CPU0 is automatically disabled later by
|
|
|
|
* syscore_suspend() -> timekeeping_suspend() -> tick_suspend() -> ...
|
x86/hyperv: Initialize clockevents earlier in CPU onlining
Hyper-V has historically initialized stimer-based clockevents late in the
process of onlining a CPU because clockevents depend on stimer
interrupts. In the original Hyper-V design, stimer interrupts generate a
VMbus message, so the VMbus machinery must be running first, and VMbus
can't be initialized until relatively late. On x86/64, LAPIC timer based
clockevents are used during early initialization before VMbus and
stimer-based clockevents are ready, and again during CPU offlining after
the stimer clockevents have been shut down.
Unfortunately, this design creates problems when offlining CPUs for
hibernation or other purposes. stimer-based clockevents are shut down
relatively early in the offlining process, so clockevents_unbind_device()
must be used to fallback to the LAPIC-based clockevents for the remainder
of the offlining process. Furthermore, the late initialization and early
shutdown of stimer-based clockevents doesn't work well on ARM64 since there
is no other timer like the LAPIC to fallback to. So CPU onlining and
offlining doesn't work properly.
Fix this by recognizing that stimer Direct Mode is the normal path for
newer versions of Hyper-V on x86/64, and the only path on other
architectures. With stimer Direct Mode, stimer interrupts don't require any
VMbus machinery. stimer clockevents can be initialized and shut down
consistent with how it is done for other clockevent devices. While the old
VMbus-based stimer interrupts must still be supported for backward
compatibility on x86, that mode of operation can be treated as legacy.
So add a new Hyper-V stimer entry in the CPU hotplug state list, and use
that new state when in Direct Mode. Update the Hyper-V clocksource driver
to allocate and initialize stimer clockevents earlier during boot. Update
Hyper-V initialization and the VMbus driver to use this new design. As a
result, the LAPIC timer is no longer used during boot or CPU
onlining/offlining and clockevents_unbind_device() is not called. But
retain the old design as a legacy implementation for older versions of
Hyper-V that don't support Direct Mode.
Signed-off-by: Michael Kelley <mikelley@microsoft.com>
Signed-off-by: Thomas Gleixner <tglx@linutronix.de>
Tested-by: Dexuan Cui <decui@microsoft.com>
Reviewed-by: Dexuan Cui <decui@microsoft.com>
Link: https://lkml.kernel.org/r/1573607467-9456-1-git-send-email-mikelley@microsoft.com
2019-11-13 01:11:49 +00:00
|
|
|
* -> clockevents_shutdown() -> ... -> hv_ce_shutdown()
|
|
|
|
* 3) a warning would be triggered if we call
|
|
|
|
* clockevents_unbind_device(), which may sleep, in an
|
|
|
|
* interrupts-disabled context.
|
2019-09-05 23:01:16 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
hv_synic_disable_regs(0);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void hv_synic_resume(void)
|
|
|
|
{
|
|
|
|
hv_synic_enable_regs(0);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Note: we don't need to call hv_stimer_init(0), because the timer
|
|
|
|
* on CPU0 is not unbound in hv_synic_suspend(), and the timer is
|
|
|
|
* automatically re-enabled in timekeeping_resume().
|
|
|
|
*/
|
|
|
|
}
|
|
|
|
|
|
|
|
/* The callbacks run only on CPU0, with irqs_disabled. */
|
|
|
|
static struct syscore_ops hv_synic_syscore_ops = {
|
|
|
|
.suspend = hv_synic_suspend,
|
|
|
|
.resume = hv_synic_resume,
|
|
|
|
};
|
|
|
|
|
2011-06-06 22:49:39 +00:00
|
|
|
static int __init hv_acpi_init(void)
|
2011-03-15 22:03:32 +00:00
|
|
|
{
|
2011-07-15 20:38:56 +00:00
|
|
|
int ret, t;
|
2011-04-29 20:45:15 +00:00
|
|
|
|
2017-12-22 18:19:02 +00:00
|
|
|
if (!hv_is_hyperv_initialized())
|
2012-08-17 10:52:43 +00:00
|
|
|
return -ENODEV;
|
|
|
|
|
2011-04-29 20:45:15 +00:00
|
|
|
init_completion(&probe_event);
|
|
|
|
|
|
|
|
/*
|
2015-12-15 00:01:46 +00:00
|
|
|
* Get ACPI resources first.
|
2011-04-29 20:45:15 +00:00
|
|
|
*/
|
2011-06-06 22:49:40 +00:00
|
|
|
ret = acpi_bus_register_driver(&vmbus_acpi_driver);
|
|
|
|
|
2011-04-29 20:45:15 +00:00
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
|
2011-07-15 20:38:56 +00:00
|
|
|
t = wait_for_completion_timeout(&probe_event, 5*HZ);
|
|
|
|
if (t == 0) {
|
|
|
|
ret = -ETIMEDOUT;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
2019-10-03 21:01:49 +00:00
|
|
|
hv_debug_init();
|
2011-04-29 20:45:15 +00:00
|
|
|
|
2015-12-15 00:01:46 +00:00
|
|
|
ret = vmbus_bus_init();
|
2011-06-16 20:16:38 +00:00
|
|
|
if (ret)
|
2011-07-15 20:38:56 +00:00
|
|
|
goto cleanup;
|
|
|
|
|
2015-08-01 23:08:07 +00:00
|
|
|
hv_setup_kexec_handler(hv_kexec_handler);
|
2015-08-01 23:08:09 +00:00
|
|
|
hv_setup_crash_handler(hv_crash_handler);
|
2015-08-01 23:08:07 +00:00
|
|
|
|
2019-09-05 23:01:16 +00:00
|
|
|
register_syscore_ops(&hv_synic_syscore_ops);
|
|
|
|
|
2011-07-15 20:38:56 +00:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
acpi_bus_unregister_driver(&vmbus_acpi_driver);
|
2011-12-01 17:59:34 +00:00
|
|
|
hv_acpi_dev = NULL;
|
2011-06-16 20:16:38 +00:00
|
|
|
return ret;
|
2011-03-15 22:03:32 +00:00
|
|
|
}
|
|
|
|
|
2011-12-12 17:29:17 +00:00
|
|
|
static void __exit vmbus_exit(void)
|
|
|
|
{
|
2015-02-27 19:25:55 +00:00
|
|
|
int cpu;
|
|
|
|
|
2019-09-05 23:01:16 +00:00
|
|
|
unregister_syscore_ops(&hv_synic_syscore_ops);
|
|
|
|
|
2015-08-01 23:08:07 +00:00
|
|
|
hv_remove_kexec_handler();
|
2015-08-01 23:08:09 +00:00
|
|
|
hv_remove_crash_handler();
|
2015-02-27 19:25:54 +00:00
|
|
|
vmbus_connection.conn_state = DISCONNECTED;
|
2019-07-01 04:25:56 +00:00
|
|
|
hv_stimer_global_cleanup();
|
2015-04-23 04:31:32 +00:00
|
|
|
vmbus_disconnect();
|
2014-03-05 12:42:14 +00:00
|
|
|
hv_remove_vmbus_irq();
|
2017-02-12 06:02:19 +00:00
|
|
|
for_each_online_cpu(cpu) {
|
|
|
|
struct hv_per_cpu_context *hv_cpu
|
|
|
|
= per_cpu_ptr(hv_context.cpu_context, cpu);
|
|
|
|
|
|
|
|
tasklet_kill(&hv_cpu->msg_dpc);
|
|
|
|
}
|
2019-10-03 21:01:49 +00:00
|
|
|
hv_debug_rm_all_dir();
|
|
|
|
|
2011-12-12 17:29:17 +00:00
|
|
|
vmbus_free_channels();
|
2020-04-06 00:15:06 +00:00
|
|
|
kfree(vmbus_connection.channels);
|
2017-02-12 06:02:19 +00:00
|
|
|
|
2015-08-01 23:08:20 +00:00
|
|
|
if (ms_hyperv.misc_features & HV_FEATURE_GUEST_CRASH_MSR_AVAILABLE) {
|
2018-07-08 02:56:51 +00:00
|
|
|
kmsg_dump_unregister(&hv_kmsg_dumper);
|
2015-08-01 23:08:10 +00:00
|
|
|
unregister_die_notifier(&hyperv_die_block);
|
2015-04-23 04:31:29 +00:00
|
|
|
atomic_notifier_chain_unregister(&panic_notifier_list,
|
|
|
|
&hyperv_panic_block);
|
|
|
|
}
|
2018-07-08 02:56:51 +00:00
|
|
|
|
|
|
|
free_page((unsigned long)hv_panic_page);
|
2018-07-28 21:58:46 +00:00
|
|
|
unregister_sysctl_table(hv_ctl_table_hdr);
|
|
|
|
hv_ctl_table_hdr = NULL;
|
2011-12-12 17:29:17 +00:00
|
|
|
bus_unregister(&hv_bus);
|
2017-02-12 06:02:19 +00:00
|
|
|
|
2016-12-07 22:53:11 +00:00
|
|
|
cpuhp_remove_state(hyperv_cpuhp_online);
|
2015-08-01 23:08:05 +00:00
|
|
|
hv_synic_free();
|
2011-12-12 17:29:17 +00:00
|
|
|
acpi_bus_unregister_driver(&vmbus_acpi_driver);
|
|
|
|
}
|
|
|
|
|
2011-03-15 22:03:32 +00:00
|
|
|
|
2009-09-02 14:11:14 +00:00
|
|
|
MODULE_LICENSE("GPL");
|
2019-04-23 03:47:27 +00:00
|
|
|
MODULE_DESCRIPTION("Microsoft Hyper-V VMBus Driver");
|
2009-07-13 23:02:34 +00:00
|
|
|
|
2011-10-24 18:28:12 +00:00
|
|
|
subsys_initcall(hv_acpi_init);
|
2011-12-12 17:29:17 +00:00
|
|
|
module_exit(vmbus_exit);
|