interrupt_gcc.S 2.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899
  1. /*
  2. * Copyright (c) 2006-2021, RT-Thread Development Team
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. *
  6. * Change Logs:
  7. * Date Author Notes
  8. * 2018/10/02 Bernard The first version
  9. * 2018/12/27 Jesven Add SMP schedule
  10. * 2021/02/02 lizhirui Add userspace support
  11. * 2021/12/24 JasonHu Add user setting save/restore
  12. * 2022/10/22 Shell Support kernel mode RVV;
  13. * Rewrite trap handling routine
  14. */
  15. #include "cpuport.h"
  16. #include "encoding.h"
  17. #include "stackframe.h"
  18. .align 2
  19. .global trap_entry
  20. .global debug_check_sp
  21. trap_entry:
  22. // distingush exception from kernel or user
  23. csrrw sp, sscratch, sp
  24. bnez sp, _save_context
  25. // BE REALLY careful with sscratch,
  26. // if it's wrong, we could looping here forever
  27. // or accessing random memory and seeing things totally
  28. // messy after a long time and don't even know why
  29. _from_kernel:
  30. csrr sp, sscratch
  31. j _save_context
  32. _save_context:
  33. SAVE_ALL
  34. // clear sscratch to say 'now in kernel mode'
  35. csrw sscratch, zero
  36. RESTORE_SYS_GP
  37. // now we are ready to enter interrupt / excepiton handler
  38. _distinguish_syscall:
  39. csrr t0, scause
  40. #ifdef RT_USING_SMART
  41. // TODO swap 8 with config macro name
  42. li t1, 8
  43. bne t0, t1, _handle_interrupt_and_exception
  44. call syscall_entry
  45. // syscall never return here
  46. #endif
  47. _handle_interrupt_and_exception:
  48. mv a0, t0
  49. csrrc a1, stval, zero
  50. csrr a2, sepc
  51. // sp as exception frame pointer
  52. mv a3, sp
  53. call handle_trap
  54. _interrupt_exit:
  55. la s0, rt_thread_switch_interrupt_flag
  56. lw s2, 0(s0)
  57. beqz s2, _resume_execution
  58. sw zero, 0(s0)
  59. _context_switch:
  60. la t0, rt_interrupt_from_thread
  61. LOAD a0, 0(t0)
  62. la t0, rt_interrupt_to_thread
  63. LOAD a1, 0(t0)
  64. csrr t0, sstatus
  65. andi t0, t0, ~SSTATUS_SPIE
  66. csrw sstatus, t0
  67. jal rt_hw_context_switch
  68. _resume_execution:
  69. #ifdef RT_USING_SMART
  70. LOAD t0, FRAME_OFF_SSTATUS(sp)
  71. andi t0, t0, SSTATUS_SPP
  72. bnez t0, _resume_kernel
  73. call arch_ret_to_user
  74. #endif
  75. _resume_kernel:
  76. RESTORE_ALL
  77. csrw sscratch, zero
  78. sret
  79. .global rt_hw_interrupt_enable
  80. rt_hw_interrupt_enable:
  81. csrs sstatus, a0 /* restore to old csr */
  82. jr ra
  83. .global rt_hw_interrupt_disable
  84. rt_hw_interrupt_disable:
  85. csrrci a0, sstatus, 2 /* clear SIE */
  86. jr ra