signal-common.h 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152
  1. /*
  2. * This file is subject to the terms and conditions of the GNU General Public
  3. * License. See the file "COPYING" in the main directory of this archive
  4. * for more details.
  5. *
  6. * Copyright (C) 1991, 1992 Linus Torvalds
  7. * Copyright (C) 1994 - 2000 Ralf Baechle
  8. * Copyright (C) 1999, 2000 Silicon Graphics, Inc.
  9. */
  10. static inline int
  11. setup_sigcontext(struct pt_regs *regs, struct sigcontext __user *sc)
  12. {
  13. int err = 0;
  14. int i;
  15. err |= __put_user(regs->cp0_epc, &sc->sc_pc);
  16. err |= __put_user(0, &sc->sc_regs[0]);
  17. for (i = 1; i < 32; i++)
  18. err |= __put_user(regs->regs[i], &sc->sc_regs[i]);
  19. err |= __put_user(regs->hi, &sc->sc_mdhi);
  20. err |= __put_user(regs->lo, &sc->sc_mdlo);
  21. if (cpu_has_dsp) {
  22. err |= __put_user(mfhi1(), &sc->sc_hi1);
  23. err |= __put_user(mflo1(), &sc->sc_lo1);
  24. err |= __put_user(mfhi2(), &sc->sc_hi2);
  25. err |= __put_user(mflo2(), &sc->sc_lo2);
  26. err |= __put_user(mfhi3(), &sc->sc_hi3);
  27. err |= __put_user(mflo3(), &sc->sc_lo3);
  28. err |= __put_user(rddsp(DSP_MASK), &sc->sc_dsp);
  29. }
  30. err |= __put_user(!!used_math(), &sc->sc_used_math);
  31. if (used_math()) {
  32. /*
  33. * Save FPU state to signal context. Signal handler
  34. * will "inherit" current FPU state.
  35. */
  36. preempt_disable();
  37. if (!is_fpu_owner()) {
  38. own_fpu();
  39. restore_fp(current);
  40. }
  41. err |= save_fp_context(sc);
  42. preempt_enable();
  43. }
  44. return err;
  45. }
  46. static inline int
  47. restore_sigcontext(struct pt_regs *regs, struct sigcontext __user *sc)
  48. {
  49. unsigned int used_math;
  50. unsigned long treg;
  51. int err = 0;
  52. int i;
  53. /* Always make any pending restarted system calls return -EINTR */
  54. current_thread_info()->restart_block.fn = do_no_restart_syscall;
  55. err |= __get_user(regs->cp0_epc, &sc->sc_pc);
  56. err |= __get_user(regs->hi, &sc->sc_mdhi);
  57. err |= __get_user(regs->lo, &sc->sc_mdlo);
  58. if (cpu_has_dsp) {
  59. err |= __get_user(treg, &sc->sc_hi1); mthi1(treg);
  60. err |= __get_user(treg, &sc->sc_lo1); mtlo1(treg);
  61. err |= __get_user(treg, &sc->sc_hi2); mthi2(treg);
  62. err |= __get_user(treg, &sc->sc_lo2); mtlo2(treg);
  63. err |= __get_user(treg, &sc->sc_hi3); mthi3(treg);
  64. err |= __get_user(treg, &sc->sc_lo3); mtlo3(treg);
  65. err |= __get_user(treg, &sc->sc_dsp); wrdsp(treg, DSP_MASK);
  66. }
  67. for (i = 1; i < 32; i++)
  68. err |= __get_user(regs->regs[i], &sc->sc_regs[i]);
  69. err |= __get_user(used_math, &sc->sc_used_math);
  70. conditional_used_math(used_math);
  71. preempt_disable();
  72. if (used_math()) {
  73. /* restore fpu context if we have used it before */
  74. own_fpu();
  75. err |= restore_fp_context(sc);
  76. } else {
  77. /* signal handler may have used FPU. Give it up. */
  78. lose_fpu();
  79. }
  80. preempt_enable();
  81. return err;
  82. }
  83. /*
  84. * Determine which stack to use..
  85. */
  86. static inline void __user *
  87. get_sigframe(struct k_sigaction *ka, struct pt_regs *regs, size_t frame_size)
  88. {
  89. unsigned long sp;
  90. /* Default to using normal stack */
  91. sp = regs->regs[29];
  92. /*
  93. * FPU emulator may have it's own trampoline active just
  94. * above the user stack, 16-bytes before the next lowest
  95. * 16 byte boundary. Try to avoid trashing it.
  96. */
  97. sp -= 32;
  98. /* This is the X/Open sanctioned signal stack switching. */
  99. if ((ka->sa.sa_flags & SA_ONSTACK) && (sas_ss_flags (sp) == 0))
  100. sp = current->sas_ss_sp + current->sas_ss_size;
  101. return (void __user *)((sp - frame_size) & (ICACHE_REFILLS_WORKAROUND_WAR ? ~(cpu_icache_line_size()-1) : ALMASK));
  102. }
  103. static inline int install_sigtramp(unsigned int __user *tramp,
  104. unsigned int syscall)
  105. {
  106. int err;
  107. /*
  108. * Set up the return code ...
  109. *
  110. * li v0, __NR__foo_sigreturn
  111. * syscall
  112. */
  113. err = __put_user(0x24020000 + syscall, tramp + 0);
  114. err |= __put_user(0x0000000c , tramp + 1);
  115. if (ICACHE_REFILLS_WORKAROUND_WAR) {
  116. err |= __put_user(0, tramp + 2);
  117. err |= __put_user(0, tramp + 3);
  118. err |= __put_user(0, tramp + 4);
  119. err |= __put_user(0, tramp + 5);
  120. err |= __put_user(0, tramp + 6);
  121. err |= __put_user(0, tramp + 7);
  122. }
  123. flush_cache_sigtramp((unsigned long) tramp);
  124. return err;
  125. }