Drivers: hv: vmbus: Always handle the VMBus messages on CPU0
[ Upstream commit 8a857c55420f29da4fc131adc22b12d474c48f4c ] A Linux guest have to pick a "connect CPU" to communicate with the Hyper-V host. This CPU can not be taken offline because Hyper-V does not provide a way to change that CPU assignment. Current code sets the connect CPU to whatever CPU ends up running the function vmbus_negotiate_version(), and this will generate problems if that CPU is taken offine. Establish CPU0 as the connect CPU, and add logics to prevents the connect CPU from being taken offline. We could pick some other CPU, and we could pick that "other CPU" dynamically if there was a reason to do so at some point in the future. But for now, #defining the connect CPU to 0 is the most straightforward and least complex solution. While on this, add inline comments explaining "why" offer and rescind messages should not be handled by a same serialized work queue. Suggested-by: Dexuan Cui <decui@microsoft.com> Signed-off-by: Andrea Parri (Microsoft) <parri.andrea@gmail.com> Reviewed-by: Vitaly Kuznetsov <vkuznets@redhat.com> Link: https://lore.kernel.org/r/20200406001514.19876-2-parri.andrea@gmail.com Reviewed-by: Michael Kelley <mikelley@microsoft.com> Signed-off-by: Wei Liu <wei.liu@kernel.org> Signed-off-by: Sasha Levin <sashal@kernel.org>
This commit is contained in:
parent
51d2957b21
commit
04ab099d1e
@ -67,7 +67,6 @@ static __u32 vmbus_get_next_version(__u32 current_version)
|
|||||||
int vmbus_negotiate_version(struct vmbus_channel_msginfo *msginfo, u32 version)
|
int vmbus_negotiate_version(struct vmbus_channel_msginfo *msginfo, u32 version)
|
||||||
{
|
{
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
unsigned int cur_cpu;
|
|
||||||
struct vmbus_channel_initiate_contact *msg;
|
struct vmbus_channel_initiate_contact *msg;
|
||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
|
|
||||||
@ -100,24 +99,7 @@ int vmbus_negotiate_version(struct vmbus_channel_msginfo *msginfo, u32 version)
|
|||||||
|
|
||||||
msg->monitor_page1 = virt_to_phys(vmbus_connection.monitor_pages[0]);
|
msg->monitor_page1 = virt_to_phys(vmbus_connection.monitor_pages[0]);
|
||||||
msg->monitor_page2 = virt_to_phys(vmbus_connection.monitor_pages[1]);
|
msg->monitor_page2 = virt_to_phys(vmbus_connection.monitor_pages[1]);
|
||||||
/*
|
msg->target_vcpu = hv_cpu_number_to_vp_number(VMBUS_CONNECT_CPU);
|
||||||
* We want all channel messages to be delivered on CPU 0.
|
|
||||||
* This has been the behavior pre-win8. This is not
|
|
||||||
* perf issue and having all channel messages delivered on CPU 0
|
|
||||||
* would be ok.
|
|
||||||
* For post win8 hosts, we support receiving channel messagges on
|
|
||||||
* all the CPUs. This is needed for kexec to work correctly where
|
|
||||||
* the CPU attempting to connect may not be CPU 0.
|
|
||||||
*/
|
|
||||||
if (version >= VERSION_WIN8_1) {
|
|
||||||
cur_cpu = get_cpu();
|
|
||||||
msg->target_vcpu = hv_cpu_number_to_vp_number(cur_cpu);
|
|
||||||
vmbus_connection.connect_cpu = cur_cpu;
|
|
||||||
put_cpu();
|
|
||||||
} else {
|
|
||||||
msg->target_vcpu = 0;
|
|
||||||
vmbus_connection.connect_cpu = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Add to list before we send the request since we may
|
* Add to list before we send the request since we may
|
||||||
|
@ -249,6 +249,13 @@ int hv_synic_cleanup(unsigned int cpu)
|
|||||||
bool channel_found = false;
|
bool channel_found = false;
|
||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Hyper-V does not provide a way to change the connect CPU once
|
||||||
|
* it is set; we must prevent the connect CPU from going offline.
|
||||||
|
*/
|
||||||
|
if (cpu == VMBUS_CONNECT_CPU)
|
||||||
|
return -EBUSY;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Search for channels which are bound to the CPU we're about to
|
* Search for channels which are bound to the CPU we're about to
|
||||||
* cleanup. In case we find one and vmbus is still connected we need to
|
* cleanup. In case we find one and vmbus is still connected we need to
|
||||||
|
@ -212,12 +212,13 @@ enum vmbus_connect_state {
|
|||||||
|
|
||||||
#define MAX_SIZE_CHANNEL_MESSAGE HV_MESSAGE_PAYLOAD_BYTE_COUNT
|
#define MAX_SIZE_CHANNEL_MESSAGE HV_MESSAGE_PAYLOAD_BYTE_COUNT
|
||||||
|
|
||||||
struct vmbus_connection {
|
/*
|
||||||
/*
|
* The CPU that Hyper-V will interrupt for VMBUS messages, such as
|
||||||
* CPU on which the initial host contact was made.
|
* CHANNELMSG_OFFERCHANNEL and CHANNELMSG_RESCIND_CHANNELOFFER.
|
||||||
*/
|
*/
|
||||||
int connect_cpu;
|
#define VMBUS_CONNECT_CPU 0
|
||||||
|
|
||||||
|
struct vmbus_connection {
|
||||||
u32 msg_conn_id;
|
u32 msg_conn_id;
|
||||||
|
|
||||||
atomic_t offer_in_progress;
|
atomic_t offer_in_progress;
|
||||||
|
@ -1092,14 +1092,28 @@ void vmbus_on_msg_dpc(unsigned long data)
|
|||||||
/*
|
/*
|
||||||
* If we are handling the rescind message;
|
* If we are handling the rescind message;
|
||||||
* schedule the work on the global work queue.
|
* schedule the work on the global work queue.
|
||||||
|
*
|
||||||
|
* 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.
|
||||||
*/
|
*/
|
||||||
schedule_work_on(vmbus_connection.connect_cpu,
|
schedule_work_on(VMBUS_CONNECT_CPU,
|
||||||
&ctx->work);
|
&ctx->work);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case CHANNELMSG_OFFERCHANNEL:
|
case CHANNELMSG_OFFERCHANNEL:
|
||||||
atomic_inc(&vmbus_connection.offer_in_progress);
|
atomic_inc(&vmbus_connection.offer_in_progress);
|
||||||
queue_work_on(vmbus_connection.connect_cpu,
|
queue_work_on(VMBUS_CONNECT_CPU,
|
||||||
vmbus_connection.work_queue,
|
vmbus_connection.work_queue,
|
||||||
&ctx->work);
|
&ctx->work);
|
||||||
break;
|
break;
|
||||||
@ -1146,7 +1160,7 @@ static void vmbus_force_channel_rescinded(struct vmbus_channel *channel)
|
|||||||
|
|
||||||
INIT_WORK(&ctx->work, vmbus_onmessage_work);
|
INIT_WORK(&ctx->work, vmbus_onmessage_work);
|
||||||
|
|
||||||
queue_work_on(vmbus_connection.connect_cpu,
|
queue_work_on(VMBUS_CONNECT_CPU,
|
||||||
vmbus_connection.work_queue,
|
vmbus_connection.work_queue,
|
||||||
&ctx->work);
|
&ctx->work);
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user