aboutsummaryrefslogtreecommitdiffstats
path: root/arch/mn10300/mm
diff options
context:
space:
mode:
authorKautuk Consul <consul.kautuk@gmail.com>2012-12-12 10:36:39 -0500
committerDavid Howells <dhowells@redhat.com>2012-12-12 10:46:15 -0500
commit3d7b6a674e266999a90ff1e21c6c839e30200c51 (patch)
tree5c89370be6ffe8e328e5f28cf2877ce7580a37b3 /arch/mn10300/mm
parent83c2dc15ce824450e7044b9f90cd529c25747ae0 (diff)
mn10300/mm/fault.c: Port OOM changes to do_page_fault
Commit d065bd810b6deb67d4897a14bfe21f8eb526ba99 (mm: retry page fault when blocking on disk transfer) and commit 37b23e0525d393d48a7d59f870b3bc061a30ccdb (x86,mm: make pagefault killable) The above commits introduced changes into the x86 pagefault handler for making the page fault handler retryable as well as killable. These changes reduce the mmap_sem hold time, which is crucial during OOM killer invocation. Port these changes to mn10300. Signed-off-by: Kautuk Consul <consul.kautuk@gmail.com> Signed-off-by: David Howells <dhowells@redhat.com>
Diffstat (limited to 'arch/mn10300/mm')
-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;