diff options
Diffstat (limited to 'arch/i386/kernel/time.c')
| -rw-r--r-- | arch/i386/kernel/time.c | 23 |
1 files changed, 19 insertions, 4 deletions
diff --git a/arch/i386/kernel/time.c b/arch/i386/kernel/time.c index 1302e4ab3c4f..86944acfb647 100644 --- a/arch/i386/kernel/time.c +++ b/arch/i386/kernel/time.c | |||
| @@ -130,18 +130,33 @@ static int set_rtc_mmss(unsigned long nowtime) | |||
| 130 | 130 | ||
| 131 | int timer_ack; | 131 | int timer_ack; |
| 132 | 132 | ||
| 133 | #if defined(CONFIG_SMP) && defined(CONFIG_FRAME_POINTER) | ||
| 134 | unsigned long profile_pc(struct pt_regs *regs) | 133 | unsigned long profile_pc(struct pt_regs *regs) |
| 135 | { | 134 | { |
| 136 | unsigned long pc = instruction_pointer(regs); | 135 | unsigned long pc = instruction_pointer(regs); |
| 137 | 136 | ||
| 138 | if (!user_mode_vm(regs) && in_lock_functions(pc)) | 137 | #ifdef CONFIG_SMP |
| 138 | if (!user_mode_vm(regs) && in_lock_functions(pc)) { | ||
| 139 | #ifdef CONFIG_FRAME_POINTER | ||
| 139 | return *(unsigned long *)(regs->ebp + 4); | 140 | return *(unsigned long *)(regs->ebp + 4); |
| 140 | 141 | #else | |
| 142 | unsigned long *sp; | ||
| 143 | if ((regs->xcs & 3) == 0) | ||
| 144 | sp = (unsigned long *)®s->esp; | ||
| 145 | else | ||
| 146 | sp = (unsigned long *)regs->esp; | ||
| 147 | /* Return address is either directly at stack pointer | ||
| 148 | or above a saved eflags. Eflags has bits 22-31 zero, | ||
| 149 | kernel addresses don't. */ | ||
| 150 | if (sp[0] >> 22) | ||
| 151 | return sp[0]; | ||
| 152 | if (sp[1] >> 22) | ||
| 153 | return sp[1]; | ||
| 154 | #endif | ||
| 155 | } | ||
| 156 | #endif | ||
| 141 | return pc; | 157 | return pc; |
| 142 | } | 158 | } |
| 143 | EXPORT_SYMBOL(profile_pc); | 159 | EXPORT_SYMBOL(profile_pc); |
| 144 | #endif | ||
| 145 | 160 | ||
| 146 | /* | 161 | /* |
| 147 | * This is the same as the above, except we _also_ save the current | 162 | * This is the same as the above, except we _also_ save the current |
