1. 18 7月, 2015 5 次提交
    • P
      rcu: Make expedited GP CPU stoppage asynchronous · 3a6d7c64
      Peter Zijlstra 提交于
      Sequentially stopping the CPUs slows down expedited grace periods by
      at least a factor of two, based on rcutorture's grace-period-per-second
      rate.  This is a conservative measure because rcutorture uses unusually
      long RCU read-side critical sections and because rcutorture periodically
      quiesces the system in order to test RCU's ability to ramp down to and
      up from the idle state.  This commit therefore replaces the stop_one_cpu()
      with stop_one_cpu_nowait(), using an atomic-counter scheme to determine
      when all CPUs have passed through the stopped state.
      Signed-off-by: NPeter Zijlstra <peterz@infradead.org>
      Signed-off-by: NPaul E. McKenney <paulmck@linux.vnet.ibm.com>
      3a6d7c64
    • P
      rcu: Get rid of synchronize_sched_expedited()'s polling loop · 385b73c0
      Paul E. McKenney 提交于
      This commit gets rid of synchronize_sched_expedited()'s mutex_trylock()
      polling loop in favor of a funnel-locking scheme based on the rcu_node
      tree.  The work-done check is done at each level of the tree, allowing
      high-contention situations to be resolved quickly with reasonable levels
      of mutex contention.
      Signed-off-by: NPaul E. McKenney <paulmck@linux.vnet.ibm.com>
      385b73c0
    • P
      rcu: Rework synchronize_sched_expedited() counter handling · d6ada2cf
      Paul E. McKenney 提交于
      Now that synchronize_sched_expedited() have a mutex, it can use simpler
      work-already-done detection scheme.  This commit simplifies this scheme
      by using something similar to the sequence-locking counter scheme.
      A counter is incremented before and after each grace period, so that
      the counter is odd in the midst of the grace period and even otherwise.
      So if the counter has advanced to the second even number that is
      greater than or equal to the snapshot, the required grace period has
      already happened.
      Signed-off-by: NPaul E. McKenney <paulmck@linux.vnet.ibm.com>
      d6ada2cf
    • P
      rcu: Switch synchronize_sched_expedited() to stop_one_cpu() · c190c3b1
      Peter Zijlstra 提交于
      The synchronize_sched_expedited() currently invokes try_stop_cpus(),
      which schedules the stopper kthreads on each online non-idle CPU,
      and waits until all those kthreads are running before letting any
      of them stop.  This is disastrous for real-time workloads, which
      get hit with a preemption that is as long as the longest scheduling
      latency on any CPU, including any non-realtime housekeeping CPUs.
      This commit therefore switches to using stop_one_cpu() on each CPU
      in turn.  This avoids inflicting the worst-case scheduling latency
      on the worst-case CPU onto all other CPUs, and also simplifies the
      code a little bit.
      
      Follow-up commits will simplify the counter-snapshotting algorithm
      and convert a number of the counters that are now protected by the
      new ->expedited_mutex to non-atomic.
      Signed-off-by: NPeter Zijlstra <peterz@infradead.org>
      [ paulmck: Kept stop_one_cpu(), dropped disabling of "guardrails". ]
      Signed-off-by: NPaul E. McKenney <paulmck@linux.vnet.ibm.com>
      c190c3b1
    • P
      rcu: Reset rcu_fanout_leaf if out of bounds · 13bd6494
      Paul E. McKenney 提交于
      Currently if the rcu_fanout_leaf boot parameter is out of bounds (that
      is, less than RCU_FANOUT_LEAF or greater than the number of bits in an
      unsigned long), a warning is issued and execution continues with the
      out-of-bounds value.  This can result in all manner of failures, so this
      patch resets rcu_fanout_leaf to RCU_FANOUT_LEAF when an out-of-bounds
      condition is detected.
      Signed-off-by: NPaul E. McKenney <paulmck@linux.vnet.ibm.com>
      13bd6494
  2. 16 7月, 2015 8 次提交
  3. 28 5月, 2015 21 次提交
  4. 14 5月, 2015 1 次提交
  5. 15 4月, 2015 1 次提交
    • P
      rcu: Control grace-period delays directly from value · 8d7dc928
      Paul E. McKenney 提交于
      In a misguided attempt to avoid an #ifdef, the use of the
      gp_init_delay module parameter was conditioned on the corresponding
      RCU_TORTURE_TEST_SLOW_INIT Kconfig variable, using IS_ENABLED() at
      the point of use in the code.  This meant that the compiler always saw
      the delay, which meant that RCU_TORTURE_TEST_SLOW_INIT_DELAY had to be
      unconditionally defined.  This in turn caused "make oldconfig" to ask
      pointless questions about the value of RCU_TORTURE_TEST_SLOW_INIT_DELAY
      in cases where it was not even used.
      
      This commit avoids these pointless questions by defining gp_init_delay
      under #ifdef.  In one branch, gp_init_delay is initialized to
      RCU_TORTURE_TEST_SLOW_INIT_DELAY and is also a module parameter (thus
      allowing boot-time modification), and in the other branch gp_init_delay
      is a const variable initialized by default to zero.
      
      This approach also simplifies the code at the delay point by eliminating
      the IS_DEFINED().  Because gp_init_delay is constant zero in the no-delay
      case intended for production use, the "gp_init_delay > 0" check causes
      the delay to become dead code, as desired in this case.  In addition,
      this commit replaces magic constant "10" with the preprocessor variable
      PER_RCU_NODE_PERIOD, which controls the number of grace periods that
      are allowed to elapse at full speed before a delay is inserted.
      
      Reported-by: Linus Torvalds <torvalds@linux-foundation.org> Signed-off-by:
      Paul E. McKenney <paulmck@linux.vnet.ibm.com>
      8d7dc928
  6. 20 3月, 2015 2 次提交
    • P
      rcu: Associate quiescent-state reports with grace period · 654e9533
      Paul E. McKenney 提交于
      As noted in earlier commit logs, CPU hotplug operations running
      concurrently with grace-period initialization can result in a given
      leaf rcu_node structure having all CPUs offline and no blocked readers,
      but with this rcu_node structure nevertheless blocking the current
      grace period.  Therefore, the quiescent-state forcing code now checks
      for this situation and repairs it.
      
      Unfortunately, this checking can result in false positives, for example,
      when the last task has just removed itself from this leaf rcu_node
      structure, but has not yet started clearing the ->qsmask bits further
      up the structure.  This means that the grace-period kthread (which
      forces quiescent states) and some other task might be attempting to
      concurrently clear these ->qsmask bits.  This is usually not a problem:
      One of these tasks will be the first to acquire the upper-level rcu_node
      structure's lock and with therefore clear the bit, and the other task,
      seeing the bit already cleared, will stop trying to clear bits.
      
      Sadly, this means that the following unusual sequence of events -can-
      result in a problem:
      
      1.	The grace-period kthread wins, and clears the ->qsmask bits.
      
      2.	This is the last thing blocking the current grace period, so
      	that the grace-period kthread clears ->qsmask bits all the way
      	to the root and finds that the root ->qsmask field is now zero.
      
      3.	Another grace period is required, so that the grace period kthread
      	initializes it, including setting all the needed qsmask bits.
      
      4.	The leaf rcu_node structure (the one that started this whole
      	mess) is blocking this new grace period, either because it
      	has at least one online CPU or because there is at least one
      	task that had blocked within an RCU read-side critical section
      	while running on one of this leaf rcu_node structure's CPUs.
      	(And yes, that CPU might well have gone offline before the
      	grace period in step (3) above started, which can mean that
      	there is a task on the leaf rcu_node structure's ->blkd_tasks
      	list, but ->qsmask equal to zero.)
      
      5.	The other kthread didn't get around to trying to clear the upper
      	level ->qsmask bits until all the above had happened.  This means
      	that it now sees bits set in the upper-level ->qsmask field, so it
      	proceeds to clear them.  Too bad that it is doing so on behalf of
      	a quiescent state that does not apply to the current grace period!
      
      This sequence of events can result in the new grace period being too
      short.  It can also result in the new grace period ending before the
      leaf rcu_node structure's ->qsmask bits have been cleared, which will
      result in splats during initialization of the next grace period.  In
      addition, it can result in tasks blocking the new grace period still
      being queued at the start of the next grace period, which will result
      in other splats.  Sasha's testing turned up another of these splats,
      as did rcutorture testing.  (And yes, rcutorture is being adjusted to
      make these splats show up more quickly.  Which probably is having the
      undesirable side effect of making other problems show up less quickly.
      Can't have everything!)
      Reported-by: NSasha Levin <sasha.levin@oracle.com>
      Signed-off-by: NPaul E. McKenney <paulmck@linux.vnet.ibm.com>
      Cc: <stable@vger.kernel.org> # 4.0.x
      Tested-by: NSasha Levin <sasha.levin@oracle.com>
      654e9533
    • P
      rcu: Yet another fix for preemption and CPU hotplug · a77da14c
      Paul E. McKenney 提交于
      As noted earlier, the following sequence of events can occur when
      running PREEMPT_RCU and HOTPLUG_CPU on a system with a multi-level
      rcu_node combining tree:
      
      1.	A group of tasks block on CPUs corresponding to a given leaf
      	rcu_node structure while within RCU read-side critical sections.
      2.	All CPUs corrsponding to that rcu_node structure go offline.
      3.	The next grace period starts, but because there are still tasks
      	blocked, the upper-level bits corresponding to this leaf rcu_node
      	structure remain set.
      4.	All the tasks exit their RCU read-side critical sections and
      	remove themselves from the leaf rcu_node structure's list,
      	leaving it empty.
      5.	But because there now is code to check for this condition at
      	force-quiescent-state time, the upper bits are cleared and the
      	grace period completes.
      
      However, there is another complication that can occur following step 4 above:
      
      4a.	The grace period starts, and the leaf rcu_node structure's
      	gp_tasks pointer is set to NULL because there are no tasks
      	blocked on this structure.
      4b.	One of the CPUs corresponding to the leaf rcu_node structure
      	comes back online.
      4b.	An endless stream of tasks are preempted within RCU read-side
      	critical sections on this CPU, such that the ->blkd_tasks
      	list is always non-empty.
      
      The grace period will never end.
      
      This commit therefore makes the force-quiescent-state processing check only
      for absence of tasks blocking the current grace period rather than absence
      of tasks altogether.  This will cause a quiescent state to be reported if
      the current leaf rcu_node structure is not blocking the current grace period
      and its parent thinks that it is, regardless of how RCU managed to get
      itself into this state.
      Signed-off-by: NPaul E. McKenney <paulmck@linux.vnet.ibm.com>
      Cc: <stable@vger.kernel.org> # 4.0.x
      Tested-by: NSasha Levin <sasha.levin@oracle.com>
      a77da14c
  7. 13 3月, 2015 2 次提交
    • P
      rcu: Add diagnostics to grace-period cleanup · 5c60d25f
      Paul E. McKenney 提交于
      At grace-period initialization time, RCU checks that all quiescent
      states were really reported for the previous grace period.  Now that
      grace-period cleanup has been split out of grace-period initialization,
      this commit also performs those checks at grace-period cleanup time.
      Signed-off-by: NPaul E. McKenney <paulmck@linux.vnet.ibm.com>
      5c60d25f
    • P
      rcu: Handle outgoing CPUs on exit from idle loop · 88428cc5
      Paul E. McKenney 提交于
      This commit informs RCU of an outgoing CPU just before that CPU invokes
      arch_cpu_idle_dead() during its last pass through the idle loop (via a
      new CPU_DYING_IDLE notifier value).  This change means that RCU need not
      deal with outgoing CPUs passing through the scheduler after informing
      RCU that they are no longer online.  Note that removing the CPU from
      the rcu_node ->qsmaskinit bit masks is done at CPU_DYING_IDLE time,
      and orphaning callbacks is still done at CPU_DEAD time, the reason being
      that at CPU_DEAD time we have another CPU that can adopt them.
      Signed-off-by: NPaul E. McKenney <paulmck@linux.vnet.ibm.com>
      88428cc5