Commit 17b3fed1 authored by Manfred Spraul's avatar Manfred Spraul Committed by Linus Torvalds

[PATCH] rcu: simplify quiescent state detection

Based on an initial patch from Oleg Nesterov <oleg@tv-sign.ru>

rcu_data.last_qsctr is not needed.  Actually, not even a counter is needed,
just a flag that indicates that there was a quiescent state.
Signed-Off-By: default avatarManfred Spraul <manfred@colorfullife.com>
Signed-off-by: default avatarAndrew Morton <akpm@osdl.org>
Signed-off-by: default avatarLinus Torvalds <torvalds@osdl.org>
parent 2f803905
...@@ -87,9 +87,7 @@ static inline int rcu_batch_after(long a, long b) ...@@ -87,9 +87,7 @@ static inline int rcu_batch_after(long a, long b)
struct rcu_data { struct rcu_data {
/* 1) quiescent state handling : */ /* 1) quiescent state handling : */
long quiescbatch; /* Batch # for grace period */ long quiescbatch; /* Batch # for grace period */
long qsctr; /* User-mode/idle loop etc. */ int passed_quiesc; /* User-mode/idle loop etc. */
long last_qsctr; /* value of qsctr at beginning */
/* of rcu grace period */
int qs_pending; /* core waits for quiesc state */ int qs_pending; /* core waits for quiesc state */
/* 2) batch handling */ /* 2) batch handling */
...@@ -109,17 +107,20 @@ extern struct rcu_ctrlblk rcu_ctrlblk; ...@@ -109,17 +107,20 @@ extern struct rcu_ctrlblk rcu_ctrlblk;
extern struct rcu_ctrlblk rcu_bh_ctrlblk; extern struct rcu_ctrlblk rcu_bh_ctrlblk;
/* /*
* Increment the quiscent state counter. * Increment the quiescent state counter.
* The counter is a bit degenerated: We do not need to know
* how many quiescent states passed, just if there was at least
* one since the start of the grace period. Thus just a flag.
*/ */
static inline void rcu_qsctr_inc(int cpu) static inline void rcu_qsctr_inc(int cpu)
{ {
struct rcu_data *rdp = &per_cpu(rcu_data, cpu); struct rcu_data *rdp = &per_cpu(rcu_data, cpu);
rdp->qsctr++; rdp->passed_quiesc = 1;
} }
static inline void rcu_bh_qsctr_inc(int cpu) static inline void rcu_bh_qsctr_inc(int cpu)
{ {
struct rcu_data *rdp = &per_cpu(rcu_bh_data, cpu); struct rcu_data *rdp = &per_cpu(rcu_bh_data, cpu);
rdp->qsctr++; rdp->passed_quiesc = 1;
} }
static inline int __rcu_pending(struct rcu_ctrlblk *rcp, static inline int __rcu_pending(struct rcu_ctrlblk *rcp,
......
...@@ -219,9 +219,9 @@ static void rcu_check_quiescent_state(struct rcu_ctrlblk *rcp, ...@@ -219,9 +219,9 @@ static void rcu_check_quiescent_state(struct rcu_ctrlblk *rcp,
struct rcu_state *rsp, struct rcu_data *rdp) struct rcu_state *rsp, struct rcu_data *rdp)
{ {
if (rdp->quiescbatch != rcp->cur) { if (rdp->quiescbatch != rcp->cur) {
/* new grace period: record qsctr value. */ /* start new grace period: */
rdp->qs_pending = 1; rdp->qs_pending = 1;
rdp->last_qsctr = rdp->qsctr; rdp->passed_quiesc = 0;
rdp->quiescbatch = rcp->cur; rdp->quiescbatch = rcp->cur;
return; return;
} }
...@@ -234,11 +234,10 @@ static void rcu_check_quiescent_state(struct rcu_ctrlblk *rcp, ...@@ -234,11 +234,10 @@ static void rcu_check_quiescent_state(struct rcu_ctrlblk *rcp,
return; return;
/* /*
* Races with local timer interrupt - in the worst case * Was there a quiescent state since the beginning of the grace
* we may miss one quiescent state of that CPU. That is * period? If no, then exit and wait for the next call.
* tolerable. So no need to disable interrupts.
*/ */
if (rdp->qsctr == rdp->last_qsctr) if (!rdp->passed_quiesc)
return; return;
rdp->qs_pending = 0; rdp->qs_pending = 0;
......
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