Blame view

arch/arm/mm/fault.c 13.9 KB
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1
2
3
4
5
6
7
8
9
10
  /*
   *  linux/arch/arm/mm/fault.c
   *
   *  Copyright (C) 1995  Linus Torvalds
   *  Modifications for ARM processor (c) 1995-2004 Russell King
   *
   * This program is free software; you can redistribute it and/or modify
   * it under the terms of the GNU General Public License version 2 as
   * published by the Free Software Foundation.
   */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
11
12
  #include <linux/module.h>
  #include <linux/signal.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
13
  #include <linux/mm.h>
67306da61   Russell King   [ARM] Ensure linu...
14
  #include <linux/hardirq.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
15
  #include <linux/init.h>
25ce1dd71   Nicolas Pitre   ARM kprobes: add ...
16
  #include <linux/kprobes.h>
33fa9b132   Russell King   [ARM] Convert asm...
17
  #include <linux/uaccess.h>
252d4c276   Nicolas Pitre   [ARM] remove bogu...
18
  #include <linux/page-flags.h>
412bb0a62   Catalin Marinas   Include linux/sch...
19
  #include <linux/sched.h>
65cec8e3d   Russell King   ARM: implement hi...
20
  #include <linux/highmem.h>
7ada189f5   Jamie Iles   ARM: 5900/2: arm:...
21
  #include <linux/perf_event.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
22

5a567d78c   Jamie Iles   ARM: 7115/4: move...
23
  #include <asm/exception.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
24
25
26
  #include <asm/system.h>
  #include <asm/pgtable.h>
  #include <asm/tlbflush.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
27
28
  
  #include "fault.h"
09529f7a1   Catalin Marinas   nommu: Fix the fa...
29
  #ifdef CONFIG_MMU
25ce1dd71   Nicolas Pitre   ARM kprobes: add ...
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
  
  #ifdef CONFIG_KPROBES
  static inline int notify_page_fault(struct pt_regs *regs, unsigned int fsr)
  {
  	int ret = 0;
  
  	if (!user_mode(regs)) {
  		/* kprobe_running() needs smp_processor_id() */
  		preempt_disable();
  		if (kprobe_running() && kprobe_fault_handler(regs, fsr))
  			ret = 1;
  		preempt_enable();
  	}
  
  	return ret;
  }
  #else
  static inline int notify_page_fault(struct pt_regs *regs, unsigned int fsr)
  {
  	return 0;
  }
  #endif
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
52
53
54
55
56
57
58
59
60
61
62
63
64
65
  /*
   * This is useful to dump out the page tables associated with
   * 'addr' in mm 'mm'.
   */
  void show_pte(struct mm_struct *mm, unsigned long addr)
  {
  	pgd_t *pgd;
  
  	if (!mm)
  		mm = &init_mm;
  
  	printk(KERN_ALERT "pgd = %p
  ", mm->pgd);
  	pgd = pgd_offset(mm, addr);
29a38193c   Will Deacon   ARM: 6674/1: LPAE...
66
67
  	printk(KERN_ALERT "[%08lx] *pgd=%08llx",
  			addr, (long long)pgd_val(*pgd));
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
68
69
  
  	do {
516295e5a   Russell King   ARM: pgtable: add...
70
  		pud_t *pud;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
71
72
73
74
75
76
77
78
79
80
  		pmd_t *pmd;
  		pte_t *pte;
  
  		if (pgd_none(*pgd))
  			break;
  
  		if (pgd_bad(*pgd)) {
  			printk("(bad)");
  			break;
  		}
516295e5a   Russell King   ARM: pgtable: add...
81
82
  		pud = pud_offset(pgd, addr);
  		if (PTRS_PER_PUD != 1)
140d5dc1d   Catalin Marinas   ARM: 7000/1: LPAE...
83
  			printk(", *pud=%08llx", (long long)pud_val(*pud));
516295e5a   Russell King   ARM: pgtable: add...
84
85
86
87
88
89
90
91
92
93
  
  		if (pud_none(*pud))
  			break;
  
  		if (pud_bad(*pud)) {
  			printk("(bad)");
  			break;
  		}
  
  		pmd = pmd_offset(pud, addr);
da46c79a5   Nicolas Pitre   [ARM] 5272/1: rem...
94
  		if (PTRS_PER_PMD != 1)
29a38193c   Will Deacon   ARM: 6674/1: LPAE...
95
  			printk(", *pmd=%08llx", (long long)pmd_val(*pmd));
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
96
97
98
99
100
101
102
103
  
  		if (pmd_none(*pmd))
  			break;
  
  		if (pmd_bad(*pmd)) {
  			printk("(bad)");
  			break;
  		}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
104
  		/* We must not map this if we have highmem enabled */
252d4c276   Nicolas Pitre   [ARM] remove bogu...
105
106
  		if (PageHighMem(pfn_to_page(pmd_val(*pmd) >> PAGE_SHIFT)))
  			break;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
107
  		pte = pte_offset_map(pmd, addr);
29a38193c   Will Deacon   ARM: 6674/1: LPAE...
108
  		printk(", *pte=%08llx", (long long)pte_val(*pte));
f7b8156d1   Catalin Marinas   ARM: LPAE: Add fa...
109
  #ifndef CONFIG_ARM_LPAE
29a38193c   Will Deacon   ARM: 6674/1: LPAE...
110
111
  		printk(", *ppte=%08llx",
  		       (long long)pte_val(pte[PTE_HWTABLE_PTRS]));
f7b8156d1   Catalin Marinas   ARM: LPAE: Add fa...
112
  #endif
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
113
  		pte_unmap(pte);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
114
115
116
117
118
  	} while(0);
  
  	printk("
  ");
  }
09529f7a1   Catalin Marinas   nommu: Fix the fa...
119
120
121
122
  #else					/* CONFIG_MMU */
  void show_pte(struct mm_struct *mm, unsigned long addr)
  { }
  #endif					/* CONFIG_MMU */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
  
  /*
   * Oops.  The kernel tried to access some page that wasn't present.
   */
  static void
  __do_kernel_fault(struct mm_struct *mm, unsigned long addr, unsigned int fsr,
  		  struct pt_regs *regs)
  {
  	/*
  	 * Are we prepared to handle this kernel fault?
  	 */
  	if (fixup_exception(regs))
  		return;
  
  	/*
  	 * No handler, we'll have to terminate things with extreme prejudice.
  	 */
  	bust_spinlocks(1);
  	printk(KERN_ALERT
  		"Unable to handle kernel %s at virtual address %08lx
  ",
  		(addr < PAGE_SIZE) ? "NULL pointer dereference" :
  		"paging request", addr);
  
  	show_pte(mm, addr);
  	die("Oops", regs, fsr);
  	bust_spinlocks(0);
  	do_exit(SIGKILL);
  }
  
  /*
   * Something tried to access memory that isn't in our memory map..
   * User mode accesses just cause a SIGSEGV
   */
  static void
  __do_user_fault(struct task_struct *tsk, unsigned long addr,
2d137c24e   Andrew Morton   [PATCH] arm: fix ...
159
160
  		unsigned int fsr, unsigned int sig, int code,
  		struct pt_regs *regs)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
161
162
163
164
165
  {
  	struct siginfo si;
  
  #ifdef CONFIG_DEBUG_USER
  	if (user_debug & UDBG_SEGV) {
2d137c24e   Andrew Morton   [PATCH] arm: fix ...
166
167
168
  		printk(KERN_DEBUG "%s: unhandled page fault (%d) at 0x%08lx, code 0x%03x
  ",
  		       tsk->comm, sig, addr, fsr);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
169
170
171
172
173
174
175
176
  		show_pte(tsk->mm, addr);
  		show_regs(regs);
  	}
  #endif
  
  	tsk->thread.address = addr;
  	tsk->thread.error_code = fsr;
  	tsk->thread.trap_no = 14;
2d137c24e   Andrew Morton   [PATCH] arm: fix ...
177
  	si.si_signo = sig;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
178
179
180
  	si.si_errno = 0;
  	si.si_code = code;
  	si.si_addr = (void __user *)addr;
2d137c24e   Andrew Morton   [PATCH] arm: fix ...
181
  	force_sig_info(sig, &si, tsk);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
182
  }
e5beac371   Russell King   [ARM] do_bad_area...
183
  void do_bad_area(unsigned long addr, unsigned int fsr, struct pt_regs *regs)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
184
  {
e5beac371   Russell King   [ARM] do_bad_area...
185
186
  	struct task_struct *tsk = current;
  	struct mm_struct *mm = tsk->active_mm;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
187
188
189
190
191
  	/*
  	 * If we are in kernel mode at this point, we
  	 * have no context to handle this fault with.
  	 */
  	if (user_mode(regs))
2d137c24e   Andrew Morton   [PATCH] arm: fix ...
192
  		__do_user_fault(tsk, addr, fsr, SIGSEGV, SEGV_MAPERR, regs);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
193
194
195
  	else
  		__do_kernel_fault(mm, addr, fsr, regs);
  }
09529f7a1   Catalin Marinas   nommu: Fix the fa...
196
  #ifdef CONFIG_MMU
5c72fc5ca   Nick Piggin   arm: fix up handl...
197
198
  #define VM_FAULT_BADMAP		0x010000
  #define VM_FAULT_BADACCESS	0x020000
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
199

d374bf14a   Russell King   ARM: Separate out...
200
201
202
203
204
205
206
207
208
209
210
  /*
   * Check that the permissions on the VMA allow for the fault which occurred.
   * If we encountered a write fault, we must have write permission, otherwise
   * we allow any permission.
   */
  static inline bool access_error(unsigned int fsr, struct vm_area_struct *vma)
  {
  	unsigned int mask = VM_READ | VM_WRITE | VM_EXEC;
  
  	if (fsr & FSR_WRITE)
  		mask = VM_WRITE;
df297bf6c   Russell King   ARM: Add support ...
211
212
  	if (fsr & FSR_LNX_PF)
  		mask = VM_EXEC;
d374bf14a   Russell King   ARM: Separate out...
213
214
215
216
217
  
  	return vma->vm_flags & mask ? false : true;
  }
  
  static int __kprobes
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
218
  __do_page_fault(struct mm_struct *mm, unsigned long addr, unsigned int fsr,
8878a539f   Kautuk Consul   ARM: 7178/1: faul...
219
  		unsigned int flags, struct task_struct *tsk)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
220
221
  {
  	struct vm_area_struct *vma;
d374bf14a   Russell King   ARM: Separate out...
222
  	int fault;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
223
224
225
  
  	vma = find_vma(mm, addr);
  	fault = VM_FAULT_BADMAP;
d374bf14a   Russell King   ARM: Separate out...
226
  	if (unlikely(!vma))
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
227
  		goto out;
d374bf14a   Russell King   ARM: Separate out...
228
  	if (unlikely(vma->vm_start > addr))
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
229
230
231
232
233
234
235
  		goto check_stack;
  
  	/*
  	 * Ok, we have a good vm_area for this
  	 * memory access, so we can handle it.
  	 */
  good_area:
d374bf14a   Russell King   ARM: Separate out...
236
237
  	if (access_error(fsr, vma)) {
  		fault = VM_FAULT_BADACCESS;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
238
  		goto out;
d374bf14a   Russell King   ARM: Separate out...
239
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
240

8878a539f   Kautuk Consul   ARM: 7178/1: faul...
241
  	return handle_mm_fault(mm, vma, addr & PAGE_MASK, flags);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
242

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
243
244
245
246
247
248
  check_stack:
  	if (vma->vm_flags & VM_GROWSDOWN && !expand_stack(vma, addr))
  		goto good_area;
  out:
  	return fault;
  }
785d3cd28   Nicolas Pitre   ARM kprobes: prev...
249
  static int __kprobes
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
250
251
252
253
  do_page_fault(unsigned long addr, unsigned int fsr, struct pt_regs *regs)
  {
  	struct task_struct *tsk;
  	struct mm_struct *mm;
2d137c24e   Andrew Morton   [PATCH] arm: fix ...
254
  	int fault, sig, code;
8878a539f   Kautuk Consul   ARM: 7178/1: faul...
255
256
257
  	int write = fsr & FSR_WRITE;
  	unsigned int flags = FAULT_FLAG_ALLOW_RETRY | FAULT_FLAG_KILLABLE |
  				(write ? FAULT_FLAG_WRITE : 0);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
258

25ce1dd71   Nicolas Pitre   ARM kprobes: add ...
259
260
  	if (notify_page_fault(regs, fsr))
  		return 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
261
262
  	tsk = current;
  	mm  = tsk->mm;
02fe2845d   Russell King   ARM: entry: avoid...
263
264
265
  	/* Enable interrupts if they were enabled in the parent context. */
  	if (interrupts_enabled(regs))
  		local_irq_enable();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
266
267
268
269
  	/*
  	 * If we're in an interrupt or have no user
  	 * context, we must not take the fault..
  	 */
6edaf68a8   Peter Zijlstra   [PATCH] mm: arch ...
270
  	if (in_atomic() || !mm)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
271
  		goto no_context;
840ff6a4f   Russell King   [ARM] Prevent dea...
272
273
274
275
276
277
278
279
  	/*
  	 * As per x86, we may deadlock here.  However, since the kernel only
  	 * validly references user space from well defined areas of the code,
  	 * we can bug out early if this is from code which shouldn't.
  	 */
  	if (!down_read_trylock(&mm->mmap_sem)) {
  		if (!user_mode(regs) && !search_exception_tables(regs->ARM_pc))
  			goto no_context;
8878a539f   Kautuk Consul   ARM: 7178/1: faul...
280
  retry:
840ff6a4f   Russell King   [ARM] Prevent dea...
281
  		down_read(&mm->mmap_sem);
bf4569922   Russell King   ARM: Ensure corre...
282
283
284
285
286
287
288
  	} else {
  		/*
  		 * The above down_read_trylock() might have succeeded in
  		 * which case, we'll have missed the might_sleep() from
  		 * down_read()
  		 */
  		might_sleep();
1d2127123   Imre Deak   ARM: 5742/1: ARM:...
289
290
291
292
293
  #ifdef CONFIG_DEBUG_VM
  		if (!user_mode(regs) &&
  		    !search_exception_tables(regs->ARM_pc))
  			goto no_context;
  #endif
840ff6a4f   Russell King   [ARM] Prevent dea...
294
  	}
8878a539f   Kautuk Consul   ARM: 7178/1: faul...
295
296
297
298
299
300
301
302
303
304
305
306
307
308
  	fault = __do_page_fault(mm, addr, fsr, flags, tsk);
  
  	/* If we need to retry but a fatal signal is pending, handle the
  	 * signal first. We do not need to release the mmap_sem because
  	 * it would already be released in __lock_page_or_retry in
  	 * mm/filemap.c. */
  	if ((fault & VM_FAULT_RETRY) && fatal_signal_pending(current))
  		return 0;
  
  	/*
  	 * Major/minor page fault accounting is only done on the
  	 * initial attempt. If we go through a retry, it is extremely
  	 * likely that the page will be found in page cache at that point.
  	 */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
309

a8b0ca17b   Peter Zijlstra   perf: Remove the ...
310
  	perf_sw_event(PERF_COUNT_SW_PAGE_FAULTS, 1, regs, addr);
8878a539f   Kautuk Consul   ARM: 7178/1: faul...
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
  	if (flags & FAULT_FLAG_ALLOW_RETRY) {
  		if (fault & VM_FAULT_MAJOR) {
  			tsk->maj_flt++;
  			perf_sw_event(PERF_COUNT_SW_PAGE_FAULTS_MAJ, 1,
  					regs, addr);
  		} else {
  			tsk->min_flt++;
  			perf_sw_event(PERF_COUNT_SW_PAGE_FAULTS_MIN, 1,
  					regs, addr);
  		}
  		if (fault & VM_FAULT_RETRY) {
  			/* Clear FAULT_FLAG_ALLOW_RETRY to avoid any risk
  			* of starvation. */
  			flags &= ~FAULT_FLAG_ALLOW_RETRY;
  			goto retry;
  		}
  	}
  
  	up_read(&mm->mmap_sem);
7ada189f5   Jamie Iles   ARM: 5900/2: arm:...
330

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
331
  	/*
ff2afb9df   Russell King   [PATCH] ARM: Fix ...
332
  	 * Handle the "normal" case first - VM_FAULT_MAJOR / VM_FAULT_MINOR
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
333
  	 */
5c72fc5ca   Nick Piggin   arm: fix up handl...
334
  	if (likely(!(fault & (VM_FAULT_ERROR | VM_FAULT_BADMAP | VM_FAULT_BADACCESS))))
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
335
  		return 0;
b42c6344b   Russell King   ARM: Update page ...
336
337
338
339
340
341
342
343
344
  	if (fault & VM_FAULT_OOM) {
  		/*
  		 * We ran out of memory, call the OOM killer, and return to
  		 * userspace (which will retry the fault, or kill us if we
  		 * got oom-killed)
  		 */
  		pagefault_out_of_memory();
  		return 0;
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
345
  	/*
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
346
347
348
349
350
  	 * If we are in kernel mode at this point, we
  	 * have no context to handle this fault with.
  	 */
  	if (!user_mode(regs))
  		goto no_context;
83c54070e   Nick Piggin   mm: fault feedbac...
351
  	if (fault & VM_FAULT_SIGBUS) {
2d137c24e   Andrew Morton   [PATCH] arm: fix ...
352
353
354
355
356
357
  		/*
  		 * We had some memory, but were unable to
  		 * successfully fix up this page fault.
  		 */
  		sig = SIGBUS;
  		code = BUS_ADRERR;
83c54070e   Nick Piggin   mm: fault feedbac...
358
  	} else {
2d137c24e   Andrew Morton   [PATCH] arm: fix ...
359
360
361
362
363
364
365
  		/*
  		 * Something tried to access memory that
  		 * isn't in our memory map..
  		 */
  		sig = SIGSEGV;
  		code = fault == VM_FAULT_BADACCESS ?
  			SEGV_ACCERR : SEGV_MAPERR;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
366
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
367

2d137c24e   Andrew Morton   [PATCH] arm: fix ...
368
369
  	__do_user_fault(tsk, addr, fsr, sig, code, regs);
  	return 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
370
371
372
373
374
  
  no_context:
  	__do_kernel_fault(mm, addr, fsr, regs);
  	return 0;
  }
09529f7a1   Catalin Marinas   nommu: Fix the fa...
375
376
377
378
379
380
381
  #else					/* CONFIG_MMU */
  static int
  do_page_fault(unsigned long addr, unsigned int fsr, struct pt_regs *regs)
  {
  	return 0;
  }
  #endif					/* CONFIG_MMU */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
  
  /*
   * First Level Translation Fault Handler
   *
   * We enter here because the first level page table doesn't contain
   * a valid entry for the address.
   *
   * If the address is in kernel space (>= TASK_SIZE), then we are
   * probably faulting in the vmalloc() area.
   *
   * If the init_task's first level page tables contains the relevant
   * entry, we copy the it to this task.  If not, we send the process
   * a signal, fixup the exception, or oops the kernel.
   *
   * NOTE! We MUST NOT take any locks for this case. We may be in an
   * interrupt or a critical region, and should only copy the information
   * from the master page table, nothing more.
   */
09529f7a1   Catalin Marinas   nommu: Fix the fa...
400
  #ifdef CONFIG_MMU
785d3cd28   Nicolas Pitre   ARM kprobes: prev...
401
  static int __kprobes
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
402
403
404
  do_translation_fault(unsigned long addr, unsigned int fsr,
  		     struct pt_regs *regs)
  {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
405
406
  	unsigned int index;
  	pgd_t *pgd, *pgd_k;
516295e5a   Russell King   ARM: pgtable: add...
407
  	pud_t *pud, *pud_k;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
408
409
410
411
  	pmd_t *pmd, *pmd_k;
  
  	if (addr < TASK_SIZE)
  		return do_page_fault(addr, fsr, regs);
5e27fb78d   Anfei   ARM: 6166/1: Prop...
412
413
  	if (user_mode(regs))
  		goto bad_area;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
414
415
416
417
418
419
420
421
422
423
  	index = pgd_index(addr);
  
  	/*
  	 * FIXME: CP15 C1 is write only on ARMv3 architectures.
  	 */
  	pgd = cpu_get_pgd() + index;
  	pgd_k = init_mm.pgd + index;
  
  	if (pgd_none(*pgd_k))
  		goto bad_area;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
424
425
  	if (!pgd_present(*pgd))
  		set_pgd(pgd, *pgd_k);
516295e5a   Russell King   ARM: pgtable: add...
426
427
428
429
430
431
432
433
434
435
  	pud = pud_offset(pgd, addr);
  	pud_k = pud_offset(pgd_k, addr);
  
  	if (pud_none(*pud_k))
  		goto bad_area;
  	if (!pud_present(*pud))
  		set_pud(pud, *pud_k);
  
  	pmd = pmd_offset(pud, addr);
  	pmd_k = pmd_offset(pud_k, addr);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
436

f7b8156d1   Catalin Marinas   ARM: LPAE: Add fa...
437
438
439
440
441
442
  #ifdef CONFIG_ARM_LPAE
  	/*
  	 * Only one hardware entry per PMD with LPAE.
  	 */
  	index = 0;
  #else
33a9c41bf   Kirill A. Shutemov   ARM: 6255/1: Work...
443
444
445
446
447
448
449
450
451
  	/*
  	 * On ARM one Linux PGD entry contains two hardware entries (see page
  	 * tables layout in pgtable.h). We normally guarantee that we always
  	 * fill both L1 entries. But create_mapping() doesn't follow the rule.
  	 * It can create inidividual L1 entries, so here we have to call
  	 * pmd_none() check for the entry really corresponded to address, not
  	 * for the first of pair.
  	 */
  	index = (addr >> SECTION_SHIFT) & 1;
f7b8156d1   Catalin Marinas   ARM: LPAE: Add fa...
452
  #endif
33a9c41bf   Kirill A. Shutemov   ARM: 6255/1: Work...
453
  	if (pmd_none(pmd_k[index]))
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
454
455
456
457
458
459
  		goto bad_area;
  
  	copy_pmd(pmd, pmd_k);
  	return 0;
  
  bad_area:
e5beac371   Russell King   [ARM] do_bad_area...
460
  	do_bad_area(addr, fsr, regs);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
461
462
  	return 0;
  }
09529f7a1   Catalin Marinas   nommu: Fix the fa...
463
464
465
466
467
468
469
470
  #else					/* CONFIG_MMU */
  static int
  do_translation_fault(unsigned long addr, unsigned int fsr,
  		     struct pt_regs *regs)
  {
  	return 0;
  }
  #endif					/* CONFIG_MMU */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
471
472
473
474
475
476
477
478
  
  /*
   * Some section permission faults need to be handled gracefully.
   * They can happen due to a __{get,put}_user during an oops.
   */
  static int
  do_sect_fault(unsigned long addr, unsigned int fsr, struct pt_regs *regs)
  {
e5beac371   Russell King   [ARM] do_bad_area...
479
  	do_bad_area(addr, fsr, regs);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
480
481
482
483
484
485
486
487
488
489
490
  	return 0;
  }
  
  /*
   * This abort handler always returns "fault".
   */
  static int
  do_bad(unsigned long addr, unsigned int fsr, struct pt_regs *regs)
  {
  	return 1;
  }
136848d4c   Catalin Marinas   ARM: LPAE: Move t...
491
  struct fsr_info {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
492
493
  	int	(*fn)(unsigned long addr, unsigned int fsr, struct pt_regs *regs);
  	int	sig;
cfb0810ea   Russell King   [PATCH] ARM: Don'...
494
  	int	code;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
495
  	const char *name;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
496
  };
136848d4c   Catalin Marinas   ARM: LPAE: Move t...
497
  /* FSR definition */
f7b8156d1   Catalin Marinas   ARM: LPAE: Add fa...
498
499
500
  #ifdef CONFIG_ARM_LPAE
  #include "fsr-3level.c"
  #else
136848d4c   Catalin Marinas   ARM: LPAE: Move t...
501
  #include "fsr-2level.c"
f7b8156d1   Catalin Marinas   ARM: LPAE: Add fa...
502
  #endif
136848d4c   Catalin Marinas   ARM: LPAE: Move t...
503

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
504
505
  void __init
  hook_fault_code(int nr, int (*fn)(unsigned long, unsigned int, struct pt_regs *),
6338a6aa7   Kirill A. Shutemov   ARM: 6269/1: Add ...
506
  		int sig, int code, const char *name)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
507
  {
6338a6aa7   Kirill A. Shutemov   ARM: 6269/1: Add ...
508
509
510
511
512
513
514
  	if (nr < 0 || nr >= ARRAY_SIZE(fsr_info))
  		BUG();
  
  	fsr_info[nr].fn   = fn;
  	fsr_info[nr].sig  = sig;
  	fsr_info[nr].code = code;
  	fsr_info[nr].name = name;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
515
516
517
518
519
  }
  
  /*
   * Dispatch a data abort to the relevant handler.
   */
7ab3f8d59   Russell King   [ARM] Add ability...
520
  asmlinkage void __exception
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
521
522
  do_DataAbort(unsigned long addr, unsigned int fsr, struct pt_regs *regs)
  {
c88d6aa71   Russell King   ARM: Provide defi...
523
  	const struct fsr_info *inf = fsr_info + fsr_fs(fsr);
cfb0810ea   Russell King   [PATCH] ARM: Don'...
524
  	struct siginfo info;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
525

df297bf6c   Russell King   ARM: Add support ...
526
  	if (!inf->fn(addr, fsr & ~FSR_LNX_PF, regs))
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
527
528
529
530
531
  		return;
  
  	printk(KERN_ALERT "Unhandled fault: %s (0x%03x) at 0x%08lx
  ",
  		inf->name, fsr, addr);
cfb0810ea   Russell King   [PATCH] ARM: Don'...
532
533
534
535
536
  
  	info.si_signo = inf->sig;
  	info.si_errno = 0;
  	info.si_code  = inf->code;
  	info.si_addr  = (void __user *)addr;
1eeb66a1b   Christoph Hellwig   move die notifier...
537
  	arm_notify_die("", regs, &info, fsr, 0);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
538
  }
3a4b5dca5   Will Deacon   ARM: 6355/1: hw-b...
539
540
541
542
543
544
545
546
547
548
549
550
  void __init
  hook_ifault_code(int nr, int (*fn)(unsigned long, unsigned int, struct pt_regs *),
  		 int sig, int code, const char *name)
  {
  	if (nr < 0 || nr >= ARRAY_SIZE(ifsr_info))
  		BUG();
  
  	ifsr_info[nr].fn   = fn;
  	ifsr_info[nr].sig  = sig;
  	ifsr_info[nr].code = code;
  	ifsr_info[nr].name = name;
  }
7ab3f8d59   Russell King   [ARM] Add ability...
551
  asmlinkage void __exception
4fb284743   Kirill A. Shutemov   ARM: 5727/1: Pass...
552
  do_PrefetchAbort(unsigned long addr, unsigned int ifsr, struct pt_regs *regs)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
553
  {
d25ef8b86   Kirill A. Shutemov   ARM: 5728/1: Prop...
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
  	const struct fsr_info *inf = ifsr_info + fsr_fs(ifsr);
  	struct siginfo info;
  
  	if (!inf->fn(addr, ifsr | FSR_LNX_PF, regs))
  		return;
  
  	printk(KERN_ALERT "Unhandled prefetch abort: %s (0x%03x) at 0x%08lx
  ",
  		inf->name, ifsr, addr);
  
  	info.si_signo = inf->sig;
  	info.si_errno = 0;
  	info.si_code  = inf->code;
  	info.si_addr  = (void __user *)addr;
  	arm_notify_die("", regs, &info, ifsr, 0);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
569
  }
f7b8156d1   Catalin Marinas   ARM: LPAE: Add fa...
570
  #ifndef CONFIG_ARM_LPAE
993bf4ec8   Kirill A. Shutemov   ARM: 6256/1: Chec...
571
572
573
574
575
576
  static int __init exceptions_init(void)
  {
  	if (cpu_architecture() >= CPU_ARCH_ARMv6) {
  		hook_fault_code(4, do_translation_fault, SIGSEGV, SEGV_MAPERR,
  				"I-cache maintenance fault");
  	}
b8ab5397b   Kirill A. Shutemov   ARM: 6268/1: ARMv...
577
578
579
580
581
582
583
584
585
586
  	if (cpu_architecture() >= CPU_ARCH_ARMv7) {
  		/*
  		 * TODO: Access flag faults introduced in ARMv6K.
  		 * Runtime check for 'K' extension is needed
  		 */
  		hook_fault_code(3, do_bad, SIGSEGV, SEGV_MAPERR,
  				"section access flag fault");
  		hook_fault_code(6, do_bad, SIGSEGV, SEGV_MAPERR,
  				"section access flag fault");
  	}
993bf4ec8   Kirill A. Shutemov   ARM: 6256/1: Chec...
587
588
589
590
  	return 0;
  }
  
  arch_initcall(exceptions_init);
f7b8156d1   Catalin Marinas   ARM: LPAE: Add fa...
591
  #endif