Commit 449ca62c authored by Vitaly Kuznetsov's avatar Vitaly Kuznetsov Committed by Tim Gardner

hv_netvsc: untangle the pointer mess

BugLink: http://bugs.launchpad.net/bugs/1616677

We have the following structures keeping netvsc adapter state:
- struct net_device
- struct net_device_context
- struct netvsc_device
- struct rndis_device
- struct hv_device
and there are pointers/dependencies between them:
- struct net_device_context is contained in struct net_device
- struct hv_device has driver_data pointer which points to
  'struct net_device' OR 'struct netvsc_device' depending on driver's
  state (!).
- struct net_device_context has a pointer to 'struct hv_device'.
- struct netvsc_device has pointers to 'struct hv_device' and
  'struct net_device_context'.
- struct rndis_device has a pointer to 'struct netvsc_device'.

Different functions get different structures as parameters and use these
pointers for traveling. The problem is (in addition to keeping in mind
this complex graph) that some of these structures (struct netvsc_device
and struct rndis_device) are being removed and re-created on mtu change
(as we implement it as re-creation of hyper-v device) so our travel using
these pointers is dangerous.

Simplify this to a the following:
- add struct netvsc_device pointer to struct net_device_context (which is
  a part of struct net_device and thus never disappears)
- remove struct hv_device and struct net_device_context pointers from
  struct netvsc_device
- replace pointer to 'struct netvsc_device' with pointer to
  'struct net_device'.
- always keep 'struct net_device' in hv_device driver_data.

We'll end up with the following 'circular' structure:

net_device:
 [net_device_context] -> netvsc_device -> rndis_device -> net_device
                      -> hv_device -> net_device

On MTU change we'll be removing the 'netvsc_device -> rndis_device'
branch and re-creating it making the synchronization easier.

There is one additional redundant pointer left, it is struct net_device
link in struct netvsc_device, it is going to be removed in a separate
commit.
Signed-off-by: default avatarVitaly Kuznetsov <vkuznets@redhat.com>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
(cherry picked from commit 3d541ac5)
Signed-off-by: default avatarTim Gardner <tim.gardner@canonical.com>
Acked-by: default avatarBrad Figg <brad.figg@canonical.com>
Acked-by: default avatarKamal Mostafa <kamal@canonical.com>
parent e3d4997b
...@@ -158,7 +158,7 @@ enum rndis_device_state { ...@@ -158,7 +158,7 @@ enum rndis_device_state {
}; };
struct rndis_device { struct rndis_device {
struct netvsc_device *net_dev; struct net_device *ndev;
enum rndis_device_state state; enum rndis_device_state state;
bool link_state; bool link_state;
...@@ -173,6 +173,7 @@ struct rndis_device { ...@@ -173,6 +173,7 @@ struct rndis_device {
/* Interface */ /* Interface */
struct rndis_message; struct rndis_message;
struct netvsc_device;
int netvsc_device_add(struct hv_device *device, void *additional_info); int netvsc_device_add(struct hv_device *device, void *additional_info);
int netvsc_device_remove(struct hv_device *device); int netvsc_device_remove(struct hv_device *device);
int netvsc_send(struct hv_device *device, int netvsc_send(struct hv_device *device,
...@@ -653,6 +654,8 @@ struct garp_wrk { ...@@ -653,6 +654,8 @@ struct garp_wrk {
struct net_device_context { struct net_device_context {
/* point back to our device context */ /* point back to our device context */
struct hv_device *device_ctx; struct hv_device *device_ctx;
/* netvsc_device */
struct netvsc_device *nvdev;
/* reconfigure work */ /* reconfigure work */
struct delayed_work dwork; struct delayed_work dwork;
/* last reconfig time */ /* last reconfig time */
...@@ -679,8 +682,6 @@ struct net_device_context { ...@@ -679,8 +682,6 @@ struct net_device_context {
/* Per netvsc device */ /* Per netvsc device */
struct netvsc_device { struct netvsc_device {
struct hv_device *dev;
u32 nvsp_version; u32 nvsp_version;
atomic_t num_outstanding_sends; atomic_t num_outstanding_sends;
...@@ -734,9 +735,6 @@ struct netvsc_device { ...@@ -734,9 +735,6 @@ struct netvsc_device {
u32 max_pkt; /* max number of pkt in one send, e.g. 8 */ u32 max_pkt; /* max number of pkt in one send, e.g. 8 */
u32 pkt_align; /* alignment bytes, e.g. 8 */ u32 pkt_align; /* alignment bytes, e.g. 8 */
/* The net device context */
struct net_device_context *nd_ctx;
/* 1: allocated, serial number is valid. 0: not allocated */ /* 1: allocated, serial number is valid. 0: not allocated */
u32 vf_alloc; u32 vf_alloc;
/* Serial number of the VF to team with */ /* Serial number of the VF to team with */
......
...@@ -40,7 +40,9 @@ ...@@ -40,7 +40,9 @@
void netvsc_switch_datapath(struct netvsc_device *nv_dev, bool vf) void netvsc_switch_datapath(struct netvsc_device *nv_dev, bool vf)
{ {
struct nvsp_message *init_pkt = &nv_dev->channel_init_pkt; struct nvsp_message *init_pkt = &nv_dev->channel_init_pkt;
struct hv_device *dev = nv_dev->dev; struct net_device *ndev = nv_dev->ndev;
struct net_device_context *net_device_ctx = netdev_priv(ndev);
struct hv_device *dev = net_device_ctx->device_ctx;
memset(init_pkt, 0, sizeof(struct nvsp_message)); memset(init_pkt, 0, sizeof(struct nvsp_message));
init_pkt->hdr.msg_type = NVSP_MSG4_TYPE_SWITCH_DATA_PATH; init_pkt->hdr.msg_type = NVSP_MSG4_TYPE_SWITCH_DATA_PATH;
...@@ -62,6 +64,7 @@ static struct netvsc_device *alloc_net_device(struct hv_device *device) ...@@ -62,6 +64,7 @@ static struct netvsc_device *alloc_net_device(struct hv_device *device)
{ {
struct netvsc_device *net_device; struct netvsc_device *net_device;
struct net_device *ndev = hv_get_drvdata(device); struct net_device *ndev = hv_get_drvdata(device);
struct net_device_context *net_device_ctx = netdev_priv(ndev);
net_device = kzalloc(sizeof(struct netvsc_device), GFP_KERNEL); net_device = kzalloc(sizeof(struct netvsc_device), GFP_KERNEL);
if (!net_device) if (!net_device)
...@@ -77,7 +80,6 @@ static struct netvsc_device *alloc_net_device(struct hv_device *device) ...@@ -77,7 +80,6 @@ static struct netvsc_device *alloc_net_device(struct hv_device *device)
net_device->destroy = false; net_device->destroy = false;
atomic_set(&net_device->open_cnt, 0); atomic_set(&net_device->open_cnt, 0);
atomic_set(&net_device->vf_use_cnt, 0); atomic_set(&net_device->vf_use_cnt, 0);
net_device->dev = device;
net_device->ndev = ndev; net_device->ndev = ndev;
net_device->max_pkt = RNDIS_MAX_PKT_DEFAULT; net_device->max_pkt = RNDIS_MAX_PKT_DEFAULT;
net_device->pkt_align = RNDIS_PKT_ALIGN_DEFAULT; net_device->pkt_align = RNDIS_PKT_ALIGN_DEFAULT;
...@@ -85,7 +87,8 @@ static struct netvsc_device *alloc_net_device(struct hv_device *device) ...@@ -85,7 +87,8 @@ static struct netvsc_device *alloc_net_device(struct hv_device *device)
net_device->vf_netdev = NULL; net_device->vf_netdev = NULL;
net_device->vf_inject = false; net_device->vf_inject = false;
hv_set_drvdata(device, net_device); net_device_ctx->nvdev = net_device;
return net_device; return net_device;
} }
...@@ -97,9 +100,10 @@ static void free_netvsc_device(struct netvsc_device *nvdev) ...@@ -97,9 +100,10 @@ static void free_netvsc_device(struct netvsc_device *nvdev)
static struct netvsc_device *get_outbound_net_device(struct hv_device *device) static struct netvsc_device *get_outbound_net_device(struct hv_device *device)
{ {
struct netvsc_device *net_device; struct net_device *ndev = hv_get_drvdata(device);
struct net_device_context *net_device_ctx = netdev_priv(ndev);
struct netvsc_device *net_device = net_device_ctx->nvdev;
net_device = hv_get_drvdata(device);
if (net_device && net_device->destroy) if (net_device && net_device->destroy)
net_device = NULL; net_device = NULL;
...@@ -108,9 +112,9 @@ static struct netvsc_device *get_outbound_net_device(struct hv_device *device) ...@@ -108,9 +112,9 @@ static struct netvsc_device *get_outbound_net_device(struct hv_device *device)
static struct netvsc_device *get_inbound_net_device(struct hv_device *device) static struct netvsc_device *get_inbound_net_device(struct hv_device *device)
{ {
struct netvsc_device *net_device; struct net_device *ndev = hv_get_drvdata(device);
struct net_device_context *net_device_ctx = netdev_priv(ndev);
net_device = hv_get_drvdata(device); struct netvsc_device *net_device = net_device_ctx->nvdev;
if (!net_device) if (!net_device)
goto get_in_err; goto get_in_err;
...@@ -124,11 +128,13 @@ static struct netvsc_device *get_inbound_net_device(struct hv_device *device) ...@@ -124,11 +128,13 @@ static struct netvsc_device *get_inbound_net_device(struct hv_device *device)
} }
static int netvsc_destroy_buf(struct netvsc_device *net_device) static int netvsc_destroy_buf(struct hv_device *device)
{ {
struct nvsp_message *revoke_packet; struct nvsp_message *revoke_packet;
int ret = 0; int ret = 0;
struct net_device *ndev = net_device->ndev; struct net_device *ndev = hv_get_drvdata(device);
struct net_device_context *net_device_ctx = netdev_priv(ndev);
struct netvsc_device *net_device = net_device_ctx->nvdev;
/* /*
* If we got a section count, it means we received a * If we got a section count, it means we received a
...@@ -146,7 +152,7 @@ static int netvsc_destroy_buf(struct netvsc_device *net_device) ...@@ -146,7 +152,7 @@ static int netvsc_destroy_buf(struct netvsc_device *net_device)
revoke_packet->msg.v1_msg. revoke_packet->msg.v1_msg.
revoke_recv_buf.id = NETVSC_RECEIVE_BUFFER_ID; revoke_recv_buf.id = NETVSC_RECEIVE_BUFFER_ID;
ret = vmbus_sendpacket(net_device->dev->channel, ret = vmbus_sendpacket(device->channel,
revoke_packet, revoke_packet,
sizeof(struct nvsp_message), sizeof(struct nvsp_message),
(unsigned long)revoke_packet, (unsigned long)revoke_packet,
...@@ -164,8 +170,8 @@ static int netvsc_destroy_buf(struct netvsc_device *net_device) ...@@ -164,8 +170,8 @@ static int netvsc_destroy_buf(struct netvsc_device *net_device)
/* Teardown the gpadl on the vsp end */ /* Teardown the gpadl on the vsp end */
if (net_device->recv_buf_gpadl_handle) { if (net_device->recv_buf_gpadl_handle) {
ret = vmbus_teardown_gpadl(net_device->dev->channel, ret = vmbus_teardown_gpadl(device->channel,
net_device->recv_buf_gpadl_handle); net_device->recv_buf_gpadl_handle);
/* If we failed here, we might as well return and have a leak /* If we failed here, we might as well return and have a leak
* rather than continue and a bugchk * rather than continue and a bugchk
...@@ -206,7 +212,7 @@ static int netvsc_destroy_buf(struct netvsc_device *net_device) ...@@ -206,7 +212,7 @@ static int netvsc_destroy_buf(struct netvsc_device *net_device)
revoke_packet->msg.v1_msg.revoke_send_buf.id = revoke_packet->msg.v1_msg.revoke_send_buf.id =
NETVSC_SEND_BUFFER_ID; NETVSC_SEND_BUFFER_ID;
ret = vmbus_sendpacket(net_device->dev->channel, ret = vmbus_sendpacket(device->channel,
revoke_packet, revoke_packet,
sizeof(struct nvsp_message), sizeof(struct nvsp_message),
(unsigned long)revoke_packet, (unsigned long)revoke_packet,
...@@ -222,7 +228,7 @@ static int netvsc_destroy_buf(struct netvsc_device *net_device) ...@@ -222,7 +228,7 @@ static int netvsc_destroy_buf(struct netvsc_device *net_device)
} }
/* Teardown the gpadl on the vsp end */ /* Teardown the gpadl on the vsp end */
if (net_device->send_buf_gpadl_handle) { if (net_device->send_buf_gpadl_handle) {
ret = vmbus_teardown_gpadl(net_device->dev->channel, ret = vmbus_teardown_gpadl(device->channel,
net_device->send_buf_gpadl_handle); net_device->send_buf_gpadl_handle);
/* If we failed here, we might as well return and have a leak /* If we failed here, we might as well return and have a leak
...@@ -434,7 +440,7 @@ static int netvsc_init_buf(struct hv_device *device) ...@@ -434,7 +440,7 @@ static int netvsc_init_buf(struct hv_device *device)
goto exit; goto exit;
cleanup: cleanup:
netvsc_destroy_buf(net_device); netvsc_destroy_buf(device);
exit: exit:
return ret; return ret;
...@@ -565,9 +571,9 @@ static int netvsc_connect_vsp(struct hv_device *device) ...@@ -565,9 +571,9 @@ static int netvsc_connect_vsp(struct hv_device *device)
return ret; return ret;
} }
static void netvsc_disconnect_vsp(struct netvsc_device *net_device) static void netvsc_disconnect_vsp(struct hv_device *device)
{ {
netvsc_destroy_buf(net_device); netvsc_destroy_buf(device);
} }
/* /*
...@@ -575,24 +581,13 @@ static void netvsc_disconnect_vsp(struct netvsc_device *net_device) ...@@ -575,24 +581,13 @@ static void netvsc_disconnect_vsp(struct netvsc_device *net_device)
*/ */
int netvsc_device_remove(struct hv_device *device) int netvsc_device_remove(struct hv_device *device)
{ {
struct netvsc_device *net_device; struct net_device *ndev = hv_get_drvdata(device);
unsigned long flags; struct net_device_context *net_device_ctx = netdev_priv(ndev);
struct netvsc_device *net_device = net_device_ctx->nvdev;
net_device = hv_get_drvdata(device);
netvsc_disconnect_vsp(net_device);
/* netvsc_disconnect_vsp(device);
* Since we have already drained, we don't need to busy wait
* as was done in final_release_stor_device()
* Note that we cannot set the ext pointer to NULL until
* we have drained - to drain the outgoing packets, we need to
* allow incoming packets.
*/
spin_lock_irqsave(&device->channel->inbound_lock, flags); net_device_ctx->nvdev = NULL;
hv_set_drvdata(device, NULL);
spin_unlock_irqrestore(&device->channel->inbound_lock, flags);
/* /*
* At this point, no one should be accessing net_device * At this point, no one should be accessing net_device
...@@ -640,12 +635,11 @@ static void netvsc_send_completion(struct netvsc_device *net_device, ...@@ -640,12 +635,11 @@ static void netvsc_send_completion(struct netvsc_device *net_device,
{ {
struct nvsp_message *nvsp_packet; struct nvsp_message *nvsp_packet;
struct hv_netvsc_packet *nvsc_packet; struct hv_netvsc_packet *nvsc_packet;
struct net_device *ndev; struct net_device *ndev = hv_get_drvdata(device);
struct net_device_context *net_device_ctx = netdev_priv(ndev);
u32 send_index; u32 send_index;
struct sk_buff *skb; struct sk_buff *skb;
ndev = net_device->ndev;
nvsp_packet = (struct nvsp_message *)((unsigned long)packet + nvsp_packet = (struct nvsp_message *)((unsigned long)packet +
(packet->offset8 << 3)); (packet->offset8 << 3));
...@@ -690,7 +684,7 @@ static void netvsc_send_completion(struct netvsc_device *net_device, ...@@ -690,7 +684,7 @@ static void netvsc_send_completion(struct netvsc_device *net_device,
wake_up(&net_device->wait_drain); wake_up(&net_device->wait_drain);
if (netif_tx_queue_stopped(netdev_get_tx_queue(ndev, q_idx)) && if (netif_tx_queue_stopped(netdev_get_tx_queue(ndev, q_idx)) &&
!net_device->nd_ctx->start_remove && !net_device_ctx->start_remove &&
(hv_ringbuf_avail_percent(&channel->outbound) > (hv_ringbuf_avail_percent(&channel->outbound) >
RING_AVAIL_PERCENT_HIWATER || queue_sends < 1)) RING_AVAIL_PERCENT_HIWATER || queue_sends < 1))
netif_tx_wake_queue(netdev_get_tx_queue( netif_tx_wake_queue(netdev_get_tx_queue(
...@@ -1264,17 +1258,7 @@ int netvsc_device_add(struct hv_device *device, void *additional_info) ...@@ -1264,17 +1258,7 @@ int netvsc_device_add(struct hv_device *device, void *additional_info)
net_device->ring_size = ring_size; net_device->ring_size = ring_size;
/* ndev = hv_get_drvdata(device);
* Coming into this function, struct net_device * is
* registered as the driver private data.
* In alloc_net_device(), we register struct netvsc_device *
* as the driver private data and stash away struct net_device *
* in struct netvsc_device *.
*/
ndev = net_device->ndev;
/* Add netvsc_device context to netvsc_device */
net_device->nd_ctx = netdev_priv(ndev);
/* Initialize the NetVSC channel extension */ /* Initialize the NetVSC channel extension */
init_completion(&net_device->channel_init_wait); init_completion(&net_device->channel_init_wait);
......
...@@ -70,7 +70,7 @@ static void do_set_multicast(struct work_struct *w) ...@@ -70,7 +70,7 @@ static void do_set_multicast(struct work_struct *w)
struct netvsc_device *nvdev; struct netvsc_device *nvdev;
struct rndis_device *rdev; struct rndis_device *rdev;
nvdev = hv_get_drvdata(ndevctx->device_ctx); nvdev = ndevctx->nvdev;
if (nvdev == NULL || nvdev->ndev == NULL) if (nvdev == NULL || nvdev->ndev == NULL)
return; return;
...@@ -99,7 +99,7 @@ static int netvsc_open(struct net_device *net) ...@@ -99,7 +99,7 @@ static int netvsc_open(struct net_device *net)
{ {
struct net_device_context *net_device_ctx = netdev_priv(net); struct net_device_context *net_device_ctx = netdev_priv(net);
struct hv_device *device_obj = net_device_ctx->device_ctx; struct hv_device *device_obj = net_device_ctx->device_ctx;
struct netvsc_device *nvdev; struct netvsc_device *nvdev = net_device_ctx->nvdev;
struct rndis_device *rdev; struct rndis_device *rdev;
int ret = 0; int ret = 0;
...@@ -114,7 +114,6 @@ static int netvsc_open(struct net_device *net) ...@@ -114,7 +114,6 @@ static int netvsc_open(struct net_device *net)
netif_tx_wake_all_queues(net); netif_tx_wake_all_queues(net);
nvdev = hv_get_drvdata(device_obj);
rdev = nvdev->extension; rdev = nvdev->extension;
if (!rdev->link_state) if (!rdev->link_state)
netif_carrier_on(net); netif_carrier_on(net);
...@@ -126,7 +125,7 @@ static int netvsc_close(struct net_device *net) ...@@ -126,7 +125,7 @@ static int netvsc_close(struct net_device *net)
{ {
struct net_device_context *net_device_ctx = netdev_priv(net); struct net_device_context *net_device_ctx = netdev_priv(net);
struct hv_device *device_obj = net_device_ctx->device_ctx; struct hv_device *device_obj = net_device_ctx->device_ctx;
struct netvsc_device *nvdev = hv_get_drvdata(device_obj); struct netvsc_device *nvdev = net_device_ctx->nvdev;
int ret; int ret;
u32 aread, awrite, i, msec = 10, retry = 0, retry_max = 20; u32 aread, awrite, i, msec = 10, retry = 0, retry_max = 20;
struct vmbus_channel *chn; struct vmbus_channel *chn;
...@@ -205,8 +204,7 @@ static u16 netvsc_select_queue(struct net_device *ndev, struct sk_buff *skb, ...@@ -205,8 +204,7 @@ static u16 netvsc_select_queue(struct net_device *ndev, struct sk_buff *skb,
void *accel_priv, select_queue_fallback_t fallback) void *accel_priv, select_queue_fallback_t fallback)
{ {
struct net_device_context *net_device_ctx = netdev_priv(ndev); struct net_device_context *net_device_ctx = netdev_priv(ndev);
struct hv_device *hdev = net_device_ctx->device_ctx; struct netvsc_device *nvsc_dev = net_device_ctx->nvdev;
struct netvsc_device *nvsc_dev = hv_get_drvdata(hdev);
u32 hash; u32 hash;
u16 q_idx = 0; u16 q_idx = 0;
...@@ -580,7 +578,6 @@ void netvsc_linkstatus_callback(struct hv_device *device_obj, ...@@ -580,7 +578,6 @@ void netvsc_linkstatus_callback(struct hv_device *device_obj,
struct rndis_indicate_status *indicate = &resp->msg.indicate_status; struct rndis_indicate_status *indicate = &resp->msg.indicate_status;
struct net_device *net; struct net_device *net;
struct net_device_context *ndev_ctx; struct net_device_context *ndev_ctx;
struct netvsc_device *net_device;
struct netvsc_reconfig *event; struct netvsc_reconfig *event;
unsigned long flags; unsigned long flags;
...@@ -590,8 +587,7 @@ void netvsc_linkstatus_callback(struct hv_device *device_obj, ...@@ -590,8 +587,7 @@ void netvsc_linkstatus_callback(struct hv_device *device_obj,
indicate->status != RNDIS_STATUS_MEDIA_DISCONNECT) indicate->status != RNDIS_STATUS_MEDIA_DISCONNECT)
return; return;
net_device = hv_get_drvdata(device_obj); net = hv_get_drvdata(device_obj);
net = net_device->ndev;
if (!net || net->reg_state != NETREG_REGISTERED) if (!net || net->reg_state != NETREG_REGISTERED)
return; return;
...@@ -659,16 +655,15 @@ int netvsc_recv_callback(struct hv_device *device_obj, ...@@ -659,16 +655,15 @@ int netvsc_recv_callback(struct hv_device *device_obj,
struct vmbus_channel *channel, struct vmbus_channel *channel,
u16 vlan_tci) u16 vlan_tci)
{ {
struct net_device *net; struct net_device *net = hv_get_drvdata(device_obj);
struct net_device_context *net_device_ctx; struct net_device_context *net_device_ctx = netdev_priv(net);
struct sk_buff *skb; struct sk_buff *skb;
struct sk_buff *vf_skb; struct sk_buff *vf_skb;
struct netvsc_stats *rx_stats; struct netvsc_stats *rx_stats;
struct netvsc_device *netvsc_dev = hv_get_drvdata(device_obj); struct netvsc_device *netvsc_dev = net_device_ctx->nvdev;
u32 bytes_recvd = packet->total_data_buflen; u32 bytes_recvd = packet->total_data_buflen;
int ret = 0; int ret = 0;
net = netvsc_dev->ndev;
if (!net || net->reg_state != NETREG_REGISTERED) if (!net || net->reg_state != NETREG_REGISTERED)
return NVSP_STAT_FAIL; return NVSP_STAT_FAIL;
...@@ -743,8 +738,7 @@ static void netvsc_get_channels(struct net_device *net, ...@@ -743,8 +738,7 @@ static void netvsc_get_channels(struct net_device *net,
struct ethtool_channels *channel) struct ethtool_channels *channel)
{ {
struct net_device_context *net_device_ctx = netdev_priv(net); struct net_device_context *net_device_ctx = netdev_priv(net);
struct hv_device *dev = net_device_ctx->device_ctx; struct netvsc_device *nvdev = net_device_ctx->nvdev;
struct netvsc_device *nvdev = hv_get_drvdata(dev);
if (nvdev) { if (nvdev) {
channel->max_combined = nvdev->max_chn; channel->max_combined = nvdev->max_chn;
...@@ -757,7 +751,7 @@ static int netvsc_set_channels(struct net_device *net, ...@@ -757,7 +751,7 @@ static int netvsc_set_channels(struct net_device *net,
{ {
struct net_device_context *net_device_ctx = netdev_priv(net); struct net_device_context *net_device_ctx = netdev_priv(net);
struct hv_device *dev = net_device_ctx->device_ctx; struct hv_device *dev = net_device_ctx->device_ctx;
struct netvsc_device *nvdev = hv_get_drvdata(dev); struct netvsc_device *nvdev = net_device_ctx->nvdev;
struct netvsc_device_info device_info; struct netvsc_device_info device_info;
u32 num_chn; u32 num_chn;
u32 max_chn; u32 max_chn;
...@@ -798,9 +792,6 @@ static int netvsc_set_channels(struct net_device *net, ...@@ -798,9 +792,6 @@ static int netvsc_set_channels(struct net_device *net,
nvdev->num_chn = channels->combined_count; nvdev->num_chn = channels->combined_count;
net_device_ctx->device_ctx = dev;
hv_set_drvdata(dev, net);
memset(&device_info, 0, sizeof(device_info)); memset(&device_info, 0, sizeof(device_info));
device_info.num_chn = nvdev->num_chn; /* passed to RNDIS */ device_info.num_chn = nvdev->num_chn; /* passed to RNDIS */
device_info.ring_size = ring_size; device_info.ring_size = ring_size;
...@@ -815,7 +806,7 @@ static int netvsc_set_channels(struct net_device *net, ...@@ -815,7 +806,7 @@ static int netvsc_set_channels(struct net_device *net,
goto recover; goto recover;
} }
nvdev = hv_get_drvdata(dev); nvdev = net_device_ctx->nvdev;
ret = netif_set_real_num_tx_queues(net, nvdev->num_chn); ret = netif_set_real_num_tx_queues(net, nvdev->num_chn);
if (ret) { if (ret) {
...@@ -908,8 +899,8 @@ static int netvsc_set_settings(struct net_device *dev, struct ethtool_cmd *cmd) ...@@ -908,8 +899,8 @@ static int netvsc_set_settings(struct net_device *dev, struct ethtool_cmd *cmd)
static int netvsc_change_mtu(struct net_device *ndev, int mtu) static int netvsc_change_mtu(struct net_device *ndev, int mtu)
{ {
struct net_device_context *ndevctx = netdev_priv(ndev); struct net_device_context *ndevctx = netdev_priv(ndev);
struct hv_device *hdev = ndevctx->device_ctx; struct netvsc_device *nvdev = ndevctx->nvdev;
struct netvsc_device *nvdev = hv_get_drvdata(hdev); struct hv_device *hdev = ndevctx->device_ctx;
struct netvsc_device_info device_info; struct netvsc_device_info device_info;
int limit = ETH_DATA_LEN; int limit = ETH_DATA_LEN;
u32 num_chn; u32 num_chn;
...@@ -935,9 +926,6 @@ static int netvsc_change_mtu(struct net_device *ndev, int mtu) ...@@ -935,9 +926,6 @@ static int netvsc_change_mtu(struct net_device *ndev, int mtu)
ndev->mtu = mtu; ndev->mtu = mtu;
ndevctx->device_ctx = hdev;
hv_set_drvdata(hdev, ndev);
memset(&device_info, 0, sizeof(device_info)); memset(&device_info, 0, sizeof(device_info));
device_info.ring_size = ring_size; device_info.ring_size = ring_size;
device_info.num_chn = num_chn; device_info.num_chn = num_chn;
...@@ -1076,7 +1064,7 @@ static void netvsc_link_change(struct work_struct *w) ...@@ -1076,7 +1064,7 @@ static void netvsc_link_change(struct work_struct *w)
if (ndev_ctx->start_remove) if (ndev_ctx->start_remove)
goto out_unlock; goto out_unlock;
net_device = hv_get_drvdata(ndev_ctx->device_ctx); net_device = ndev_ctx->nvdev;
rdev = net_device->extension; rdev = net_device->extension;
net = net_device->ndev; net = net_device->ndev;
...@@ -1201,7 +1189,7 @@ static struct netvsc_device *get_netvsc_device(char *mac) ...@@ -1201,7 +1189,7 @@ static struct netvsc_device *get_netvsc_device(char *mac)
if (netvsc_ctx == NULL) if (netvsc_ctx == NULL)
return NULL; return NULL;
return hv_get_drvdata(netvsc_ctx->device_ctx); return netvsc_ctx->nvdev;
} }
static int netvsc_register_vf(struct net_device *vf_netdev) static int netvsc_register_vf(struct net_device *vf_netdev)
...@@ -1407,7 +1395,7 @@ static int netvsc_probe(struct hv_device *dev, ...@@ -1407,7 +1395,7 @@ static int netvsc_probe(struct hv_device *dev,
} }
memcpy(net->dev_addr, device_info.mac_adr, ETH_ALEN); memcpy(net->dev_addr, device_info.mac_adr, ETH_ALEN);
nvdev = hv_get_drvdata(dev); nvdev = net_device_ctx->nvdev;
netif_set_real_num_tx_queues(net, nvdev->num_chn); netif_set_real_num_tx_queues(net, nvdev->num_chn);
netif_set_real_num_rx_queues(net, nvdev->num_chn); netif_set_real_num_rx_queues(net, nvdev->num_chn);
...@@ -1429,8 +1417,7 @@ static int netvsc_remove(struct hv_device *dev) ...@@ -1429,8 +1417,7 @@ static int netvsc_remove(struct hv_device *dev)
struct net_device_context *ndev_ctx; struct net_device_context *ndev_ctx;
struct netvsc_device *net_device; struct netvsc_device *net_device;
net_device = hv_get_drvdata(dev); net = hv_get_drvdata(dev);
net = net_device->ndev;
if (net == NULL) { if (net == NULL) {
dev_err(&dev->device, "No net device to remove\n"); dev_err(&dev->device, "No net device to remove\n");
...@@ -1439,6 +1426,8 @@ static int netvsc_remove(struct hv_device *dev) ...@@ -1439,6 +1426,8 @@ static int netvsc_remove(struct hv_device *dev)
ndev_ctx = netdev_priv(net); ndev_ctx = netdev_priv(net);
net_device = ndev_ctx->nvdev;
ndev_ctx->start_remove = true; ndev_ctx->start_remove = true;
cancel_delayed_work_sync(&ndev_ctx->dwork); cancel_delayed_work_sync(&ndev_ctx->dwork);
...@@ -1455,6 +1444,8 @@ static int netvsc_remove(struct hv_device *dev) ...@@ -1455,6 +1444,8 @@ static int netvsc_remove(struct hv_device *dev)
*/ */
rndis_filter_device_remove(dev); rndis_filter_device_remove(dev);
hv_set_drvdata(dev, NULL);
netvsc_free_netdev(net); netvsc_free_netdev(net);
return 0; return 0;
} }
......
...@@ -126,11 +126,7 @@ static void put_rndis_request(struct rndis_device *dev, ...@@ -126,11 +126,7 @@ static void put_rndis_request(struct rndis_device *dev,
static void dump_rndis_message(struct hv_device *hv_dev, static void dump_rndis_message(struct hv_device *hv_dev,
struct rndis_message *rndis_msg) struct rndis_message *rndis_msg)
{ {
struct net_device *netdev; struct net_device *netdev = hv_get_drvdata(hv_dev);
struct netvsc_device *net_device;
net_device = hv_get_drvdata(hv_dev);
netdev = net_device->ndev;
switch (rndis_msg->ndis_msg_type) { switch (rndis_msg->ndis_msg_type) {
case RNDIS_MSG_PACKET: case RNDIS_MSG_PACKET:
...@@ -211,6 +207,7 @@ static int rndis_filter_send_request(struct rndis_device *dev, ...@@ -211,6 +207,7 @@ static int rndis_filter_send_request(struct rndis_device *dev,
struct hv_netvsc_packet *packet; struct hv_netvsc_packet *packet;
struct hv_page_buffer page_buf[2]; struct hv_page_buffer page_buf[2];
struct hv_page_buffer *pb = page_buf; struct hv_page_buffer *pb = page_buf;
struct net_device_context *net_device_ctx = netdev_priv(dev->ndev);
/* Setup the packet to send it */ /* Setup the packet to send it */
packet = &req->pkt; packet = &req->pkt;
...@@ -236,7 +233,7 @@ static int rndis_filter_send_request(struct rndis_device *dev, ...@@ -236,7 +233,7 @@ static int rndis_filter_send_request(struct rndis_device *dev,
pb[0].len; pb[0].len;
} }
ret = netvsc_send(dev->net_dev->dev, packet, NULL, &pb, NULL); ret = netvsc_send(net_device_ctx->device_ctx, packet, NULL, &pb, NULL);
return ret; return ret;
} }
...@@ -262,9 +259,7 @@ static void rndis_filter_receive_response(struct rndis_device *dev, ...@@ -262,9 +259,7 @@ static void rndis_filter_receive_response(struct rndis_device *dev,
struct rndis_request *request = NULL; struct rndis_request *request = NULL;
bool found = false; bool found = false;
unsigned long flags; unsigned long flags;
struct net_device *ndev; struct net_device *ndev = dev->ndev;
ndev = dev->net_dev->ndev;
spin_lock_irqsave(&dev->request_lock, flags); spin_lock_irqsave(&dev->request_lock, flags);
list_for_each_entry(request, &dev->req_list, list_ent) { list_for_each_entry(request, &dev->req_list, list_ent) {
...@@ -355,6 +350,7 @@ static int rndis_filter_receive_data(struct rndis_device *dev, ...@@ -355,6 +350,7 @@ static int rndis_filter_receive_data(struct rndis_device *dev,
struct ndis_pkt_8021q_info *vlan; struct ndis_pkt_8021q_info *vlan;
struct ndis_tcp_ip_checksum_info *csum_info; struct ndis_tcp_ip_checksum_info *csum_info;
u16 vlan_tci = 0; u16 vlan_tci = 0;
struct net_device_context *net_device_ctx = netdev_priv(dev->ndev);
rndis_pkt = &msg->msg.pkt; rndis_pkt = &msg->msg.pkt;
...@@ -368,7 +364,7 @@ static int rndis_filter_receive_data(struct rndis_device *dev, ...@@ -368,7 +364,7 @@ static int rndis_filter_receive_data(struct rndis_device *dev,
* should be the data packet size plus the trailer padding size * should be the data packet size plus the trailer padding size
*/ */
if (pkt->total_data_buflen < rndis_pkt->data_len) { if (pkt->total_data_buflen < rndis_pkt->data_len) {
netdev_err(dev->net_dev->ndev, "rndis message buffer " netdev_err(dev->ndev, "rndis message buffer "
"overflow detected (got %u, min %u)" "overflow detected (got %u, min %u)"
"...dropping this message!\n", "...dropping this message!\n",
pkt->total_data_buflen, rndis_pkt->data_len); pkt->total_data_buflen, rndis_pkt->data_len);
...@@ -390,7 +386,7 @@ static int rndis_filter_receive_data(struct rndis_device *dev, ...@@ -390,7 +386,7 @@ static int rndis_filter_receive_data(struct rndis_device *dev,
} }
csum_info = rndis_get_ppi(rndis_pkt, TCPIP_CHKSUM_PKTINFO); csum_info = rndis_get_ppi(rndis_pkt, TCPIP_CHKSUM_PKTINFO);
return netvsc_recv_callback(dev->net_dev->dev, pkt, data, return netvsc_recv_callback(net_device_ctx->device_ctx, pkt, data,
csum_info, channel, vlan_tci); csum_info, channel, vlan_tci);
} }
...@@ -399,10 +395,11 @@ int rndis_filter_receive(struct hv_device *dev, ...@@ -399,10 +395,11 @@ int rndis_filter_receive(struct hv_device *dev,
void **data, void **data,
struct vmbus_channel *channel) struct vmbus_channel *channel)
{ {
struct netvsc_device *net_dev = hv_get_drvdata(dev); struct net_device *ndev = hv_get_drvdata(dev);
struct net_device_context *net_device_ctx = netdev_priv(ndev);
struct netvsc_device *net_dev = net_device_ctx->nvdev;
struct rndis_device *rndis_dev; struct rndis_device *rndis_dev;
struct rndis_message *rndis_msg; struct rndis_message *rndis_msg;
struct net_device *ndev;
int ret = 0; int ret = 0;
if (!net_dev) { if (!net_dev) {
...@@ -410,8 +407,6 @@ int rndis_filter_receive(struct hv_device *dev, ...@@ -410,8 +407,6 @@ int rndis_filter_receive(struct hv_device *dev,
goto exit; goto exit;
} }
ndev = net_dev->ndev;
/* Make sure the rndis device state is initialized */ /* Make sure the rndis device state is initialized */
if (!net_dev->extension) { if (!net_dev->extension) {
netdev_err(ndev, "got rndis message but no rndis device - " netdev_err(ndev, "got rndis message but no rndis device - "
...@@ -430,7 +425,7 @@ int rndis_filter_receive(struct hv_device *dev, ...@@ -430,7 +425,7 @@ int rndis_filter_receive(struct hv_device *dev,
rndis_msg = *data; rndis_msg = *data;
if (netif_msg_rx_err(net_dev->nd_ctx)) if (netif_msg_rx_err(net_device_ctx))
dump_rndis_message(dev, rndis_msg); dump_rndis_message(dev, rndis_msg);
switch (rndis_msg->ndis_msg_type) { switch (rndis_msg->ndis_msg_type) {
...@@ -550,9 +545,10 @@ static int rndis_filter_query_device_mac(struct rndis_device *dev) ...@@ -550,9 +545,10 @@ static int rndis_filter_query_device_mac(struct rndis_device *dev)
int rndis_filter_set_device_mac(struct hv_device *hdev, char *mac) int rndis_filter_set_device_mac(struct hv_device *hdev, char *mac)
{ {
struct netvsc_device *nvdev = hv_get_drvdata(hdev); struct net_device *ndev = hv_get_drvdata(hdev);
struct net_device_context *net_device_ctx = netdev_priv(ndev);
struct netvsc_device *nvdev = net_device_ctx->nvdev;
struct rndis_device *rdev = nvdev->extension; struct rndis_device *rdev = nvdev->extension;
struct net_device *ndev = nvdev->ndev;
struct rndis_request *request; struct rndis_request *request;
struct rndis_set_request *set; struct rndis_set_request *set;
struct rndis_config_parameter_info *cpi; struct rndis_config_parameter_info *cpi;
...@@ -629,9 +625,10 @@ static int ...@@ -629,9 +625,10 @@ static int
rndis_filter_set_offload_params(struct hv_device *hdev, rndis_filter_set_offload_params(struct hv_device *hdev,
struct ndis_offload_params *req_offloads) struct ndis_offload_params *req_offloads)
{ {
struct netvsc_device *nvdev = hv_get_drvdata(hdev); struct net_device *ndev = hv_get_drvdata(hdev);
struct net_device_context *net_device_ctx = netdev_priv(ndev);
struct netvsc_device *nvdev = net_device_ctx->nvdev;
struct rndis_device *rdev = nvdev->extension; struct rndis_device *rdev = nvdev->extension;
struct net_device *ndev = nvdev->ndev;
struct rndis_request *request; struct rndis_request *request;
struct rndis_set_request *set; struct rndis_set_request *set;
struct ndis_offload_params *offload_params; struct ndis_offload_params *offload_params;
...@@ -703,7 +700,7 @@ u8 netvsc_hash_key[HASH_KEYLEN] = { ...@@ -703,7 +700,7 @@ u8 netvsc_hash_key[HASH_KEYLEN] = {
static int rndis_filter_set_rss_param(struct rndis_device *rdev, int num_queue) static int rndis_filter_set_rss_param(struct rndis_device *rdev, int num_queue)
{ {
struct net_device *ndev = rdev->net_dev->ndev; struct net_device *ndev = rdev->ndev;
struct rndis_request *request; struct rndis_request *request;
struct rndis_set_request *set; struct rndis_set_request *set;
struct rndis_set_complete *set_complete; struct rndis_set_complete *set_complete;
...@@ -799,9 +796,7 @@ int rndis_filter_set_packet_filter(struct rndis_device *dev, u32 new_filter) ...@@ -799,9 +796,7 @@ int rndis_filter_set_packet_filter(struct rndis_device *dev, u32 new_filter)
u32 status; u32 status;
int ret; int ret;
unsigned long t; unsigned long t;
struct net_device *ndev; struct net_device *ndev = dev->ndev;
ndev = dev->net_dev->ndev;
request = get_rndis_request(dev, RNDIS_MSG_SET, request = get_rndis_request(dev, RNDIS_MSG_SET,
RNDIS_MESSAGE_SIZE(struct rndis_set_request) + RNDIS_MESSAGE_SIZE(struct rndis_set_request) +
...@@ -856,7 +851,8 @@ static int rndis_filter_init_device(struct rndis_device *dev) ...@@ -856,7 +851,8 @@ static int rndis_filter_init_device(struct rndis_device *dev)
u32 status; u32 status;
int ret; int ret;
unsigned long t; unsigned long t;
struct netvsc_device *nvdev = dev->net_dev; struct net_device_context *net_device_ctx = netdev_priv(dev->ndev);
struct netvsc_device *nvdev = net_device_ctx->nvdev;
request = get_rndis_request(dev, RNDIS_MSG_INIT, request = get_rndis_request(dev, RNDIS_MSG_INIT,
RNDIS_MESSAGE_SIZE(struct rndis_initialize_request)); RNDIS_MESSAGE_SIZE(struct rndis_initialize_request));
...@@ -879,7 +875,6 @@ static int rndis_filter_init_device(struct rndis_device *dev) ...@@ -879,7 +875,6 @@ static int rndis_filter_init_device(struct rndis_device *dev)
goto cleanup; goto cleanup;
} }
t = wait_for_completion_timeout(&request->wait_event, 5*HZ); t = wait_for_completion_timeout(&request->wait_event, 5*HZ);
if (t == 0) { if (t == 0) {
...@@ -910,8 +905,9 @@ static void rndis_filter_halt_device(struct rndis_device *dev) ...@@ -910,8 +905,9 @@ static void rndis_filter_halt_device(struct rndis_device *dev)
{ {
struct rndis_request *request; struct rndis_request *request;
struct rndis_halt_request *halt; struct rndis_halt_request *halt;
struct netvsc_device *nvdev = dev->net_dev; struct net_device_context *net_device_ctx = netdev_priv(dev->ndev);
struct hv_device *hdev = nvdev->dev; struct netvsc_device *nvdev = net_device_ctx->nvdev;
struct hv_device *hdev = net_device_ctx->device_ctx;
ulong flags; ulong flags;
/* Attempt to do a rndis device halt */ /* Attempt to do a rndis device halt */
...@@ -979,13 +975,14 @@ static int rndis_filter_close_device(struct rndis_device *dev) ...@@ -979,13 +975,14 @@ static int rndis_filter_close_device(struct rndis_device *dev)
static void netvsc_sc_open(struct vmbus_channel *new_sc) static void netvsc_sc_open(struct vmbus_channel *new_sc)
{ {
struct netvsc_device *nvscdev; struct net_device *ndev =
hv_get_drvdata(new_sc->primary_channel->device_obj);
struct net_device_context *net_device_ctx = netdev_priv(ndev);
struct netvsc_device *nvscdev = net_device_ctx->nvdev;
u16 chn_index = new_sc->offermsg.offer.sub_channel_index; u16 chn_index = new_sc->offermsg.offer.sub_channel_index;
int ret; int ret;
unsigned long flags; unsigned long flags;
nvscdev = hv_get_drvdata(new_sc->primary_channel->device_obj);
if (chn_index >= nvscdev->num_chn) if (chn_index >= nvscdev->num_chn)
return; return;
...@@ -1010,6 +1007,8 @@ int rndis_filter_device_add(struct hv_device *dev, ...@@ -1010,6 +1007,8 @@ int rndis_filter_device_add(struct hv_device *dev,
void *additional_info) void *additional_info)
{ {
int ret; int ret;
struct net_device *net = hv_get_drvdata(dev);
struct net_device_context *net_device_ctx = netdev_priv(net);
struct netvsc_device *net_device; struct netvsc_device *net_device;
struct rndis_device *rndis_device; struct rndis_device *rndis_device;
struct netvsc_device_info *device_info = additional_info; struct netvsc_device_info *device_info = additional_info;
...@@ -1040,16 +1039,15 @@ int rndis_filter_device_add(struct hv_device *dev, ...@@ -1040,16 +1039,15 @@ int rndis_filter_device_add(struct hv_device *dev,
return ret; return ret;
} }
/* Initialize the rndis device */ /* Initialize the rndis device */
net_device = hv_get_drvdata(dev); net_device = net_device_ctx->nvdev;
net_device->max_chn = 1; net_device->max_chn = 1;
net_device->num_chn = 1; net_device->num_chn = 1;
spin_lock_init(&net_device->sc_lock); spin_lock_init(&net_device->sc_lock);
net_device->extension = rndis_device; net_device->extension = rndis_device;
rndis_device->net_dev = net_device; rndis_device->ndev = net;
/* Send the rndis initialization message */ /* Send the rndis initialization message */
ret = rndis_filter_init_device(rndis_device); ret = rndis_filter_init_device(rndis_device);
...@@ -1198,7 +1196,9 @@ int rndis_filter_device_add(struct hv_device *dev, ...@@ -1198,7 +1196,9 @@ int rndis_filter_device_add(struct hv_device *dev,
void rndis_filter_device_remove(struct hv_device *dev) void rndis_filter_device_remove(struct hv_device *dev)
{ {
struct netvsc_device *net_dev = hv_get_drvdata(dev); struct net_device *ndev = hv_get_drvdata(dev);
struct net_device_context *net_device_ctx = netdev_priv(ndev);
struct netvsc_device *net_dev = net_device_ctx->nvdev;
struct rndis_device *rndis_dev = net_dev->extension; struct rndis_device *rndis_dev = net_dev->extension;
unsigned long t; unsigned long t;
...@@ -1224,7 +1224,9 @@ void rndis_filter_device_remove(struct hv_device *dev) ...@@ -1224,7 +1224,9 @@ void rndis_filter_device_remove(struct hv_device *dev)
int rndis_filter_open(struct hv_device *dev) int rndis_filter_open(struct hv_device *dev)
{ {
struct netvsc_device *net_device = hv_get_drvdata(dev); struct net_device *ndev = hv_get_drvdata(dev);
struct net_device_context *net_device_ctx = netdev_priv(ndev);
struct netvsc_device *net_device = net_device_ctx->nvdev;
if (!net_device) if (!net_device)
return -EINVAL; return -EINVAL;
...@@ -1237,7 +1239,9 @@ int rndis_filter_open(struct hv_device *dev) ...@@ -1237,7 +1239,9 @@ int rndis_filter_open(struct hv_device *dev)
int rndis_filter_close(struct hv_device *dev) int rndis_filter_close(struct hv_device *dev)
{ {
struct netvsc_device *nvdev = hv_get_drvdata(dev); struct net_device *ndev = hv_get_drvdata(dev);
struct net_device_context *net_device_ctx = netdev_priv(ndev);
struct netvsc_device *nvdev = net_device_ctx->nvdev;
if (!nvdev) if (!nvdev)
return -EINVAL; return -EINVAL;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment