提交 4e65f6e8 编写于 作者: K K. Y. Srinivasan 提交者: Greg Kroah-Hartman

Staging: hv: util: Properly handle util services in the util driver

Now, properly handle util services in the util driver and eliminate code
that will not be necessary. In the current code, util services were
all handled not as other vmbus devices (net, block) but rather through
special handling (channel setup etc.). In this patch we handle all
services using the standard Linux Driver Model.
Signed-off-by: NK. Y. Srinivasan <kys@microsoft.com>
Signed-off-by: NHaiyang Zhang <haiyangz@microsoft.com>
Signed-off-by: NGreg Kroah-Hartman <gregkh@suse.de>
上级 a29b643c
...@@ -181,24 +181,6 @@ void chn_cb_negotiate(void *context) ...@@ -181,24 +181,6 @@ void chn_cb_negotiate(void *context)
struct icmsg_hdr *icmsghdrp; struct icmsg_hdr *icmsghdrp;
struct icmsg_negotiate *negop = NULL; struct icmsg_negotiate *negop = NULL;
if (channel->util_index >= 0) {
/*
* This is a properly initialized util channel.
* Route this callback appropriately and setup state
* so that we don't need to reroute again.
*/
if (hv_cb_utils[channel->util_index].callback != NULL) {
/*
* The util driver has established a handler for
* this service; do the magic.
*/
channel->onchannel_callback =
hv_cb_utils[channel->util_index].callback;
(hv_cb_utils[channel->util_index].callback)(channel);
return;
}
}
buflen = PAGE_SIZE; buflen = PAGE_SIZE;
buf = kmalloc(buflen, GFP_ATOMIC); buf = kmalloc(buflen, GFP_ATOMIC);
...@@ -348,7 +330,6 @@ static void vmbus_process_offer(struct work_struct *work) ...@@ -348,7 +330,6 @@ static void vmbus_process_offer(struct work_struct *work)
struct vmbus_channel *channel; struct vmbus_channel *channel;
bool fnew = true; bool fnew = true;
int ret; int ret;
int cnt;
unsigned long flags; unsigned long flags;
/* The next possible work is rescind handling */ /* The next possible work is rescind handling */
...@@ -410,23 +391,6 @@ static void vmbus_process_offer(struct work_struct *work) ...@@ -410,23 +391,6 @@ static void vmbus_process_offer(struct work_struct *work)
* can cleanup properly * can cleanup properly
*/ */
newchannel->state = CHANNEL_OPEN_STATE; newchannel->state = CHANNEL_OPEN_STATE;
newchannel->util_index = -1; /* Invalid index */
/* Open IC channels */
for (cnt = 0; cnt < MAX_MSG_TYPES; cnt++) {
if (!uuid_le_cmp(newchannel->offermsg.offer.if_type,
hv_cb_utils[cnt].data) &&
vmbus_open(newchannel, 2 * PAGE_SIZE,
2 * PAGE_SIZE, NULL, 0,
chn_cb_negotiate,
newchannel) == 0) {
hv_cb_utils[cnt].channel = newchannel;
newchannel->util_index = cnt;
pr_info("%s\n", hv_cb_utils[cnt].log_msg);
}
}
} }
} }
......
...@@ -177,6 +177,13 @@ kvp_respond_to_host(char *key, char *value, int error) ...@@ -177,6 +177,13 @@ kvp_respond_to_host(char *key, char *value, int error)
channel = kvp_transaction.recv_channel; channel = kvp_transaction.recv_channel;
req_id = kvp_transaction.recv_req_id; req_id = kvp_transaction.recv_req_id;
if (channel->onchannel_callback == NULL)
/*
* We have raced with util driver being unloaded;
* silently return.
*/
return;
icmsghdrp = (struct icmsg_hdr *) icmsghdrp = (struct icmsg_hdr *)
&recv_buffer[sizeof(struct vmbuspipe_hdr)]; &recv_buffer[sizeof(struct vmbuspipe_hdr)];
kvp_msg = (struct hv_kvp_msg *) kvp_msg = (struct hv_kvp_msg *)
......
...@@ -269,19 +269,32 @@ static int util_probe(struct hv_device *dev, ...@@ -269,19 +269,32 @@ static int util_probe(struct hv_device *dev,
if (srv->util_init) { if (srv->util_init) {
ret = srv->util_init(srv); ret = srv->util_init(srv);
if (ret) { if (ret) {
kfree(srv->recv_buffer); ret = -ENODEV;
return -ENODEV; goto error1;
} }
} }
ret = vmbus_open(dev->channel, 2 * PAGE_SIZE, 2 * PAGE_SIZE, NULL, 0,
srv->util_cb, dev->channel);
if (ret)
goto error;
hv_set_drvdata(dev, srv); hv_set_drvdata(dev, srv);
return 0; return 0;
error:
if (srv->util_deinit)
srv->util_deinit();
error1:
kfree(srv->recv_buffer);
return ret;
} }
static int util_remove(struct hv_device *dev) static int util_remove(struct hv_device *dev)
{ {
struct hv_util_service *srv = hv_get_drvdata(dev); struct hv_util_service *srv = hv_get_drvdata(dev);
vmbus_close(dev->channel);
if (srv->util_deinit) if (srv->util_deinit)
srv->util_deinit(); srv->util_deinit();
kfree(srv->recv_buffer); kfree(srv->recv_buffer);
...@@ -321,51 +334,15 @@ static struct hv_driver util_drv = { ...@@ -321,51 +334,15 @@ static struct hv_driver util_drv = {
static int __init init_hyperv_utils(void) static int __init init_hyperv_utils(void)
{ {
int ret;
pr_info("Registering HyperV Utility Driver\n"); pr_info("Registering HyperV Utility Driver\n");
return vmbus_driver_register(&util_drv);
ret = vmbus_driver_register(&util_drv);
if (ret != 0)
return ret;
hv_cb_utils[HV_SHUTDOWN_MSG].callback = &shutdown_onchannelcallback;
hv_cb_utils[HV_TIMESYNC_MSG].callback = &timesync_onchannelcallback;
hv_cb_utils[HV_HEARTBEAT_MSG].callback = &heartbeat_onchannelcallback;
hv_cb_utils[HV_KVP_MSG].callback = &hv_kvp_onchannelcallback;
return 0;
} }
static void exit_hyperv_utils(void) static void exit_hyperv_utils(void)
{ {
pr_info("De-Registered HyperV Utility Driver\n"); pr_info("De-Registered HyperV Utility Driver\n");
if (hv_cb_utils[HV_SHUTDOWN_MSG].channel != NULL)
hv_cb_utils[HV_SHUTDOWN_MSG].channel->onchannel_callback =
&chn_cb_negotiate;
hv_cb_utils[HV_SHUTDOWN_MSG].callback = NULL;
if (hv_cb_utils[HV_TIMESYNC_MSG].channel != NULL)
hv_cb_utils[HV_TIMESYNC_MSG].channel->onchannel_callback =
&chn_cb_negotiate;
hv_cb_utils[HV_TIMESYNC_MSG].callback = NULL;
if (hv_cb_utils[HV_HEARTBEAT_MSG].channel != NULL)
hv_cb_utils[HV_HEARTBEAT_MSG].channel->onchannel_callback =
&chn_cb_negotiate;
hv_cb_utils[HV_HEARTBEAT_MSG].callback = NULL;
if (hv_cb_utils[HV_KVP_MSG].channel != NULL)
hv_cb_utils[HV_KVP_MSG].channel->onchannel_callback =
&chn_cb_negotiate;
hv_cb_utils[HV_KVP_MSG].callback = NULL;
vmbus_driver_unregister(&util_drv); vmbus_driver_unregister(&util_drv);
} }
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册