diff options
| author | David Miller <davem@davemloft.net> | 2011-12-02 11:52:18 -0500 |
|---|---|---|
| committer | David S. Miller <davem@davemloft.net> | 2011-12-05 15:20:19 -0500 |
| commit | a4757123aeadf450b5b3c5f51f214660e20477f3 (patch) | |
| tree | 075d43113136926357be2863b2efb3b6a099e553 /drivers/net/ethernet/chelsio/cxgb3 | |
| parent | 51d45974515c35cd401f6194a6e728a2d1c3e3c6 (diff) | |
cxgb3: Rework t3_l2t_get to take a dst_entry instead of a neighbour.
This way we consolidate the RCU locking down into the place where it
actually matters, and also we can make the code handle
dst_get_neighbour_noref() returning NULL properly.
Signed-off-by: David S. Miller <davem@davemloft.net>
Diffstat (limited to 'drivers/net/ethernet/chelsio/cxgb3')
| -rw-r--r-- | drivers/net/ethernet/chelsio/cxgb3/cxgb3_offload.c | 2 | ||||
| -rw-r--r-- | drivers/net/ethernet/chelsio/cxgb3/l2t.c | 27 | ||||
| -rw-r--r-- | drivers/net/ethernet/chelsio/cxgb3/l2t.h | 2 |
3 files changed, 22 insertions, 9 deletions
diff --git a/drivers/net/ethernet/chelsio/cxgb3/cxgb3_offload.c b/drivers/net/ethernet/chelsio/cxgb3/cxgb3_offload.c index 6ed9f87db8ea..596cfe32d79e 100644 --- a/drivers/net/ethernet/chelsio/cxgb3/cxgb3_offload.c +++ b/drivers/net/ethernet/chelsio/cxgb3/cxgb3_offload.c | |||
| @@ -1132,7 +1132,7 @@ static void cxgb_redirect(struct dst_entry *old, struct dst_entry *new) | |||
| 1132 | } | 1132 | } |
| 1133 | 1133 | ||
| 1134 | /* Add new L2T entry */ | 1134 | /* Add new L2T entry */ |
| 1135 | e = t3_l2t_get(tdev, dst_get_neighbour_noref(new), newdev); | 1135 | e = t3_l2t_get(tdev, new, newdev); |
| 1136 | if (!e) { | 1136 | if (!e) { |
| 1137 | printk(KERN_ERR "%s: couldn't allocate new l2t entry!\n", | 1137 | printk(KERN_ERR "%s: couldn't allocate new l2t entry!\n", |
| 1138 | __func__); | 1138 | __func__); |
diff --git a/drivers/net/ethernet/chelsio/cxgb3/l2t.c b/drivers/net/ethernet/chelsio/cxgb3/l2t.c index 70fec8b1140f..3fa3c8833ed7 100644 --- a/drivers/net/ethernet/chelsio/cxgb3/l2t.c +++ b/drivers/net/ethernet/chelsio/cxgb3/l2t.c | |||
| @@ -298,18 +298,31 @@ static inline void reuse_entry(struct l2t_entry *e, struct neighbour *neigh) | |||
| 298 | spin_unlock(&e->lock); | 298 | spin_unlock(&e->lock); |
| 299 | } | 299 | } |
| 300 | 300 | ||
| 301 | struct l2t_entry *t3_l2t_get(struct t3cdev *cdev, struct neighbour *neigh, | 301 | struct l2t_entry *t3_l2t_get(struct t3cdev *cdev, struct dst_entry *dst, |
| 302 | struct net_device *dev) | 302 | struct net_device *dev) |
| 303 | { | 303 | { |
| 304 | struct l2t_entry *e = NULL; | 304 | struct l2t_entry *e = NULL; |
| 305 | struct neighbour *neigh; | ||
| 306 | struct port_info *p; | ||
| 305 | struct l2t_data *d; | 307 | struct l2t_data *d; |
| 306 | int hash; | 308 | int hash; |
| 307 | u32 addr = *(u32 *) neigh->primary_key; | 309 | u32 addr; |
| 308 | int ifidx = neigh->dev->ifindex; | 310 | int ifidx; |
| 309 | struct port_info *p = netdev_priv(dev); | 311 | int smt_idx; |
| 310 | int smt_idx = p->port_id; | ||
| 311 | 312 | ||
| 312 | rcu_read_lock(); | 313 | rcu_read_lock(); |
| 314 | neigh = dst_get_neighbour_noref(dst); | ||
| 315 | if (!neigh) | ||
| 316 | goto done_rcu; | ||
| 317 | |||
| 318 | addr = *(u32 *) neigh->primary_key; | ||
| 319 | ifidx = neigh->dev->ifindex; | ||
| 320 | |||
| 321 | if (!dev) | ||
| 322 | dev = neigh->dev; | ||
| 323 | p = netdev_priv(dev); | ||
| 324 | smt_idx = p->port_id; | ||
| 325 | |||
| 313 | d = L2DATA(cdev); | 326 | d = L2DATA(cdev); |
| 314 | if (!d) | 327 | if (!d) |
| 315 | goto done_rcu; | 328 | goto done_rcu; |
| @@ -323,7 +336,7 @@ struct l2t_entry *t3_l2t_get(struct t3cdev *cdev, struct neighbour *neigh, | |||
| 323 | l2t_hold(d, e); | 336 | l2t_hold(d, e); |
| 324 | if (atomic_read(&e->refcnt) == 1) | 337 | if (atomic_read(&e->refcnt) == 1) |
| 325 | reuse_entry(e, neigh); | 338 | reuse_entry(e, neigh); |
| 326 | goto done; | 339 | goto done_unlock; |
| 327 | } | 340 | } |
| 328 | 341 | ||
| 329 | /* Need to allocate a new entry */ | 342 | /* Need to allocate a new entry */ |
| @@ -344,7 +357,7 @@ struct l2t_entry *t3_l2t_get(struct t3cdev *cdev, struct neighbour *neigh, | |||
| 344 | e->vlan = VLAN_NONE; | 357 | e->vlan = VLAN_NONE; |
| 345 | spin_unlock(&e->lock); | 358 | spin_unlock(&e->lock); |
| 346 | } | 359 | } |
| 347 | done: | 360 | done_unlock: |
| 348 | write_unlock_bh(&d->lock); | 361 | write_unlock_bh(&d->lock); |
| 349 | done_rcu: | 362 | done_rcu: |
| 350 | rcu_read_unlock(); | 363 | rcu_read_unlock(); |
diff --git a/drivers/net/ethernet/chelsio/cxgb3/l2t.h b/drivers/net/ethernet/chelsio/cxgb3/l2t.h index c5f54796e2cb..c4e864369751 100644 --- a/drivers/net/ethernet/chelsio/cxgb3/l2t.h +++ b/drivers/net/ethernet/chelsio/cxgb3/l2t.h | |||
| @@ -109,7 +109,7 @@ static inline void set_arp_failure_handler(struct sk_buff *skb, | |||
| 109 | 109 | ||
| 110 | void t3_l2e_free(struct l2t_data *d, struct l2t_entry *e); | 110 | void t3_l2e_free(struct l2t_data *d, struct l2t_entry *e); |
| 111 | void t3_l2t_update(struct t3cdev *dev, struct neighbour *neigh); | 111 | void t3_l2t_update(struct t3cdev *dev, struct neighbour *neigh); |
| 112 | struct l2t_entry *t3_l2t_get(struct t3cdev *cdev, struct neighbour *neigh, | 112 | struct l2t_entry *t3_l2t_get(struct t3cdev *cdev, struct dst_entry *dst, |
| 113 | struct net_device *dev); | 113 | struct net_device *dev); |
| 114 | int t3_l2t_send_slow(struct t3cdev *dev, struct sk_buff *skb, | 114 | int t3_l2t_send_slow(struct t3cdev *dev, struct sk_buff *skb, |
| 115 | struct l2t_entry *e); | 115 | struct l2t_entry *e); |
