This source file includes following definitions.
- die_if_kernel
- do_entArith
- do_entIF
- do_entUna
- do_entUnaUser
- do_entSys
- trap_init
1
2
3
4
5
6
7
8
9
10
11 #include <linux/mm.h>
12 #include <linux/sched.h>
13 #include <linux/tty.h>
14 #include <linux/config.h>
15
16 #include <asm/gentrap.h>
17 #include <asm/segment.h>
18 #include <asm/unaligned.h>
19
20 void die_if_kernel(char * str, struct pt_regs * regs, long err)
21 {
22 long i;
23 unsigned long sp;
24 unsigned int * pc;
25
26 if (regs->ps & 8)
27 return;
28 printk("%s(%d): %s %ld\n", current->comm, current->pid, str, err);
29 sp = (unsigned long) (regs+1);
30 printk("pc = [<%lx>] ps = %04lx\n", regs->pc, regs->ps);
31 printk("rp = [<%lx>] sp = %lx\n", regs->r26, sp);
32 printk("r0=%lx r1=%lx r2=%lx r3=%lx\n",
33 regs->r0, regs->r1, regs->r2, regs->r3);
34 printk("r8=%lx\n", regs->r8);
35 printk("r16=%lx r17=%lx r18=%lx r19=%lx\n",
36 regs->r16, regs->r17, regs->r18, regs->r19);
37 printk("r20=%lx r21=%lx r22=%lx r23=%lx\n",
38 regs->r20, regs->r21, regs->r22, regs->r23);
39 printk("r24=%lx r25=%lx r26=%lx r27=%lx\n",
40 regs->r24, regs->r25, regs->r26, regs->r27);
41 printk("r28=%lx r29=%lx r30=%lx\n",
42 regs->r28, regs->gp, sp);
43 printk("Code:");
44 pc = (unsigned int *) regs->pc;
45 for (i = -3; i < 6; i++)
46 printk("%c%08x%c",i?' ':'<',pc[i],i?' ':'>');
47 printk("\n");
48 do_exit(SIGSEGV);
49 }
50
51 asmlinkage void do_entArith(unsigned long summary, unsigned long write_mask,
52 unsigned long a2, unsigned long a3,
53 unsigned long a4, unsigned long a5,
54 struct pt_regs regs)
55 {
56 if ((summary & 1)) {
57 extern long alpha_fp_emul_imprecise (struct pt_regs * regs,
58 unsigned long write_mask);
59
60
61
62
63 if (alpha_fp_emul_imprecise(®s, write_mask)) {
64 return;
65 }
66 }
67 printk("Arithmetic trap at %016lx: %02lx %016lx\n",
68 regs.pc, summary, write_mask);
69 die_if_kernel("Arithmetic fault", ®s, 0);
70 send_sig(SIGFPE, current, 1);
71 }
72
73 asmlinkage void do_entIF(unsigned long type, unsigned long a1, unsigned long a2,
74 unsigned long a3, unsigned long a4, unsigned long a5,
75 struct pt_regs regs)
76 {
77 extern int ptrace_cancel_bpt (struct task_struct *who);
78
79 die_if_kernel("Instruction fault", ®s, type);
80 switch (type) {
81 case 0:
82 if (ptrace_cancel_bpt(current)) {
83 regs.pc -= 4;
84 }
85 send_sig(SIGTRAP, current, 1);
86 break;
87
88 case 2:
89
90
91
92
93
94
95
96 switch ((long) regs.r16) {
97 case GEN_INTOVF: case GEN_INTDIV: case GEN_FLTOVF:
98 case GEN_FLTDIV: case GEN_FLTUND: case GEN_FLTINV:
99 case GEN_FLTINE:
100 send_sig(SIGFPE, current, 1);
101 break;
102
103 case GEN_DECOVF:
104 case GEN_DECDIV:
105 case GEN_DECINV:
106 case GEN_ROPRAND:
107 case GEN_ASSERTERR:
108 case GEN_NULPTRERR:
109 case GEN_STKOVF:
110 case GEN_STRLENERR:
111 case GEN_SUBSTRERR:
112 case GEN_RANGERR:
113 case GEN_SUBRNG:
114 case GEN_SUBRNG1:
115 case GEN_SUBRNG2:
116 case GEN_SUBRNG3:
117 case GEN_SUBRNG4:
118 case GEN_SUBRNG5:
119 case GEN_SUBRNG6:
120 case GEN_SUBRNG7:
121 send_sig(SIGILL, current, 1);
122 break;
123 }
124 break;
125
126 case 1:
127 case 3:
128 send_sig(SIGILL, current, 1);
129 break;
130
131 case 4:
132 #ifdef CONFIG_ALPHA_NEED_ROUNDING_EMULATION
133 {
134 extern long alpha_fp_emul (unsigned long pc);
135 unsigned int opcode;
136
137
138 opcode = get_user((__u32*)(regs.pc - 4)) >> 26;
139 if (opcode == 0x16) {
140
141
142
143
144 if (!alpha_fp_emul(regs.pc - 4))
145 send_sig(SIGFPE, current, 1);
146 break;
147 }
148 }
149 #endif
150 send_sig(SIGILL, current, 1);
151 break;
152
153 default:
154 panic("do_entIF: unexpected instruction-fault type");
155 }
156 }
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171 struct allregs {
172 unsigned long regs[32];
173 unsigned long ps, pc, gp, a0, a1, a2;
174 };
175
176 struct unaligned_stat {
177 unsigned long count, va, pc;
178 } unaligned[2];
179
180 asmlinkage void do_entUna(void * va, unsigned long opcode, unsigned long reg,
181 unsigned long a3, unsigned long a4, unsigned long a5,
182 struct allregs regs)
183 {
184 static int cnt = 0;
185 static long last_time = 0;
186
187 if (cnt >= 5 && jiffies - last_time > 5*HZ) {
188 cnt = 0;
189 }
190 if (++cnt < 5) {
191 printk("kernel: unaligned trap at %016lx: %p %lx %ld\n",
192 regs.pc - 4, va, opcode, reg);
193 }
194 last_time = jiffies;
195
196 ++unaligned[0].count;
197 unaligned[0].va = (unsigned long) va - 4;
198 unaligned[0].pc = regs.pc;
199
200
201 if (reg >= 16 && reg <= 18)
202 reg += 19;
203 switch (opcode) {
204 case 0x28:
205 *(reg+regs.regs) = (int) ldl_u(va);
206 return;
207 case 0x29:
208 *(reg+regs.regs) = ldq_u(va);
209 return;
210 case 0x2c:
211 stl_u(*(reg+regs.regs), va);
212 return;
213 case 0x2d:
214 stq_u(*(reg+regs.regs), va);
215 return;
216 }
217 printk("Bad unaligned kernel access at %016lx: %p %lx %ld\n",
218 regs.pc, va, opcode, reg);
219 do_exit(SIGSEGV);
220 }
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242 asmlinkage void do_entUnaUser(void * va, unsigned long opcode, unsigned long reg,
243 unsigned long * frame)
244 {
245 long dir, size;
246 unsigned long *reg_addr, *pc_addr, usp, zero = 0;
247 static int cnt = 0;
248 static long last_time = 0;
249
250 pc_addr = frame + 7 + 20 + 1;
251
252 if (cnt >= 5 && jiffies - last_time > 5*HZ) {
253 cnt = 0;
254 }
255 if (++cnt < 5) {
256 printk("%s(%d): unaligned trap at %016lx: %p %lx %ld\n",
257 current->comm, current->pid,
258 *pc_addr - 4, va, opcode, reg);
259 }
260 last_time = jiffies;
261
262 ++unaligned[1].count;
263 unaligned[1].va = (unsigned long) va - 4;
264 unaligned[1].pc = *pc_addr;
265
266 dir = VERIFY_READ;
267 if (opcode & 0x4) {
268
269 dir = VERIFY_WRITE;
270 }
271 size = 4;
272 if (opcode & 0x1) {
273
274 size = 8;
275 }
276 if (verify_area(dir, va, size)) {
277 *pc_addr -= 4;
278 send_sig(SIGSEGV, current, 1);
279 return;
280 }
281
282 reg_addr = frame;
283 if (opcode >= 0x28) {
284
285 if (reg < 9) {
286 reg_addr += 7 + reg;
287 } else if (reg < 16) {
288 reg_addr += (reg - 9);
289 } else if (reg < 19) {
290 reg_addr += 7 + 20 + 3 + (reg - 16);
291 } else if (reg < 29) {
292 reg_addr += 7 + 9 + (reg - 19);
293 } else {
294 switch (reg) {
295 case 29:
296 reg_addr += 7 + 20 + 2;
297 break;
298 case 30:
299 usp = rdusp();
300 reg_addr = &usp;
301 break;
302 case 31:
303 reg_addr = &zero;
304 break;
305 }
306 }
307 }
308
309 switch (opcode) {
310 case 0x23: alpha_write_fp_reg(reg, ldq_u(va)); break;
311 case 0x27: stq_u(alpha_read_fp_reg(reg), va); break;
312
313 case 0x28: *reg_addr = (int) ldl_u(va); break;
314 case 0x29: *reg_addr = ldq_u(va); break;
315 case 0x2c: stl_u(*reg_addr, va); break;
316 case 0x2d: stq_u(*reg_addr, va); break;
317 default:
318 *pc_addr -= 4;
319 send_sig(SIGBUS, current, 1);
320 return;
321 }
322
323 if (opcode >= 0x28 && reg == 30 && dir == VERIFY_WRITE) {
324 wrusp(usp);
325 }
326 }
327
328
329
330
331
332
333
334
335
336
337
338
339 asmlinkage long do_entSys(unsigned long a0, unsigned long a1, unsigned long a2,
340 unsigned long a3, unsigned long a4, unsigned long a5, struct pt_regs regs)
341 {
342 if (regs.r0 != 112)
343 printk("<sc %ld(%lx,%lx,%lx)>", regs.r0, a0, a1, a2);
344 return -1;
345 }
346
347 extern asmlinkage void entMM(void);
348 extern asmlinkage void entIF(void);
349 extern asmlinkage void entArith(void);
350 extern asmlinkage void entUna(void);
351 extern asmlinkage void entSys(void);
352
353 void trap_init(void)
354 {
355 unsigned long gptr;
356
357
358
359
360 __asm__("br %0,___tmp\n"
361 "___tmp:\tldgp %0,0(%0)"
362 : "=r" (gptr));
363 wrkgp(gptr);
364
365 wrent(entArith, 1);
366 wrent(entMM, 2);
367 wrent(entIF, 3);
368 wrent(entUna, 4);
369 wrent(entSys, 5);
370 }