diff options
| author | Haiyang Zhang <haiyangz@microsoft.com> | 2014-03-04 17:11:06 -0500 |
|---|---|---|
| committer | David S. Miller <davem@davemloft.net> | 2014-03-05 20:40:25 -0500 |
| commit | 1b07da516ee25250f458c76c012ebe4cd677a84f (patch) | |
| tree | ac4eef27d05f7b53022a2274292b40c7d0733d0e | |
| parent | 48330e08fa168395b9fd9f369f06cca1df204361 (diff) | |
hyperv: Move state setting for link query
It moves the state setting for query into rndis_filter_receive_response().
All callbacks including query-complete and status-callback are synchronized
by channel->inbound_lock. This prevents pentential race between them.
Signed-off-by: Haiyang Zhang <haiyangz@microsoft.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
| -rw-r--r-- | drivers/net/hyperv/netvsc_drv.c | 4 | ||||
| -rw-r--r-- | drivers/net/hyperv/rndis_filter.c | 21 |
2 files changed, 24 insertions, 1 deletions
diff --git a/drivers/net/hyperv/netvsc_drv.c b/drivers/net/hyperv/netvsc_drv.c index 7141a1937360..d6fce9750b95 100644 --- a/drivers/net/hyperv/netvsc_drv.c +++ b/drivers/net/hyperv/netvsc_drv.c | |||
| @@ -442,6 +442,8 @@ static int netvsc_probe(struct hv_device *dev, | |||
| 442 | if (!net) | 442 | if (!net) |
| 443 | return -ENOMEM; | 443 | return -ENOMEM; |
| 444 | 444 | ||
| 445 | netif_carrier_off(net); | ||
| 446 | |||
| 445 | net_device_ctx = netdev_priv(net); | 447 | net_device_ctx = netdev_priv(net); |
| 446 | net_device_ctx->device_ctx = dev; | 448 | net_device_ctx->device_ctx = dev; |
| 447 | hv_set_drvdata(dev, net); | 449 | hv_set_drvdata(dev, net); |
| @@ -473,6 +475,8 @@ static int netvsc_probe(struct hv_device *dev, | |||
| 473 | pr_err("Unable to register netdev.\n"); | 475 | pr_err("Unable to register netdev.\n"); |
| 474 | rndis_filter_device_remove(dev); | 476 | rndis_filter_device_remove(dev); |
| 475 | free_netdev(net); | 477 | free_netdev(net); |
| 478 | } else { | ||
| 479 | schedule_delayed_work(&net_device_ctx->dwork, 0); | ||
| 476 | } | 480 | } |
| 477 | 481 | ||
| 478 | return ret; | 482 | return ret; |
diff --git a/drivers/net/hyperv/rndis_filter.c b/drivers/net/hyperv/rndis_filter.c index 1084e5de3ceb..b54fd257652b 100644 --- a/drivers/net/hyperv/rndis_filter.c +++ b/drivers/net/hyperv/rndis_filter.c | |||
| @@ -243,6 +243,22 @@ static int rndis_filter_send_request(struct rndis_device *dev, | |||
| 243 | return ret; | 243 | return ret; |
| 244 | } | 244 | } |
| 245 | 245 | ||
| 246 | static void rndis_set_link_state(struct rndis_device *rdev, | ||
| 247 | struct rndis_request *request) | ||
| 248 | { | ||
| 249 | u32 link_status; | ||
| 250 | struct rndis_query_complete *query_complete; | ||
| 251 | |||
| 252 | query_complete = &request->response_msg.msg.query_complete; | ||
| 253 | |||
| 254 | if (query_complete->status == RNDIS_STATUS_SUCCESS && | ||
| 255 | query_complete->info_buflen == sizeof(u32)) { | ||
| 256 | memcpy(&link_status, (void *)((unsigned long)query_complete + | ||
| 257 | query_complete->info_buf_offset), sizeof(u32)); | ||
| 258 | rdev->link_state = link_status != 0; | ||
| 259 | } | ||
| 260 | } | ||
| 261 | |||
| 246 | static void rndis_filter_receive_response(struct rndis_device *dev, | 262 | static void rndis_filter_receive_response(struct rndis_device *dev, |
| 247 | struct rndis_message *resp) | 263 | struct rndis_message *resp) |
| 248 | { | 264 | { |
| @@ -272,6 +288,10 @@ static void rndis_filter_receive_response(struct rndis_device *dev, | |||
| 272 | sizeof(struct rndis_message) + RNDIS_EXT_LEN) { | 288 | sizeof(struct rndis_message) + RNDIS_EXT_LEN) { |
| 273 | memcpy(&request->response_msg, resp, | 289 | memcpy(&request->response_msg, resp, |
| 274 | resp->msg_len); | 290 | resp->msg_len); |
| 291 | if (request->request_msg.ndis_msg_type == | ||
| 292 | RNDIS_MSG_QUERY && request->request_msg.msg. | ||
| 293 | query_req.oid == RNDIS_OID_GEN_MEDIA_CONNECT_STATUS) | ||
| 294 | rndis_set_link_state(dev, request); | ||
| 275 | } else { | 295 | } else { |
| 276 | netdev_err(ndev, | 296 | netdev_err(ndev, |
| 277 | "rndis response buffer overflow " | 297 | "rndis response buffer overflow " |
| @@ -620,7 +640,6 @@ static int rndis_filter_query_device_link_status(struct rndis_device *dev) | |||
| 620 | ret = rndis_filter_query_device(dev, | 640 | ret = rndis_filter_query_device(dev, |
| 621 | RNDIS_OID_GEN_MEDIA_CONNECT_STATUS, | 641 | RNDIS_OID_GEN_MEDIA_CONNECT_STATUS, |
| 622 | &link_status, &size); | 642 | &link_status, &size); |
| 623 | dev->link_state = (link_status != 0) ? true : false; | ||
| 624 | 643 | ||
| 625 | return ret; | 644 | return ret; |
| 626 | } | 645 | } |
