fixup.S 2.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293
  1. /*
  2. * Linux/PA-RISC Project (http://www.parisc-linux.org/)
  3. *
  4. * Copyright (C) 2004 Randolph Chung <tausq@debian.org>
  5. *
  6. * This program is free software; you can redistribute it and/or modify
  7. * it under the terms of the GNU General Public License as published by
  8. * the Free Software Foundation; either version 2, or (at your option)
  9. * any later version.
  10. *
  11. * This program is distributed in the hope that it will be useful,
  12. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  13. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  14. * GNU General Public License for more details.
  15. *
  16. * You should have received a copy of the GNU General Public License
  17. * along with this program; if not, write to the Free Software
  18. * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
  19. *
  20. * Fixup routines for kernel exception handling.
  21. */
  22. #include <asm/asm-offsets.h>
  23. #include <asm/assembly.h>
  24. #include <asm/errno.h>
  25. #include <linux/linkage.h>
  26. #ifdef CONFIG_SMP
  27. .macro get_fault_ip t1 t2
  28. addil LT%__per_cpu_offset,%r27
  29. LDREG RT%__per_cpu_offset(%r1),\t1
  30. /* t2 = smp_processor_id() */
  31. mfctl 30,\t2
  32. ldw TI_CPU(\t2),\t2
  33. #ifdef CONFIG_64BIT
  34. extrd,u \t2,63,32,\t2
  35. #endif
  36. /* t2 = &__per_cpu_offset[smp_processor_id()]; */
  37. LDREGX \t2(\t1),\t2
  38. addil LT%exception_data,%r27
  39. LDREG RT%exception_data(%r1),\t1
  40. /* t1 = &__get_cpu_var(exception_data) */
  41. add,l \t1,\t2,\t1
  42. /* t1 = t1->fault_ip */
  43. LDREG EXCDATA_IP(\t1), \t1
  44. .endm
  45. #else
  46. .macro get_fault_ip t1 t2
  47. /* t1 = &__get_cpu_var(exception_data) */
  48. addil LT%exception_data,%r27
  49. LDREG RT%exception_data(%r1),\t2
  50. /* t1 = t2->fault_ip */
  51. LDREG EXCDATA_IP(\t2), \t1
  52. .endm
  53. #endif
  54. .level LEVEL
  55. .text
  56. .section .fixup, "ax"
  57. /* get_user() fixups, store -EFAULT in r8, and 0 in r9 */
  58. ENTRY(fixup_get_user_skip_1)
  59. get_fault_ip %r1,%r8
  60. ldo 4(%r1), %r1
  61. ldi -EFAULT, %r8
  62. bv %r0(%r1)
  63. copy %r0, %r9
  64. ENDPROC(fixup_get_user_skip_1)
  65. ENTRY(fixup_get_user_skip_2)
  66. get_fault_ip %r1,%r8
  67. ldo 8(%r1), %r1
  68. ldi -EFAULT, %r8
  69. bv %r0(%r1)
  70. copy %r0, %r9
  71. ENDPROC(fixup_get_user_skip_2)
  72. /* put_user() fixups, store -EFAULT in r8 */
  73. ENTRY(fixup_put_user_skip_1)
  74. get_fault_ip %r1,%r8
  75. ldo 4(%r1), %r1
  76. bv %r0(%r1)
  77. ldi -EFAULT, %r8
  78. ENDPROC(fixup_put_user_skip_1)
  79. ENTRY(fixup_put_user_skip_2)
  80. get_fault_ip %r1,%r8
  81. ldo 8(%r1), %r1
  82. bv %r0(%r1)
  83. ldi -EFAULT, %r8
  84. ENDPROC(fixup_put_user_skip_2)