Commit a78d4a2a authored by Uladzislau Rezki (Sony)'s avatar Uladzislau Rezki (Sony) Committed by Paul E. McKenney

kvfree_rcu: Refactor kfree_rcu_monitor()

Currently we have three functions which depend on each other.
Two of them are quite tiny and the last one where the most
work is done. All of them are related to queuing RCU batches
to reclaim objects after a GP.

1. kfree_rcu_monitor(). It consist of few lines. It acquires a spin-lock
   and calls kfree_rcu_drain_unlock().

2. kfree_rcu_drain_unlock(). It also consists of few lines of code. It
   calls queue_kfree_rcu_work() to queue the batch.  If this fails,
   it rearms the monitor work to try again later.

3. queue_kfree_rcu_work(). This provides the bulk of the functionality,
   attempting to start a new batch to free objects after a GP.

Since there are no external users of functions [2] and [3], both
can eliminated by moving all logic directly into [1], which both
shrinks and simplifies the code.

Also replace comments which start with "/*" to "//" format to make it
unified across the file.
Signed-off-by: default avatarUladzislau Rezki (Sony) <urezki@gmail.com>
Signed-off-by: default avatarPaul E. McKenney <paulmck@kernel.org>
parent d8628f35
...@@ -3379,29 +3379,26 @@ static void kfree_rcu_work(struct work_struct *work) ...@@ -3379,29 +3379,26 @@ static void kfree_rcu_work(struct work_struct *work)
} }
/* /*
* Schedule the kfree batch RCU work to run in workqueue context after a GP. * This function is invoked after the KFREE_DRAIN_JIFFIES timeout.
*
* This function is invoked by kfree_rcu_monitor() when the KFREE_DRAIN_JIFFIES
* timeout has been reached.
*/ */
static inline bool queue_kfree_rcu_work(struct kfree_rcu_cpu *krcp) static void kfree_rcu_monitor(struct work_struct *work)
{ {
struct kfree_rcu_cpu_work *krwp; struct kfree_rcu_cpu *krcp = container_of(work,
bool repeat = false; struct kfree_rcu_cpu, monitor_work.work);
unsigned long flags;
int i, j; int i, j;
lockdep_assert_held(&krcp->lock); raw_spin_lock_irqsave(&krcp->lock, flags);
// Attempt to start a new batch.
for (i = 0; i < KFREE_N_BATCHES; i++) { for (i = 0; i < KFREE_N_BATCHES; i++) {
krwp = &(krcp->krw_arr[i]); struct kfree_rcu_cpu_work *krwp = &(krcp->krw_arr[i]);
/* // Try to detach bkvhead or head and attach it over any
* Try to detach bkvhead or head and attach it over any // available corresponding free channel. It can be that
* available corresponding free channel. It can be that // a previous RCU batch is in progress, it means that
* a previous RCU batch is in progress, it means that // immediately to queue another one is not possible so
* immediately to queue another one is not possible so // in that case the monitor work is rearmed.
* return false to tell caller to retry.
*/
if ((krcp->bkvhead[0] && !krwp->bkvhead_free[0]) || if ((krcp->bkvhead[0] && !krwp->bkvhead_free[0]) ||
(krcp->bkvhead[1] && !krwp->bkvhead_free[1]) || (krcp->bkvhead[1] && !krwp->bkvhead_free[1]) ||
(krcp->head && !krwp->head_free)) { (krcp->head && !krwp->head_free)) {
...@@ -3423,57 +3420,28 @@ static inline bool queue_kfree_rcu_work(struct kfree_rcu_cpu *krcp) ...@@ -3423,57 +3420,28 @@ static inline bool queue_kfree_rcu_work(struct kfree_rcu_cpu *krcp)
WRITE_ONCE(krcp->count, 0); WRITE_ONCE(krcp->count, 0);
/* // One work is per one batch, so there are three
* One work is per one batch, so there are three // "free channels", the batch can handle. It can
* "free channels", the batch can handle. It can // be that the work is in the pending state when
* be that the work is in the pending state when // channels have been detached following by each
* channels have been detached following by each // other.
* other.
*/
queue_rcu_work(system_wq, &krwp->rcu_work); queue_rcu_work(system_wq, &krwp->rcu_work);
} }
// Repeat if any "free" corresponding channel is still busy.
if (krcp->bkvhead[0] || krcp->bkvhead[1] || krcp->head)
repeat = true;
} }
return !repeat; // If there is nothing to detach, it means that our job is
} // successfully done here. In case of having at least one
// of the channels that is still busy we should rearm the
static inline void kfree_rcu_drain_unlock(struct kfree_rcu_cpu *krcp, // work to repeat an attempt. Because previous batches are
unsigned long flags) // still in progress.
{ if (!krcp->bkvhead[0] && !krcp->bkvhead[1] && !krcp->head)
// Attempt to start a new batch.
if (queue_kfree_rcu_work(krcp)) {
// Success! Our job is done here.
krcp->monitor_todo = false; krcp->monitor_todo = false;
raw_spin_unlock_irqrestore(&krcp->lock, flags); else
return; schedule_delayed_work(&krcp->monitor_work, KFREE_DRAIN_JIFFIES);
}
// Previous RCU batch still in progress, try again later.
schedule_delayed_work(&krcp->monitor_work, KFREE_DRAIN_JIFFIES);
raw_spin_unlock_irqrestore(&krcp->lock, flags); raw_spin_unlock_irqrestore(&krcp->lock, flags);
} }
/*
* This function is invoked after the KFREE_DRAIN_JIFFIES timeout.
* It invokes kfree_rcu_drain_unlock() to attempt to start another batch.
*/
static void kfree_rcu_monitor(struct work_struct *work)
{
unsigned long flags;
struct kfree_rcu_cpu *krcp = container_of(work, struct kfree_rcu_cpu,
monitor_work.work);
raw_spin_lock_irqsave(&krcp->lock, flags);
if (krcp->monitor_todo)
kfree_rcu_drain_unlock(krcp, flags);
else
raw_spin_unlock_irqrestore(&krcp->lock, flags);
}
static enum hrtimer_restart static enum hrtimer_restart
schedule_page_work_fn(struct hrtimer *t) schedule_page_work_fn(struct hrtimer *t)
{ {
......
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