risc_isr_handler.c 2.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107
  1. #include <mips/cpu.h>
  2. #include <mips/hal.h>
  3. #include "hv_chip_Config.h"
  4. #ifdef HV_CONFIG_HARD_FLOAT
  5. /* pointer to the current and old fpu context for lazy context switching */
  6. struct fp64ctx g_stCtx;
  7. /* fpu context of current running task */
  8. struct fp64ctx *g_pstCurrentCtx;
  9. /* fpu context of last task that executed fpu */
  10. struct fp64ctx *g_pstOldCtx;
  11. #endif
  12. void _mips_handle_exception(struct gpctx *pstCtx, int iException)
  13. {
  14. #ifdef HV_CONFIG_HARD_FLOAT
  15. if ( iException == EXC_CPU )
  16. {
  17. mips_bissr(SR_CU1);
  18. pstCtx->status |= SR_CU1;
  19. if ( !g_pstCurrentCtx )
  20. {
  21. g_pstCurrentCtx = &g_stCtx;
  22. if ( g_pstCurrentCtx == NULL )
  23. {
  24. Hv_Chip_DebugUartPuts("assert failed at boot_main _mips_handle_exception.\n");
  25. }
  26. }
  27. /* this means no one exec'd fpu since we last run */
  28. if ( g_pstOldCtx == g_pstCurrentCtx )
  29. {
  30. return;
  31. }
  32. if ( g_pstOldCtx )
  33. {
  34. _fpctx_save( &g_pstOldCtx->fp );
  35. }
  36. _fpctx_load( &g_pstCurrentCtx->fp );
  37. /* next fpu exception must save our context as it's not necessarily the
  38. * next context switch will cause fpu exception and it's very hard for
  39. * any future task to determine which was the last one that performed
  40. * fpu operations. so by saving this pointer now we give this knowledge
  41. * to that future task */
  42. g_pstOldCtx = g_pstCurrentCtx;
  43. }
  44. else
  45. {
  46. __exception_handle( pstCtx, iException );
  47. }
  48. #else
  49. if ( iException == EXC_CPU )
  50. {
  51. mips_bissr(SR_CU1);
  52. pstCtx->status |= SR_CU1;
  53. Hv_Chip_DebugUartPuts("FPU exception.\n");
  54. }
  55. __exception_handle( pstCtx, iException );
  56. #endif
  57. }
  58. __attribute__((long_call)) extern void _risc_interrup_handler(int num);
  59. HV_IRQ_FUNCTION _risc_isr_sw0(void)
  60. {
  61. _risc_interrup_handler(0);
  62. }
  63. HV_IRQ_FUNCTION _risc_isr_sw1(void)
  64. {
  65. _risc_interrup_handler(1);
  66. }
  67. HV_IRQ_FUNCTION _risc_isr_hw0(void)
  68. {
  69. _risc_interrup_handler(2);
  70. }
  71. HV_IRQ_FUNCTION _risc_isr_hw1(void)
  72. {
  73. _risc_interrup_handler(3);
  74. }
  75. HV_IRQ_FUNCTION _risc_isr_hw2(void)
  76. {
  77. _risc_interrup_handler(4);
  78. }
  79. HV_IRQ_FUNCTION _risc_isr_hw3(void)
  80. {
  81. _risc_interrup_handler(5);
  82. }
  83. HV_IRQ_FUNCTION _risc_isr_hw4(void)
  84. {
  85. _risc_interrup_handler(6);
  86. }
  87. HV_IRQ_FUNCTION _risc_isr_hw5(void)
  88. {
  89. _risc_interrup_handler(7);
  90. }