diff options
Diffstat (limited to 'drivers')
-rw-r--r-- | drivers/hv/connection.c | 165 |
1 files changed, 111 insertions, 54 deletions
diff --git a/drivers/hv/connection.c b/drivers/hv/connection.c index d1019a770ad7..2b56a3f47b30 100644 --- a/drivers/hv/connection.c +++ b/drivers/hv/connection.c | |||
@@ -40,15 +40,111 @@ struct vmbus_connection vmbus_connection = { | |||
40 | }; | 40 | }; |
41 | 41 | ||
42 | /* | 42 | /* |
43 | * VMBUS version is 32 bit entity broken up into | ||
44 | * two 16 bit quantities: major_number. minor_number. | ||
45 | * | ||
46 | * 0 . 13 (Windows Server 2008) | ||
47 | * 1 . 1 (Windows 7) | ||
48 | * 2 . 4 (Windows 8) | ||
49 | */ | ||
50 | |||
51 | #define VERSION_WS2008 ((0 << 16) | (13)) | ||
52 | #define VERSION_WIN7 ((1 << 16) | (1)) | ||
53 | #define VERSION_WIN8 ((2 << 16) | (4)) | ||
54 | |||
55 | #define VERSION_INVAL -1 | ||
56 | |||
57 | static __u32 vmbus_get_next_version(__u32 current_version) | ||
58 | { | ||
59 | switch (current_version) { | ||
60 | case (VERSION_WIN7): | ||
61 | return VERSION_WS2008; | ||
62 | |||
63 | case (VERSION_WIN8): | ||
64 | return VERSION_WIN7; | ||
65 | |||
66 | case (VERSION_WS2008): | ||
67 | default: | ||
68 | return VERSION_INVAL; | ||
69 | } | ||
70 | } | ||
71 | |||
72 | static int vmbus_negotiate_version(struct vmbus_channel_msginfo *msginfo, | ||
73 | __u32 version) | ||
74 | { | ||
75 | int ret = 0; | ||
76 | struct vmbus_channel_initiate_contact *msg; | ||
77 | unsigned long flags; | ||
78 | int t; | ||
79 | |||
80 | init_completion(&msginfo->waitevent); | ||
81 | |||
82 | msg = (struct vmbus_channel_initiate_contact *)msginfo->msg; | ||
83 | |||
84 | msg->header.msgtype = CHANNELMSG_INITIATE_CONTACT; | ||
85 | msg->vmbus_version_requested = version; | ||
86 | msg->interrupt_page = virt_to_phys(vmbus_connection.int_page); | ||
87 | msg->monitor_page1 = virt_to_phys(vmbus_connection.monitor_pages); | ||
88 | msg->monitor_page2 = virt_to_phys( | ||
89 | (void *)((unsigned long)vmbus_connection.monitor_pages + | ||
90 | PAGE_SIZE)); | ||
91 | |||
92 | /* | ||
93 | * Add to list before we send the request since we may | ||
94 | * receive the response before returning from this routine | ||
95 | */ | ||
96 | spin_lock_irqsave(&vmbus_connection.channelmsg_lock, flags); | ||
97 | list_add_tail(&msginfo->msglistentry, | ||
98 | &vmbus_connection.chn_msg_list); | ||
99 | |||
100 | spin_unlock_irqrestore(&vmbus_connection.channelmsg_lock, flags); | ||
101 | |||
102 | ret = vmbus_post_msg(msg, | ||
103 | sizeof(struct vmbus_channel_initiate_contact)); | ||
104 | if (ret != 0) { | ||
105 | spin_lock_irqsave(&vmbus_connection.channelmsg_lock, flags); | ||
106 | list_del(&msginfo->msglistentry); | ||
107 | spin_unlock_irqrestore(&vmbus_connection.channelmsg_lock, | ||
108 | flags); | ||
109 | return ret; | ||
110 | } | ||
111 | |||
112 | /* Wait for the connection response */ | ||
113 | t = wait_for_completion_timeout(&msginfo->waitevent, 5*HZ); | ||
114 | if (t == 0) { | ||
115 | spin_lock_irqsave(&vmbus_connection.channelmsg_lock, | ||
116 | flags); | ||
117 | list_del(&msginfo->msglistentry); | ||
118 | spin_unlock_irqrestore(&vmbus_connection.channelmsg_lock, | ||
119 | flags); | ||
120 | return -ETIMEDOUT; | ||
121 | } | ||
122 | |||
123 | spin_lock_irqsave(&vmbus_connection.channelmsg_lock, flags); | ||
124 | list_del(&msginfo->msglistentry); | ||
125 | spin_unlock_irqrestore(&vmbus_connection.channelmsg_lock, flags); | ||
126 | |||
127 | /* Check if successful */ | ||
128 | if (msginfo->response.version_response.version_supported) { | ||
129 | vmbus_connection.conn_state = CONNECTED; | ||
130 | } else { | ||
131 | pr_err("Unable to connect, " | ||
132 | "Version %d not supported by Hyper-V\n", | ||
133 | version); | ||
134 | return -ECONNREFUSED; | ||
135 | } | ||
136 | |||
137 | return ret; | ||
138 | } | ||
139 | |||
140 | /* | ||
43 | * vmbus_connect - Sends a connect request on the partition service connection | 141 | * vmbus_connect - Sends a connect request on the partition service connection |
44 | */ | 142 | */ |
45 | int vmbus_connect(void) | 143 | int vmbus_connect(void) |
46 | { | 144 | { |
47 | int ret = 0; | 145 | int ret = 0; |
48 | int t; | ||
49 | struct vmbus_channel_msginfo *msginfo = NULL; | 146 | struct vmbus_channel_msginfo *msginfo = NULL; |
50 | struct vmbus_channel_initiate_contact *msg; | 147 | __u32 version; |
51 | unsigned long flags; | ||
52 | 148 | ||
53 | /* Initialize the vmbus connection */ | 149 | /* Initialize the vmbus connection */ |
54 | vmbus_connection.conn_state = CONNECTING; | 150 | vmbus_connection.conn_state = CONNECTING; |
@@ -99,64 +195,25 @@ int vmbus_connect(void) | |||
99 | goto cleanup; | 195 | goto cleanup; |
100 | } | 196 | } |
101 | 197 | ||
102 | init_completion(&msginfo->waitevent); | ||
103 | |||
104 | msg = (struct vmbus_channel_initiate_contact *)msginfo->msg; | ||
105 | |||
106 | msg->header.msgtype = CHANNELMSG_INITIATE_CONTACT; | ||
107 | msg->vmbus_version_requested = VMBUS_REVISION_NUMBER; | ||
108 | msg->interrupt_page = virt_to_phys(vmbus_connection.int_page); | ||
109 | msg->monitor_page1 = virt_to_phys(vmbus_connection.monitor_pages); | ||
110 | msg->monitor_page2 = virt_to_phys( | ||
111 | (void *)((unsigned long)vmbus_connection.monitor_pages + | ||
112 | PAGE_SIZE)); | ||
113 | |||
114 | /* | 198 | /* |
115 | * Add to list before we send the request since we may | 199 | * Negotiate a compatible VMBUS version number with the |
116 | * receive the response before returning from this routine | 200 | * host. We start with the highest number we can support |
201 | * and work our way down until we negotiate a compatible | ||
202 | * version. | ||
117 | */ | 203 | */ |
118 | spin_lock_irqsave(&vmbus_connection.channelmsg_lock, flags); | ||
119 | list_add_tail(&msginfo->msglistentry, | ||
120 | &vmbus_connection.chn_msg_list); | ||
121 | |||
122 | spin_unlock_irqrestore(&vmbus_connection.channelmsg_lock, flags); | ||
123 | 204 | ||
124 | ret = vmbus_post_msg(msg, | 205 | version = VERSION_WS2008; |
125 | sizeof(struct vmbus_channel_initiate_contact)); | ||
126 | if (ret != 0) { | ||
127 | spin_lock_irqsave(&vmbus_connection.channelmsg_lock, flags); | ||
128 | list_del(&msginfo->msglistentry); | ||
129 | spin_unlock_irqrestore(&vmbus_connection.channelmsg_lock, | ||
130 | flags); | ||
131 | goto cleanup; | ||
132 | } | ||
133 | 206 | ||
134 | /* Wait for the connection response */ | 207 | do { |
135 | t = wait_for_completion_timeout(&msginfo->waitevent, 5*HZ); | 208 | ret = vmbus_negotiate_version(msginfo, version); |
136 | if (t == 0) { | 209 | if (ret == 0) |
137 | spin_lock_irqsave(&vmbus_connection.channelmsg_lock, | 210 | break; |
138 | flags); | ||
139 | list_del(&msginfo->msglistentry); | ||
140 | spin_unlock_irqrestore(&vmbus_connection.channelmsg_lock, | ||
141 | flags); | ||
142 | ret = -ETIMEDOUT; | ||
143 | goto cleanup; | ||
144 | } | ||
145 | 211 | ||
146 | spin_lock_irqsave(&vmbus_connection.channelmsg_lock, flags); | 212 | version = vmbus_get_next_version(version); |
147 | list_del(&msginfo->msglistentry); | 213 | } while (version != VERSION_INVAL); |
148 | spin_unlock_irqrestore(&vmbus_connection.channelmsg_lock, flags); | ||
149 | 214 | ||
150 | /* Check if successful */ | 215 | if (version == VERSION_INVAL) |
151 | if (msginfo->response.version_response.version_supported) { | ||
152 | vmbus_connection.conn_state = CONNECTED; | ||
153 | } else { | ||
154 | pr_err("Unable to connect, " | ||
155 | "Version %d not supported by Hyper-V\n", | ||
156 | VMBUS_REVISION_NUMBER); | ||
157 | ret = -ECONNREFUSED; | ||
158 | goto cleanup; | 216 | goto cleanup; |
159 | } | ||
160 | 217 | ||
161 | kfree(msginfo); | 218 | kfree(msginfo); |
162 | return 0; | 219 | return 0; |