diff options
author | Li Zhong <zhong@linux.vnet.ibm.com> | 2011-12-18 11:03:04 -0500 |
---|---|---|
committer | Benjamin Herrenschmidt <benh@kernel.crashing.org> | 2012-01-02 20:09:27 -0500 |
commit | e4f387d8db3ba3c2dae4d8bdfe7bb5f4fe1bcb0d (patch) | |
tree | 148f0524d6f8413dbc5ed09e9bf056e2592649e6 | |
parent | 1bb0b7d21584b3f878e2bc880db62351ddee5185 (diff) |
powerpc: Fix unpaired probe_hcall_entry and probe_hcall_exit
Unpaired calling of probe_hcall_entry and probe_hcall_exit might happen
as following, which could cause incorrect preempt count.
__trace_hcall_entry => trace_hcall_entry -> probe_hcall_entry =>
get_cpu_var => preempt_disable
__trace_hcall_exit => trace_hcall_exit -> probe_hcall_exit =>
put_cpu_var => preempt_enable
where:
A => B and A -> B means A calls B, but
=> means A will call B through function name, and B will definitely be
called.
-> means A will call B through function pointer, so B might not be
called if the function pointer is not set.
So error happens when only one of probe_hcall_entry and probe_hcall_exit
get called during a hcall.
This patch tries to move the preempt count operations from
probe_hcall_entry and probe_hcall_exit to its callers.
Reported-by: Paul E. McKenney <paulmck@linux.vnet.ibm.com>
Signed-off-by: Li Zhong <zhong@linux.vnet.ibm.com>
Tested-by: Paul E. McKenney <paulmck@linux.vnet.ibm.com>
CC: stable@kernel.org [v2.6.32+]
Signed-off-by: Benjamin Herrenschmidt <benh@kernel.crashing.org>
-rw-r--r-- | arch/powerpc/platforms/pseries/hvCall_inst.c | 4 | ||||
-rw-r--r-- | arch/powerpc/platforms/pseries/lpar.c | 2 |
2 files changed, 3 insertions, 3 deletions
diff --git a/arch/powerpc/platforms/pseries/hvCall_inst.c b/arch/powerpc/platforms/pseries/hvCall_inst.c index f106662f4381..c9311cfdfcac 100644 --- a/arch/powerpc/platforms/pseries/hvCall_inst.c +++ b/arch/powerpc/platforms/pseries/hvCall_inst.c | |||
@@ -109,7 +109,7 @@ static void probe_hcall_entry(void *ignored, unsigned long opcode, unsigned long | |||
109 | if (opcode > MAX_HCALL_OPCODE) | 109 | if (opcode > MAX_HCALL_OPCODE) |
110 | return; | 110 | return; |
111 | 111 | ||
112 | h = &get_cpu_var(hcall_stats)[opcode / 4]; | 112 | h = &__get_cpu_var(hcall_stats)[opcode / 4]; |
113 | h->tb_start = mftb(); | 113 | h->tb_start = mftb(); |
114 | h->purr_start = mfspr(SPRN_PURR); | 114 | h->purr_start = mfspr(SPRN_PURR); |
115 | } | 115 | } |
@@ -126,8 +126,6 @@ static void probe_hcall_exit(void *ignored, unsigned long opcode, unsigned long | |||
126 | h->num_calls++; | 126 | h->num_calls++; |
127 | h->tb_total += mftb() - h->tb_start; | 127 | h->tb_total += mftb() - h->tb_start; |
128 | h->purr_total += mfspr(SPRN_PURR) - h->purr_start; | 128 | h->purr_total += mfspr(SPRN_PURR) - h->purr_start; |
129 | |||
130 | put_cpu_var(hcall_stats); | ||
131 | } | 129 | } |
132 | 130 | ||
133 | static int __init hcall_inst_init(void) | 131 | static int __init hcall_inst_init(void) |
diff --git a/arch/powerpc/platforms/pseries/lpar.c b/arch/powerpc/platforms/pseries/lpar.c index 27a49508b410..dc36ea6c7727 100644 --- a/arch/powerpc/platforms/pseries/lpar.c +++ b/arch/powerpc/platforms/pseries/lpar.c | |||
@@ -554,6 +554,7 @@ void __trace_hcall_entry(unsigned long opcode, unsigned long *args) | |||
554 | goto out; | 554 | goto out; |
555 | 555 | ||
556 | (*depth)++; | 556 | (*depth)++; |
557 | preempt_disable(); | ||
557 | trace_hcall_entry(opcode, args); | 558 | trace_hcall_entry(opcode, args); |
558 | (*depth)--; | 559 | (*depth)--; |
559 | 560 | ||
@@ -576,6 +577,7 @@ void __trace_hcall_exit(long opcode, unsigned long retval, | |||
576 | 577 | ||
577 | (*depth)++; | 578 | (*depth)++; |
578 | trace_hcall_exit(opcode, retval, retbuf); | 579 | trace_hcall_exit(opcode, retval, retbuf); |
580 | preempt_enable(); | ||
579 | (*depth)--; | 581 | (*depth)--; |
580 | 582 | ||
581 | out: | 583 | out: |