2 * Handle unaligned accesses by emulation.
4 * This file is subject to the terms and conditions of the GNU General Public
5 * License. See the file "COPYING" in the main directory of this archive
8 * Copyright (C) 1996, 1998, 1999, 2002 by Ralf Baechle
9 * Copyright (C) 1999 Silicon Graphics, Inc.
11 * This file contains exception handler for address error exception with the
12 * special capability to execute faulting instructions in software. The
13 * handler does not try to handle the case when the program counter points
14 * to an address not aligned to a word boundary.
16 * Putting data to unaligned addresses is a bad practice even on Intel where
17 * only the performance is affected. Much worse is that such code is non-
18 * portable. Due to several programs that die on MIPS due to alignment
19 * problems I decided to implement this handler anyway though I originally
20 * didn't intend to do this at all for user code.
22 * For now I enable fixing of address errors by default to make life easier.
23 * I however intend to disable this somewhen in the future when the alignment
24 * problems with user programs have been fixed. For programmers this is the
27 * Fixing address errors is a per process option. The option is inherited
28 * across fork(2) and execve(2) calls. If you really want to use the
29 * option in your user programs - I discourage the use of the software
30 * emulation strongly - use the following code in your userland stuff:
32 * #include <sys/sysmips.h>
35 * sysmips(MIPS_FIXADE, x);
38 * The argument x is 0 for disabling software emulation, enabled otherwise.
40 * Below a little program to play around with this feature.
43 * #include <sys/sysmips.h>
46 * unsigned char bar[8];
49 * main(int argc, char *argv[])
51 * struct foo x = {0, 1, 2, 3, 4, 5, 6, 7};
52 * unsigned int *p = (unsigned int *) (x.bar + 3);
56 * sysmips(MIPS_FIXADE, atoi(argv[1]));
58 * printf("*p = %08lx\n", *p);
62 * for(i = 0; i <= 7; i++)
63 * printf("%02x ", x.bar[i]);
67 * Coprocessor loads are not supported; I think this case is unimportant
70 * TODO: Handle ndc (attempted store to doubleword in uncached memory)
71 * exception for the R6000.
72 * A store crossing a page boundary might be executed only partially.
73 * Undo the partial store in this case.
75 #include <linux/config.h>
77 #include <linux/module.h>
78 #include <linux/signal.h>
79 #include <linux/smp.h>
80 #include <linux/smp_lock.h>
83 #include <asm/branch.h>
84 #include <asm/byteorder.h>
86 #include <asm/uaccess.h>
87 #include <asm/system.h>
89 #define STR(x) __STR(x)
93 unsigned long unaligned_instructions
;
96 static inline int emulate_load_store_insn(struct pt_regs
*regs
,
97 void __user
*addr
, unsigned int __user
*pc
,
98 unsigned long **regptr
, unsigned long *newvalue
)
100 union mips_instruction insn
;
108 * This load never faults.
110 __get_user(insn
.word
, pc
);
112 switch (insn
.i_format
.opcode
) {
114 * These are instructions that a compiler doesn't generate. We
115 * can assume therefore that the code is MIPS-aware and
116 * really buggy. Emulating these instructions would break the
125 * For these instructions the only way to create an address
126 * error is an attempted access to kernel/supervisor address
143 * The remaining opcodes are the ones that are really of interest.
146 if (!access_ok(VERIFY_READ
, addr
, 2))
149 __asm__
__volatile__ (".set\tnoat\n"
151 "1:\tlb\t%0, 0(%2)\n"
152 "2:\tlbu\t$1, 1(%2)\n\t"
154 #ifdef __LITTLE_ENDIAN
155 "1:\tlb\t%0, 1(%2)\n"
156 "2:\tlbu\t$1, 0(%2)\n\t"
162 ".section\t.fixup,\"ax\"\n\t"
166 ".section\t__ex_table,\"a\"\n\t"
167 STR(PTR
)"\t1b, 4b\n\t"
168 STR(PTR
)"\t2b, 4b\n\t"
170 : "=&r" (value
), "=r" (res
)
171 : "r" (addr
), "i" (-EFAULT
));
175 *regptr
= ®s
->regs
[insn
.i_format
.rt
];
179 if (!access_ok(VERIFY_READ
, addr
, 4))
182 __asm__
__volatile__ (
184 "1:\tlwl\t%0, (%2)\n"
185 "2:\tlwr\t%0, 3(%2)\n\t"
187 #ifdef __LITTLE_ENDIAN
188 "1:\tlwl\t%0, 3(%2)\n"
189 "2:\tlwr\t%0, (%2)\n\t"
192 "3:\t.section\t.fixup,\"ax\"\n\t"
196 ".section\t__ex_table,\"a\"\n\t"
197 STR(PTR
)"\t1b, 4b\n\t"
198 STR(PTR
)"\t2b, 4b\n\t"
200 : "=&r" (value
), "=r" (res
)
201 : "r" (addr
), "i" (-EFAULT
));
205 *regptr
= ®s
->regs
[insn
.i_format
.rt
];
209 if (!access_ok(VERIFY_READ
, addr
, 2))
212 __asm__
__volatile__ (
215 "1:\tlbu\t%0, 0(%2)\n"
216 "2:\tlbu\t$1, 1(%2)\n\t"
218 #ifdef __LITTLE_ENDIAN
219 "1:\tlbu\t%0, 1(%2)\n"
220 "2:\tlbu\t$1, 0(%2)\n\t"
226 ".section\t.fixup,\"ax\"\n\t"
230 ".section\t__ex_table,\"a\"\n\t"
231 STR(PTR
)"\t1b, 4b\n\t"
232 STR(PTR
)"\t2b, 4b\n\t"
234 : "=&r" (value
), "=r" (res
)
235 : "r" (addr
), "i" (-EFAULT
));
239 *regptr
= ®s
->regs
[insn
.i_format
.rt
];
245 * A 32-bit kernel might be running on a 64-bit processor. But
246 * if we're on a 32-bit processor and an i-cache incoherency
247 * or race makes us see a 64-bit instruction here the sdl/sdr
248 * would blow up, so for now we don't handle unaligned 64-bit
249 * instructions on 32-bit kernels.
251 if (!access_ok(VERIFY_READ
, addr
, 4))
254 __asm__
__volatile__ (
256 "1:\tlwl\t%0, (%2)\n"
257 "2:\tlwr\t%0, 3(%2)\n\t"
259 #ifdef __LITTLE_ENDIAN
260 "1:\tlwl\t%0, 3(%2)\n"
261 "2:\tlwr\t%0, (%2)\n\t"
263 "dsll\t%0, %0, 32\n\t"
264 "dsrl\t%0, %0, 32\n\t"
266 "3:\t.section\t.fixup,\"ax\"\n\t"
270 ".section\t__ex_table,\"a\"\n\t"
271 STR(PTR
)"\t1b, 4b\n\t"
272 STR(PTR
)"\t2b, 4b\n\t"
274 : "=&r" (value
), "=r" (res
)
275 : "r" (addr
), "i" (-EFAULT
));
279 *regptr
= ®s
->regs
[insn
.i_format
.rt
];
281 #endif /* CONFIG_64BIT */
283 /* Cannot handle 64-bit instructions in 32-bit kernel */
289 * A 32-bit kernel might be running on a 64-bit processor. But
290 * if we're on a 32-bit processor and an i-cache incoherency
291 * or race makes us see a 64-bit instruction here the sdl/sdr
292 * would blow up, so for now we don't handle unaligned 64-bit
293 * instructions on 32-bit kernels.
295 if (!access_ok(VERIFY_READ
, addr
, 8))
298 __asm__
__volatile__ (
300 "1:\tldl\t%0, (%2)\n"
301 "2:\tldr\t%0, 7(%2)\n\t"
303 #ifdef __LITTLE_ENDIAN
304 "1:\tldl\t%0, 7(%2)\n"
305 "2:\tldr\t%0, (%2)\n\t"
308 "3:\t.section\t.fixup,\"ax\"\n\t"
312 ".section\t__ex_table,\"a\"\n\t"
313 STR(PTR
)"\t1b, 4b\n\t"
314 STR(PTR
)"\t2b, 4b\n\t"
316 : "=&r" (value
), "=r" (res
)
317 : "r" (addr
), "i" (-EFAULT
));
321 *regptr
= ®s
->regs
[insn
.i_format
.rt
];
323 #endif /* CONFIG_64BIT */
325 /* Cannot handle 64-bit instructions in 32-bit kernel */
329 if (!access_ok(VERIFY_WRITE
, addr
, 2))
332 value
= regs
->regs
[insn
.i_format
.rt
];
333 __asm__
__volatile__ (
336 "1:\tsb\t%1, 1(%2)\n\t"
338 "2:\tsb\t$1, 0(%2)\n\t"
341 #ifdef __LITTLE_ENDIAN
343 "1:\tsb\t%1, 0(%2)\n\t"
345 "2:\tsb\t$1, 1(%2)\n\t"
350 ".section\t.fixup,\"ax\"\n\t"
354 ".section\t__ex_table,\"a\"\n\t"
355 STR(PTR
)"\t1b, 4b\n\t"
356 STR(PTR
)"\t2b, 4b\n\t"
359 : "r" (value
), "r" (addr
), "i" (-EFAULT
));
365 if (!access_ok(VERIFY_WRITE
, addr
, 4))
368 value
= regs
->regs
[insn
.i_format
.rt
];
369 __asm__
__volatile__ (
372 "2:\tswr\t%1, 3(%2)\n\t"
374 #ifdef __LITTLE_ENDIAN
375 "1:\tswl\t%1, 3(%2)\n"
376 "2:\tswr\t%1, (%2)\n\t"
380 ".section\t.fixup,\"ax\"\n\t"
384 ".section\t__ex_table,\"a\"\n\t"
385 STR(PTR
)"\t1b, 4b\n\t"
386 STR(PTR
)"\t2b, 4b\n\t"
389 : "r" (value
), "r" (addr
), "i" (-EFAULT
));
397 * A 32-bit kernel might be running on a 64-bit processor. But
398 * if we're on a 32-bit processor and an i-cache incoherency
399 * or race makes us see a 64-bit instruction here the sdl/sdr
400 * would blow up, so for now we don't handle unaligned 64-bit
401 * instructions on 32-bit kernels.
403 if (!access_ok(VERIFY_WRITE
, addr
, 8))
406 value
= regs
->regs
[insn
.i_format
.rt
];
407 __asm__
__volatile__ (
410 "2:\tsdr\t%1, 7(%2)\n\t"
412 #ifdef __LITTLE_ENDIAN
413 "1:\tsdl\t%1, 7(%2)\n"
414 "2:\tsdr\t%1, (%2)\n\t"
418 ".section\t.fixup,\"ax\"\n\t"
422 ".section\t__ex_table,\"a\"\n\t"
423 STR(PTR
)"\t1b, 4b\n\t"
424 STR(PTR
)"\t2b, 4b\n\t"
427 : "r" (value
), "r" (addr
), "i" (-EFAULT
));
431 #endif /* CONFIG_64BIT */
433 /* Cannot handle 64-bit instructions in 32-bit kernel */
441 * I herewith declare: this does not happen. So send SIGBUS.
450 * These are the coprocessor 2 load/stores. The current
451 * implementations don't use cp2 and cp2 should always be
452 * disabled in c0_status. So send SIGILL.
453 * (No longer true: The Sony Praystation uses cp2 for
454 * 3D matrix operations. Dunno if that thingy has a MMU ...)
458 * Pheeee... We encountered an yet unknown instruction or
459 * cache coherence problem. Die sucker, die ...
464 #ifdef CONFIG_PROC_FS
465 unaligned_instructions
++;
471 /* Did we have an exception handler installed? */
472 if (fixup_exception(regs
))
475 die_if_kernel ("Unhandled kernel unaligned access", regs
);
476 send_sig(SIGSEGV
, current
, 1);
481 die_if_kernel("Unhandled kernel unaligned access", regs
);
482 send_sig(SIGBUS
, current
, 1);
487 die_if_kernel("Unhandled kernel unaligned access or invalid instruction", regs
);
488 send_sig(SIGILL
, current
, 1);
493 asmlinkage
void do_ade(struct pt_regs
*regs
)
495 unsigned long *regptr
, newval
;
496 extern int do_dsemulret(struct pt_regs
*);
497 unsigned int __user
*pc
;
501 * Address errors may be deliberately induced by the FPU emulator to
502 * retake control of the CPU after executing the instruction in the
503 * delay slot of an emulated branch.
505 /* Terminate if exception was recognized as a delay slot return */
506 if (do_dsemulret(regs
))
509 /* Otherwise handle as normal */
512 * Did we catch a fault trying to load an instruction?
513 * Or are we running in MIPS16 mode?
515 if ((regs
->cp0_badvaddr
== regs
->cp0_epc
) || (regs
->cp0_epc
& 0x1))
518 pc
= (unsigned int __user
*) exception_epc(regs
);
519 if ((current
->thread
.mflags
& MF_FIXADE
) == 0)
523 * Do branch emulation only if we didn't forward the exception.
524 * This is all so but ugly ...
527 if (!user_mode(regs
))
529 if (!emulate_load_store_insn(regs
, (void __user
*)regs
->cp0_badvaddr
, pc
,
531 compute_return_epc(regs
);
533 * Now that branch is evaluated, update the dest
534 * register if necessary
544 die_if_kernel("Kernel unaligned instruction access", regs
);
545 force_sig(SIGBUS
, current
);
548 * XXX On return from the signal handler we should advance the epc