aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorTom Herbert <tom@herbertland.com>2015-08-05 12:39:27 -0400
committerDavid S. Miller <davem@davemloft.net>2015-08-07 18:56:56 -0400
commit10e4ea75149d11883a9e04c3b32ee1d7600d481e (patch)
tree4cf760912a8c9e3447619d6cdaeeac0835e9eaae
parente05176a3283822bd32a1f3d929ce2050232299a8 (diff)
net: Fix race condition in store_rps_map
There is a race condition in store_rps_map that allows jump label count in rps_needed to go below zero. This can happen when concurrently attempting to set and a clear map. Scenario: 1. rps_needed count is zero 2. New map is assigned by setting thread, but rps_needed count _not_ yet incremented (rps_needed count still zero) 2. Map is cleared by second thread, old_map set to that just assigned 3. Second thread performs static_key_slow_dec, rps_needed count now goes negative Fix is to increment or decrement rps_needed under the spinlock. Signed-off-by: Tom Herbert <tom@herbertland.com> Signed-off-by: David S. Miller <davem@davemloft.net>
-rw-r--r--net/core/net-sysfs.c11
1 files changed, 7 insertions, 4 deletions
diff --git a/net/core/net-sysfs.c b/net/core/net-sysfs.c
index 194c1d03b2b3..39ec6949c1e6 100644
--- a/net/core/net-sysfs.c
+++ b/net/core/net-sysfs.c
@@ -726,14 +726,17 @@ static ssize_t store_rps_map(struct netdev_rx_queue *queue,
726 old_map = rcu_dereference_protected(queue->rps_map, 726 old_map = rcu_dereference_protected(queue->rps_map,
727 lockdep_is_held(&rps_map_lock)); 727 lockdep_is_held(&rps_map_lock));
728 rcu_assign_pointer(queue->rps_map, map); 728 rcu_assign_pointer(queue->rps_map, map);
729 spin_unlock(&rps_map_lock);
730 729
731 if (map) 730 if (map)
732 static_key_slow_inc(&rps_needed); 731 static_key_slow_inc(&rps_needed);
733 if (old_map) { 732 if (old_map)
734 kfree_rcu(old_map, rcu);
735 static_key_slow_dec(&rps_needed); 733 static_key_slow_dec(&rps_needed);
736 } 734
735 spin_unlock(&rps_map_lock);
736
737 if (old_map)
738 kfree_rcu(old_map, rcu);
739
737 free_cpumask_var(mask); 740 free_cpumask_var(mask);
738 return len; 741 return len;
739} 742}