diff options
-rw-r--r-- | arch/parisc/kernel/unwind.c | 43 | ||||
-rw-r--r-- | drivers/parisc/led.c | 6 | ||||
-rw-r--r-- | include/asm-parisc/system.h | 1 | ||||
-rw-r--r-- | mm/mmap.c | 9 |
4 files changed, 43 insertions, 16 deletions
diff --git a/arch/parisc/kernel/unwind.c b/arch/parisc/kernel/unwind.c index e70f57e27643..322167737de7 100644 --- a/arch/parisc/kernel/unwind.c +++ b/arch/parisc/kernel/unwind.c | |||
@@ -16,6 +16,8 @@ | |||
16 | 16 | ||
17 | #include <asm/uaccess.h> | 17 | #include <asm/uaccess.h> |
18 | #include <asm/assembly.h> | 18 | #include <asm/assembly.h> |
19 | #include <asm/asm-offsets.h> | ||
20 | #include <asm/ptrace.h> | ||
19 | 21 | ||
20 | #include <asm/unwind.h> | 22 | #include <asm/unwind.h> |
21 | 23 | ||
@@ -26,6 +28,8 @@ | |||
26 | #define dbg(x...) | 28 | #define dbg(x...) |
27 | #endif | 29 | #endif |
28 | 30 | ||
31 | #define KERNEL_START (KERNEL_BINARY_TEXT_START - 0x1000) | ||
32 | |||
29 | extern struct unwind_table_entry __start___unwind[]; | 33 | extern struct unwind_table_entry __start___unwind[]; |
30 | extern struct unwind_table_entry __stop___unwind[]; | 34 | extern struct unwind_table_entry __stop___unwind[]; |
31 | 35 | ||
@@ -197,6 +201,29 @@ static int unwind_init(void) | |||
197 | return 0; | 201 | return 0; |
198 | } | 202 | } |
199 | 203 | ||
204 | #ifdef CONFIG_64BIT | ||
205 | #define get_func_addr(fptr) fptr[2] | ||
206 | #else | ||
207 | #define get_func_addr(fptr) fptr[0] | ||
208 | #endif | ||
209 | |||
210 | static int unwind_special(struct unwind_frame_info *info, unsigned long pc, int frame_size) | ||
211 | { | ||
212 | void handle_interruption(int, struct pt_regs *); | ||
213 | static unsigned long *hi = (unsigned long)&handle_interruption; | ||
214 | |||
215 | if (pc == get_func_addr(hi)) { | ||
216 | struct pt_regs *regs = (struct pt_regs *)(info->sp - frame_size - PT_SZ_ALGN); | ||
217 | dbg("Unwinding through handle_interruption()\n"); | ||
218 | info->prev_sp = regs->gr[30]; | ||
219 | info->prev_ip = regs->iaoq[0]; | ||
220 | |||
221 | return 1; | ||
222 | } | ||
223 | |||
224 | return 0; | ||
225 | } | ||
226 | |||
200 | static void unwind_frame_regs(struct unwind_frame_info *info) | 227 | static void unwind_frame_regs(struct unwind_frame_info *info) |
201 | { | 228 | { |
202 | const struct unwind_table_entry *e; | 229 | const struct unwind_table_entry *e; |
@@ -310,13 +337,15 @@ static void unwind_frame_regs(struct unwind_frame_info *info) | |||
310 | } | 337 | } |
311 | } | 338 | } |
312 | 339 | ||
313 | info->prev_sp = info->sp - frame_size; | 340 | if (!unwind_special(info, e->region_start, frame_size)) { |
314 | if (e->Millicode) | 341 | info->prev_sp = info->sp - frame_size; |
315 | info->rp = info->r31; | 342 | if (e->Millicode) |
316 | else if (rpoffset) | 343 | info->rp = info->r31; |
317 | info->rp = *(unsigned long *)(info->prev_sp - rpoffset); | 344 | else if (rpoffset) |
318 | info->prev_ip = info->rp; | 345 | info->rp = *(unsigned long *)(info->prev_sp - rpoffset); |
319 | info->rp = 0; | 346 | info->prev_ip = info->rp; |
347 | info->rp = 0; | ||
348 | } | ||
320 | 349 | ||
321 | dbg("analyzing func @ %lx, setting prev_sp=%lx " | 350 | dbg("analyzing func @ %lx, setting prev_sp=%lx " |
322 | "prev_ip=%lx npc=%lx\n", info->ip, info->prev_sp, | 351 | "prev_ip=%lx npc=%lx\n", info->ip, info->prev_sp, |
diff --git a/drivers/parisc/led.c b/drivers/parisc/led.c index 98be2880757d..e5d7ed92d6f7 100644 --- a/drivers/parisc/led.c +++ b/drivers/parisc/led.c | |||
@@ -195,12 +195,6 @@ static int led_proc_write(struct file *file, const char *buf, | |||
195 | 195 | ||
196 | cur = lbuf; | 196 | cur = lbuf; |
197 | 197 | ||
198 | /* skip initial spaces */ | ||
199 | while (*cur && isspace(*cur)) | ||
200 | { | ||
201 | cur++; | ||
202 | } | ||
203 | |||
204 | switch ((long)data) | 198 | switch ((long)data) |
205 | { | 199 | { |
206 | case LED_NOLCD: | 200 | case LED_NOLCD: |
diff --git a/include/asm-parisc/system.h b/include/asm-parisc/system.h index 7e9afa720d43..21fbfc5afd02 100644 --- a/include/asm-parisc/system.h +++ b/include/asm-parisc/system.h | |||
@@ -188,7 +188,6 @@ static inline void set_eiem(unsigned long val) | |||
188 | # define __lock_aligned __attribute__((__section__(".data.lock_aligned"))) | 188 | # define __lock_aligned __attribute__((__section__(".data.lock_aligned"))) |
189 | #endif | 189 | #endif |
190 | 190 | ||
191 | #define KERNEL_START (0x10100000 - 0x1000) | ||
192 | #define arch_align_stack(x) (x) | 191 | #define arch_align_stack(x) (x) |
193 | 192 | ||
194 | #endif | 193 | #endif |
@@ -1536,9 +1536,14 @@ int expand_upwards(struct vm_area_struct *vma, unsigned long address) | |||
1536 | * vma->vm_start/vm_end cannot change under us because the caller | 1536 | * vma->vm_start/vm_end cannot change under us because the caller |
1537 | * is required to hold the mmap_sem in read mode. We need the | 1537 | * is required to hold the mmap_sem in read mode. We need the |
1538 | * anon_vma lock to serialize against concurrent expand_stacks. | 1538 | * anon_vma lock to serialize against concurrent expand_stacks. |
1539 | * Also guard against wrapping around to address 0. | ||
1539 | */ | 1540 | */ |
1540 | address += 4 + PAGE_SIZE - 1; | 1541 | if (address < PAGE_ALIGN(address+4)) |
1541 | address &= PAGE_MASK; | 1542 | address = PAGE_ALIGN(address+4); |
1543 | else { | ||
1544 | anon_vma_unlock(vma); | ||
1545 | return -ENOMEM; | ||
1546 | } | ||
1542 | error = 0; | 1547 | error = 0; |
1543 | 1548 | ||
1544 | /* Somebody else might have raced and expanded it already */ | 1549 | /* Somebody else might have raced and expanded it already */ |