blob: 3f0a95290e140daa99880cfffe8d0545dcaa7e0b [file] [log] [blame]
Hank Janssen3e7ee492009-07-13 16:02:34 -07001/*
Hank Janssen3e7ee492009-07-13 16:02:34 -07002 * Copyright (c) 2009, Microsoft Corporation.
3 *
4 * This program is free software; you can redistribute it and/or modify it
5 * under the terms and conditions of the GNU General Public License,
6 * version 2, as published by the Free Software Foundation.
7 *
8 * This program is distributed in the hope it will be useful, but WITHOUT
9 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
10 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
11 * more details.
12 *
13 * You should have received a copy of the GNU General Public License along with
14 * this program; if not, write to the Free Software Foundation, Inc., 59 Temple
15 * Place - Suite 330, Boston, MA 02111-1307 USA.
16 *
17 * Authors:
18 * Haiyang Zhang <haiyangz@microsoft.com>
19 * Hank Janssen <hjanssen@microsoft.com>
K. Y. Srinivasanb0069f42011-04-29 13:45:15 -070020 * K. Y. Srinivasan <kys@microsoft.com>
K. Y. Srinivasan52e5c1c2011-03-15 15:03:34 -070021 *
Hank Janssen3e7ee492009-07-13 16:02:34 -070022 */
Hank Janssen0a466182011-03-29 13:58:47 -070023#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
24
Hank Janssen3e7ee492009-07-13 16:02:34 -070025#include <linux/init.h>
26#include <linux/module.h>
27#include <linux/device.h>
Hank Janssen3e7ee492009-07-13 16:02:34 -070028#include <linux/interrupt.h>
29#include <linux/sysctl.h>
Tejun Heo5a0e3ad2010-03-24 17:04:11 +090030#include <linux/slab.h>
K. Y. Srinivasanb0069f42011-04-29 13:45:15 -070031#include <linux/acpi.h>
Haiyang Zhang8b5d6d32010-05-28 23:22:44 +000032#include <linux/completion.h>
Greg Kroah-Hartman46a97192011-10-04 12:29:52 -070033#include <linux/hyperv.h>
K. Y. Srinivasanb0209502012-12-01 06:46:54 -080034#include <linux/kernel_stat.h>
Greg Kroah-Hartman407dd162011-10-11 08:36:44 -060035#include <asm/hyperv.h>
Jason Wang1f94ea82012-08-31 13:32:44 +080036#include <asm/hypervisor.h>
K. Y. Srinivasan302a3c02013-02-17 11:30:44 -080037#include <asm/mshyperv.h>
K. Y. Srinivasan0f2a6612011-05-12 19:34:28 -070038#include "hyperv_vmbus.h"
Hank Janssen3e7ee492009-07-13 16:02:34 -070039
K. Y. Srinivasan607c1a12011-06-06 15:49:39 -070040static struct acpi_device *hv_acpi_dev;
K. Y. Srinivasan1168ac22011-03-15 15:03:32 -070041
K. Y. Srinivasan59c0e4f2011-04-29 13:45:06 -070042static struct tasklet_struct msg_dpc;
K. Y. Srinivasan71a66552011-04-29 13:45:04 -070043static struct completion probe_event;
K. Y. Srinivasanb0069f42011-04-29 13:45:15 -070044static int irq;
Hank Janssen3e7ee492009-07-13 16:02:34 -070045
K. Y. Srinivasancf6a2ea2011-12-01 09:59:34 -080046static int vmbus_exists(void)
47{
48 if (hv_acpi_dev == NULL)
49 return -ENODEV;
50
51 return 0;
52}
53
Olaf Heringfd776ba2011-09-02 18:25:56 +020054#define VMBUS_ALIAS_LEN ((sizeof((struct hv_vmbus_device_id *)0)->guid) * 2)
55static void print_alias_name(struct hv_device *hv_dev, char *alias_name)
56{
57 int i;
58 for (i = 0; i < VMBUS_ALIAS_LEN; i += 2)
59 sprintf(&alias_name[i], "%02x", hv_dev->dev_type.b[i/2]);
60}
61
Greg Kroah-Hartman76c52bb2013-09-13 11:32:56 -070062static u8 channel_monitor_group(struct vmbus_channel *channel)
63{
64 return (u8)channel->offermsg.monitorid / 32;
65}
66
67static u8 channel_monitor_offset(struct vmbus_channel *channel)
68{
69 return (u8)channel->offermsg.monitorid % 32;
70}
71
72static u32 channel_pending(struct vmbus_channel *channel,
73 struct hv_monitor_page *monitor_page)
74{
75 u8 monitor_group = channel_monitor_group(channel);
76 return monitor_page->trigger_group[monitor_group].pending;
77}
78
Greg Kroah-Hartman1cee2722013-09-13 11:32:57 -070079static u32 channel_latency(struct vmbus_channel *channel,
80 struct hv_monitor_page *monitor_page)
81{
82 u8 monitor_group = channel_monitor_group(channel);
83 u8 monitor_offset = channel_monitor_offset(channel);
84 return monitor_page->latency[monitor_group][monitor_offset];
85}
86
Greg Kroah-Hartman4947c742013-09-13 11:32:58 -070087static u32 channel_conn_id(struct vmbus_channel *channel,
88 struct hv_monitor_page *monitor_page)
89{
90 u8 monitor_group = channel_monitor_group(channel);
91 u8 monitor_offset = channel_monitor_offset(channel);
92 return monitor_page->parameter[monitor_group][monitor_offset].connectionid.u.id;
93}
94
Greg Kroah-Hartman03f3a912013-09-13 11:32:49 -070095static ssize_t id_show(struct device *dev, struct device_attribute *dev_attr,
96 char *buf)
97{
98 struct hv_device *hv_dev = device_to_hv_device(dev);
99
100 if (!hv_dev->channel)
101 return -ENODEV;
102 return sprintf(buf, "%d\n", hv_dev->channel->offermsg.child_relid);
103}
104static DEVICE_ATTR_RO(id);
105
Greg Kroah-Hartmana8fb5f32013-09-13 11:32:50 -0700106static ssize_t state_show(struct device *dev, struct device_attribute *dev_attr,
107 char *buf)
108{
109 struct hv_device *hv_dev = device_to_hv_device(dev);
110
111 if (!hv_dev->channel)
112 return -ENODEV;
113 return sprintf(buf, "%d\n", hv_dev->channel->state);
114}
115static DEVICE_ATTR_RO(state);
116
Greg Kroah-Hartman5ffd00e2013-09-13 11:32:51 -0700117static ssize_t monitor_id_show(struct device *dev,
118 struct device_attribute *dev_attr, char *buf)
119{
120 struct hv_device *hv_dev = device_to_hv_device(dev);
121
122 if (!hv_dev->channel)
123 return -ENODEV;
124 return sprintf(buf, "%d\n", hv_dev->channel->offermsg.monitorid);
125}
126static DEVICE_ATTR_RO(monitor_id);
127
Greg Kroah-Hartman68234c02013-09-13 11:32:53 -0700128static ssize_t class_id_show(struct device *dev,
129 struct device_attribute *dev_attr, char *buf)
130{
131 struct hv_device *hv_dev = device_to_hv_device(dev);
132
133 if (!hv_dev->channel)
134 return -ENODEV;
135 return sprintf(buf, "{%pUl}\n",
136 hv_dev->channel->offermsg.offer.if_type.b);
137}
138static DEVICE_ATTR_RO(class_id);
139
Greg Kroah-Hartman7c55e1d2013-09-13 11:32:54 -0700140static ssize_t device_id_show(struct device *dev,
141 struct device_attribute *dev_attr, char *buf)
142{
143 struct hv_device *hv_dev = device_to_hv_device(dev);
144
145 if (!hv_dev->channel)
146 return -ENODEV;
147 return sprintf(buf, "{%pUl}\n",
148 hv_dev->channel->offermsg.offer.if_instance.b);
149}
150static DEVICE_ATTR_RO(device_id);
151
Greg Kroah-Hartman647fa372013-09-13 11:32:52 -0700152static ssize_t modalias_show(struct device *dev,
153 struct device_attribute *dev_attr, char *buf)
154{
155 struct hv_device *hv_dev = device_to_hv_device(dev);
156 char alias_name[VMBUS_ALIAS_LEN + 1];
157
158 print_alias_name(hv_dev, alias_name);
159 return sprintf(buf, "vmbus:%s\n", alias_name);
160}
161static DEVICE_ATTR_RO(modalias);
162
Greg Kroah-Hartman76c52bb2013-09-13 11:32:56 -0700163static ssize_t server_monitor_pending_show(struct device *dev,
164 struct device_attribute *dev_attr,
165 char *buf)
166{
167 struct hv_device *hv_dev = device_to_hv_device(dev);
168
169 if (!hv_dev->channel)
170 return -ENODEV;
171 return sprintf(buf, "%d\n",
172 channel_pending(hv_dev->channel,
173 vmbus_connection.monitor_pages[1]));
174}
175static DEVICE_ATTR_RO(server_monitor_pending);
176
177static ssize_t client_monitor_pending_show(struct device *dev,
178 struct device_attribute *dev_attr,
179 char *buf)
180{
181 struct hv_device *hv_dev = device_to_hv_device(dev);
182
183 if (!hv_dev->channel)
184 return -ENODEV;
185 return sprintf(buf, "%d\n",
186 channel_pending(hv_dev->channel,
187 vmbus_connection.monitor_pages[1]));
188}
189static DEVICE_ATTR_RO(client_monitor_pending);
Greg Kroah-Hartman68234c02013-09-13 11:32:53 -0700190
Greg Kroah-Hartman1cee2722013-09-13 11:32:57 -0700191static ssize_t server_monitor_latency_show(struct device *dev,
192 struct device_attribute *dev_attr,
193 char *buf)
194{
195 struct hv_device *hv_dev = device_to_hv_device(dev);
196
197 if (!hv_dev->channel)
198 return -ENODEV;
199 return sprintf(buf, "%d\n",
200 channel_latency(hv_dev->channel,
201 vmbus_connection.monitor_pages[0]));
202}
203static DEVICE_ATTR_RO(server_monitor_latency);
204
205static ssize_t client_monitor_latency_show(struct device *dev,
206 struct device_attribute *dev_attr,
207 char *buf)
208{
209 struct hv_device *hv_dev = device_to_hv_device(dev);
210
211 if (!hv_dev->channel)
212 return -ENODEV;
213 return sprintf(buf, "%d\n",
214 channel_latency(hv_dev->channel,
215 vmbus_connection.monitor_pages[1]));
216}
217static DEVICE_ATTR_RO(client_monitor_latency);
218
Greg Kroah-Hartman4947c742013-09-13 11:32:58 -0700219static ssize_t server_monitor_conn_id_show(struct device *dev,
220 struct device_attribute *dev_attr,
221 char *buf)
222{
223 struct hv_device *hv_dev = device_to_hv_device(dev);
224
225 if (!hv_dev->channel)
226 return -ENODEV;
227 return sprintf(buf, "%d\n",
228 channel_conn_id(hv_dev->channel,
229 vmbus_connection.monitor_pages[0]));
230}
231static DEVICE_ATTR_RO(server_monitor_conn_id);
232
233static ssize_t client_monitor_conn_id_show(struct device *dev,
234 struct device_attribute *dev_attr,
235 char *buf)
236{
237 struct hv_device *hv_dev = device_to_hv_device(dev);
238
239 if (!hv_dev->channel)
240 return -ENODEV;
241 return sprintf(buf, "%d\n",
242 channel_conn_id(hv_dev->channel,
243 vmbus_connection.monitor_pages[1]));
244}
245static DEVICE_ATTR_RO(client_monitor_conn_id);
246
Greg Kroah-Hartman98f4c652013-09-13 11:33:01 -0700247static ssize_t out_intr_mask_show(struct device *dev,
248 struct device_attribute *dev_attr, char *buf)
249{
250 struct hv_device *hv_dev = device_to_hv_device(dev);
251 struct hv_ring_buffer_debug_info outbound;
252
253 if (!hv_dev->channel)
254 return -ENODEV;
255 hv_ringbuffer_get_debuginfo(&hv_dev->channel->outbound, &outbound);
256 return sprintf(buf, "%d\n", outbound.current_interrupt_mask);
257}
258static DEVICE_ATTR_RO(out_intr_mask);
259
260static ssize_t out_read_index_show(struct device *dev,
261 struct device_attribute *dev_attr, char *buf)
262{
263 struct hv_device *hv_dev = device_to_hv_device(dev);
264 struct hv_ring_buffer_debug_info outbound;
265
266 if (!hv_dev->channel)
267 return -ENODEV;
268 hv_ringbuffer_get_debuginfo(&hv_dev->channel->outbound, &outbound);
269 return sprintf(buf, "%d\n", outbound.current_read_index);
270}
271static DEVICE_ATTR_RO(out_read_index);
272
273static ssize_t out_write_index_show(struct device *dev,
274 struct device_attribute *dev_attr,
275 char *buf)
276{
277 struct hv_device *hv_dev = device_to_hv_device(dev);
278 struct hv_ring_buffer_debug_info outbound;
279
280 if (!hv_dev->channel)
281 return -ENODEV;
282 hv_ringbuffer_get_debuginfo(&hv_dev->channel->outbound, &outbound);
283 return sprintf(buf, "%d\n", outbound.current_write_index);
284}
285static DEVICE_ATTR_RO(out_write_index);
286
287static ssize_t out_read_bytes_avail_show(struct device *dev,
288 struct device_attribute *dev_attr,
289 char *buf)
290{
291 struct hv_device *hv_dev = device_to_hv_device(dev);
292 struct hv_ring_buffer_debug_info outbound;
293
294 if (!hv_dev->channel)
295 return -ENODEV;
296 hv_ringbuffer_get_debuginfo(&hv_dev->channel->outbound, &outbound);
297 return sprintf(buf, "%d\n", outbound.bytes_avail_toread);
298}
299static DEVICE_ATTR_RO(out_read_bytes_avail);
300
301static ssize_t out_write_bytes_avail_show(struct device *dev,
302 struct device_attribute *dev_attr,
303 char *buf)
304{
305 struct hv_device *hv_dev = device_to_hv_device(dev);
306 struct hv_ring_buffer_debug_info outbound;
307
308 if (!hv_dev->channel)
309 return -ENODEV;
310 hv_ringbuffer_get_debuginfo(&hv_dev->channel->outbound, &outbound);
311 return sprintf(buf, "%d\n", outbound.bytes_avail_towrite);
312}
313static DEVICE_ATTR_RO(out_write_bytes_avail);
314
315static ssize_t in_intr_mask_show(struct device *dev,
316 struct device_attribute *dev_attr, char *buf)
317{
318 struct hv_device *hv_dev = device_to_hv_device(dev);
319 struct hv_ring_buffer_debug_info inbound;
320
321 if (!hv_dev->channel)
322 return -ENODEV;
323 hv_ringbuffer_get_debuginfo(&hv_dev->channel->inbound, &inbound);
324 return sprintf(buf, "%d\n", inbound.current_interrupt_mask);
325}
326static DEVICE_ATTR_RO(in_intr_mask);
327
328static ssize_t in_read_index_show(struct device *dev,
329 struct device_attribute *dev_attr, char *buf)
330{
331 struct hv_device *hv_dev = device_to_hv_device(dev);
332 struct hv_ring_buffer_debug_info inbound;
333
334 if (!hv_dev->channel)
335 return -ENODEV;
336 hv_ringbuffer_get_debuginfo(&hv_dev->channel->inbound, &inbound);
337 return sprintf(buf, "%d\n", inbound.current_read_index);
338}
339static DEVICE_ATTR_RO(in_read_index);
340
341static ssize_t in_write_index_show(struct device *dev,
342 struct device_attribute *dev_attr, char *buf)
343{
344 struct hv_device *hv_dev = device_to_hv_device(dev);
345 struct hv_ring_buffer_debug_info inbound;
346
347 if (!hv_dev->channel)
348 return -ENODEV;
349 hv_ringbuffer_get_debuginfo(&hv_dev->channel->inbound, &inbound);
350 return sprintf(buf, "%d\n", inbound.current_write_index);
351}
352static DEVICE_ATTR_RO(in_write_index);
353
354static ssize_t in_read_bytes_avail_show(struct device *dev,
355 struct device_attribute *dev_attr,
356 char *buf)
357{
358 struct hv_device *hv_dev = device_to_hv_device(dev);
359 struct hv_ring_buffer_debug_info inbound;
360
361 if (!hv_dev->channel)
362 return -ENODEV;
363 hv_ringbuffer_get_debuginfo(&hv_dev->channel->inbound, &inbound);
364 return sprintf(buf, "%d\n", inbound.bytes_avail_toread);
365}
366static DEVICE_ATTR_RO(in_read_bytes_avail);
367
368static ssize_t in_write_bytes_avail_show(struct device *dev,
369 struct device_attribute *dev_attr,
370 char *buf)
371{
372 struct hv_device *hv_dev = device_to_hv_device(dev);
373 struct hv_ring_buffer_debug_info inbound;
374
375 if (!hv_dev->channel)
376 return -ENODEV;
377 hv_ringbuffer_get_debuginfo(&hv_dev->channel->inbound, &inbound);
378 return sprintf(buf, "%d\n", inbound.bytes_avail_towrite);
379}
380static DEVICE_ATTR_RO(in_write_bytes_avail);
381
382/* Set up per device attributes in /sys/bus/vmbus/devices/<bus device> */
Greg Kroah-Hartman03f3a912013-09-13 11:32:49 -0700383static struct attribute *vmbus_attrs[] = {
384 &dev_attr_id.attr,
Greg Kroah-Hartmana8fb5f32013-09-13 11:32:50 -0700385 &dev_attr_state.attr,
Greg Kroah-Hartman5ffd00e2013-09-13 11:32:51 -0700386 &dev_attr_monitor_id.attr,
Greg Kroah-Hartman68234c02013-09-13 11:32:53 -0700387 &dev_attr_class_id.attr,
Greg Kroah-Hartman7c55e1d2013-09-13 11:32:54 -0700388 &dev_attr_device_id.attr,
Greg Kroah-Hartman647fa372013-09-13 11:32:52 -0700389 &dev_attr_modalias.attr,
Greg Kroah-Hartman76c52bb2013-09-13 11:32:56 -0700390 &dev_attr_server_monitor_pending.attr,
391 &dev_attr_client_monitor_pending.attr,
Greg Kroah-Hartman1cee2722013-09-13 11:32:57 -0700392 &dev_attr_server_monitor_latency.attr,
393 &dev_attr_client_monitor_latency.attr,
Greg Kroah-Hartman4947c742013-09-13 11:32:58 -0700394 &dev_attr_server_monitor_conn_id.attr,
395 &dev_attr_client_monitor_conn_id.attr,
Greg Kroah-Hartman98f4c652013-09-13 11:33:01 -0700396 &dev_attr_out_intr_mask.attr,
397 &dev_attr_out_read_index.attr,
398 &dev_attr_out_write_index.attr,
399 &dev_attr_out_read_bytes_avail.attr,
400 &dev_attr_out_write_bytes_avail.attr,
401 &dev_attr_in_intr_mask.attr,
402 &dev_attr_in_read_index.attr,
403 &dev_attr_in_write_index.attr,
404 &dev_attr_in_read_bytes_avail.attr,
405 &dev_attr_in_write_bytes_avail.attr,
Greg Kroah-Hartman03f3a912013-09-13 11:32:49 -0700406 NULL,
407};
408ATTRIBUTE_GROUPS(vmbus);
409
K. Y. Srinivasanadde2482011-03-15 15:03:37 -0700410/*
411 * vmbus_uevent - add uevent for our device
412 *
413 * This routine is invoked when a device is added or removed on the vmbus to
414 * generate a uevent to udev in the userspace. The udev will then look at its
415 * rule and the uevent generated here to load the appropriate driver
K. Y. Srinivasan0ddda662011-08-25 09:48:38 -0700416 *
417 * The alias string will be of the form vmbus:guid where guid is the string
418 * representation of the device guid (each byte of the guid will be
419 * represented with two hex characters.
K. Y. Srinivasanadde2482011-03-15 15:03:37 -0700420 */
421static int vmbus_uevent(struct device *device, struct kobj_uevent_env *env)
422{
423 struct hv_device *dev = device_to_hv_device(device);
Olaf Heringfd776ba2011-09-02 18:25:56 +0200424 int ret;
425 char alias_name[VMBUS_ALIAS_LEN + 1];
K. Y. Srinivasanadde2482011-03-15 15:03:37 -0700426
Olaf Heringfd776ba2011-09-02 18:25:56 +0200427 print_alias_name(dev, alias_name);
K. Y. Srinivasan0ddda662011-08-25 09:48:38 -0700428 ret = add_uevent_var(env, "MODALIAS=vmbus:%s", alias_name);
429 return ret;
K. Y. Srinivasanadde2482011-03-15 15:03:37 -0700430}
431
K. Y. Srinivasan5841a822011-08-25 09:48:39 -0700432static uuid_le null_guid;
433
434static inline bool is_null_guid(const __u8 *guid)
435{
436 if (memcmp(guid, &null_guid, sizeof(uuid_le)))
437 return false;
438 return true;
439}
440
K. Y. Srinivasan3037a7b2011-09-13 10:59:37 -0700441/*
442 * Return a matching hv_vmbus_device_id pointer.
443 * If there is no match, return NULL.
444 */
445static const struct hv_vmbus_device_id *hv_vmbus_get_id(
446 const struct hv_vmbus_device_id *id,
447 __u8 *guid)
448{
449 for (; !is_null_guid(id->guid); id++)
450 if (!memcmp(&id->guid, guid, sizeof(uuid_le)))
451 return id;
452
453 return NULL;
454}
455
456
K. Y. Srinivasanb7fc1472011-03-15 15:03:38 -0700457
458/*
459 * vmbus_match - Attempt to match the specified device to the specified driver
460 */
461static int vmbus_match(struct device *device, struct device_driver *driver)
462{
K. Y. Srinivasanb7fc1472011-03-15 15:03:38 -0700463 struct hv_driver *drv = drv_to_hv_drv(driver);
K. Y. Srinivasane8e27042011-06-06 15:50:04 -0700464 struct hv_device *hv_dev = device_to_hv_device(device);
K. Y. Srinivasanb7fc1472011-03-15 15:03:38 -0700465
K. Y. Srinivasan3037a7b2011-09-13 10:59:37 -0700466 if (hv_vmbus_get_id(drv->id_table, hv_dev->dev_type.b))
467 return 1;
K. Y. Srinivasande632a22011-04-26 09:20:24 -0700468
K. Y. Srinivasan5841a822011-08-25 09:48:39 -0700469 return 0;
K. Y. Srinivasanb7fc1472011-03-15 15:03:38 -0700470}
471
K. Y. Srinivasanf1f0d672011-03-15 15:03:39 -0700472/*
473 * vmbus_probe - Add the new vmbus's child device
474 */
475static int vmbus_probe(struct device *child_device)
476{
477 int ret = 0;
478 struct hv_driver *drv =
479 drv_to_hv_drv(child_device->driver);
K. Y. Srinivasan9efd21e2011-04-29 13:45:10 -0700480 struct hv_device *dev = device_to_hv_device(child_device);
K. Y. Srinivasan84946892011-09-13 10:59:38 -0700481 const struct hv_vmbus_device_id *dev_id;
K. Y. Srinivasanf1f0d672011-03-15 15:03:39 -0700482
K. Y. Srinivasan84946892011-09-13 10:59:38 -0700483 dev_id = hv_vmbus_get_id(drv->id_table, dev->dev_type.b);
K. Y. Srinivasan9efd21e2011-04-29 13:45:10 -0700484 if (drv->probe) {
K. Y. Srinivasan84946892011-09-13 10:59:38 -0700485 ret = drv->probe(dev, dev_id);
K. Y. Srinivasanb14a7b32011-04-29 13:45:03 -0700486 if (ret != 0)
Hank Janssen0a466182011-03-29 13:58:47 -0700487 pr_err("probe failed for device %s (%d)\n",
488 dev_name(child_device), ret);
K. Y. Srinivasanf1f0d672011-03-15 15:03:39 -0700489
K. Y. Srinivasanf1f0d672011-03-15 15:03:39 -0700490 } else {
Hank Janssen0a466182011-03-29 13:58:47 -0700491 pr_err("probe not set for driver %s\n",
492 dev_name(child_device));
K. Y. Srinivasan6de925b2011-06-06 15:50:07 -0700493 ret = -ENODEV;
K. Y. Srinivasanf1f0d672011-03-15 15:03:39 -0700494 }
495 return ret;
496}
497
K. Y. Srinivasanc5dce3d2011-03-15 15:03:40 -0700498/*
499 * vmbus_remove - Remove a vmbus device
500 */
501static int vmbus_remove(struct device *child_device)
502{
K. Y. Srinivasand4372172011-09-13 10:59:44 -0700503 struct hv_driver *drv = drv_to_hv_drv(child_device->driver);
K. Y. Srinivasan415b0232011-04-29 13:45:12 -0700504 struct hv_device *dev = device_to_hv_device(child_device);
K. Y. Srinivasanc5dce3d2011-03-15 15:03:40 -0700505
K. Y. Srinivasand4372172011-09-13 10:59:44 -0700506 if (drv->remove)
507 drv->remove(dev);
508 else
509 pr_err("remove not set for driver %s\n",
510 dev_name(child_device));
K. Y. Srinivasanc5dce3d2011-03-15 15:03:40 -0700511
512 return 0;
513}
514
K. Y. Srinivasaneb1bb252011-03-15 15:03:41 -0700515
516/*
517 * vmbus_shutdown - Shutdown a vmbus device
518 */
519static void vmbus_shutdown(struct device *child_device)
520{
521 struct hv_driver *drv;
K. Y. Srinivasanca6887f2011-04-29 13:45:14 -0700522 struct hv_device *dev = device_to_hv_device(child_device);
K. Y. Srinivasaneb1bb252011-03-15 15:03:41 -0700523
524
525 /* The device may not be attached yet */
526 if (!child_device->driver)
527 return;
528
529 drv = drv_to_hv_drv(child_device->driver);
530
K. Y. Srinivasanca6887f2011-04-29 13:45:14 -0700531 if (drv->shutdown)
532 drv->shutdown(dev);
K. Y. Srinivasaneb1bb252011-03-15 15:03:41 -0700533
534 return;
535}
536
K. Y. Srinivasan086e7a52011-03-15 15:03:42 -0700537
538/*
539 * vmbus_device_release - Final callback release of the vmbus child device
540 */
541static void vmbus_device_release(struct device *device)
542{
K. Y. Srinivasane8e27042011-06-06 15:50:04 -0700543 struct hv_device *hv_dev = device_to_hv_device(device);
K. Y. Srinivasan086e7a52011-03-15 15:03:42 -0700544
K. Y. Srinivasane8e27042011-06-06 15:50:04 -0700545 kfree(hv_dev);
K. Y. Srinivasan086e7a52011-03-15 15:03:42 -0700546
547}
548
Bill Pemberton454f18a2009-07-27 16:47:24 -0400549/* The one and only one */
K. Y. Srinivasan9adcac52011-04-29 13:45:08 -0700550static struct bus_type hv_bus = {
551 .name = "vmbus",
552 .match = vmbus_match,
553 .shutdown = vmbus_shutdown,
554 .remove = vmbus_remove,
555 .probe = vmbus_probe,
556 .uevent = vmbus_uevent,
Greg Kroah-Hartman03f3a912013-09-13 11:32:49 -0700557 .dev_groups = vmbus_groups,
Hank Janssen3e7ee492009-07-13 16:02:34 -0700558};
559
Timo Teräsbf6506f2010-12-15 20:48:08 +0200560struct onmessage_work_context {
561 struct work_struct work;
562 struct hv_message msg;
563};
564
565static void vmbus_onmessage_work(struct work_struct *work)
566{
567 struct onmessage_work_context *ctx;
568
569 ctx = container_of(work, struct onmessage_work_context,
570 work);
571 vmbus_onmessage(&ctx->msg);
572 kfree(ctx);
573}
574
K. Y. Srinivasan62c10592011-03-10 14:04:53 -0800575static void vmbus_on_msg_dpc(unsigned long data)
Greg Kroah-Hartman36199a92010-12-02 11:59:22 -0800576{
577 int cpu = smp_processor_id();
578 void *page_addr = hv_context.synic_message_page[cpu];
579 struct hv_message *msg = (struct hv_message *)page_addr +
580 VMBUS_MESSAGE_SINT;
Timo Teräsbf6506f2010-12-15 20:48:08 +0200581 struct onmessage_work_context *ctx;
Greg Kroah-Hartman36199a92010-12-02 11:59:22 -0800582
583 while (1) {
584 if (msg->header.message_type == HVMSG_NONE) {
585 /* no msg */
586 break;
587 } else {
Timo Teräsbf6506f2010-12-15 20:48:08 +0200588 ctx = kmalloc(sizeof(*ctx), GFP_ATOMIC);
589 if (ctx == NULL)
Greg Kroah-Hartman36199a92010-12-02 11:59:22 -0800590 continue;
Timo Teräsbf6506f2010-12-15 20:48:08 +0200591 INIT_WORK(&ctx->work, vmbus_onmessage_work);
592 memcpy(&ctx->msg, msg, sizeof(*msg));
Haiyang Zhangda9fcb72011-01-26 12:12:14 -0800593 queue_work(vmbus_connection.work_queue, &ctx->work);
Greg Kroah-Hartman36199a92010-12-02 11:59:22 -0800594 }
595
596 msg->header.message_type = HVMSG_NONE;
597
598 /*
599 * Make sure the write to MessageType (ie set to
600 * HVMSG_NONE) happens before we read the
601 * MessagePending and EOMing. Otherwise, the EOMing
602 * will not deliver any more messages since there is
603 * no empty slot
604 */
Jason Wang35848f62013-06-18 13:04:23 +0800605 mb();
Greg Kroah-Hartman36199a92010-12-02 11:59:22 -0800606
607 if (msg->header.message_flags.msg_pending) {
608 /*
609 * This will cause message queue rescan to
610 * possibly deliver another msg from the
611 * hypervisor
612 */
613 wrmsrl(HV_X64_MSR_EOM, 0);
614 }
615 }
616}
617
Thomas Gleixner76d388c2014-03-05 13:42:14 +0100618static void vmbus_isr(void)
Greg Kroah-Hartman36199a92010-12-02 11:59:22 -0800619{
Greg Kroah-Hartman36199a92010-12-02 11:59:22 -0800620 int cpu = smp_processor_id();
621 void *page_addr;
622 struct hv_message *msg;
623 union hv_synic_event_flags *event;
K. Y. Srinivasanae4636e2011-08-27 11:31:35 -0700624 bool handled = false;
Greg Kroah-Hartman36199a92010-12-02 11:59:22 -0800625
K. Y. Srinivasan5ab05952012-12-01 06:46:55 -0800626 page_addr = hv_context.synic_event_page[cpu];
627 if (page_addr == NULL)
Thomas Gleixner76d388c2014-03-05 13:42:14 +0100628 return;
K. Y. Srinivasan5ab05952012-12-01 06:46:55 -0800629
630 event = (union hv_synic_event_flags *)page_addr +
631 VMBUS_MESSAGE_SINT;
K. Y. Srinivasan7341d902011-08-31 14:35:56 -0700632 /*
633 * Check for events before checking for messages. This is the order
634 * in which events and messages are checked in Windows guests on
635 * Hyper-V, and the Windows team suggested we do the same.
636 */
Greg Kroah-Hartman36199a92010-12-02 11:59:22 -0800637
K. Y. Srinivasan6552ecd2012-12-01 06:46:49 -0800638 if ((vmbus_proto_version == VERSION_WS2008) ||
639 (vmbus_proto_version == VERSION_WIN7)) {
Greg Kroah-Hartman36199a92010-12-02 11:59:22 -0800640
K. Y. Srinivasan6552ecd2012-12-01 06:46:49 -0800641 /* Since we are a child, we only need to check bit 0 */
642 if (sync_test_and_clear_bit(0,
643 (unsigned long *) &event->flags32[0])) {
644 handled = true;
645 }
646 } else {
647 /*
648 * Our host is win8 or above. The signaling mechanism
649 * has changed and we can directly look at the event page.
650 * If bit n is set then we have an interrup on the channel
651 * whose id is n.
652 */
K. Y. Srinivasanae4636e2011-08-27 11:31:35 -0700653 handled = true;
K. Y. Srinivasan793be9c2011-03-15 15:03:43 -0700654 }
K. Y. Srinivasanae4636e2011-08-27 11:31:35 -0700655
K. Y. Srinivasan6552ecd2012-12-01 06:46:49 -0800656 if (handled)
K. Y. Srinivasandb11f122012-12-01 06:46:53 -0800657 tasklet_schedule(hv_context.event_dpc[cpu]);
K. Y. Srinivasan6552ecd2012-12-01 06:46:49 -0800658
659
K. Y. Srinivasan7341d902011-08-31 14:35:56 -0700660 page_addr = hv_context.synic_message_page[cpu];
661 msg = (struct hv_message *)page_addr + VMBUS_MESSAGE_SINT;
662
663 /* Check if there are actual msgs to be processed */
Thomas Gleixner76d388c2014-03-05 13:42:14 +0100664 if (msg->header.message_type != HVMSG_NONE)
K. Y. Srinivasan7341d902011-08-31 14:35:56 -0700665 tasklet_schedule(&msg_dpc);
K. Y. Srinivasan793be9c2011-03-15 15:03:43 -0700666}
667
Hank Janssen3e189512010-03-04 22:11:00 +0000668/*
Greg Kroah-Hartman90c99602009-09-02 07:11:14 -0700669 * vmbus_bus_init -Main vmbus driver initialization routine.
670 *
671 * Here, we
Lars Lindley0686e4f2010-03-11 23:51:23 +0100672 * - initialize the vmbus driver context
Lars Lindley0686e4f2010-03-11 23:51:23 +0100673 * - invoke the vmbus hv main init routine
674 * - get the irq resource
Lars Lindley0686e4f2010-03-11 23:51:23 +0100675 * - retrieve the channel offers
Greg Kroah-Hartman90c99602009-09-02 07:11:14 -0700676 */
K. Y. Srinivasan9aaa9952011-06-06 15:49:37 -0700677static int vmbus_bus_init(int irq)
Hank Janssen3e7ee492009-07-13 16:02:34 -0700678{
Greg Kroah-Hartman90c99602009-09-02 07:11:14 -0700679 int ret;
Hank Janssen3e7ee492009-07-13 16:02:34 -0700680
Greg Kroah-Hartman6d26e38fa22010-12-02 12:08:08 -0800681 /* Hypervisor initialization...setup hypercall page..etc */
682 ret = hv_init();
Greg Kroah-Hartman90c99602009-09-02 07:11:14 -0700683 if (ret != 0) {
Hank Janssen0a466182011-03-29 13:58:47 -0700684 pr_err("Unable to initialize the hypervisor - 0x%x\n", ret);
K. Y. Srinivasand6c1c5d2011-06-06 15:50:08 -0700685 return ret;
Hank Janssen3e7ee492009-07-13 16:02:34 -0700686 }
687
K. Y. Srinivasan59c0e4f2011-04-29 13:45:06 -0700688 tasklet_init(&msg_dpc, vmbus_on_msg_dpc, 0);
Hank Janssen3e7ee492009-07-13 16:02:34 -0700689
K. Y. Srinivasan9adcac52011-04-29 13:45:08 -0700690 ret = bus_register(&hv_bus);
K. Y. Srinivasand6c1c5d2011-06-06 15:50:08 -0700691 if (ret)
K. Y. Srinivasan8b9987e2011-08-31 14:35:55 -0700692 goto err_cleanup;
Hank Janssen3e7ee492009-07-13 16:02:34 -0700693
Thomas Gleixner76d388c2014-03-05 13:42:14 +0100694 hv_setup_vmbus_irq(vmbus_isr);
Hank Janssen3e7ee492009-07-13 16:02:34 -0700695
Jason Wang2608fb62013-06-19 11:28:10 +0800696 ret = hv_synic_alloc();
697 if (ret)
698 goto err_alloc;
K. Y. Srinivasan800b6902011-03-15 15:03:33 -0700699 /*
K. Y. Srinivasan302a3c02013-02-17 11:30:44 -0800700 * Initialize the per-cpu interrupt state and
K. Y. Srinivasan800b6902011-03-15 15:03:33 -0700701 * connect to the host.
702 */
K. Y. Srinivasan302a3c02013-02-17 11:30:44 -0800703 on_each_cpu(hv_synic_init, NULL, 1);
K. Y. Srinivasan800b6902011-03-15 15:03:33 -0700704 ret = vmbus_connect();
K. Y. Srinivasan8b9987e2011-08-31 14:35:55 -0700705 if (ret)
Jason Wang2608fb62013-06-19 11:28:10 +0800706 goto err_alloc;
K. Y. Srinivasan800b6902011-03-15 15:03:33 -0700707
Greg Kroah-Hartman2d6e8822010-12-02 08:50:58 -0800708 vmbus_request_offers();
Haiyang Zhang8b5d6d32010-05-28 23:22:44 +0000709
K. Y. Srinivasand6c1c5d2011-06-06 15:50:08 -0700710 return 0;
K. Y. Srinivasan8b9987e2011-08-31 14:35:55 -0700711
Jason Wang2608fb62013-06-19 11:28:10 +0800712err_alloc:
713 hv_synic_free();
Thomas Gleixner76d388c2014-03-05 13:42:14 +0100714 hv_remove_vmbus_irq();
K. Y. Srinivasan8b9987e2011-08-31 14:35:55 -0700715
K. Y. Srinivasan8b9987e2011-08-31 14:35:55 -0700716 bus_unregister(&hv_bus);
717
718err_cleanup:
719 hv_cleanup();
720
721 return ret;
Hank Janssen3e7ee492009-07-13 16:02:34 -0700722}
723
Greg Kroah-Hartman90c99602009-09-02 07:11:14 -0700724/**
Greg Kroah-Hartman768fa212011-08-25 15:07:32 -0700725 * __vmbus_child_driver_register - Register a vmbus's driver
726 * @drv: Pointer to driver structure you want to register
727 * @owner: owner module of the drv
728 * @mod_name: module name string
Hank Janssen3e189512010-03-04 22:11:00 +0000729 *
730 * Registers the given driver with Linux through the 'driver_register()' call
Greg Kroah-Hartman768fa212011-08-25 15:07:32 -0700731 * and sets up the hyper-v vmbus handling for this driver.
Hank Janssen3e189512010-03-04 22:11:00 +0000732 * It will return the state of the 'driver_register()' call.
733 *
Greg Kroah-Hartman90c99602009-09-02 07:11:14 -0700734 */
Greg Kroah-Hartman768fa212011-08-25 15:07:32 -0700735int __vmbus_driver_register(struct hv_driver *hv_driver, struct module *owner, const char *mod_name)
Hank Janssen3e7ee492009-07-13 16:02:34 -0700736{
Bill Pemberton5d48a1c2009-07-27 16:47:36 -0400737 int ret;
Hank Janssen3e7ee492009-07-13 16:02:34 -0700738
Greg Kroah-Hartman768fa212011-08-25 15:07:32 -0700739 pr_info("registering driver %s\n", hv_driver->name);
Hank Janssen3e7ee492009-07-13 16:02:34 -0700740
K. Y. Srinivasancf6a2ea2011-12-01 09:59:34 -0800741 ret = vmbus_exists();
742 if (ret < 0)
743 return ret;
744
Greg Kroah-Hartman768fa212011-08-25 15:07:32 -0700745 hv_driver->driver.name = hv_driver->name;
746 hv_driver->driver.owner = owner;
747 hv_driver->driver.mod_name = mod_name;
748 hv_driver->driver.bus = &hv_bus;
Hank Janssen3e7ee492009-07-13 16:02:34 -0700749
Greg Kroah-Hartman768fa212011-08-25 15:07:32 -0700750 ret = driver_register(&hv_driver->driver);
Hank Janssen3e7ee492009-07-13 16:02:34 -0700751
Bill Pemberton5d48a1c2009-07-27 16:47:36 -0400752 return ret;
Hank Janssen3e7ee492009-07-13 16:02:34 -0700753}
Greg Kroah-Hartman768fa212011-08-25 15:07:32 -0700754EXPORT_SYMBOL_GPL(__vmbus_driver_register);
Hank Janssen3e7ee492009-07-13 16:02:34 -0700755
Greg Kroah-Hartman90c99602009-09-02 07:11:14 -0700756/**
Greg Kroah-Hartman768fa212011-08-25 15:07:32 -0700757 * vmbus_driver_unregister() - Unregister a vmbus's driver
758 * @drv: Pointer to driver structure you want to un-register
Hank Janssen3e189512010-03-04 22:11:00 +0000759 *
Greg Kroah-Hartman768fa212011-08-25 15:07:32 -0700760 * Un-register the given driver that was previous registered with a call to
761 * vmbus_driver_register()
Greg Kroah-Hartman90c99602009-09-02 07:11:14 -0700762 */
Greg Kroah-Hartman768fa212011-08-25 15:07:32 -0700763void vmbus_driver_unregister(struct hv_driver *hv_driver)
Hank Janssen3e7ee492009-07-13 16:02:34 -0700764{
Greg Kroah-Hartman768fa212011-08-25 15:07:32 -0700765 pr_info("unregistering driver %s\n", hv_driver->name);
Hank Janssen3e7ee492009-07-13 16:02:34 -0700766
K. Y. Srinivasancf6a2ea2011-12-01 09:59:34 -0800767 if (!vmbus_exists())
K. Y. Srinivasan8f257a12011-12-27 13:49:37 -0800768 driver_unregister(&hv_driver->driver);
Hank Janssen3e7ee492009-07-13 16:02:34 -0700769}
Greg Kroah-Hartman768fa212011-08-25 15:07:32 -0700770EXPORT_SYMBOL_GPL(vmbus_driver_unregister);
Hank Janssen3e7ee492009-07-13 16:02:34 -0700771
Hank Janssen3e189512010-03-04 22:11:00 +0000772/*
K. Y. Srinivasanf2c73012011-09-08 07:24:12 -0700773 * vmbus_device_create - Creates and registers a new child device
Hank Janssen3e189512010-03-04 22:11:00 +0000774 * on the vmbus.
Greg Kroah-Hartman90c99602009-09-02 07:11:14 -0700775 */
K. Y. Srinivasanf2c73012011-09-08 07:24:12 -0700776struct hv_device *vmbus_device_create(uuid_le *type,
K. Y. Srinivasan358d2ee2011-08-25 09:48:28 -0700777 uuid_le *instance,
Greg Kroah-Hartman89733aa2010-12-02 08:22:41 -0800778 struct vmbus_channel *channel)
Hank Janssen3e7ee492009-07-13 16:02:34 -0700779{
Nicolas Palix3d3b5512009-07-28 17:32:53 +0200780 struct hv_device *child_device_obj;
Hank Janssen3e7ee492009-07-13 16:02:34 -0700781
K. Y. Srinivasan6bad88da2011-03-07 13:35:48 -0800782 child_device_obj = kzalloc(sizeof(struct hv_device), GFP_KERNEL);
783 if (!child_device_obj) {
Hank Janssen0a466182011-03-29 13:58:47 -0700784 pr_err("Unable to allocate device object for child device\n");
Hank Janssen3e7ee492009-07-13 16:02:34 -0700785 return NULL;
786 }
787
Greg Kroah-Hartmancae5b842010-10-21 09:05:27 -0700788 child_device_obj->channel = channel;
K. Y. Srinivasan358d2ee2011-08-25 09:48:28 -0700789 memcpy(&child_device_obj->dev_type, type, sizeof(uuid_le));
Haiyang Zhangca623ad2011-01-26 12:12:11 -0800790 memcpy(&child_device_obj->dev_instance, instance,
K. Y. Srinivasan358d2ee2011-08-25 09:48:28 -0700791 sizeof(uuid_le));
Hank Janssen3e7ee492009-07-13 16:02:34 -0700792
Hank Janssen3e7ee492009-07-13 16:02:34 -0700793
Hank Janssen3e7ee492009-07-13 16:02:34 -0700794 return child_device_obj;
795}
796
Hank Janssen3e189512010-03-04 22:11:00 +0000797/*
K. Y. Srinivasan227942812011-09-08 07:24:13 -0700798 * vmbus_device_register - Register the child device
Greg Kroah-Hartman90c99602009-09-02 07:11:14 -0700799 */
K. Y. Srinivasan227942812011-09-08 07:24:13 -0700800int vmbus_device_register(struct hv_device *child_device_obj)
Hank Janssen3e7ee492009-07-13 16:02:34 -0700801{
Greg Kroah-Hartman90c99602009-09-02 07:11:14 -0700802 int ret = 0;
K. Y. Srinivasan6bad88da2011-03-07 13:35:48 -0800803
Bill Pembertonf4888412009-07-29 17:00:12 -0400804 static atomic_t device_num = ATOMIC_INIT(0);
Hank Janssen3e7ee492009-07-13 16:02:34 -0700805
K. Y. Srinivasan6bad88da2011-03-07 13:35:48 -0800806 dev_set_name(&child_device_obj->device, "vmbus_0_%d",
Greg Kroah-Hartman90c99602009-09-02 07:11:14 -0700807 atomic_inc_return(&device_num));
Hank Janssen3e7ee492009-07-13 16:02:34 -0700808
K. Y. Srinivasan0bce28b2011-08-27 11:31:39 -0700809 child_device_obj->device.bus = &hv_bus;
K. Y. Srinivasan607c1a12011-06-06 15:49:39 -0700810 child_device_obj->device.parent = &hv_acpi_dev->dev;
K. Y. Srinivasan6bad88da2011-03-07 13:35:48 -0800811 child_device_obj->device.release = vmbus_device_release;
Hank Janssen3e7ee492009-07-13 16:02:34 -0700812
Greg Kroah-Hartman90c99602009-09-02 07:11:14 -0700813 /*
814 * Register with the LDM. This will kick off the driver/device
815 * binding...which will eventually call vmbus_match() and vmbus_probe()
816 */
K. Y. Srinivasan6bad88da2011-03-07 13:35:48 -0800817 ret = device_register(&child_device_obj->device);
Hank Janssen3e7ee492009-07-13 16:02:34 -0700818
Hank Janssen3e7ee492009-07-13 16:02:34 -0700819 if (ret)
Hank Janssen0a466182011-03-29 13:58:47 -0700820 pr_err("Unable to register child device\n");
Hank Janssen3e7ee492009-07-13 16:02:34 -0700821 else
Fernando Soto84672362013-06-14 23:13:35 +0000822 pr_debug("child device %s registered\n",
Hank Janssen0a466182011-03-29 13:58:47 -0700823 dev_name(&child_device_obj->device));
Hank Janssen3e7ee492009-07-13 16:02:34 -0700824
Hank Janssen3e7ee492009-07-13 16:02:34 -0700825 return ret;
826}
827
Hank Janssen3e189512010-03-04 22:11:00 +0000828/*
K. Y. Srinivasan696453b2011-09-08 07:24:14 -0700829 * vmbus_device_unregister - Remove the specified child device
Hank Janssen3e189512010-03-04 22:11:00 +0000830 * from the vmbus.
Greg Kroah-Hartman90c99602009-09-02 07:11:14 -0700831 */
K. Y. Srinivasan696453b2011-09-08 07:24:14 -0700832void vmbus_device_unregister(struct hv_device *device_obj)
Hank Janssen3e7ee492009-07-13 16:02:34 -0700833{
Fernando Soto84672362013-06-14 23:13:35 +0000834 pr_debug("child device %s unregistered\n",
835 dev_name(&device_obj->device));
836
Greg Kroah-Hartman90c99602009-09-02 07:11:14 -0700837 /*
838 * Kick off the process of unregistering the device.
839 * This will call vmbus_remove() and eventually vmbus_device_release()
840 */
K. Y. Srinivasan6bad88da2011-03-07 13:35:48 -0800841 device_unregister(&device_obj->device);
Hank Janssen3e7ee492009-07-13 16:02:34 -0700842}
843
Hank Janssen3e7ee492009-07-13 16:02:34 -0700844
K. Y. Srinivasanb0069f42011-04-29 13:45:15 -0700845/*
846 * VMBUS is an acpi enumerated device. Get the the IRQ information
847 * from DSDT.
848 */
849
850static acpi_status vmbus_walk_resources(struct acpi_resource *res, void *irq)
851{
852
853 if (res->type == ACPI_RESOURCE_TYPE_IRQ) {
854 struct acpi_resource_irq *irqp;
855 irqp = &res->data.irq;
856
857 *((unsigned int *)irq) = irqp->interrupts[0];
858 }
859
860 return AE_OK;
861}
862
863static int vmbus_acpi_add(struct acpi_device *device)
864{
865 acpi_status result;
866
K. Y. Srinivasan607c1a12011-06-06 15:49:39 -0700867 hv_acpi_dev = device;
868
K. Y. Srinivasan0a4425b2011-08-27 11:31:38 -0700869 result = acpi_walk_resources(device->handle, METHOD_NAME__CRS,
870 vmbus_walk_resources, &irq);
K. Y. Srinivasanb0069f42011-04-29 13:45:15 -0700871
872 if (ACPI_FAILURE(result)) {
873 complete(&probe_event);
874 return -ENODEV;
875 }
876 complete(&probe_event);
877 return 0;
878}
879
880static const struct acpi_device_id vmbus_acpi_device_ids[] = {
881 {"VMBUS", 0},
K. Y. Srinivasan9d7b18d2011-06-06 15:49:42 -0700882 {"VMBus", 0},
K. Y. Srinivasanb0069f42011-04-29 13:45:15 -0700883 {"", 0},
884};
885MODULE_DEVICE_TABLE(acpi, vmbus_acpi_device_ids);
886
887static struct acpi_driver vmbus_acpi_driver = {
888 .name = "vmbus",
889 .ids = vmbus_acpi_device_ids,
890 .ops = {
891 .add = vmbus_acpi_add,
892 },
893};
894
K. Y. Srinivasan607c1a12011-06-06 15:49:39 -0700895static int __init hv_acpi_init(void)
K. Y. Srinivasan1168ac22011-03-15 15:03:32 -0700896{
K. Y. Srinivasan2dda95f2011-07-15 13:38:56 -0700897 int ret, t;
K. Y. Srinivasanb0069f42011-04-29 13:45:15 -0700898
Jason Wang1f94ea82012-08-31 13:32:44 +0800899 if (x86_hyper != &x86_hyper_ms_hyperv)
Jason Wang05929692012-08-17 18:52:43 +0800900 return -ENODEV;
901
K. Y. Srinivasanb0069f42011-04-29 13:45:15 -0700902 init_completion(&probe_event);
903
904 /*
905 * Get irq resources first.
906 */
K. Y. Srinivasan02466042011-06-06 15:49:40 -0700907 ret = acpi_bus_register_driver(&vmbus_acpi_driver);
908
K. Y. Srinivasanb0069f42011-04-29 13:45:15 -0700909 if (ret)
910 return ret;
911
K. Y. Srinivasan2dda95f2011-07-15 13:38:56 -0700912 t = wait_for_completion_timeout(&probe_event, 5*HZ);
913 if (t == 0) {
914 ret = -ETIMEDOUT;
915 goto cleanup;
916 }
K. Y. Srinivasanb0069f42011-04-29 13:45:15 -0700917
918 if (irq <= 0) {
K. Y. Srinivasan2dda95f2011-07-15 13:38:56 -0700919 ret = -ENODEV;
920 goto cleanup;
K. Y. Srinivasanb0069f42011-04-29 13:45:15 -0700921 }
922
K. Y. Srinivasan91fd7992011-06-16 13:16:38 -0700923 ret = vmbus_bus_init(irq);
924 if (ret)
K. Y. Srinivasan2dda95f2011-07-15 13:38:56 -0700925 goto cleanup;
926
927 return 0;
928
929cleanup:
930 acpi_bus_unregister_driver(&vmbus_acpi_driver);
K. Y. Srinivasancf6a2ea2011-12-01 09:59:34 -0800931 hv_acpi_dev = NULL;
K. Y. Srinivasan91fd7992011-06-16 13:16:38 -0700932 return ret;
K. Y. Srinivasan1168ac22011-03-15 15:03:32 -0700933}
934
K. Y. Srinivasan93e5bd02011-12-12 09:29:17 -0800935static void __exit vmbus_exit(void)
936{
Thomas Gleixner76d388c2014-03-05 13:42:14 +0100937 hv_remove_vmbus_irq();
K. Y. Srinivasan93e5bd02011-12-12 09:29:17 -0800938 vmbus_free_channels();
939 bus_unregister(&hv_bus);
940 hv_cleanup();
941 acpi_bus_unregister_driver(&vmbus_acpi_driver);
942}
943
K. Y. Srinivasan1168ac22011-03-15 15:03:32 -0700944
Greg Kroah-Hartman90c99602009-09-02 07:11:14 -0700945MODULE_LICENSE("GPL");
Hank Janssen3e7ee492009-07-13 16:02:34 -0700946
K. Y. Srinivasan43d4e112011-10-24 11:28:12 -0700947subsys_initcall(hv_acpi_init);
K. Y. Srinivasan93e5bd02011-12-12 09:29:17 -0800948module_exit(vmbus_exit);