xref: /openbmc/linux/arch/m68k/mm/fault.c (revision b6dcefde)
1 /*
2  *  linux/arch/m68k/mm/fault.c
3  *
4  *  Copyright (C) 1995  Hamish Macdonald
5  */
6 
7 #include <linux/mman.h>
8 #include <linux/mm.h>
9 #include <linux/kernel.h>
10 #include <linux/ptrace.h>
11 #include <linux/interrupt.h>
12 #include <linux/module.h>
13 
14 #include <asm/setup.h>
15 #include <asm/traps.h>
16 #include <asm/system.h>
17 #include <asm/uaccess.h>
18 #include <asm/pgalloc.h>
19 
20 extern void die_if_kernel(char *, struct pt_regs *, long);
21 extern const int frame_extra_sizes[]; /* in m68k/kernel/signal.c */
22 
23 int send_fault_sig(struct pt_regs *regs)
24 {
25 	siginfo_t siginfo = { 0, 0, 0, };
26 
27 	siginfo.si_signo = current->thread.signo;
28 	siginfo.si_code = current->thread.code;
29 	siginfo.si_addr = (void *)current->thread.faddr;
30 #ifdef DEBUG
31 	printk("send_fault_sig: %p,%d,%d\n", siginfo.si_addr, siginfo.si_signo, siginfo.si_code);
32 #endif
33 
34 	if (user_mode(regs)) {
35 		force_sig_info(siginfo.si_signo,
36 			       &siginfo, current);
37 	} else {
38 		const struct exception_table_entry *fixup;
39 
40 		/* Are we prepared to handle this kernel fault? */
41 		if ((fixup = search_exception_tables(regs->pc))) {
42 			struct pt_regs *tregs;
43 			/* Create a new four word stack frame, discarding the old
44 			   one.  */
45 			regs->stkadj = frame_extra_sizes[regs->format];
46 			tregs =	(struct pt_regs *)((ulong)regs + regs->stkadj);
47 			tregs->vector = regs->vector;
48 			tregs->format = 0;
49 			tregs->pc = fixup->fixup;
50 			tregs->sr = regs->sr;
51 			return -1;
52 		}
53 
54 		//if (siginfo.si_signo == SIGBUS)
55 		//	force_sig_info(siginfo.si_signo,
56 		//		       &siginfo, current);
57 
58 		/*
59 		 * Oops. The kernel tried to access some bad page. We'll have to
60 		 * terminate things with extreme prejudice.
61 		 */
62 		if ((unsigned long)siginfo.si_addr < PAGE_SIZE)
63 			printk(KERN_ALERT "Unable to handle kernel NULL pointer dereference");
64 		else
65 			printk(KERN_ALERT "Unable to handle kernel access");
66 		printk(" at virtual address %p\n", siginfo.si_addr);
67 		die_if_kernel("Oops", regs, 0 /*error_code*/);
68 		do_exit(SIGKILL);
69 	}
70 
71 	return 1;
72 }
73 
74 /*
75  * This routine handles page faults.  It determines the problem, and
76  * then passes it off to one of the appropriate routines.
77  *
78  * error_code:
79  *	bit 0 == 0 means no page found, 1 means protection fault
80  *	bit 1 == 0 means read, 1 means write
81  *
82  * If this routine detects a bad access, it returns 1, otherwise it
83  * returns 0.
84  */
85 int do_page_fault(struct pt_regs *regs, unsigned long address,
86 			      unsigned long error_code)
87 {
88 	struct mm_struct *mm = current->mm;
89 	struct vm_area_struct * vma;
90 	int write, fault;
91 
92 #ifdef DEBUG
93 	printk ("do page fault:\nregs->sr=%#x, regs->pc=%#lx, address=%#lx, %ld, %p\n",
94 		regs->sr, regs->pc, address, error_code,
95 		current->mm->pgd);
96 #endif
97 
98 	/*
99 	 * If we're in an interrupt or have no user
100 	 * context, we must not take the fault..
101 	 */
102 	if (in_atomic() || !mm)
103 		goto no_context;
104 
105 	down_read(&mm->mmap_sem);
106 
107 	vma = find_vma(mm, address);
108 	if (!vma)
109 		goto map_err;
110 	if (vma->vm_flags & VM_IO)
111 		goto acc_err;
112 	if (vma->vm_start <= address)
113 		goto good_area;
114 	if (!(vma->vm_flags & VM_GROWSDOWN))
115 		goto map_err;
116 	if (user_mode(regs)) {
117 		/* Accessing the stack below usp is always a bug.  The
118 		   "+ 256" is there due to some instructions doing
119 		   pre-decrement on the stack and that doesn't show up
120 		   until later.  */
121 		if (address + 256 < rdusp())
122 			goto map_err;
123 	}
124 	if (expand_stack(vma, address))
125 		goto map_err;
126 
127 /*
128  * Ok, we have a good vm_area for this memory access, so
129  * we can handle it..
130  */
131 good_area:
132 #ifdef DEBUG
133 	printk("do_page_fault: good_area\n");
134 #endif
135 	write = 0;
136 	switch (error_code & 3) {
137 		default:	/* 3: write, present */
138 			/* fall through */
139 		case 2:		/* write, not present */
140 			if (!(vma->vm_flags & VM_WRITE))
141 				goto acc_err;
142 			write++;
143 			break;
144 		case 1:		/* read, present */
145 			goto acc_err;
146 		case 0:		/* read, not present */
147 			if (!(vma->vm_flags & (VM_READ | VM_EXEC | VM_WRITE)))
148 				goto acc_err;
149 	}
150 
151 	/*
152 	 * If for any reason at all we couldn't handle the fault,
153 	 * make sure we exit gracefully rather than endlessly redo
154 	 * the fault.
155 	 */
156 
157  survive:
158 	fault = handle_mm_fault(mm, vma, address, write ? FAULT_FLAG_WRITE : 0);
159 #ifdef DEBUG
160 	printk("handle_mm_fault returns %d\n",fault);
161 #endif
162 	if (unlikely(fault & VM_FAULT_ERROR)) {
163 		if (fault & VM_FAULT_OOM)
164 			goto out_of_memory;
165 		else if (fault & VM_FAULT_SIGBUS)
166 			goto bus_err;
167 		BUG();
168 	}
169 	if (fault & VM_FAULT_MAJOR)
170 		current->maj_flt++;
171 	else
172 		current->min_flt++;
173 
174 	up_read(&mm->mmap_sem);
175 	return 0;
176 
177 /*
178  * We ran out of memory, or some other thing happened to us that made
179  * us unable to handle the page fault gracefully.
180  */
181 out_of_memory:
182 	up_read(&mm->mmap_sem);
183 	if (is_global_init(current)) {
184 		yield();
185 		down_read(&mm->mmap_sem);
186 		goto survive;
187 	}
188 
189 	printk("VM: killing process %s\n", current->comm);
190 	if (user_mode(regs))
191 		do_group_exit(SIGKILL);
192 
193 no_context:
194 	current->thread.signo = SIGBUS;
195 	current->thread.faddr = address;
196 	return send_fault_sig(regs);
197 
198 bus_err:
199 	current->thread.signo = SIGBUS;
200 	current->thread.code = BUS_ADRERR;
201 	current->thread.faddr = address;
202 	goto send_sig;
203 
204 map_err:
205 	current->thread.signo = SIGSEGV;
206 	current->thread.code = SEGV_MAPERR;
207 	current->thread.faddr = address;
208 	goto send_sig;
209 
210 acc_err:
211 	current->thread.signo = SIGSEGV;
212 	current->thread.code = SEGV_ACCERR;
213 	current->thread.faddr = address;
214 
215 send_sig:
216 	up_read(&mm->mmap_sem);
217 	return send_fault_sig(regs);
218 }
219