diff options
Diffstat (limited to 'net')
| -rw-r--r-- | net/core/dev.c | 7 | ||||
| -rw-r--r-- | net/core/flow.c | 2 | ||||
| -rw-r--r-- | net/core/rtnetlink.c | 4 | ||||
| -rw-r--r-- | net/ipv6/addrconf.c | 27 | ||||
| -rw-r--r-- | net/ipv6/ip6_input.c | 12 | ||||
| -rw-r--r-- | net/key/af_key.c | 1 | ||||
| -rw-r--r-- | net/mac80211/iface.c | 35 | ||||
| -rw-r--r-- | net/mac80211/mesh.c | 3 | ||||
| -rw-r--r-- | net/mac80211/mlme.c | 6 | ||||
| -rw-r--r-- | net/mac80211/rx.c | 14 | ||||
| -rw-r--r-- | net/mac80211/sta_info.c | 12 | ||||
| -rw-r--r-- | net/sched/sch_cbq.c | 5 | ||||
| -rw-r--r-- | net/sched/sch_fq_codel.c | 2 | ||||
| -rw-r--r-- | net/sched/sch_generic.c | 2 | ||||
| -rw-r--r-- | net/vmw_vsock/af_vsock.c | 6 | ||||
| -rw-r--r-- | net/vmw_vsock/vmci_transport.c | 31 | ||||
| -rw-r--r-- | net/vmw_vsock/vsock_addr.c | 10 | ||||
| -rw-r--r-- | net/vmw_vsock/vsock_addr.h | 2 | ||||
| -rw-r--r-- | net/wireless/core.c | 64 | ||||
| -rw-r--r-- | net/wireless/core.h | 3 | ||||
| -rw-r--r-- | net/wireless/nl80211.c | 52 | ||||
| -rw-r--r-- | net/wireless/scan.c | 24 | ||||
| -rw-r--r-- | net/wireless/sme.c | 6 | ||||
| -rw-r--r-- | net/wireless/trace.h | 5 | ||||
| -rw-r--r-- | net/wireless/wext-sme.c | 6 | ||||
| -rw-r--r-- | net/xfrm/xfrm_replay.c | 66 |
26 files changed, 298 insertions, 109 deletions
diff --git a/net/core/dev.c b/net/core/dev.c index b13e5c766c11..13e6447f0398 100644 --- a/net/core/dev.c +++ b/net/core/dev.c | |||
| @@ -1624,7 +1624,6 @@ int dev_forward_skb(struct net_device *dev, struct sk_buff *skb) | |||
| 1624 | } | 1624 | } |
| 1625 | 1625 | ||
| 1626 | skb_orphan(skb); | 1626 | skb_orphan(skb); |
| 1627 | nf_reset(skb); | ||
| 1628 | 1627 | ||
| 1629 | if (unlikely(!is_skb_forwardable(dev, skb))) { | 1628 | if (unlikely(!is_skb_forwardable(dev, skb))) { |
| 1630 | atomic_long_inc(&dev->rx_dropped); | 1629 | atomic_long_inc(&dev->rx_dropped); |
| @@ -3314,6 +3313,7 @@ int netdev_rx_handler_register(struct net_device *dev, | |||
| 3314 | if (dev->rx_handler) | 3313 | if (dev->rx_handler) |
| 3315 | return -EBUSY; | 3314 | return -EBUSY; |
| 3316 | 3315 | ||
| 3316 | /* Note: rx_handler_data must be set before rx_handler */ | ||
| 3317 | rcu_assign_pointer(dev->rx_handler_data, rx_handler_data); | 3317 | rcu_assign_pointer(dev->rx_handler_data, rx_handler_data); |
| 3318 | rcu_assign_pointer(dev->rx_handler, rx_handler); | 3318 | rcu_assign_pointer(dev->rx_handler, rx_handler); |
| 3319 | 3319 | ||
| @@ -3334,6 +3334,11 @@ void netdev_rx_handler_unregister(struct net_device *dev) | |||
| 3334 | 3334 | ||
| 3335 | ASSERT_RTNL(); | 3335 | ASSERT_RTNL(); |
| 3336 | RCU_INIT_POINTER(dev->rx_handler, NULL); | 3336 | RCU_INIT_POINTER(dev->rx_handler, NULL); |
| 3337 | /* a reader seeing a non NULL rx_handler in a rcu_read_lock() | ||
| 3338 | * section has a guarantee to see a non NULL rx_handler_data | ||
| 3339 | * as well. | ||
| 3340 | */ | ||
| 3341 | synchronize_net(); | ||
| 3337 | RCU_INIT_POINTER(dev->rx_handler_data, NULL); | 3342 | RCU_INIT_POINTER(dev->rx_handler_data, NULL); |
| 3338 | } | 3343 | } |
| 3339 | EXPORT_SYMBOL_GPL(netdev_rx_handler_unregister); | 3344 | EXPORT_SYMBOL_GPL(netdev_rx_handler_unregister); |
diff --git a/net/core/flow.c b/net/core/flow.c index c56ea6f7f6c7..2bfd081c59f7 100644 --- a/net/core/flow.c +++ b/net/core/flow.c | |||
| @@ -328,7 +328,7 @@ static void flow_cache_flush_per_cpu(void *data) | |||
| 328 | struct flow_flush_info *info = data; | 328 | struct flow_flush_info *info = data; |
| 329 | struct tasklet_struct *tasklet; | 329 | struct tasklet_struct *tasklet; |
| 330 | 330 | ||
| 331 | tasklet = this_cpu_ptr(&info->cache->percpu->flush_tasklet); | 331 | tasklet = &this_cpu_ptr(info->cache->percpu)->flush_tasklet; |
| 332 | tasklet->data = (unsigned long)info; | 332 | tasklet->data = (unsigned long)info; |
| 333 | tasklet_schedule(tasklet); | 333 | tasklet_schedule(tasklet); |
| 334 | } | 334 | } |
diff --git a/net/core/rtnetlink.c b/net/core/rtnetlink.c index 5fb8d7e47294..b65441da74ab 100644 --- a/net/core/rtnetlink.c +++ b/net/core/rtnetlink.c | |||
| @@ -496,8 +496,10 @@ static int rtnl_link_fill(struct sk_buff *skb, const struct net_device *dev) | |||
| 496 | } | 496 | } |
| 497 | if (ops->fill_info) { | 497 | if (ops->fill_info) { |
| 498 | data = nla_nest_start(skb, IFLA_INFO_DATA); | 498 | data = nla_nest_start(skb, IFLA_INFO_DATA); |
| 499 | if (data == NULL) | 499 | if (data == NULL) { |
| 500 | err = -EMSGSIZE; | ||
| 500 | goto err_cancel_link; | 501 | goto err_cancel_link; |
| 502 | } | ||
| 501 | err = ops->fill_info(skb, dev); | 503 | err = ops->fill_info(skb, dev); |
| 502 | if (err < 0) | 504 | if (err < 0) |
| 503 | goto err_cancel_data; | 505 | goto err_cancel_data; |
diff --git a/net/ipv6/addrconf.c b/net/ipv6/addrconf.c index 26512250e095..a459c4f5b769 100644 --- a/net/ipv6/addrconf.c +++ b/net/ipv6/addrconf.c | |||
| @@ -2529,6 +2529,9 @@ static void sit_add_v4_addrs(struct inet6_dev *idev) | |||
| 2529 | static void init_loopback(struct net_device *dev) | 2529 | static void init_loopback(struct net_device *dev) |
| 2530 | { | 2530 | { |
| 2531 | struct inet6_dev *idev; | 2531 | struct inet6_dev *idev; |
| 2532 | struct net_device *sp_dev; | ||
| 2533 | struct inet6_ifaddr *sp_ifa; | ||
| 2534 | struct rt6_info *sp_rt; | ||
| 2532 | 2535 | ||
| 2533 | /* ::1 */ | 2536 | /* ::1 */ |
| 2534 | 2537 | ||
| @@ -2540,6 +2543,30 @@ static void init_loopback(struct net_device *dev) | |||
| 2540 | } | 2543 | } |
| 2541 | 2544 | ||
| 2542 | add_addr(idev, &in6addr_loopback, 128, IFA_HOST); | 2545 | add_addr(idev, &in6addr_loopback, 128, IFA_HOST); |
| 2546 | |||
| 2547 | /* Add routes to other interface's IPv6 addresses */ | ||
| 2548 | for_each_netdev(dev_net(dev), sp_dev) { | ||
| 2549 | if (!strcmp(sp_dev->name, dev->name)) | ||
| 2550 | continue; | ||
| 2551 | |||
| 2552 | idev = __in6_dev_get(sp_dev); | ||
| 2553 | if (!idev) | ||
| 2554 | continue; | ||
| 2555 | |||
| 2556 | read_lock_bh(&idev->lock); | ||
| 2557 | list_for_each_entry(sp_ifa, &idev->addr_list, if_list) { | ||
| 2558 | |||
| 2559 | if (sp_ifa->flags & (IFA_F_DADFAILED | IFA_F_TENTATIVE)) | ||
| 2560 | continue; | ||
| 2561 | |||
| 2562 | sp_rt = addrconf_dst_alloc(idev, &sp_ifa->addr, 0); | ||
| 2563 | |||
| 2564 | /* Failure cases are ignored */ | ||
| 2565 | if (!IS_ERR(sp_rt)) | ||
| 2566 | ip6_ins_rt(sp_rt); | ||
| 2567 | } | ||
| 2568 | read_unlock_bh(&idev->lock); | ||
| 2569 | } | ||
| 2543 | } | 2570 | } |
| 2544 | 2571 | ||
| 2545 | static void addrconf_add_linklocal(struct inet6_dev *idev, const struct in6_addr *addr) | 2572 | static void addrconf_add_linklocal(struct inet6_dev *idev, const struct in6_addr *addr) |
diff --git a/net/ipv6/ip6_input.c b/net/ipv6/ip6_input.c index e33fe0ab2568..2bab2aa59745 100644 --- a/net/ipv6/ip6_input.c +++ b/net/ipv6/ip6_input.c | |||
| @@ -118,6 +118,18 @@ int ipv6_rcv(struct sk_buff *skb, struct net_device *dev, struct packet_type *pt | |||
| 118 | ipv6_addr_loopback(&hdr->daddr)) | 118 | ipv6_addr_loopback(&hdr->daddr)) |
| 119 | goto err; | 119 | goto err; |
| 120 | 120 | ||
| 121 | /* RFC4291 Errata ID: 3480 | ||
| 122 | * Interface-Local scope spans only a single interface on a | ||
| 123 | * node and is useful only for loopback transmission of | ||
| 124 | * multicast. Packets with interface-local scope received | ||
| 125 | * from another node must be discarded. | ||
| 126 | */ | ||
| 127 | if (!(skb->pkt_type == PACKET_LOOPBACK || | ||
| 128 | dev->flags & IFF_LOOPBACK) && | ||
| 129 | ipv6_addr_is_multicast(&hdr->daddr) && | ||
| 130 | IPV6_ADDR_MC_SCOPE(&hdr->daddr) == 1) | ||
| 131 | goto err; | ||
| 132 | |||
| 121 | /* RFC4291 2.7 | 133 | /* RFC4291 2.7 |
| 122 | * Nodes must not originate a packet to a multicast address whose scope | 134 | * Nodes must not originate a packet to a multicast address whose scope |
| 123 | * field contains the reserved value 0; if such a packet is received, it | 135 | * field contains the reserved value 0; if such a packet is received, it |
diff --git a/net/key/af_key.c b/net/key/af_key.c index 8555f331ea60..5b1e5af25713 100644 --- a/net/key/af_key.c +++ b/net/key/af_key.c | |||
| @@ -2693,6 +2693,7 @@ static int key_notify_policy_flush(const struct km_event *c) | |||
| 2693 | hdr->sadb_msg_pid = c->portid; | 2693 | hdr->sadb_msg_pid = c->portid; |
| 2694 | hdr->sadb_msg_version = PF_KEY_V2; | 2694 | hdr->sadb_msg_version = PF_KEY_V2; |
| 2695 | hdr->sadb_msg_errno = (uint8_t) 0; | 2695 | hdr->sadb_msg_errno = (uint8_t) 0; |
| 2696 | hdr->sadb_msg_satype = SADB_SATYPE_UNSPEC; | ||
| 2696 | hdr->sadb_msg_len = (sizeof(struct sadb_msg) / sizeof(uint64_t)); | 2697 | hdr->sadb_msg_len = (sizeof(struct sadb_msg) / sizeof(uint64_t)); |
| 2697 | pfkey_broadcast(skb_out, GFP_ATOMIC, BROADCAST_ALL, NULL, c->net); | 2698 | pfkey_broadcast(skb_out, GFP_ATOMIC, BROADCAST_ALL, NULL, c->net); |
| 2698 | return 0; | 2699 | return 0; |
diff --git a/net/mac80211/iface.c b/net/mac80211/iface.c index baaa8608e52d..3bfe2612c8c2 100644 --- a/net/mac80211/iface.c +++ b/net/mac80211/iface.c | |||
| @@ -349,21 +349,19 @@ static void ieee80211_set_default_queues(struct ieee80211_sub_if_data *sdata) | |||
| 349 | static int ieee80211_add_virtual_monitor(struct ieee80211_local *local) | 349 | static int ieee80211_add_virtual_monitor(struct ieee80211_local *local) |
| 350 | { | 350 | { |
| 351 | struct ieee80211_sub_if_data *sdata; | 351 | struct ieee80211_sub_if_data *sdata; |
| 352 | int ret = 0; | 352 | int ret; |
| 353 | 353 | ||
| 354 | if (!(local->hw.flags & IEEE80211_HW_WANT_MONITOR_VIF)) | 354 | if (!(local->hw.flags & IEEE80211_HW_WANT_MONITOR_VIF)) |
| 355 | return 0; | 355 | return 0; |
| 356 | 356 | ||
| 357 | mutex_lock(&local->iflist_mtx); | 357 | ASSERT_RTNL(); |
| 358 | 358 | ||
| 359 | if (local->monitor_sdata) | 359 | if (local->monitor_sdata) |
| 360 | goto out_unlock; | 360 | return 0; |
| 361 | 361 | ||
| 362 | sdata = kzalloc(sizeof(*sdata) + local->hw.vif_data_size, GFP_KERNEL); | 362 | sdata = kzalloc(sizeof(*sdata) + local->hw.vif_data_size, GFP_KERNEL); |
| 363 | if (!sdata) { | 363 | if (!sdata) |
| 364 | ret = -ENOMEM; | 364 | return -ENOMEM; |
| 365 | goto out_unlock; | ||
| 366 | } | ||
| 367 | 365 | ||
| 368 | /* set up data */ | 366 | /* set up data */ |
| 369 | sdata->local = local; | 367 | sdata->local = local; |
| @@ -377,13 +375,13 @@ static int ieee80211_add_virtual_monitor(struct ieee80211_local *local) | |||
| 377 | if (WARN_ON(ret)) { | 375 | if (WARN_ON(ret)) { |
| 378 | /* ok .. stupid driver, it asked for this! */ | 376 | /* ok .. stupid driver, it asked for this! */ |
| 379 | kfree(sdata); | 377 | kfree(sdata); |
| 380 | goto out_unlock; | 378 | return ret; |
| 381 | } | 379 | } |
| 382 | 380 | ||
| 383 | ret = ieee80211_check_queues(sdata); | 381 | ret = ieee80211_check_queues(sdata); |
| 384 | if (ret) { | 382 | if (ret) { |
| 385 | kfree(sdata); | 383 | kfree(sdata); |
| 386 | goto out_unlock; | 384 | return ret; |
| 387 | } | 385 | } |
| 388 | 386 | ||
| 389 | ret = ieee80211_vif_use_channel(sdata, &local->monitor_chandef, | 387 | ret = ieee80211_vif_use_channel(sdata, &local->monitor_chandef, |
| @@ -391,13 +389,14 @@ static int ieee80211_add_virtual_monitor(struct ieee80211_local *local) | |||
| 391 | if (ret) { | 389 | if (ret) { |
| 392 | drv_remove_interface(local, sdata); | 390 | drv_remove_interface(local, sdata); |
| 393 | kfree(sdata); | 391 | kfree(sdata); |
| 394 | goto out_unlock; | 392 | return ret; |
| 395 | } | 393 | } |
| 396 | 394 | ||
| 395 | mutex_lock(&local->iflist_mtx); | ||
| 397 | rcu_assign_pointer(local->monitor_sdata, sdata); | 396 | rcu_assign_pointer(local->monitor_sdata, sdata); |
| 398 | out_unlock: | ||
| 399 | mutex_unlock(&local->iflist_mtx); | 397 | mutex_unlock(&local->iflist_mtx); |
| 400 | return ret; | 398 | |
| 399 | return 0; | ||
| 401 | } | 400 | } |
| 402 | 401 | ||
| 403 | static void ieee80211_del_virtual_monitor(struct ieee80211_local *local) | 402 | static void ieee80211_del_virtual_monitor(struct ieee80211_local *local) |
| @@ -407,14 +406,20 @@ static void ieee80211_del_virtual_monitor(struct ieee80211_local *local) | |||
| 407 | if (!(local->hw.flags & IEEE80211_HW_WANT_MONITOR_VIF)) | 406 | if (!(local->hw.flags & IEEE80211_HW_WANT_MONITOR_VIF)) |
| 408 | return; | 407 | return; |
| 409 | 408 | ||
| 409 | ASSERT_RTNL(); | ||
| 410 | |||
| 410 | mutex_lock(&local->iflist_mtx); | 411 | mutex_lock(&local->iflist_mtx); |
| 411 | 412 | ||
| 412 | sdata = rcu_dereference_protected(local->monitor_sdata, | 413 | sdata = rcu_dereference_protected(local->monitor_sdata, |
| 413 | lockdep_is_held(&local->iflist_mtx)); | 414 | lockdep_is_held(&local->iflist_mtx)); |
| 414 | if (!sdata) | 415 | if (!sdata) { |
| 415 | goto out_unlock; | 416 | mutex_unlock(&local->iflist_mtx); |
| 417 | return; | ||
| 418 | } | ||
| 416 | 419 | ||
| 417 | rcu_assign_pointer(local->monitor_sdata, NULL); | 420 | rcu_assign_pointer(local->monitor_sdata, NULL); |
| 421 | mutex_unlock(&local->iflist_mtx); | ||
| 422 | |||
| 418 | synchronize_net(); | 423 | synchronize_net(); |
| 419 | 424 | ||
| 420 | ieee80211_vif_release_channel(sdata); | 425 | ieee80211_vif_release_channel(sdata); |
| @@ -422,8 +427,6 @@ static void ieee80211_del_virtual_monitor(struct ieee80211_local *local) | |||
| 422 | drv_remove_interface(local, sdata); | 427 | drv_remove_interface(local, sdata); |
| 423 | 428 | ||
| 424 | kfree(sdata); | 429 | kfree(sdata); |
| 425 | out_unlock: | ||
| 426 | mutex_unlock(&local->iflist_mtx); | ||
| 427 | } | 430 | } |
| 428 | 431 | ||
| 429 | /* | 432 | /* |
diff --git a/net/mac80211/mesh.c b/net/mac80211/mesh.c index 29ce2aa87e7b..4749b3858695 100644 --- a/net/mac80211/mesh.c +++ b/net/mac80211/mesh.c | |||
| @@ -1060,7 +1060,8 @@ void ieee80211_mesh_notify_scan_completed(struct ieee80211_local *local) | |||
| 1060 | 1060 | ||
| 1061 | rcu_read_lock(); | 1061 | rcu_read_lock(); |
| 1062 | list_for_each_entry_rcu(sdata, &local->interfaces, list) | 1062 | list_for_each_entry_rcu(sdata, &local->interfaces, list) |
| 1063 | if (ieee80211_vif_is_mesh(&sdata->vif)) | 1063 | if (ieee80211_vif_is_mesh(&sdata->vif) && |
| 1064 | ieee80211_sdata_running(sdata)) | ||
| 1064 | ieee80211_queue_work(&local->hw, &sdata->work); | 1065 | ieee80211_queue_work(&local->hw, &sdata->work); |
| 1065 | rcu_read_unlock(); | 1066 | rcu_read_unlock(); |
| 1066 | } | 1067 | } |
diff --git a/net/mac80211/mlme.c b/net/mac80211/mlme.c index 141577412d84..82cc30318a86 100644 --- a/net/mac80211/mlme.c +++ b/net/mac80211/mlme.c | |||
| @@ -3608,8 +3608,10 @@ void ieee80211_mlme_notify_scan_completed(struct ieee80211_local *local) | |||
| 3608 | 3608 | ||
| 3609 | /* Restart STA timers */ | 3609 | /* Restart STA timers */ |
| 3610 | rcu_read_lock(); | 3610 | rcu_read_lock(); |
| 3611 | list_for_each_entry_rcu(sdata, &local->interfaces, list) | 3611 | list_for_each_entry_rcu(sdata, &local->interfaces, list) { |
| 3612 | ieee80211_restart_sta_timer(sdata); | 3612 | if (ieee80211_sdata_running(sdata)) |
| 3613 | ieee80211_restart_sta_timer(sdata); | ||
| 3614 | } | ||
| 3613 | rcu_read_unlock(); | 3615 | rcu_read_unlock(); |
| 3614 | } | 3616 | } |
| 3615 | 3617 | ||
diff --git a/net/mac80211/rx.c b/net/mac80211/rx.c index bb73ed2d20b9..c6844ad080be 100644 --- a/net/mac80211/rx.c +++ b/net/mac80211/rx.c | |||
| @@ -2675,7 +2675,19 @@ ieee80211_rx_h_action_return(struct ieee80211_rx_data *rx) | |||
| 2675 | 2675 | ||
| 2676 | memset(nskb->cb, 0, sizeof(nskb->cb)); | 2676 | memset(nskb->cb, 0, sizeof(nskb->cb)); |
| 2677 | 2677 | ||
| 2678 | ieee80211_tx_skb(rx->sdata, nskb); | 2678 | if (rx->sdata->vif.type == NL80211_IFTYPE_P2P_DEVICE) { |
| 2679 | struct ieee80211_tx_info *info = IEEE80211_SKB_CB(nskb); | ||
| 2680 | |||
| 2681 | info->flags = IEEE80211_TX_CTL_TX_OFFCHAN | | ||
| 2682 | IEEE80211_TX_INTFL_OFFCHAN_TX_OK | | ||
| 2683 | IEEE80211_TX_CTL_NO_CCK_RATE; | ||
| 2684 | if (local->hw.flags & IEEE80211_HW_QUEUE_CONTROL) | ||
| 2685 | info->hw_queue = | ||
| 2686 | local->hw.offchannel_tx_hw_queue; | ||
| 2687 | } | ||
| 2688 | |||
| 2689 | __ieee80211_tx_skb_tid_band(rx->sdata, nskb, 7, | ||
| 2690 | status->band); | ||
| 2679 | } | 2691 | } |
| 2680 | dev_kfree_skb(rx->skb); | 2692 | dev_kfree_skb(rx->skb); |
| 2681 | return RX_QUEUED; | 2693 | return RX_QUEUED; |
diff --git a/net/mac80211/sta_info.c b/net/mac80211/sta_info.c index a79ce820cb50..238a0cca320e 100644 --- a/net/mac80211/sta_info.c +++ b/net/mac80211/sta_info.c | |||
| @@ -766,6 +766,7 @@ int __must_check __sta_info_destroy(struct sta_info *sta) | |||
| 766 | struct ieee80211_local *local; | 766 | struct ieee80211_local *local; |
| 767 | struct ieee80211_sub_if_data *sdata; | 767 | struct ieee80211_sub_if_data *sdata; |
| 768 | int ret, i; | 768 | int ret, i; |
| 769 | bool have_key = false; | ||
| 769 | 770 | ||
| 770 | might_sleep(); | 771 | might_sleep(); |
| 771 | 772 | ||
| @@ -793,12 +794,19 @@ int __must_check __sta_info_destroy(struct sta_info *sta) | |||
| 793 | list_del_rcu(&sta->list); | 794 | list_del_rcu(&sta->list); |
| 794 | 795 | ||
| 795 | mutex_lock(&local->key_mtx); | 796 | mutex_lock(&local->key_mtx); |
| 796 | for (i = 0; i < NUM_DEFAULT_KEYS; i++) | 797 | for (i = 0; i < NUM_DEFAULT_KEYS; i++) { |
| 797 | __ieee80211_key_free(key_mtx_dereference(local, sta->gtk[i])); | 798 | __ieee80211_key_free(key_mtx_dereference(local, sta->gtk[i])); |
| 798 | if (sta->ptk) | 799 | have_key = true; |
| 800 | } | ||
| 801 | if (sta->ptk) { | ||
| 799 | __ieee80211_key_free(key_mtx_dereference(local, sta->ptk)); | 802 | __ieee80211_key_free(key_mtx_dereference(local, sta->ptk)); |
| 803 | have_key = true; | ||
| 804 | } | ||
| 800 | mutex_unlock(&local->key_mtx); | 805 | mutex_unlock(&local->key_mtx); |
| 801 | 806 | ||
| 807 | if (!have_key) | ||
| 808 | synchronize_net(); | ||
| 809 | |||
| 802 | sta->dead = true; | 810 | sta->dead = true; |
| 803 | 811 | ||
| 804 | local->num_sta--; | 812 | local->num_sta--; |
diff --git a/net/sched/sch_cbq.c b/net/sched/sch_cbq.c index 13aa47aa2ffb..1bc210ffcba2 100644 --- a/net/sched/sch_cbq.c +++ b/net/sched/sch_cbq.c | |||
| @@ -962,8 +962,11 @@ cbq_dequeue(struct Qdisc *sch) | |||
| 962 | cbq_update(q); | 962 | cbq_update(q); |
| 963 | if ((incr -= incr2) < 0) | 963 | if ((incr -= incr2) < 0) |
| 964 | incr = 0; | 964 | incr = 0; |
| 965 | q->now += incr; | ||
| 966 | } else { | ||
| 967 | if (now > q->now) | ||
| 968 | q->now = now; | ||
| 965 | } | 969 | } |
| 966 | q->now += incr; | ||
| 967 | q->now_rt = now; | 970 | q->now_rt = now; |
| 968 | 971 | ||
| 969 | for (;;) { | 972 | for (;;) { |
diff --git a/net/sched/sch_fq_codel.c b/net/sched/sch_fq_codel.c index 4e606fcb2534..55786283a3df 100644 --- a/net/sched/sch_fq_codel.c +++ b/net/sched/sch_fq_codel.c | |||
| @@ -195,7 +195,7 @@ static int fq_codel_enqueue(struct sk_buff *skb, struct Qdisc *sch) | |||
| 195 | flow->deficit = q->quantum; | 195 | flow->deficit = q->quantum; |
| 196 | flow->dropped = 0; | 196 | flow->dropped = 0; |
| 197 | } | 197 | } |
| 198 | if (++sch->q.qlen < sch->limit) | 198 | if (++sch->q.qlen <= sch->limit) |
| 199 | return NET_XMIT_SUCCESS; | 199 | return NET_XMIT_SUCCESS; |
| 200 | 200 | ||
| 201 | q->drop_overlimit++; | 201 | q->drop_overlimit++; |
diff --git a/net/sched/sch_generic.c b/net/sched/sch_generic.c index ffad48109a22..eac7e0ee23c1 100644 --- a/net/sched/sch_generic.c +++ b/net/sched/sch_generic.c | |||
| @@ -904,7 +904,7 @@ void psched_ratecfg_precompute(struct psched_ratecfg *r, u32 rate) | |||
| 904 | u64 mult; | 904 | u64 mult; |
| 905 | int shift; | 905 | int shift; |
| 906 | 906 | ||
| 907 | r->rate_bps = rate << 3; | 907 | r->rate_bps = (u64)rate << 3; |
| 908 | r->shift = 0; | 908 | r->shift = 0; |
| 909 | r->mult = 1; | 909 | r->mult = 1; |
| 910 | /* | 910 | /* |
diff --git a/net/vmw_vsock/af_vsock.c b/net/vmw_vsock/af_vsock.c index ca511c4f388a..d8079daf1bde 100644 --- a/net/vmw_vsock/af_vsock.c +++ b/net/vmw_vsock/af_vsock.c | |||
| @@ -207,7 +207,7 @@ static struct sock *__vsock_find_bound_socket(struct sockaddr_vm *addr) | |||
| 207 | struct vsock_sock *vsk; | 207 | struct vsock_sock *vsk; |
| 208 | 208 | ||
| 209 | list_for_each_entry(vsk, vsock_bound_sockets(addr), bound_table) | 209 | list_for_each_entry(vsk, vsock_bound_sockets(addr), bound_table) |
| 210 | if (vsock_addr_equals_addr_any(addr, &vsk->local_addr)) | 210 | if (addr->svm_port == vsk->local_addr.svm_port) |
| 211 | return sk_vsock(vsk); | 211 | return sk_vsock(vsk); |
| 212 | 212 | ||
| 213 | return NULL; | 213 | return NULL; |
| @@ -220,8 +220,8 @@ static struct sock *__vsock_find_connected_socket(struct sockaddr_vm *src, | |||
| 220 | 220 | ||
| 221 | list_for_each_entry(vsk, vsock_connected_sockets(src, dst), | 221 | list_for_each_entry(vsk, vsock_connected_sockets(src, dst), |
| 222 | connected_table) { | 222 | connected_table) { |
| 223 | if (vsock_addr_equals_addr(src, &vsk->remote_addr) | 223 | if (vsock_addr_equals_addr(src, &vsk->remote_addr) && |
| 224 | && vsock_addr_equals_addr(dst, &vsk->local_addr)) { | 224 | dst->svm_port == vsk->local_addr.svm_port) { |
| 225 | return sk_vsock(vsk); | 225 | return sk_vsock(vsk); |
| 226 | } | 226 | } |
| 227 | } | 227 | } |
diff --git a/net/vmw_vsock/vmci_transport.c b/net/vmw_vsock/vmci_transport.c index a70ace83a153..1f6508e249ae 100644 --- a/net/vmw_vsock/vmci_transport.c +++ b/net/vmw_vsock/vmci_transport.c | |||
| @@ -464,19 +464,16 @@ static struct sock *vmci_transport_get_pending( | |||
| 464 | struct vsock_sock *vlistener; | 464 | struct vsock_sock *vlistener; |
| 465 | struct vsock_sock *vpending; | 465 | struct vsock_sock *vpending; |
| 466 | struct sock *pending; | 466 | struct sock *pending; |
| 467 | struct sockaddr_vm src; | ||
| 468 | |||
| 469 | vsock_addr_init(&src, pkt->dg.src.context, pkt->src_port); | ||
| 467 | 470 | ||
| 468 | vlistener = vsock_sk(listener); | 471 | vlistener = vsock_sk(listener); |
| 469 | 472 | ||
| 470 | list_for_each_entry(vpending, &vlistener->pending_links, | 473 | list_for_each_entry(vpending, &vlistener->pending_links, |
| 471 | pending_links) { | 474 | pending_links) { |
| 472 | struct sockaddr_vm src; | ||
| 473 | struct sockaddr_vm dst; | ||
| 474 | |||
| 475 | vsock_addr_init(&src, pkt->dg.src.context, pkt->src_port); | ||
| 476 | vsock_addr_init(&dst, pkt->dg.dst.context, pkt->dst_port); | ||
| 477 | |||
| 478 | if (vsock_addr_equals_addr(&src, &vpending->remote_addr) && | 475 | if (vsock_addr_equals_addr(&src, &vpending->remote_addr) && |
| 479 | vsock_addr_equals_addr(&dst, &vpending->local_addr)) { | 476 | pkt->dst_port == vpending->local_addr.svm_port) { |
| 480 | pending = sk_vsock(vpending); | 477 | pending = sk_vsock(vpending); |
| 481 | sock_hold(pending); | 478 | sock_hold(pending); |
| 482 | goto found; | 479 | goto found; |
| @@ -739,10 +736,15 @@ static int vmci_transport_recv_stream_cb(void *data, struct vmci_datagram *dg) | |||
| 739 | */ | 736 | */ |
| 740 | bh_lock_sock(sk); | 737 | bh_lock_sock(sk); |
| 741 | 738 | ||
| 742 | if (!sock_owned_by_user(sk) && sk->sk_state == SS_CONNECTED) | 739 | if (!sock_owned_by_user(sk)) { |
| 743 | vmci_trans(vsk)->notify_ops->handle_notify_pkt( | 740 | /* The local context ID may be out of date, update it. */ |
| 744 | sk, pkt, true, &dst, &src, | 741 | vsk->local_addr.svm_cid = dst.svm_cid; |
| 745 | &bh_process_pkt); | 742 | |
| 743 | if (sk->sk_state == SS_CONNECTED) | ||
| 744 | vmci_trans(vsk)->notify_ops->handle_notify_pkt( | ||
| 745 | sk, pkt, true, &dst, &src, | ||
| 746 | &bh_process_pkt); | ||
| 747 | } | ||
| 746 | 748 | ||
| 747 | bh_unlock_sock(sk); | 749 | bh_unlock_sock(sk); |
| 748 | 750 | ||
| @@ -902,6 +904,9 @@ static void vmci_transport_recv_pkt_work(struct work_struct *work) | |||
| 902 | 904 | ||
| 903 | lock_sock(sk); | 905 | lock_sock(sk); |
| 904 | 906 | ||
| 907 | /* The local context ID may be out of date. */ | ||
| 908 | vsock_sk(sk)->local_addr.svm_cid = pkt->dg.dst.context; | ||
| 909 | |||
| 905 | switch (sk->sk_state) { | 910 | switch (sk->sk_state) { |
| 906 | case SS_LISTEN: | 911 | case SS_LISTEN: |
| 907 | vmci_transport_recv_listen(sk, pkt); | 912 | vmci_transport_recv_listen(sk, pkt); |
| @@ -958,6 +963,10 @@ static int vmci_transport_recv_listen(struct sock *sk, | |||
| 958 | pending = vmci_transport_get_pending(sk, pkt); | 963 | pending = vmci_transport_get_pending(sk, pkt); |
| 959 | if (pending) { | 964 | if (pending) { |
| 960 | lock_sock(pending); | 965 | lock_sock(pending); |
| 966 | |||
| 967 | /* The local context ID may be out of date. */ | ||
| 968 | vsock_sk(pending)->local_addr.svm_cid = pkt->dg.dst.context; | ||
| 969 | |||
| 961 | switch (pending->sk_state) { | 970 | switch (pending->sk_state) { |
| 962 | case SS_CONNECTING: | 971 | case SS_CONNECTING: |
| 963 | err = vmci_transport_recv_connecting_server(sk, | 972 | err = vmci_transport_recv_connecting_server(sk, |
diff --git a/net/vmw_vsock/vsock_addr.c b/net/vmw_vsock/vsock_addr.c index b7df1aea7c59..ec2611b4ea0e 100644 --- a/net/vmw_vsock/vsock_addr.c +++ b/net/vmw_vsock/vsock_addr.c | |||
| @@ -64,16 +64,6 @@ bool vsock_addr_equals_addr(const struct sockaddr_vm *addr, | |||
| 64 | } | 64 | } |
| 65 | EXPORT_SYMBOL_GPL(vsock_addr_equals_addr); | 65 | EXPORT_SYMBOL_GPL(vsock_addr_equals_addr); |
| 66 | 66 | ||
| 67 | bool vsock_addr_equals_addr_any(const struct sockaddr_vm *addr, | ||
| 68 | const struct sockaddr_vm *other) | ||
| 69 | { | ||
| 70 | return (addr->svm_cid == VMADDR_CID_ANY || | ||
| 71 | other->svm_cid == VMADDR_CID_ANY || | ||
| 72 | addr->svm_cid == other->svm_cid) && | ||
| 73 | addr->svm_port == other->svm_port; | ||
| 74 | } | ||
| 75 | EXPORT_SYMBOL_GPL(vsock_addr_equals_addr_any); | ||
| 76 | |||
| 77 | int vsock_addr_cast(const struct sockaddr *addr, | 67 | int vsock_addr_cast(const struct sockaddr *addr, |
| 78 | size_t len, struct sockaddr_vm **out_addr) | 68 | size_t len, struct sockaddr_vm **out_addr) |
| 79 | { | 69 | { |
diff --git a/net/vmw_vsock/vsock_addr.h b/net/vmw_vsock/vsock_addr.h index cdfbcefdf843..9ccd5316eac0 100644 --- a/net/vmw_vsock/vsock_addr.h +++ b/net/vmw_vsock/vsock_addr.h | |||
| @@ -24,8 +24,6 @@ bool vsock_addr_bound(const struct sockaddr_vm *addr); | |||
| 24 | void vsock_addr_unbind(struct sockaddr_vm *addr); | 24 | void vsock_addr_unbind(struct sockaddr_vm *addr); |
| 25 | bool vsock_addr_equals_addr(const struct sockaddr_vm *addr, | 25 | bool vsock_addr_equals_addr(const struct sockaddr_vm *addr, |
| 26 | const struct sockaddr_vm *other); | 26 | const struct sockaddr_vm *other); |
| 27 | bool vsock_addr_equals_addr_any(const struct sockaddr_vm *addr, | ||
| 28 | const struct sockaddr_vm *other); | ||
| 29 | int vsock_addr_cast(const struct sockaddr *addr, size_t len, | 27 | int vsock_addr_cast(const struct sockaddr *addr, size_t len, |
| 30 | struct sockaddr_vm **out_addr); | 28 | struct sockaddr_vm **out_addr); |
| 31 | 29 | ||
diff --git a/net/wireless/core.c b/net/wireless/core.c index ea4155fe9733..6ddf74f0ae1e 100644 --- a/net/wireless/core.c +++ b/net/wireless/core.c | |||
| @@ -212,6 +212,39 @@ static void cfg80211_rfkill_poll(struct rfkill *rfkill, void *data) | |||
| 212 | rdev_rfkill_poll(rdev); | 212 | rdev_rfkill_poll(rdev); |
| 213 | } | 213 | } |
| 214 | 214 | ||
| 215 | void cfg80211_stop_p2p_device(struct cfg80211_registered_device *rdev, | ||
| 216 | struct wireless_dev *wdev) | ||
| 217 | { | ||
| 218 | lockdep_assert_held(&rdev->devlist_mtx); | ||
| 219 | lockdep_assert_held(&rdev->sched_scan_mtx); | ||
| 220 | |||
| 221 | if (WARN_ON(wdev->iftype != NL80211_IFTYPE_P2P_DEVICE)) | ||
| 222 | return; | ||
| 223 | |||
| 224 | if (!wdev->p2p_started) | ||
| 225 | return; | ||
| 226 | |||
| 227 | rdev_stop_p2p_device(rdev, wdev); | ||
| 228 | wdev->p2p_started = false; | ||
| 229 | |||
| 230 | rdev->opencount--; | ||
| 231 | |||
| 232 | if (rdev->scan_req && rdev->scan_req->wdev == wdev) { | ||
| 233 | bool busy = work_busy(&rdev->scan_done_wk); | ||
| 234 | |||
| 235 | /* | ||
| 236 | * If the work isn't pending or running (in which case it would | ||
| 237 | * be waiting for the lock we hold) the driver didn't properly | ||
| 238 | * cancel the scan when the interface was removed. In this case | ||
| 239 | * warn and leak the scan request object to not crash later. | ||
| 240 | */ | ||
| 241 | WARN_ON(!busy); | ||
| 242 | |||
| 243 | rdev->scan_req->aborted = true; | ||
| 244 | ___cfg80211_scan_done(rdev, !busy); | ||
| 245 | } | ||
| 246 | } | ||
| 247 | |||
| 215 | static int cfg80211_rfkill_set_block(void *data, bool blocked) | 248 | static int cfg80211_rfkill_set_block(void *data, bool blocked) |
| 216 | { | 249 | { |
| 217 | struct cfg80211_registered_device *rdev = data; | 250 | struct cfg80211_registered_device *rdev = data; |
| @@ -221,7 +254,8 @@ static int cfg80211_rfkill_set_block(void *data, bool blocked) | |||
| 221 | return 0; | 254 | return 0; |
| 222 | 255 | ||
| 223 | rtnl_lock(); | 256 | rtnl_lock(); |
| 224 | mutex_lock(&rdev->devlist_mtx); | 257 | |
| 258 | /* read-only iteration need not hold the devlist_mtx */ | ||
| 225 | 259 | ||
| 226 | list_for_each_entry(wdev, &rdev->wdev_list, list) { | 260 | list_for_each_entry(wdev, &rdev->wdev_list, list) { |
| 227 | if (wdev->netdev) { | 261 | if (wdev->netdev) { |
| @@ -231,18 +265,18 @@ static int cfg80211_rfkill_set_block(void *data, bool blocked) | |||
| 231 | /* otherwise, check iftype */ | 265 | /* otherwise, check iftype */ |
| 232 | switch (wdev->iftype) { | 266 | switch (wdev->iftype) { |
| 233 | case NL80211_IFTYPE_P2P_DEVICE: | 267 | case NL80211_IFTYPE_P2P_DEVICE: |
| 234 | if (!wdev->p2p_started) | 268 | /* but this requires it */ |
| 235 | break; | 269 | mutex_lock(&rdev->devlist_mtx); |
| 236 | rdev_stop_p2p_device(rdev, wdev); | 270 | mutex_lock(&rdev->sched_scan_mtx); |
| 237 | wdev->p2p_started = false; | 271 | cfg80211_stop_p2p_device(rdev, wdev); |
| 238 | rdev->opencount--; | 272 | mutex_unlock(&rdev->sched_scan_mtx); |
| 273 | mutex_unlock(&rdev->devlist_mtx); | ||
| 239 | break; | 274 | break; |
| 240 | default: | 275 | default: |
| 241 | break; | 276 | break; |
| 242 | } | 277 | } |
| 243 | } | 278 | } |
| 244 | 279 | ||
| 245 | mutex_unlock(&rdev->devlist_mtx); | ||
| 246 | rtnl_unlock(); | 280 | rtnl_unlock(); |
| 247 | 281 | ||
| 248 | return 0; | 282 | return 0; |
| @@ -745,17 +779,13 @@ static void wdev_cleanup_work(struct work_struct *work) | |||
| 745 | wdev = container_of(work, struct wireless_dev, cleanup_work); | 779 | wdev = container_of(work, struct wireless_dev, cleanup_work); |
| 746 | rdev = wiphy_to_dev(wdev->wiphy); | 780 | rdev = wiphy_to_dev(wdev->wiphy); |
| 747 | 781 | ||
| 748 | cfg80211_lock_rdev(rdev); | 782 | mutex_lock(&rdev->sched_scan_mtx); |
| 749 | 783 | ||
| 750 | if (WARN_ON(rdev->scan_req && rdev->scan_req->wdev == wdev)) { | 784 | if (WARN_ON(rdev->scan_req && rdev->scan_req->wdev == wdev)) { |
| 751 | rdev->scan_req->aborted = true; | 785 | rdev->scan_req->aborted = true; |
| 752 | ___cfg80211_scan_done(rdev, true); | 786 | ___cfg80211_scan_done(rdev, true); |
| 753 | } | 787 | } |
| 754 | 788 | ||
| 755 | cfg80211_unlock_rdev(rdev); | ||
| 756 | |||
| 757 | mutex_lock(&rdev->sched_scan_mtx); | ||
| 758 | |||
| 759 | if (WARN_ON(rdev->sched_scan_req && | 789 | if (WARN_ON(rdev->sched_scan_req && |
| 760 | rdev->sched_scan_req->dev == wdev->netdev)) { | 790 | rdev->sched_scan_req->dev == wdev->netdev)) { |
| 761 | __cfg80211_stop_sched_scan(rdev, false); | 791 | __cfg80211_stop_sched_scan(rdev, false); |
| @@ -781,21 +811,19 @@ void cfg80211_unregister_wdev(struct wireless_dev *wdev) | |||
| 781 | return; | 811 | return; |
| 782 | 812 | ||
| 783 | mutex_lock(&rdev->devlist_mtx); | 813 | mutex_lock(&rdev->devlist_mtx); |
| 814 | mutex_lock(&rdev->sched_scan_mtx); | ||
| 784 | list_del_rcu(&wdev->list); | 815 | list_del_rcu(&wdev->list); |
| 785 | rdev->devlist_generation++; | 816 | rdev->devlist_generation++; |
| 786 | 817 | ||
| 787 | switch (wdev->iftype) { | 818 | switch (wdev->iftype) { |
| 788 | case NL80211_IFTYPE_P2P_DEVICE: | 819 | case NL80211_IFTYPE_P2P_DEVICE: |
| 789 | if (!wdev->p2p_started) | 820 | cfg80211_stop_p2p_device(rdev, wdev); |
| 790 | break; | ||
| 791 | rdev_stop_p2p_device(rdev, wdev); | ||
| 792 | wdev->p2p_started = false; | ||
| 793 | rdev->opencount--; | ||
| 794 | break; | 821 | break; |
| 795 | default: | 822 | default: |
| 796 | WARN_ON_ONCE(1); | 823 | WARN_ON_ONCE(1); |
| 797 | break; | 824 | break; |
| 798 | } | 825 | } |
| 826 | mutex_unlock(&rdev->sched_scan_mtx); | ||
| 799 | mutex_unlock(&rdev->devlist_mtx); | 827 | mutex_unlock(&rdev->devlist_mtx); |
| 800 | } | 828 | } |
| 801 | EXPORT_SYMBOL(cfg80211_unregister_wdev); | 829 | EXPORT_SYMBOL(cfg80211_unregister_wdev); |
| @@ -936,6 +964,7 @@ static int cfg80211_netdev_notifier_call(struct notifier_block *nb, | |||
| 936 | cfg80211_update_iface_num(rdev, wdev->iftype, 1); | 964 | cfg80211_update_iface_num(rdev, wdev->iftype, 1); |
| 937 | cfg80211_lock_rdev(rdev); | 965 | cfg80211_lock_rdev(rdev); |
| 938 | mutex_lock(&rdev->devlist_mtx); | 966 | mutex_lock(&rdev->devlist_mtx); |
| 967 | mutex_lock(&rdev->sched_scan_mtx); | ||
| 939 | wdev_lock(wdev); | 968 | wdev_lock(wdev); |
| 940 | switch (wdev->iftype) { | 969 | switch (wdev->iftype) { |
| 941 | #ifdef CONFIG_CFG80211_WEXT | 970 | #ifdef CONFIG_CFG80211_WEXT |
| @@ -967,6 +996,7 @@ static int cfg80211_netdev_notifier_call(struct notifier_block *nb, | |||
| 967 | break; | 996 | break; |
| 968 | } | 997 | } |
| 969 | wdev_unlock(wdev); | 998 | wdev_unlock(wdev); |
| 999 | mutex_unlock(&rdev->sched_scan_mtx); | ||
| 970 | rdev->opencount++; | 1000 | rdev->opencount++; |
| 971 | mutex_unlock(&rdev->devlist_mtx); | 1001 | mutex_unlock(&rdev->devlist_mtx); |
| 972 | cfg80211_unlock_rdev(rdev); | 1002 | cfg80211_unlock_rdev(rdev); |
diff --git a/net/wireless/core.h b/net/wireless/core.h index 3aec0e429d8a..5845c2b37aa8 100644 --- a/net/wireless/core.h +++ b/net/wireless/core.h | |||
| @@ -503,6 +503,9 @@ int cfg80211_validate_beacon_int(struct cfg80211_registered_device *rdev, | |||
| 503 | void cfg80211_update_iface_num(struct cfg80211_registered_device *rdev, | 503 | void cfg80211_update_iface_num(struct cfg80211_registered_device *rdev, |
| 504 | enum nl80211_iftype iftype, int num); | 504 | enum nl80211_iftype iftype, int num); |
| 505 | 505 | ||
| 506 | void cfg80211_stop_p2p_device(struct cfg80211_registered_device *rdev, | ||
| 507 | struct wireless_dev *wdev); | ||
| 508 | |||
| 506 | #define CFG80211_MAX_NUM_DIFFERENT_CHANNELS 10 | 509 | #define CFG80211_MAX_NUM_DIFFERENT_CHANNELS 10 |
| 507 | 510 | ||
| 508 | #ifdef CONFIG_CFG80211_DEVELOPER_WARNINGS | 511 | #ifdef CONFIG_CFG80211_DEVELOPER_WARNINGS |
diff --git a/net/wireless/nl80211.c b/net/wireless/nl80211.c index d44ab216c0ec..58e13a8c95f9 100644 --- a/net/wireless/nl80211.c +++ b/net/wireless/nl80211.c | |||
| @@ -4702,14 +4702,19 @@ static int nl80211_trigger_scan(struct sk_buff *skb, struct genl_info *info) | |||
| 4702 | if (!rdev->ops->scan) | 4702 | if (!rdev->ops->scan) |
| 4703 | return -EOPNOTSUPP; | 4703 | return -EOPNOTSUPP; |
| 4704 | 4704 | ||
| 4705 | if (rdev->scan_req) | 4705 | mutex_lock(&rdev->sched_scan_mtx); |
| 4706 | return -EBUSY; | 4706 | if (rdev->scan_req) { |
| 4707 | err = -EBUSY; | ||
| 4708 | goto unlock; | ||
| 4709 | } | ||
| 4707 | 4710 | ||
| 4708 | if (info->attrs[NL80211_ATTR_SCAN_FREQUENCIES]) { | 4711 | if (info->attrs[NL80211_ATTR_SCAN_FREQUENCIES]) { |
| 4709 | n_channels = validate_scan_freqs( | 4712 | n_channels = validate_scan_freqs( |
| 4710 | info->attrs[NL80211_ATTR_SCAN_FREQUENCIES]); | 4713 | info->attrs[NL80211_ATTR_SCAN_FREQUENCIES]); |
| 4711 | if (!n_channels) | 4714 | if (!n_channels) { |
| 4712 | return -EINVAL; | 4715 | err = -EINVAL; |
| 4716 | goto unlock; | ||
| 4717 | } | ||
| 4713 | } else { | 4718 | } else { |
| 4714 | enum ieee80211_band band; | 4719 | enum ieee80211_band band; |
| 4715 | n_channels = 0; | 4720 | n_channels = 0; |
| @@ -4723,23 +4728,29 @@ static int nl80211_trigger_scan(struct sk_buff *skb, struct genl_info *info) | |||
| 4723 | nla_for_each_nested(attr, info->attrs[NL80211_ATTR_SCAN_SSIDS], tmp) | 4728 | nla_for_each_nested(attr, info->attrs[NL80211_ATTR_SCAN_SSIDS], tmp) |
| 4724 | n_ssids++; | 4729 | n_ssids++; |
| 4725 | 4730 | ||
| 4726 | if (n_ssids > wiphy->max_scan_ssids) | 4731 | if (n_ssids > wiphy->max_scan_ssids) { |
| 4727 | return -EINVAL; | 4732 | err = -EINVAL; |
| 4733 | goto unlock; | ||
| 4734 | } | ||
| 4728 | 4735 | ||
| 4729 | if (info->attrs[NL80211_ATTR_IE]) | 4736 | if (info->attrs[NL80211_ATTR_IE]) |
| 4730 | ie_len = nla_len(info->attrs[NL80211_ATTR_IE]); | 4737 | ie_len = nla_len(info->attrs[NL80211_ATTR_IE]); |
| 4731 | else | 4738 | else |
| 4732 | ie_len = 0; | 4739 | ie_len = 0; |
| 4733 | 4740 | ||
| 4734 | if (ie_len > wiphy->max_scan_ie_len) | 4741 | if (ie_len > wiphy->max_scan_ie_len) { |
| 4735 | return -EINVAL; | 4742 | err = -EINVAL; |
| 4743 | goto unlock; | ||
| 4744 | } | ||
| 4736 | 4745 | ||
| 4737 | request = kzalloc(sizeof(*request) | 4746 | request = kzalloc(sizeof(*request) |
| 4738 | + sizeof(*request->ssids) * n_ssids | 4747 | + sizeof(*request->ssids) * n_ssids |
| 4739 | + sizeof(*request->channels) * n_channels | 4748 | + sizeof(*request->channels) * n_channels |
| 4740 | + ie_len, GFP_KERNEL); | 4749 | + ie_len, GFP_KERNEL); |
| 4741 | if (!request) | 4750 | if (!request) { |
| 4742 | return -ENOMEM; | 4751 | err = -ENOMEM; |
| 4752 | goto unlock; | ||
| 4753 | } | ||
| 4743 | 4754 | ||
| 4744 | if (n_ssids) | 4755 | if (n_ssids) |
| 4745 | request->ssids = (void *)&request->channels[n_channels]; | 4756 | request->ssids = (void *)&request->channels[n_channels]; |
| @@ -4876,6 +4887,8 @@ static int nl80211_trigger_scan(struct sk_buff *skb, struct genl_info *info) | |||
| 4876 | kfree(request); | 4887 | kfree(request); |
| 4877 | } | 4888 | } |
| 4878 | 4889 | ||
| 4890 | unlock: | ||
| 4891 | mutex_unlock(&rdev->sched_scan_mtx); | ||
| 4879 | return err; | 4892 | return err; |
| 4880 | } | 4893 | } |
| 4881 | 4894 | ||
| @@ -7749,20 +7762,9 @@ static int nl80211_stop_p2p_device(struct sk_buff *skb, struct genl_info *info) | |||
| 7749 | if (!rdev->ops->stop_p2p_device) | 7762 | if (!rdev->ops->stop_p2p_device) |
| 7750 | return -EOPNOTSUPP; | 7763 | return -EOPNOTSUPP; |
| 7751 | 7764 | ||
| 7752 | if (!wdev->p2p_started) | 7765 | mutex_lock(&rdev->sched_scan_mtx); |
| 7753 | return 0; | 7766 | cfg80211_stop_p2p_device(rdev, wdev); |
| 7754 | 7767 | mutex_unlock(&rdev->sched_scan_mtx); | |
| 7755 | rdev_stop_p2p_device(rdev, wdev); | ||
| 7756 | wdev->p2p_started = false; | ||
| 7757 | |||
| 7758 | mutex_lock(&rdev->devlist_mtx); | ||
| 7759 | rdev->opencount--; | ||
| 7760 | mutex_unlock(&rdev->devlist_mtx); | ||
| 7761 | |||
| 7762 | if (WARN_ON(rdev->scan_req && rdev->scan_req->wdev == wdev)) { | ||
| 7763 | rdev->scan_req->aborted = true; | ||
| 7764 | ___cfg80211_scan_done(rdev, true); | ||
| 7765 | } | ||
| 7766 | 7768 | ||
| 7767 | return 0; | 7769 | return 0; |
| 7768 | } | 7770 | } |
| @@ -8486,7 +8488,7 @@ static int nl80211_add_scan_req(struct sk_buff *msg, | |||
| 8486 | struct nlattr *nest; | 8488 | struct nlattr *nest; |
| 8487 | int i; | 8489 | int i; |
| 8488 | 8490 | ||
| 8489 | ASSERT_RDEV_LOCK(rdev); | 8491 | lockdep_assert_held(&rdev->sched_scan_mtx); |
| 8490 | 8492 | ||
| 8491 | if (WARN_ON(!req)) | 8493 | if (WARN_ON(!req)) |
| 8492 | return 0; | 8494 | return 0; |
diff --git a/net/wireless/scan.c b/net/wireless/scan.c index 674aadca0079..fd99ea495b7e 100644 --- a/net/wireless/scan.c +++ b/net/wireless/scan.c | |||
| @@ -169,7 +169,7 @@ void ___cfg80211_scan_done(struct cfg80211_registered_device *rdev, bool leak) | |||
| 169 | union iwreq_data wrqu; | 169 | union iwreq_data wrqu; |
| 170 | #endif | 170 | #endif |
| 171 | 171 | ||
| 172 | ASSERT_RDEV_LOCK(rdev); | 172 | lockdep_assert_held(&rdev->sched_scan_mtx); |
| 173 | 173 | ||
| 174 | request = rdev->scan_req; | 174 | request = rdev->scan_req; |
| 175 | 175 | ||
| @@ -230,9 +230,9 @@ void __cfg80211_scan_done(struct work_struct *wk) | |||
| 230 | rdev = container_of(wk, struct cfg80211_registered_device, | 230 | rdev = container_of(wk, struct cfg80211_registered_device, |
| 231 | scan_done_wk); | 231 | scan_done_wk); |
| 232 | 232 | ||
| 233 | cfg80211_lock_rdev(rdev); | 233 | mutex_lock(&rdev->sched_scan_mtx); |
| 234 | ___cfg80211_scan_done(rdev, false); | 234 | ___cfg80211_scan_done(rdev, false); |
| 235 | cfg80211_unlock_rdev(rdev); | 235 | mutex_unlock(&rdev->sched_scan_mtx); |
| 236 | } | 236 | } |
| 237 | 237 | ||
| 238 | void cfg80211_scan_done(struct cfg80211_scan_request *request, bool aborted) | 238 | void cfg80211_scan_done(struct cfg80211_scan_request *request, bool aborted) |
| @@ -698,11 +698,6 @@ cfg80211_bss_update(struct cfg80211_registered_device *dev, | |||
| 698 | found = rb_find_bss(dev, tmp, BSS_CMP_REGULAR); | 698 | found = rb_find_bss(dev, tmp, BSS_CMP_REGULAR); |
| 699 | 699 | ||
| 700 | if (found) { | 700 | if (found) { |
| 701 | found->pub.beacon_interval = tmp->pub.beacon_interval; | ||
| 702 | found->pub.signal = tmp->pub.signal; | ||
| 703 | found->pub.capability = tmp->pub.capability; | ||
| 704 | found->ts = tmp->ts; | ||
| 705 | |||
| 706 | /* Update IEs */ | 701 | /* Update IEs */ |
| 707 | if (rcu_access_pointer(tmp->pub.proberesp_ies)) { | 702 | if (rcu_access_pointer(tmp->pub.proberesp_ies)) { |
| 708 | const struct cfg80211_bss_ies *old; | 703 | const struct cfg80211_bss_ies *old; |
| @@ -723,6 +718,8 @@ cfg80211_bss_update(struct cfg80211_registered_device *dev, | |||
| 723 | 718 | ||
| 724 | if (found->pub.hidden_beacon_bss && | 719 | if (found->pub.hidden_beacon_bss && |
| 725 | !list_empty(&found->hidden_list)) { | 720 | !list_empty(&found->hidden_list)) { |
| 721 | const struct cfg80211_bss_ies *f; | ||
| 722 | |||
| 726 | /* | 723 | /* |
| 727 | * The found BSS struct is one of the probe | 724 | * The found BSS struct is one of the probe |
| 728 | * response members of a group, but we're | 725 | * response members of a group, but we're |
| @@ -732,6 +729,10 @@ cfg80211_bss_update(struct cfg80211_registered_device *dev, | |||
| 732 | * SSID to showing it, which is confusing so | 729 | * SSID to showing it, which is confusing so |
| 733 | * drop this information. | 730 | * drop this information. |
| 734 | */ | 731 | */ |
| 732 | |||
| 733 | f = rcu_access_pointer(tmp->pub.beacon_ies); | ||
| 734 | kfree_rcu((struct cfg80211_bss_ies *)f, | ||
| 735 | rcu_head); | ||
| 735 | goto drop; | 736 | goto drop; |
| 736 | } | 737 | } |
| 737 | 738 | ||
| @@ -761,6 +762,11 @@ cfg80211_bss_update(struct cfg80211_registered_device *dev, | |||
| 761 | kfree_rcu((struct cfg80211_bss_ies *)old, | 762 | kfree_rcu((struct cfg80211_bss_ies *)old, |
| 762 | rcu_head); | 763 | rcu_head); |
| 763 | } | 764 | } |
| 765 | |||
| 766 | found->pub.beacon_interval = tmp->pub.beacon_interval; | ||
| 767 | found->pub.signal = tmp->pub.signal; | ||
| 768 | found->pub.capability = tmp->pub.capability; | ||
| 769 | found->ts = tmp->ts; | ||
| 764 | } else { | 770 | } else { |
| 765 | struct cfg80211_internal_bss *new; | 771 | struct cfg80211_internal_bss *new; |
| 766 | struct cfg80211_internal_bss *hidden; | 772 | struct cfg80211_internal_bss *hidden; |
| @@ -1056,6 +1062,7 @@ int cfg80211_wext_siwscan(struct net_device *dev, | |||
| 1056 | if (IS_ERR(rdev)) | 1062 | if (IS_ERR(rdev)) |
| 1057 | return PTR_ERR(rdev); | 1063 | return PTR_ERR(rdev); |
| 1058 | 1064 | ||
| 1065 | mutex_lock(&rdev->sched_scan_mtx); | ||
| 1059 | if (rdev->scan_req) { | 1066 | if (rdev->scan_req) { |
| 1060 | err = -EBUSY; | 1067 | err = -EBUSY; |
| 1061 | goto out; | 1068 | goto out; |
| @@ -1162,6 +1169,7 @@ int cfg80211_wext_siwscan(struct net_device *dev, | |||
| 1162 | dev_hold(dev); | 1169 | dev_hold(dev); |
| 1163 | } | 1170 | } |
| 1164 | out: | 1171 | out: |
| 1172 | mutex_unlock(&rdev->sched_scan_mtx); | ||
| 1165 | kfree(creq); | 1173 | kfree(creq); |
| 1166 | cfg80211_unlock_rdev(rdev); | 1174 | cfg80211_unlock_rdev(rdev); |
| 1167 | return err; | 1175 | return err; |
diff --git a/net/wireless/sme.c b/net/wireless/sme.c index f432bd3755b1..09d994d192ff 100644 --- a/net/wireless/sme.c +++ b/net/wireless/sme.c | |||
| @@ -85,6 +85,7 @@ static int cfg80211_conn_scan(struct wireless_dev *wdev) | |||
| 85 | ASSERT_RTNL(); | 85 | ASSERT_RTNL(); |
| 86 | ASSERT_RDEV_LOCK(rdev); | 86 | ASSERT_RDEV_LOCK(rdev); |
| 87 | ASSERT_WDEV_LOCK(wdev); | 87 | ASSERT_WDEV_LOCK(wdev); |
| 88 | lockdep_assert_held(&rdev->sched_scan_mtx); | ||
| 88 | 89 | ||
| 89 | if (rdev->scan_req) | 90 | if (rdev->scan_req) |
| 90 | return -EBUSY; | 91 | return -EBUSY; |
| @@ -320,11 +321,9 @@ void cfg80211_sme_scan_done(struct net_device *dev) | |||
| 320 | { | 321 | { |
| 321 | struct wireless_dev *wdev = dev->ieee80211_ptr; | 322 | struct wireless_dev *wdev = dev->ieee80211_ptr; |
| 322 | 323 | ||
| 323 | mutex_lock(&wiphy_to_dev(wdev->wiphy)->devlist_mtx); | ||
| 324 | wdev_lock(wdev); | 324 | wdev_lock(wdev); |
| 325 | __cfg80211_sme_scan_done(dev); | 325 | __cfg80211_sme_scan_done(dev); |
| 326 | wdev_unlock(wdev); | 326 | wdev_unlock(wdev); |
| 327 | mutex_unlock(&wiphy_to_dev(wdev->wiphy)->devlist_mtx); | ||
| 328 | } | 327 | } |
| 329 | 328 | ||
| 330 | void cfg80211_sme_rx_auth(struct net_device *dev, | 329 | void cfg80211_sme_rx_auth(struct net_device *dev, |
| @@ -924,9 +923,12 @@ int cfg80211_connect(struct cfg80211_registered_device *rdev, | |||
| 924 | int err; | 923 | int err; |
| 925 | 924 | ||
| 926 | mutex_lock(&rdev->devlist_mtx); | 925 | mutex_lock(&rdev->devlist_mtx); |
| 926 | /* might request scan - scan_mtx -> wdev_mtx dependency */ | ||
| 927 | mutex_lock(&rdev->sched_scan_mtx); | ||
| 927 | wdev_lock(dev->ieee80211_ptr); | 928 | wdev_lock(dev->ieee80211_ptr); |
| 928 | err = __cfg80211_connect(rdev, dev, connect, connkeys, NULL); | 929 | err = __cfg80211_connect(rdev, dev, connect, connkeys, NULL); |
| 929 | wdev_unlock(dev->ieee80211_ptr); | 930 | wdev_unlock(dev->ieee80211_ptr); |
| 931 | mutex_unlock(&rdev->sched_scan_mtx); | ||
| 930 | mutex_unlock(&rdev->devlist_mtx); | 932 | mutex_unlock(&rdev->devlist_mtx); |
| 931 | 933 | ||
| 932 | return err; | 934 | return err; |
diff --git a/net/wireless/trace.h b/net/wireless/trace.h index b7a531380e19..7586de77a2f8 100644 --- a/net/wireless/trace.h +++ b/net/wireless/trace.h | |||
| @@ -27,7 +27,8 @@ | |||
| 27 | #define WIPHY_PR_ARG __entry->wiphy_name | 27 | #define WIPHY_PR_ARG __entry->wiphy_name |
| 28 | 28 | ||
| 29 | #define WDEV_ENTRY __field(u32, id) | 29 | #define WDEV_ENTRY __field(u32, id) |
| 30 | #define WDEV_ASSIGN (__entry->id) = (wdev ? wdev->identifier : 0) | 30 | #define WDEV_ASSIGN (__entry->id) = (!IS_ERR_OR_NULL(wdev) \ |
| 31 | ? wdev->identifier : 0) | ||
| 31 | #define WDEV_PR_FMT "wdev(%u)" | 32 | #define WDEV_PR_FMT "wdev(%u)" |
| 32 | #define WDEV_PR_ARG (__entry->id) | 33 | #define WDEV_PR_ARG (__entry->id) |
| 33 | 34 | ||
| @@ -1778,7 +1779,7 @@ TRACE_EVENT(rdev_set_mac_acl, | |||
| 1778 | ), | 1779 | ), |
| 1779 | TP_fast_assign( | 1780 | TP_fast_assign( |
| 1780 | WIPHY_ASSIGN; | 1781 | WIPHY_ASSIGN; |
| 1781 | WIPHY_ASSIGN; | 1782 | NETDEV_ASSIGN; |
| 1782 | __entry->acl_policy = params->acl_policy; | 1783 | __entry->acl_policy = params->acl_policy; |
| 1783 | ), | 1784 | ), |
| 1784 | TP_printk(WIPHY_PR_FMT ", " NETDEV_PR_FMT ", acl policy: %d", | 1785 | TP_printk(WIPHY_PR_FMT ", " NETDEV_PR_FMT ", acl policy: %d", |
diff --git a/net/wireless/wext-sme.c b/net/wireless/wext-sme.c index fb9622f6d99c..e79cb5c0655a 100644 --- a/net/wireless/wext-sme.c +++ b/net/wireless/wext-sme.c | |||
| @@ -89,6 +89,7 @@ int cfg80211_mgd_wext_siwfreq(struct net_device *dev, | |||
| 89 | 89 | ||
| 90 | cfg80211_lock_rdev(rdev); | 90 | cfg80211_lock_rdev(rdev); |
| 91 | mutex_lock(&rdev->devlist_mtx); | 91 | mutex_lock(&rdev->devlist_mtx); |
| 92 | mutex_lock(&rdev->sched_scan_mtx); | ||
| 92 | wdev_lock(wdev); | 93 | wdev_lock(wdev); |
| 93 | 94 | ||
| 94 | if (wdev->sme_state != CFG80211_SME_IDLE) { | 95 | if (wdev->sme_state != CFG80211_SME_IDLE) { |
| @@ -135,6 +136,7 @@ int cfg80211_mgd_wext_siwfreq(struct net_device *dev, | |||
| 135 | err = cfg80211_mgd_wext_connect(rdev, wdev); | 136 | err = cfg80211_mgd_wext_connect(rdev, wdev); |
| 136 | out: | 137 | out: |
| 137 | wdev_unlock(wdev); | 138 | wdev_unlock(wdev); |
| 139 | mutex_unlock(&rdev->sched_scan_mtx); | ||
| 138 | mutex_unlock(&rdev->devlist_mtx); | 140 | mutex_unlock(&rdev->devlist_mtx); |
| 139 | cfg80211_unlock_rdev(rdev); | 141 | cfg80211_unlock_rdev(rdev); |
| 140 | return err; | 142 | return err; |
| @@ -190,6 +192,7 @@ int cfg80211_mgd_wext_siwessid(struct net_device *dev, | |||
| 190 | 192 | ||
| 191 | cfg80211_lock_rdev(rdev); | 193 | cfg80211_lock_rdev(rdev); |
| 192 | mutex_lock(&rdev->devlist_mtx); | 194 | mutex_lock(&rdev->devlist_mtx); |
| 195 | mutex_lock(&rdev->sched_scan_mtx); | ||
| 193 | wdev_lock(wdev); | 196 | wdev_lock(wdev); |
| 194 | 197 | ||
| 195 | err = 0; | 198 | err = 0; |
| @@ -223,6 +226,7 @@ int cfg80211_mgd_wext_siwessid(struct net_device *dev, | |||
| 223 | err = cfg80211_mgd_wext_connect(rdev, wdev); | 226 | err = cfg80211_mgd_wext_connect(rdev, wdev); |
| 224 | out: | 227 | out: |
| 225 | wdev_unlock(wdev); | 228 | wdev_unlock(wdev); |
| 229 | mutex_unlock(&rdev->sched_scan_mtx); | ||
| 226 | mutex_unlock(&rdev->devlist_mtx); | 230 | mutex_unlock(&rdev->devlist_mtx); |
| 227 | cfg80211_unlock_rdev(rdev); | 231 | cfg80211_unlock_rdev(rdev); |
| 228 | return err; | 232 | return err; |
| @@ -285,6 +289,7 @@ int cfg80211_mgd_wext_siwap(struct net_device *dev, | |||
| 285 | 289 | ||
| 286 | cfg80211_lock_rdev(rdev); | 290 | cfg80211_lock_rdev(rdev); |
| 287 | mutex_lock(&rdev->devlist_mtx); | 291 | mutex_lock(&rdev->devlist_mtx); |
| 292 | mutex_lock(&rdev->sched_scan_mtx); | ||
| 288 | wdev_lock(wdev); | 293 | wdev_lock(wdev); |
| 289 | 294 | ||
| 290 | if (wdev->sme_state != CFG80211_SME_IDLE) { | 295 | if (wdev->sme_state != CFG80211_SME_IDLE) { |
| @@ -313,6 +318,7 @@ int cfg80211_mgd_wext_siwap(struct net_device *dev, | |||
| 313 | err = cfg80211_mgd_wext_connect(rdev, wdev); | 318 | err = cfg80211_mgd_wext_connect(rdev, wdev); |
| 314 | out: | 319 | out: |
| 315 | wdev_unlock(wdev); | 320 | wdev_unlock(wdev); |
| 321 | mutex_unlock(&rdev->sched_scan_mtx); | ||
| 316 | mutex_unlock(&rdev->devlist_mtx); | 322 | mutex_unlock(&rdev->devlist_mtx); |
| 317 | cfg80211_unlock_rdev(rdev); | 323 | cfg80211_unlock_rdev(rdev); |
| 318 | return err; | 324 | return err; |
diff --git a/net/xfrm/xfrm_replay.c b/net/xfrm/xfrm_replay.c index 35754cc8a9e5..8dafe6d3c6e4 100644 --- a/net/xfrm/xfrm_replay.c +++ b/net/xfrm/xfrm_replay.c | |||
| @@ -334,6 +334,70 @@ static void xfrm_replay_notify_bmp(struct xfrm_state *x, int event) | |||
| 334 | x->xflags &= ~XFRM_TIME_DEFER; | 334 | x->xflags &= ~XFRM_TIME_DEFER; |
| 335 | } | 335 | } |
| 336 | 336 | ||
| 337 | static void xfrm_replay_notify_esn(struct xfrm_state *x, int event) | ||
| 338 | { | ||
| 339 | u32 seq_diff, oseq_diff; | ||
| 340 | struct km_event c; | ||
| 341 | struct xfrm_replay_state_esn *replay_esn = x->replay_esn; | ||
| 342 | struct xfrm_replay_state_esn *preplay_esn = x->preplay_esn; | ||
| 343 | |||
| 344 | /* we send notify messages in case | ||
| 345 | * 1. we updated on of the sequence numbers, and the seqno difference | ||
| 346 | * is at least x->replay_maxdiff, in this case we also update the | ||
| 347 | * timeout of our timer function | ||
| 348 | * 2. if x->replay_maxage has elapsed since last update, | ||
| 349 | * and there were changes | ||
| 350 | * | ||
| 351 | * The state structure must be locked! | ||
| 352 | */ | ||
| 353 | |||
| 354 | switch (event) { | ||
| 355 | case XFRM_REPLAY_UPDATE: | ||
| 356 | if (!x->replay_maxdiff) | ||
| 357 | break; | ||
| 358 | |||
| 359 | if (replay_esn->seq_hi == preplay_esn->seq_hi) | ||
| 360 | seq_diff = replay_esn->seq - preplay_esn->seq; | ||
| 361 | else | ||
| 362 | seq_diff = ~preplay_esn->seq + replay_esn->seq + 1; | ||
| 363 | |||
| 364 | if (replay_esn->oseq_hi == preplay_esn->oseq_hi) | ||
| 365 | oseq_diff = replay_esn->oseq - preplay_esn->oseq; | ||
| 366 | else | ||
| 367 | oseq_diff = ~preplay_esn->oseq + replay_esn->oseq + 1; | ||
| 368 | |||
| 369 | if (seq_diff < x->replay_maxdiff && | ||
| 370 | oseq_diff < x->replay_maxdiff) { | ||
| 371 | |||
| 372 | if (x->xflags & XFRM_TIME_DEFER) | ||
| 373 | event = XFRM_REPLAY_TIMEOUT; | ||
| 374 | else | ||
| 375 | return; | ||
| 376 | } | ||
| 377 | |||
| 378 | break; | ||
| 379 | |||
| 380 | case XFRM_REPLAY_TIMEOUT: | ||
| 381 | if (memcmp(x->replay_esn, x->preplay_esn, | ||
| 382 | xfrm_replay_state_esn_len(replay_esn)) == 0) { | ||
| 383 | x->xflags |= XFRM_TIME_DEFER; | ||
| 384 | return; | ||
| 385 | } | ||
| 386 | |||
| 387 | break; | ||
| 388 | } | ||
| 389 | |||
| 390 | memcpy(x->preplay_esn, x->replay_esn, | ||
| 391 | xfrm_replay_state_esn_len(replay_esn)); | ||
| 392 | c.event = XFRM_MSG_NEWAE; | ||
| 393 | c.data.aevent = event; | ||
| 394 | km_state_notify(x, &c); | ||
| 395 | |||
| 396 | if (x->replay_maxage && | ||
| 397 | !mod_timer(&x->rtimer, jiffies + x->replay_maxage)) | ||
| 398 | x->xflags &= ~XFRM_TIME_DEFER; | ||
| 399 | } | ||
| 400 | |||
| 337 | static int xfrm_replay_overflow_esn(struct xfrm_state *x, struct sk_buff *skb) | 401 | static int xfrm_replay_overflow_esn(struct xfrm_state *x, struct sk_buff *skb) |
| 338 | { | 402 | { |
| 339 | int err = 0; | 403 | int err = 0; |
| @@ -510,7 +574,7 @@ static struct xfrm_replay xfrm_replay_esn = { | |||
| 510 | .advance = xfrm_replay_advance_esn, | 574 | .advance = xfrm_replay_advance_esn, |
| 511 | .check = xfrm_replay_check_esn, | 575 | .check = xfrm_replay_check_esn, |
| 512 | .recheck = xfrm_replay_recheck_esn, | 576 | .recheck = xfrm_replay_recheck_esn, |
| 513 | .notify = xfrm_replay_notify_bmp, | 577 | .notify = xfrm_replay_notify_esn, |
| 514 | .overflow = xfrm_replay_overflow_esn, | 578 | .overflow = xfrm_replay_overflow_esn, |
| 515 | }; | 579 | }; |
| 516 | 580 | ||
