fixup.S 2.4 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091
  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 <linux/config.h>
  23. #include <asm/asm-offsets.h>
  24. #include <asm/assembly.h>
  25. #include <asm/errno.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 __LP64__
  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%per_cpu__exception_data,%r27
  39. LDREG RT%per_cpu__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%per_cpu__exception_data,%r27
  49. LDREG RT%per_cpu__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. .export fixup_get_user_skip_1
  59. fixup_get_user_skip_1:
  60. get_fault_ip %r1,%r8
  61. ldo 4(%r1), %r1
  62. ldi -EFAULT, %r8
  63. bv %r0(%r1)
  64. copy %r0, %r9
  65. .export fixup_get_user_skip_2
  66. fixup_get_user_skip_2:
  67. get_fault_ip %r1,%r8
  68. ldo 8(%r1), %r1
  69. ldi -EFAULT, %r8
  70. bv %r0(%r1)
  71. copy %r0, %r9
  72. /* put_user() fixups, store -EFAULT in r8 */
  73. .export fixup_put_user_skip_1
  74. fixup_put_user_skip_1:
  75. get_fault_ip %r1,%r8
  76. ldo 4(%r1), %r1
  77. bv %r0(%r1)
  78. ldi -EFAULT, %r8
  79. .export fixup_put_user_skip_2
  80. fixup_put_user_skip_2:
  81. get_fault_ip %r1,%r8
  82. ldo 8(%r1), %r1
  83. bv %r0(%r1)
  84. ldi -EFAULT, %r8