Commit c8d17b45 authored by Linus Torvalds's avatar Linus Torvalds

Merge git://git.kernel.org/pub/scm/linux/kernel/git/davem/net

Pull networking fixes from David Miller:

 1) Fix uninitialized struct station_info in cfg80211_wireless_stats(),
    from Johannes Berg.

 2) Revert commit attempt to fix ipv6 protocol resubmission, it adds
    regressions.

 3) Endless loops can be created in bridge port lists, fix from Nikolay
    Aleksandrov.

 4) Don't WARN_ON() if sk->sk_forward_alloc is non-zero in
    sk_clear_memalloc, it is a legal situation during swap deactivation.
    Fix from Mel Gorman.

 5) Fix order of disabling interrupts and unlocking NAPI in enic driver
    to avoid a race.  From Govindarajulu Varadarajan.

 6) High and low register writes are swapped when programming the start
    of periodic output in igb driver.  From Richard Cochran.

 7) Fix device rename handling in mpls stack, from Robert Shearman.

 8) Do not trigger compaction synchronously when optimistically trying
    to allocate an order 3 page in alloc_skb_with_frags() and
    skb_page_frag_refill().  From Shaohua Li.

 9) Authentication with COOKIE_ECHO is not handled properly in SCTP, fix
    from Marcelo Ricardo Leitner.

* git://git.kernel.org/pub/scm/linux/kernel/git/davem/net:
  Doc: networking: Fix URL for wiki.wireshark.org in udplite.txt
  sctp: allow authenticating DATA chunks that are bundled with COOKIE_ECHO
  net: don't wait for order-3 page allocation
  mpls: handle device renames for per-device sysctls
  net: igb: fix the start time for periodic output signals
  enic: fix memory leak in rq_clean
  enic: check return value for stat dump
  enic: unlock napi busy poll before unmasking intr
  net, swap: Remove a warning and clarify why sk_mem_reclaim is required when deactivating swap
  bridge: fix multicast router rlist endless loop
  tipc: disconnect socket directly after probe failure
  Revert "ipv6: Fix protocol resubmission"
  cfg80211: wext: clear sinfo struct before calling driver
parents b85dfd30 b07d4961
...@@ -20,7 +20,7 @@ ...@@ -20,7 +20,7 @@
files/UDP-Lite-HOWTO.txt files/UDP-Lite-HOWTO.txt
o The Wireshark UDP-Lite WiKi (with capture files): o The Wireshark UDP-Lite WiKi (with capture files):
http://wiki.wireshark.org/Lightweight_User_Datagram_Protocol https://wiki.wireshark.org/Lightweight_User_Datagram_Protocol
o The Protocol Spec, RFC 3828, http://www.ietf.org/rfc/rfc3828.txt o The Protocol Spec, RFC 3828, http://www.ietf.org/rfc/rfc3828.txt
......
...@@ -131,8 +131,15 @@ static void enic_get_drvinfo(struct net_device *netdev, ...@@ -131,8 +131,15 @@ static void enic_get_drvinfo(struct net_device *netdev,
{ {
struct enic *enic = netdev_priv(netdev); struct enic *enic = netdev_priv(netdev);
struct vnic_devcmd_fw_info *fw_info; struct vnic_devcmd_fw_info *fw_info;
int err;
enic_dev_fw_info(enic, &fw_info); err = enic_dev_fw_info(enic, &fw_info);
/* return only when pci_zalloc_consistent fails in vnic_dev_fw_info
* For other failures, like devcmd failure, we return previously
* recorded info.
*/
if (err == -ENOMEM)
return;
strlcpy(drvinfo->driver, DRV_NAME, sizeof(drvinfo->driver)); strlcpy(drvinfo->driver, DRV_NAME, sizeof(drvinfo->driver));
strlcpy(drvinfo->version, DRV_VERSION, sizeof(drvinfo->version)); strlcpy(drvinfo->version, DRV_VERSION, sizeof(drvinfo->version));
...@@ -181,8 +188,15 @@ static void enic_get_ethtool_stats(struct net_device *netdev, ...@@ -181,8 +188,15 @@ static void enic_get_ethtool_stats(struct net_device *netdev,
struct enic *enic = netdev_priv(netdev); struct enic *enic = netdev_priv(netdev);
struct vnic_stats *vstats; struct vnic_stats *vstats;
unsigned int i; unsigned int i;
int err;
enic_dev_stats_dump(enic, &vstats); err = enic_dev_stats_dump(enic, &vstats);
/* return only when pci_zalloc_consistent fails in vnic_dev_stats_dump
* For other failures, like devcmd failure, we return previously
* recorded stats.
*/
if (err == -ENOMEM)
return;
for (i = 0; i < enic_n_tx_stats; i++) for (i = 0; i < enic_n_tx_stats; i++)
*(data++) = ((u64 *)&vstats->tx)[enic_tx_stats[i].index]; *(data++) = ((u64 *)&vstats->tx)[enic_tx_stats[i].index];
......
...@@ -615,8 +615,15 @@ static struct rtnl_link_stats64 *enic_get_stats(struct net_device *netdev, ...@@ -615,8 +615,15 @@ static struct rtnl_link_stats64 *enic_get_stats(struct net_device *netdev,
{ {
struct enic *enic = netdev_priv(netdev); struct enic *enic = netdev_priv(netdev);
struct vnic_stats *stats; struct vnic_stats *stats;
int err;
enic_dev_stats_dump(enic, &stats); err = enic_dev_stats_dump(enic, &stats);
/* return only when pci_zalloc_consistent fails in vnic_dev_stats_dump
* For other failures, like devcmd failure, we return previously
* recorded stats.
*/
if (err == -ENOMEM)
return net_stats;
net_stats->tx_packets = stats->tx.tx_frames_ok; net_stats->tx_packets = stats->tx.tx_frames_ok;
net_stats->tx_bytes = stats->tx.tx_bytes_ok; net_stats->tx_bytes = stats->tx.tx_bytes_ok;
...@@ -1407,6 +1414,7 @@ static int enic_poll_msix_rq(struct napi_struct *napi, int budget) ...@@ -1407,6 +1414,7 @@ static int enic_poll_msix_rq(struct napi_struct *napi, int budget)
*/ */
enic_calc_int_moderation(enic, &enic->rq[rq]); enic_calc_int_moderation(enic, &enic->rq[rq]);
enic_poll_unlock_napi(&enic->rq[rq]);
if (work_done < work_to_do) { if (work_done < work_to_do) {
/* Some work done, but not enough to stay in polling, /* Some work done, but not enough to stay in polling,
...@@ -1418,7 +1426,6 @@ static int enic_poll_msix_rq(struct napi_struct *napi, int budget) ...@@ -1418,7 +1426,6 @@ static int enic_poll_msix_rq(struct napi_struct *napi, int budget)
enic_set_int_moderation(enic, &enic->rq[rq]); enic_set_int_moderation(enic, &enic->rq[rq]);
vnic_intr_unmask(&enic->intr[intr]); vnic_intr_unmask(&enic->intr[intr]);
} }
enic_poll_unlock_napi(&enic->rq[rq]);
return work_done; return work_done;
} }
......
...@@ -188,16 +188,15 @@ void vnic_rq_clean(struct vnic_rq *rq, ...@@ -188,16 +188,15 @@ void vnic_rq_clean(struct vnic_rq *rq,
struct vnic_rq_buf *buf; struct vnic_rq_buf *buf;
u32 fetch_index; u32 fetch_index;
unsigned int count = rq->ring.desc_count; unsigned int count = rq->ring.desc_count;
int i;
buf = rq->to_clean; buf = rq->to_clean;
while (vnic_rq_desc_used(rq) > 0) { for (i = 0; i < rq->ring.desc_count; i++) {
(*buf_clean)(rq, buf); (*buf_clean)(rq, buf);
buf = buf->next;
buf = rq->to_clean = buf->next;
rq->ring.desc_avail++;
} }
rq->ring.desc_avail = rq->ring.desc_count - 1;
/* Use current fetch_index as the ring starting point */ /* Use current fetch_index as the ring starting point */
fetch_index = ioread32(&rq->ctrl->fetch_index); fetch_index = ioread32(&rq->ctrl->fetch_index);
......
...@@ -538,8 +538,8 @@ static int igb_ptp_feature_enable_i210(struct ptp_clock_info *ptp, ...@@ -538,8 +538,8 @@ static int igb_ptp_feature_enable_i210(struct ptp_clock_info *ptp,
igb->perout[i].start.tv_nsec = rq->perout.start.nsec; igb->perout[i].start.tv_nsec = rq->perout.start.nsec;
igb->perout[i].period.tv_sec = ts.tv_sec; igb->perout[i].period.tv_sec = ts.tv_sec;
igb->perout[i].period.tv_nsec = ts.tv_nsec; igb->perout[i].period.tv_nsec = ts.tv_nsec;
wr32(trgttiml, rq->perout.start.sec); wr32(trgttimh, rq->perout.start.sec);
wr32(trgttimh, rq->perout.start.nsec); wr32(trgttiml, rq->perout.start.nsec);
tsauxc |= tsauxc_mask; tsauxc |= tsauxc_mask;
tsim |= tsim_mask; tsim |= tsim_mask;
} else { } else {
......
...@@ -1167,6 +1167,9 @@ static void br_multicast_add_router(struct net_bridge *br, ...@@ -1167,6 +1167,9 @@ static void br_multicast_add_router(struct net_bridge *br,
struct net_bridge_port *p; struct net_bridge_port *p;
struct hlist_node *slot = NULL; struct hlist_node *slot = NULL;
if (!hlist_unhashed(&port->rlist))
return;
hlist_for_each_entry(p, &br->router_list, rlist) { hlist_for_each_entry(p, &br->router_list, rlist) {
if ((unsigned long) port >= (unsigned long) p) if ((unsigned long) port >= (unsigned long) p)
break; break;
...@@ -1194,12 +1197,8 @@ static void br_multicast_mark_router(struct net_bridge *br, ...@@ -1194,12 +1197,8 @@ static void br_multicast_mark_router(struct net_bridge *br,
if (port->multicast_router != 1) if (port->multicast_router != 1)
return; return;
if (!hlist_unhashed(&port->rlist))
goto timer;
br_multicast_add_router(br, port); br_multicast_add_router(br, port);
timer:
mod_timer(&port->multicast_router_timer, mod_timer(&port->multicast_router_timer,
now + br->multicast_querier_interval); now + br->multicast_querier_interval);
} }
......
...@@ -4398,7 +4398,7 @@ struct sk_buff *alloc_skb_with_frags(unsigned long header_len, ...@@ -4398,7 +4398,7 @@ struct sk_buff *alloc_skb_with_frags(unsigned long header_len,
while (order) { while (order) {
if (npages >= 1 << order) { if (npages >= 1 << order) {
page = alloc_pages(gfp_mask | page = alloc_pages((gfp_mask & ~__GFP_WAIT) |
__GFP_COMP | __GFP_COMP |
__GFP_NOWARN | __GFP_NOWARN |
__GFP_NORETRY, __GFP_NORETRY,
......
...@@ -354,14 +354,11 @@ void sk_clear_memalloc(struct sock *sk) ...@@ -354,14 +354,11 @@ void sk_clear_memalloc(struct sock *sk)
/* /*
* SOCK_MEMALLOC is allowed to ignore rmem limits to ensure forward * SOCK_MEMALLOC is allowed to ignore rmem limits to ensure forward
* progress of swapping. However, if SOCK_MEMALLOC is cleared while * progress of swapping. SOCK_MEMALLOC may be cleared while
* it has rmem allocations there is a risk that the user of the * it has rmem allocations due to the last swapfile being deactivated
* socket cannot make forward progress due to exceeding the rmem * but there is a risk that the socket is unusable due to exceeding
* limits. By rights, sk_clear_memalloc() should only be called * the rmem limits. Reclaim the reserves and obey rmem limits again.
* on sockets being torn down but warn and reset the accounting if */
* that assumption breaks.
*/
if (WARN_ON(sk->sk_forward_alloc))
sk_mem_reclaim(sk); sk_mem_reclaim(sk);
} }
EXPORT_SYMBOL_GPL(sk_clear_memalloc); EXPORT_SYMBOL_GPL(sk_clear_memalloc);
...@@ -1883,7 +1880,7 @@ bool skb_page_frag_refill(unsigned int sz, struct page_frag *pfrag, gfp_t gfp) ...@@ -1883,7 +1880,7 @@ bool skb_page_frag_refill(unsigned int sz, struct page_frag *pfrag, gfp_t gfp)
pfrag->offset = 0; pfrag->offset = 0;
if (SKB_FRAG_PAGE_ORDER) { if (SKB_FRAG_PAGE_ORDER) {
pfrag->page = alloc_pages(gfp | __GFP_COMP | pfrag->page = alloc_pages((gfp & ~__GFP_WAIT) | __GFP_COMP |
__GFP_NOWARN | __GFP_NORETRY, __GFP_NOWARN | __GFP_NORETRY,
SKB_FRAG_PAGE_ORDER); SKB_FRAG_PAGE_ORDER);
if (likely(pfrag->page)) { if (likely(pfrag->page)) {
......
...@@ -212,13 +212,13 @@ static int ip6_input_finish(struct sock *sk, struct sk_buff *skb) ...@@ -212,13 +212,13 @@ static int ip6_input_finish(struct sock *sk, struct sk_buff *skb)
*/ */
rcu_read_lock(); rcu_read_lock();
resubmit:
idev = ip6_dst_idev(skb_dst(skb)); idev = ip6_dst_idev(skb_dst(skb));
if (!pskb_pull(skb, skb_transport_offset(skb))) if (!pskb_pull(skb, skb_transport_offset(skb)))
goto discard; goto discard;
nhoff = IP6CB(skb)->nhoff; nhoff = IP6CB(skb)->nhoff;
nexthdr = skb_network_header(skb)[nhoff]; nexthdr = skb_network_header(skb)[nhoff];
resubmit:
raw = raw6_local_deliver(skb, nexthdr); raw = raw6_local_deliver(skb, nexthdr);
ipprot = rcu_dereference(inet6_protos[nexthdr]); ipprot = rcu_dereference(inet6_protos[nexthdr]);
if (ipprot) { if (ipprot) {
...@@ -246,12 +246,10 @@ static int ip6_input_finish(struct sock *sk, struct sk_buff *skb) ...@@ -246,12 +246,10 @@ static int ip6_input_finish(struct sock *sk, struct sk_buff *skb)
goto discard; goto discard;
ret = ipprot->handler(skb); ret = ipprot->handler(skb);
if (ret < 0) { if (ret > 0)
nexthdr = -ret;
goto resubmit; goto resubmit;
} else if (ret == 0) { else if (ret == 0)
IP6_INC_STATS_BH(net, idev, IPSTATS_MIB_INDELIVERS); IP6_INC_STATS_BH(net, idev, IPSTATS_MIB_INDELIVERS);
}
} else { } else {
if (!raw) { if (!raw) {
if (xfrm6_policy_check(NULL, XFRM_POLICY_IN, skb)) { if (xfrm6_policy_check(NULL, XFRM_POLICY_IN, skb)) {
......
...@@ -564,6 +564,17 @@ static int mpls_dev_notify(struct notifier_block *this, unsigned long event, ...@@ -564,6 +564,17 @@ static int mpls_dev_notify(struct notifier_block *this, unsigned long event,
case NETDEV_UNREGISTER: case NETDEV_UNREGISTER:
mpls_ifdown(dev); mpls_ifdown(dev);
break; break;
case NETDEV_CHANGENAME:
mdev = mpls_dev_get(dev);
if (mdev) {
int err;
mpls_dev_sysctl_unregister(mdev);
err = mpls_dev_sysctl_register(dev, mdev);
if (err)
return notifier_from_errno(err);
}
break;
} }
return NOTIFY_OK; return NOTIFY_OK;
} }
......
...@@ -381,13 +381,14 @@ int sctp_auth_asoc_copy_shkeys(const struct sctp_endpoint *ep, ...@@ -381,13 +381,14 @@ int sctp_auth_asoc_copy_shkeys(const struct sctp_endpoint *ep,
} }
/* Public interface to creat the association shared key. /* Public interface to create the association shared key.
* See code above for the algorithm. * See code above for the algorithm.
*/ */
int sctp_auth_asoc_init_active_key(struct sctp_association *asoc, gfp_t gfp) int sctp_auth_asoc_init_active_key(struct sctp_association *asoc, gfp_t gfp)
{ {
struct sctp_auth_bytes *secret; struct sctp_auth_bytes *secret;
struct sctp_shared_key *ep_key; struct sctp_shared_key *ep_key;
struct sctp_chunk *chunk;
/* If we don't support AUTH, or peer is not capable /* If we don't support AUTH, or peer is not capable
* we don't need to do anything. * we don't need to do anything.
...@@ -410,6 +411,14 @@ int sctp_auth_asoc_init_active_key(struct sctp_association *asoc, gfp_t gfp) ...@@ -410,6 +411,14 @@ int sctp_auth_asoc_init_active_key(struct sctp_association *asoc, gfp_t gfp)
sctp_auth_key_put(asoc->asoc_shared_key); sctp_auth_key_put(asoc->asoc_shared_key);
asoc->asoc_shared_key = secret; asoc->asoc_shared_key = secret;
/* Update send queue in case any chunk already in there now
* needs authenticating
*/
list_for_each_entry(chunk, &asoc->outqueue.out_chunk_list, list) {
if (sctp_auth_send_cid(chunk->chunk_hdr->type, asoc))
chunk->auth = 1;
}
return 0; return 0;
} }
......
...@@ -2142,11 +2142,17 @@ static void tipc_sk_timeout(unsigned long data) ...@@ -2142,11 +2142,17 @@ static void tipc_sk_timeout(unsigned long data)
peer_node = tsk_peer_node(tsk); peer_node = tsk_peer_node(tsk);
if (tsk->probing_state == TIPC_CONN_PROBING) { if (tsk->probing_state == TIPC_CONN_PROBING) {
/* Previous probe not answered -> self abort */ if (!sock_owned_by_user(sk)) {
skb = tipc_msg_create(TIPC_CRITICAL_IMPORTANCE, sk->sk_socket->state = SS_DISCONNECTING;
TIPC_CONN_MSG, SHORT_H_SIZE, 0, tsk->connected = 0;
own_node, peer_node, tsk->portid, tipc_node_remove_conn(sock_net(sk), tsk_peer_node(tsk),
peer_port, TIPC_ERR_NO_PORT); tsk_peer_port(tsk));
sk->sk_state_change(sk);
} else {
/* Try again later */
sk_reset_timer(sk, &sk->sk_timer, (HZ / 20));
}
} else { } else {
skb = tipc_msg_create(CONN_MANAGER, CONN_PROBE, skb = tipc_msg_create(CONN_MANAGER, CONN_PROBE,
INT_H_SIZE, 0, peer_node, own_node, INT_H_SIZE, 0, peer_node, own_node,
......
...@@ -1333,6 +1333,8 @@ static struct iw_statistics *cfg80211_wireless_stats(struct net_device *dev) ...@@ -1333,6 +1333,8 @@ static struct iw_statistics *cfg80211_wireless_stats(struct net_device *dev)
memcpy(bssid, wdev->current_bss->pub.bssid, ETH_ALEN); memcpy(bssid, wdev->current_bss->pub.bssid, ETH_ALEN);
wdev_unlock(wdev); wdev_unlock(wdev);
memset(&sinfo, 0, sizeof(sinfo));
if (rdev_get_station(rdev, dev, bssid, &sinfo)) if (rdev_get_station(rdev, dev, bssid, &sinfo))
return NULL; return NULL;
......
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