2 * Copyright (c) 2009-2012 Microsoft Corp.
3 * Copyright (c) 2012 NetApp Inc.
4 * Copyright (c) 2012 Citrix Inc.
7 * Redistribution and use in source and binary forms, with or without
8 * modification, are permitted provided that the following conditions
10 * 1. Redistributions of source code must retain the above copyright
11 * notice unmodified, this list of conditions, and the following
13 * 2. Redistributions in binary form must reproduce the above copyright
14 * notice, this list of conditions and the following disclaimer in the
15 * documentation and/or other materials provided with the distribution.
17 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
18 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
19 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
20 * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
21 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
22 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
23 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
24 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
25 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
26 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
32 * HyperV definitions for messages that are sent between instances of the
33 * Channel Management Library in separate partitions, or in some cases,
40 #include <sys/param.h>
42 #include <sys/queue.h>
43 #include <sys/malloc.h>
44 #include <sys/kthread.h>
45 #include <sys/taskqueue.h>
46 #include <sys/systm.h>
50 #include <sys/mutex.h>
53 #include <vm/vm_param.h>
56 #include <amd64/include/xen/synch_bitops.h>
57 #include <amd64/include/atomic.h>
59 typedef uint8_t hv_bool_uint8_t;
61 #define HV_S_OK 0x00000000
62 #define HV_E_FAIL 0x80004005
63 #define HV_ERROR_NOT_SUPPORTED 0x80070032
64 #define HV_ERROR_MACHINE_LOCKED 0x800704F7
67 * VMBUS version is 32 bit, upper 16 bit for major_number and lower
68 * 16 bit for minor_number.
70 * 0.13 -- Windows Server 2008
75 #define HV_VMBUS_VERSION_WS2008 ((0 << 16) | (13))
76 #define HV_VMBUS_VERSION_WIN7 ((1 << 16) | (1))
77 #define HV_VMBUS_VERSION_WIN8 ((2 << 16) | (4))
78 #define HV_VMBUS_VERSION_WIN8_1 ((3 << 16) | (0))
80 #define HV_VMBUS_VERSION_INVALID -1
82 #define HV_VMBUS_VERSION_CURRENT HV_VMBUS_VERSION_WIN8_1
85 * Make maximum size of pipe payload of 16K
88 #define HV_MAX_PIPE_DATA_PAYLOAD (sizeof(BYTE) * 16384)
91 * Define pipe_mode values
94 #define HV_VMBUS_PIPE_TYPE_BYTE 0x00000000
95 #define HV_VMBUS_PIPE_TYPE_MESSAGE 0x00000004
98 * The size of the user defined data buffer for non-pipe offers
101 #define HV_MAX_USER_DEFINED_BYTES 120
104 * The size of the user defined data buffer for pipe offers
107 #define HV_MAX_PIPE_USER_DEFINED_BYTES 116
110 #define HV_MAX_PAGE_BUFFER_COUNT 32
111 #define HV_MAX_MULTIPAGE_BUFFER_COUNT 32
113 #define HV_ALIGN_UP(value, align) \
114 (((value) & (align-1)) ? \
115 (((value) + (align-1)) & ~(align-1) ) : (value))
117 #define HV_ALIGN_DOWN(value, align) ( (value) & ~(align-1) )
119 #define HV_NUM_PAGES_SPANNED(addr, len) \
120 ((HV_ALIGN_UP(addr+len, PAGE_SIZE) - \
121 HV_ALIGN_DOWN(addr, PAGE_SIZE)) >> PAGE_SHIFT )
123 typedef struct hv_guid {
124 unsigned char data[16];
127 int snprintf_hv_guid(char *, size_t, const hv_guid *);
129 #define HV_NIC_GUID \
130 .data = {0x63, 0x51, 0x61, 0xF8, 0x3E, 0xDF, 0xc5, 0x46, \
131 0x91, 0x3F, 0xF2, 0xD2, 0xF9, 0x65, 0xED, 0x0E}
133 #define HV_IDE_GUID \
134 .data = {0x32, 0x26, 0x41, 0x32, 0xcb, 0x86, 0xa2, 0x44, \
135 0x9b, 0x5c, 0x50, 0xd1, 0x41, 0x73, 0x54, 0xf5}
137 #define HV_SCSI_GUID \
138 .data = {0xd9, 0x63, 0x61, 0xba, 0xa1, 0x04, 0x29, 0x4d, \
139 0xb6, 0x05, 0x72, 0xe2, 0xff, 0xb1, 0xdc, 0x7f}
142 * At the center of the Channel Management library is
143 * the Channel Offer. This struct contains the
144 * fundamental information about an offer.
147 typedef struct hv_vmbus_channel_offer {
148 hv_guid interface_type;
149 hv_guid interface_instance;
150 uint64_t interrupt_latency_in_100ns_units;
151 uint32_t interface_revision;
152 uint32_t server_context_area_size; /* in bytes */
153 uint16_t channel_flags;
154 uint16_t mmio_megabytes; /* in bytes * 1024 * 1024 */
158 * Non-pipes: The user has HV_MAX_USER_DEFINED_BYTES bytes.
161 uint8_t user_defined[HV_MAX_USER_DEFINED_BYTES];
165 * Pipes: The following structure is an integrated pipe protocol, which
166 * is implemented on top of standard user-defined data. pipe
167 * clients have HV_MAX_PIPE_USER_DEFINED_BYTES left for their
172 uint8_t user_defined[HV_MAX_PIPE_USER_DEFINED_BYTES];
177 * Sub_channel_index, newly added in Win8.
179 uint16_t sub_channel_index;
182 } __packed hv_vmbus_channel_offer;
184 typedef uint32_t hv_gpadl_handle;
188 uint16_t data_offset8;
191 uint64_t transaction_id;
192 } __packed hv_vm_packet_descriptor;
194 typedef uint32_t hv_previous_packet_offset;
197 hv_previous_packet_offset previous_packet_start_offset;
198 hv_vm_packet_descriptor descriptor;
199 } __packed hv_vm_packet_header;
203 uint32_t byte_offset;
204 } __packed hv_vm_transfer_page;
207 hv_vm_packet_descriptor d;
208 uint16_t transfer_page_set_id;
209 hv_bool_uint8_t sender_owns_set;
211 uint32_t range_count;
212 hv_vm_transfer_page ranges[1];
213 } __packed hv_vm_transfer_page_packet_header;
216 hv_vm_packet_descriptor d;
219 } __packed hv_vm_gpadl_packet_header;
222 hv_vm_packet_descriptor d;
224 uint16_t transfer_page_set_id;
226 } __packed hv_vm_add_remove_transfer_page_set;
229 * This structure defines a range in guest
230 * physical space that can be made
231 * to look virtually contiguous.
236 uint32_t byte_offset;
237 uint64_t pfn_array[0];
238 } __packed hv_gpa_range;
241 * This is the format for an Establish Gpadl packet, which contains a handle
242 * by which this GPADL will be known and a set of GPA ranges associated with
243 * it. This can be converted to a MDL by the guest OS. If there are multiple
244 * GPA ranges, then the resulting MDL will be "chained," representing multiple
249 hv_vm_packet_descriptor d;
251 uint32_t range_count;
252 hv_gpa_range range[1];
253 } __packed hv_vm_establish_gpadl;
256 * This is the format for a Teardown Gpadl packet, which indicates that the
257 * GPADL handle in the Establish Gpadl packet will never be referenced again.
261 hv_vm_packet_descriptor d;
263 /* for alignment to a 8-byte boundary */
265 } __packed hv_vm_teardown_gpadl;
268 * This is the format for a GPA-Direct packet, which contains a set of GPA
269 * ranges, in addition to commands and/or data.
273 hv_vm_packet_descriptor d;
275 uint32_t range_count;
276 hv_gpa_range range[1];
277 } __packed hv_vm_data_gpa_direct;
280 * This is the format for a Additional data Packet.
283 hv_vm_packet_descriptor d;
284 uint64_t total_bytes;
285 uint32_t byte_offset;
288 } __packed hv_vm_additional_data;
291 hv_vm_packet_descriptor simple_header;
292 hv_vm_transfer_page_packet_header transfer_page_header;
293 hv_vm_gpadl_packet_header gpadl_header;
294 hv_vm_add_remove_transfer_page_set add_remove_transfer_page_header;
295 hv_vm_establish_gpadl establish_gpadl_header;
296 hv_vm_teardown_gpadl teardown_gpadl_header;
297 hv_vm_data_gpa_direct data_gpa_direct_header;
298 } __packed hv_vm_packet_largest_possible_header;
301 HV_VMBUS_PACKET_TYPE_INVALID = 0x0,
302 HV_VMBUS_PACKET_TYPES_SYNCH = 0x1,
303 HV_VMBUS_PACKET_TYPE_ADD_TRANSFER_PAGE_SET = 0x2,
304 HV_VMBUS_PACKET_TYPE_REMOVE_TRANSFER_PAGE_SET = 0x3,
305 HV_VMBUS_PACKET_TYPE_ESTABLISH_GPADL = 0x4,
306 HV_VMBUS_PACKET_TYPE_TEAR_DOWN_GPADL = 0x5,
307 HV_VMBUS_PACKET_TYPE_DATA_IN_BAND = 0x6,
308 HV_VMBUS_PACKET_TYPE_DATA_USING_TRANSFER_PAGES = 0x7,
309 HV_VMBUS_PACKET_TYPE_DATA_USING_GPADL = 0x8,
310 HV_VMBUS_PACKET_TYPE_DATA_USING_GPA_DIRECT = 0x9,
311 HV_VMBUS_PACKET_TYPE_CANCEL_REQUEST = 0xa,
312 HV_VMBUS_PACKET_TYPE_COMPLETION = 0xb,
313 HV_VMBUS_PACKET_TYPE_DATA_USING_ADDITIONAL_PACKETS = 0xc,
314 HV_VMBUS_PACKET_TYPE_ADDITIONAL_DATA = 0xd
315 } hv_vmbus_packet_type;
317 #define HV_VMBUS_DATA_PACKET_FLAG_COMPLETION_REQUESTED 1
323 HV_CHANNEL_MESSAGE_INVALID = 0,
324 HV_CHANNEL_MESSAGE_OFFER_CHANNEL = 1,
325 HV_CHANNEL_MESSAGE_RESCIND_CHANNEL_OFFER = 2,
326 HV_CHANNEL_MESSAGE_REQUEST_OFFERS = 3,
327 HV_CHANNEL_MESSAGE_ALL_OFFERS_DELIVERED = 4,
328 HV_CHANNEL_MESSAGE_OPEN_CHANNEL = 5,
329 HV_CHANNEL_MESSAGE_OPEN_CHANNEL_RESULT = 6,
330 HV_CHANNEL_MESSAGE_CLOSE_CHANNEL = 7,
331 HV_CHANNEL_MESSAGEL_GPADL_HEADER = 8,
332 HV_CHANNEL_MESSAGE_GPADL_BODY = 9,
333 HV_CHANNEL_MESSAGE_GPADL_CREATED = 10,
334 HV_CHANNEL_MESSAGE_GPADL_TEARDOWN = 11,
335 HV_CHANNEL_MESSAGE_GPADL_TORNDOWN = 12,
336 HV_CHANNEL_MESSAGE_REL_ID_RELEASED = 13,
337 HV_CHANNEL_MESSAGE_INITIATED_CONTACT = 14,
338 HV_CHANNEL_MESSAGE_VERSION_RESPONSE = 15,
339 HV_CHANNEL_MESSAGE_UNLOAD = 16,
340 HV_CHANNEL_MESSAGE_COUNT
341 } hv_vmbus_channel_msg_type;
344 hv_vmbus_channel_msg_type message_type;
346 } __packed hv_vmbus_channel_msg_header;
349 * Query VMBus Version parameters
352 hv_vmbus_channel_msg_header header;
354 } __packed hv_vmbus_channel_query_vmbus_version;
357 * VMBus Version Supported parameters
360 hv_vmbus_channel_msg_header header;
361 hv_bool_uint8_t version_supported;
362 } __packed hv_vmbus_channel_version_supported;
365 * Channel Offer parameters
368 hv_vmbus_channel_msg_header header;
369 hv_vmbus_channel_offer offer;
370 uint32_t child_rel_id;
373 * This field has been split into a bit field on Win7
376 uint8_t monitor_allocated:1;
379 * Following fields were added in win7 and higher.
380 * Make sure to check the version before accessing these fields.
382 * If "is_dedicated_interrupt" is set, we must not set the
383 * associated bit in the channel bitmap while sending the
384 * interrupt to the host.
386 * connection_id is used in signaling the host.
388 uint16_t is_dedicated_interrupt:1;
389 uint16_t reserved1:15;
390 uint32_t connection_id;
391 } __packed hv_vmbus_channel_offer_channel;
394 * Rescind Offer parameters
398 hv_vmbus_channel_msg_header header;
399 uint32_t child_rel_id;
400 } __packed hv_vmbus_channel_rescind_offer;
404 * Request Offer -- no parameters, SynIC message contains the partition ID
406 * Set Snoop -- no parameters, SynIC message contains the partition ID
408 * Clear Snoop -- no parameters, SynIC message contains the partition ID
410 * All Offers Delivered -- no parameters, SynIC message contains the
413 * Flush Client -- no parameters, SynIC message contains the partition ID
418 * Open Channel parameters
422 hv_vmbus_channel_msg_header header;
425 * Identifies the specific VMBus channel that is being opened.
427 uint32_t child_rel_id;
430 * ID making a particular open request at a channel offer unique.
435 * GPADL for the channel's ring buffer.
437 hv_gpadl_handle ring_buffer_gpadl_handle;
440 * Before win8, all incoming channel interrupts are only
441 * delivered on cpu 0. Setting this value to 0 would
442 * preserve the earlier behavior.
444 uint32_t target_vcpu;
447 * The upstream ring buffer begins at offset zero in the memory described
448 * by ring_buffer_gpadl_handle. The downstream ring buffer follows it at
449 * this offset (in pages).
451 uint32_t downstream_ring_buffer_page_offset;
454 * User-specific data to be passed along to the server endpoint.
456 uint8_t user_data[HV_MAX_USER_DEFINED_BYTES];
458 } __packed hv_vmbus_channel_open_channel;
460 typedef uint32_t hv_nt_status;
463 * Open Channel Result parameters
467 hv_vmbus_channel_msg_header header;
468 uint32_t child_rel_id;
471 } __packed hv_vmbus_channel_open_result;
474 * Close channel parameters
478 hv_vmbus_channel_msg_header header;
479 uint32_t child_rel_id;
480 } __packed hv_vmbus_channel_close_channel;
483 * Channel Message GPADL
485 #define HV_GPADL_TYPE_RING_BUFFER 1
486 #define HV_GPADL_TYPE_SERVER_SAVE_AREA 2
487 #define HV_GPADL_TYPE_TRANSACTION 8
490 * The number of PFNs in a GPADL message is defined by the number of pages
491 * that would be spanned by byte_count and byte_offset. If the implied number
492 * of PFNs won't fit in this packet, there will be a follow-up packet that
497 hv_vmbus_channel_msg_header header;
498 uint32_t child_rel_id;
500 uint16_t range_buf_len;
501 uint16_t range_count;
502 hv_gpa_range range[0];
503 } __packed hv_vmbus_channel_gpadl_header;
506 * This is the follow-up packet that contains more PFNs
509 hv_vmbus_channel_msg_header header;
510 uint32_t message_number;
513 } __packed hv_vmbus_channel_gpadl_body;
516 hv_vmbus_channel_msg_header header;
517 uint32_t child_rel_id;
519 uint32_t creation_status;
520 } __packed hv_vmbus_channel_gpadl_created;
523 hv_vmbus_channel_msg_header header;
524 uint32_t child_rel_id;
526 } __packed hv_vmbus_channel_gpadl_teardown;
529 hv_vmbus_channel_msg_header header;
531 } __packed hv_vmbus_channel_gpadl_torndown;
534 hv_vmbus_channel_msg_header header;
535 uint32_t child_rel_id;
536 } __packed hv_vmbus_channel_relid_released;
539 hv_vmbus_channel_msg_header header;
540 uint32_t vmbus_version_requested;
542 uint64_t interrupt_page;
543 uint64_t monitor_page_1;
544 uint64_t monitor_page_2;
545 } __packed hv_vmbus_channel_initiate_contact;
548 hv_vmbus_channel_msg_header header;
549 hv_bool_uint8_t version_supported;
550 } __packed hv_vmbus_channel_version_response;
552 typedef hv_vmbus_channel_msg_header hv_vmbus_channel_unload;
554 #define HW_MACADDR_LEN 6
557 * Fixme: Added to quiet "typeof" errors involving hv_vmbus.h when
558 * the including C file was compiled with "-std=c99".
561 #define typeof __typeof
565 #define NULL (void *)0
568 typedef void *hv_vmbus_handle;
570 #ifndef CONTAINING_RECORD
571 #define CONTAINING_RECORD(address, type, field) ((type *)( \
572 (uint8_t *)(address) - \
573 (uint8_t *)(&((type *)0)->field)))
574 #endif /* CONTAINING_RECORD */
577 #define container_of(ptr, type, member) ({ \
578 __typeof__( ((type *)0)->member ) *__mptr = (ptr); \
579 (type *)( (char *)__mptr - offsetof(type,member) );})
583 HV_VMBUS_IVAR_INSTANCE,
588 #define HV_VMBUS_ACCESSOR(var, ivar, type) \
589 __BUS_ACCESSOR(vmbus, var, HV_VMBUS, ivar, type)
591 HV_VMBUS_ACCESSOR(type, TYPE, const char *)
592 HV_VMBUS_ACCESSOR(devctx, DEVCTX, struct hv_device *)
596 * Common defines for Hyper-V ICs
598 #define HV_ICMSGTYPE_NEGOTIATE 0
599 #define HV_ICMSGTYPE_HEARTBEAT 1
600 #define HV_ICMSGTYPE_KVPEXCHANGE 2
601 #define HV_ICMSGTYPE_SHUTDOWN 3
602 #define HV_ICMSGTYPE_TIMESYNC 4
603 #define HV_ICMSGTYPE_VSS 5
605 #define HV_ICMSGHDRFLAG_TRANSACTION 1
606 #define HV_ICMSGHDRFLAG_REQUEST 2
607 #define HV_ICMSGHDRFLAG_RESPONSE 4
609 typedef struct hv_vmbus_pipe_hdr {
612 } __packed hv_vmbus_pipe_hdr;
614 typedef struct hv_vmbus_ic_version {
617 } __packed hv_vmbus_ic_version;
619 typedef struct hv_vmbus_icmsg_hdr {
620 hv_vmbus_ic_version icverframe;
622 hv_vmbus_ic_version icvermsg;
625 uint8_t ictransaction_id;
628 } __packed hv_vmbus_icmsg_hdr;
630 typedef struct hv_vmbus_icmsg_negotiate {
631 uint16_t icframe_vercnt;
632 uint16_t icmsg_vercnt;
634 hv_vmbus_ic_version icversion_data[1]; /* any size array */
635 } __packed hv_vmbus_icmsg_negotiate;
637 typedef struct hv_vmbus_shutdown_msg_data {
638 uint32_t reason_code;
639 uint32_t timeout_seconds;
641 uint8_t display_message[2048];
642 } __packed hv_vmbus_shutdown_msg_data;
644 typedef struct hv_vmbus_heartbeat_msg_data {
646 uint32_t reserved[8];
647 } __packed hv_vmbus_heartbeat_msg_data;
651 * offset in bytes from the start of ring data below
653 volatile uint32_t write_index;
655 * offset in bytes from the start of ring data below
657 volatile uint32_t read_index;
659 * NOTE: The interrupt_mask field is used only for channels, but
660 * vmbus connection also uses this data structure
662 volatile uint32_t interrupt_mask;
663 /* pad it to PAGE_SIZE so that data starts on a page */
664 uint8_t reserved[4084];
667 * WARNING: Ring data starts here + ring_data_start_offset
668 * !!! DO NOT place any fields below this !!!
670 uint8_t buffer[0]; /* doubles as interrupt mask */
671 } __packed hv_vmbus_ring_buffer;
677 } __packed hv_vmbus_page_buffer;
682 uint64_t pfn_array[HV_MAX_MULTIPAGE_BUFFER_COUNT];
683 } __packed hv_vmbus_multipage_buffer;
686 hv_vmbus_ring_buffer* ring_buffer;
687 uint32_t ring_size; /* Include the shared header */
688 struct mtx ring_lock;
689 uint32_t ring_data_size; /* ring_size */
690 uint32_t ring_data_start_offset;
691 } hv_vmbus_ring_buffer_info;
693 typedef void (*hv_vmbus_pfn_channel_callback)(void *context);
694 typedef void (*hv_vmbus_sc_creation_callback)(void *context);
697 HV_CHANNEL_OFFER_STATE,
698 HV_CHANNEL_OPENING_STATE,
699 HV_CHANNEL_OPEN_STATE,
700 HV_CHANNEL_OPENED_STATE,
701 HV_CHANNEL_CLOSING_NONDESTRUCTIVE_STATE,
702 } hv_vmbus_channel_state;
705 * Connection identifier type
708 uint32_t as_uint32_t;
714 } __packed hv_vmbus_connection_id;
717 * Definition of the hv_vmbus_signal_event hypercall input structure
720 hv_vmbus_connection_id connection_id;
721 uint16_t flag_number;
723 } __packed hv_vmbus_input_signal_event;
727 hv_vmbus_input_signal_event event;
728 } __packed hv_vmbus_input_signal_event_buffer;
730 typedef struct hv_vmbus_channel {
731 TAILQ_ENTRY(hv_vmbus_channel) list_entry;
732 struct hv_device* device;
733 hv_vmbus_channel_state state;
734 hv_vmbus_channel_offer_channel offer_msg;
736 * These are based on the offer_msg.monitor_id.
737 * Save it here for easy access.
739 uint8_t monitor_group;
742 uint32_t ring_buffer_gpadl_handle;
744 * Allocated memory for ring buffer
746 void* ring_buffer_pages;
747 unsigned long ring_buffer_size;
748 uint32_t ring_buffer_page_count;
752 hv_vmbus_ring_buffer_info outbound;
754 * receive from parent
756 hv_vmbus_ring_buffer_info inbound;
758 struct mtx inbound_lock;
760 hv_vmbus_pfn_channel_callback on_channel_callback;
761 void* channel_callback_context;
764 * If batched_reading is set to "true", mask the interrupt
765 * and read until the channel is empty.
766 * If batched_reading is set to "false", the channel is not
767 * going to perform batched reading.
769 * Batched reading is enabled by default; specific
770 * drivers that don't want this behavior can turn it off.
772 boolean_t batched_reading;
774 boolean_t is_dedicated_interrupt;
777 * Used as an input param for HV_CALL_SIGNAL_EVENT hypercall.
779 hv_vmbus_input_signal_event_buffer signal_event_buffer;
781 * 8-bytes aligned of the buffer above
783 hv_vmbus_input_signal_event *signal_event_param;
786 * From Win8, this field specifies the target virtual process
787 * on which to deliver the interupt from the host to guest.
788 * Before Win8, all channel interrupts would only be
789 * delivered on cpu 0. Setting this value to 0 would preserve
790 * the earlier behavior.
792 uint32_t target_vcpu;
793 /* The corresponding CPUID in the guest */
797 * Support for multi-channels.
798 * The initial offer is considered the primary channel and this
799 * offer message will indicate if the host supports multi-channels.
800 * The guest is free to ask for multi-channels to be offerred and can
801 * open these multi-channels as a normal "primary" channel. However,
802 * all multi-channels will have the same type and instance guids as the
803 * primary channel. Requests sent on a given channel will result in a
804 * response on the same channel.
808 * Multi-channel creation callback. This callback will be called in
809 * process context when a Multi-channel offer is received from the host.
810 * The guest can open the Multi-channel in the context of this callback.
812 hv_vmbus_sc_creation_callback sc_creation_callback;
817 * Link list of all the multi-channels if this is a primary channel
819 TAILQ_HEAD(, hv_vmbus_channel) sc_list_anchor;
820 TAILQ_ENTRY(hv_vmbus_channel) sc_list_entry;
823 * The primary channel this sub-channle belongs to.
824 * This will be NULL for the primary channel.
826 struct hv_vmbus_channel *primary_channel;
828 * Support per channel state for use by vmbus drivers.
830 void *per_channel_state;
834 hv_set_channel_read_state(hv_vmbus_channel* channel, boolean_t state)
836 channel->batched_reading = state;
839 typedef struct hv_device {
843 hv_vmbus_channel* channel;
848 int hv_vmbus_channel_recv_packet(
849 hv_vmbus_channel* channel,
852 uint32_t* buffer_actual_len,
853 uint64_t* request_id);
855 int hv_vmbus_channel_recv_packet_raw(
856 hv_vmbus_channel* channel,
859 uint32_t* buffer_actual_len,
860 uint64_t* request_id);
862 int hv_vmbus_channel_open(
863 hv_vmbus_channel* channel,
864 uint32_t send_ring_buffer_size,
865 uint32_t recv_ring_buffer_size,
867 uint32_t user_data_len,
868 hv_vmbus_pfn_channel_callback
869 pfn_on_channel_callback,
872 void hv_vmbus_channel_close(hv_vmbus_channel *channel);
874 int hv_vmbus_channel_send_packet(
875 hv_vmbus_channel* channel,
879 hv_vmbus_packet_type type,
882 int hv_vmbus_channel_send_packet_pagebuffer(
883 hv_vmbus_channel* channel,
884 hv_vmbus_page_buffer page_buffers[],
888 uint64_t request_id);
890 int hv_vmbus_channel_send_packet_multipagebuffer(
891 hv_vmbus_channel* channel,
892 hv_vmbus_multipage_buffer* multi_page_buffer,
895 uint64_t request_id);
897 int hv_vmbus_channel_establish_gpadl(
898 hv_vmbus_channel* channel,
899 /* must be phys and virt contiguous */
901 /* page-size multiple */
903 uint32_t* gpadl_handle);
905 int hv_vmbus_channel_teardown_gpdal(
906 hv_vmbus_channel* channel,
907 uint32_t gpadl_handle);
909 struct hv_vmbus_channel* vmbus_select_outgoing_channel(struct hv_vmbus_channel *promary);
912 * Work abstraction defines
914 typedef struct hv_work_queue {
915 struct taskqueue* queue;
917 struct sema* work_sema;
920 typedef struct hv_work_item {
922 void (*callback)(void *);
927 struct hv_work_queue* hv_work_queue_create(char* name);
929 void hv_work_queue_close(struct hv_work_queue* wq);
931 int hv_queue_work_item(
933 void (*callback)(void *),
936 * @brief Get physical address from virtual
938 static inline unsigned long
939 hv_get_phys_addr(void *virt)
942 ret = (vtophys(virt) | ((vm_offset_t) virt & PAGE_MASK));
948 * KVP related structures
951 typedef struct hv_vmbus_service {
952 hv_guid guid; /* Hyper-V GUID */
953 char *name; /* name of service */
954 boolean_t enabled; /* service enabled */
955 hv_work_queue *work_queue; /* background work queue */
958 * function to initialize service
960 int (*init)(struct hv_vmbus_service *);
963 * function to process Hyper-V messages
965 void (*callback)(void *);
968 extern uint8_t* receive_buffer[];
969 extern hv_vmbus_service service_table[];
970 extern uint32_t hv_vmbus_protocal_version;
972 void hv_kvp_callback(void *context);
973 int hv_kvp_init(hv_vmbus_service *serv);
974 void hv_kvp_deinit(void);
976 #endif /* __HYPERV_H__ */