aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--arch/mn10300/mm/fault.c33
1 files changed, 25 insertions, 8 deletions
diff --git a/arch/mn10300/mm/fault.c b/arch/mn10300/mm/fault.c
index 90f346f7392d..d48a84fd7fae 100644
--- a/arch/mn10300/mm/fault.c
+++ b/arch/mn10300/mm/fault.c
@@ -123,7 +123,8 @@ asmlinkage void do_page_fault(struct pt_regs *regs, unsigned long fault_code,
123 struct mm_struct *mm; 123 struct mm_struct *mm;
124 unsigned long page; 124 unsigned long page;
125 siginfo_t info; 125 siginfo_t info;
126 int write, fault; 126 int fault;
127 unsigned int flags = FAULT_FLAG_ALLOW_RETRY | FAULT_FLAG_KILLABLE;
127 128
128#ifdef CONFIG_GDBSTUB 129#ifdef CONFIG_GDBSTUB
129 /* handle GDB stub causing a fault */ 130 /* handle GDB stub causing a fault */
@@ -170,6 +171,7 @@ asmlinkage void do_page_fault(struct pt_regs *regs, unsigned long fault_code,
170 if (in_atomic() || !mm) 171 if (in_atomic() || !mm)
171 goto no_context; 172 goto no_context;
172 173
174retry:
173 down_read(&mm->mmap_sem); 175 down_read(&mm->mmap_sem);
174 176
175 vma = find_vma(mm, address); 177 vma = find_vma(mm, address);
@@ -220,7 +222,6 @@ asmlinkage void do_page_fault(struct pt_regs *regs, unsigned long fault_code,
220 */ 222 */
221good_area: 223good_area:
222 info.si_code = SEGV_ACCERR; 224 info.si_code = SEGV_ACCERR;
223 write = 0;
224 switch (fault_code & (MMUFCR_xFC_PGINVAL|MMUFCR_xFC_TYPE)) { 225 switch (fault_code & (MMUFCR_xFC_PGINVAL|MMUFCR_xFC_TYPE)) {
225 default: /* 3: write, present */ 226 default: /* 3: write, present */
226 case MMUFCR_xFC_TYPE_WRITE: 227 case MMUFCR_xFC_TYPE_WRITE:
@@ -232,7 +233,7 @@ good_area:
232 case MMUFCR_xFC_PGINVAL | MMUFCR_xFC_TYPE_WRITE: 233 case MMUFCR_xFC_PGINVAL | MMUFCR_xFC_TYPE_WRITE:
233 if (!(vma->vm_flags & VM_WRITE)) 234 if (!(vma->vm_flags & VM_WRITE))
234 goto bad_area; 235 goto bad_area;
235 write++; 236 flags |= FAULT_FLAG_WRITE;
236 break; 237 break;
237 238
238 /* read from protected page */ 239 /* read from protected page */
@@ -251,7 +252,11 @@ good_area:
251 * make sure we exit gracefully rather than endlessly redo 252 * make sure we exit gracefully rather than endlessly redo
252 * the fault. 253 * the fault.
253 */ 254 */
254 fault = handle_mm_fault(mm, vma, address, write ? FAULT_FLAG_WRITE : 0); 255 fault = handle_mm_fault(mm, vma, address, flags);
256
257 if ((fault & VM_FAULT_RETRY) && fatal_signal_pending(current))
258 return;
259
255 if (unlikely(fault & VM_FAULT_ERROR)) { 260 if (unlikely(fault & VM_FAULT_ERROR)) {
256 if (fault & VM_FAULT_OOM) 261 if (fault & VM_FAULT_OOM)
257 goto out_of_memory; 262 goto out_of_memory;
@@ -259,10 +264,22 @@ good_area:
259 goto do_sigbus; 264 goto do_sigbus;
260 BUG(); 265 BUG();
261 } 266 }
262 if (fault & VM_FAULT_MAJOR) 267 if (flags & FAULT_FLAG_ALLOW_RETRY) {
263 current->maj_flt++; 268 if (fault & VM_FAULT_MAJOR)
264 else 269 current->maj_flt++;
265 current->min_flt++; 270 else
271 current->min_flt++;
272 if (fault & VM_FAULT_RETRY) {
273 flags &= ~FAULT_FLAG_ALLOW_RETRY;
274
275 /* No need to up_read(&mm->mmap_sem) as we would
276 * have already released it in __lock_page_or_retry
277 * in mm/filemap.c.
278 */
279
280 goto retry;
281 }
282 }
266 283
267 up_read(&mm->mmap_sem); 284 up_read(&mm->mmap_sem);
268 return; 285 return;