risc_excpt_handler.c 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512
  1. #include <stdlib.h>
  2. #include <unistd.h>
  3. // #include <stdio.h>
  4. #include <string.h>
  5. #include <mips/cpu.h>
  6. #include <mips/fpa.h>
  7. #include <mips/hal.h>
  8. #include <mips/uhi_syscalls.h>
  9. #include "hv_chip_Config.h"
  10. /* Defined in .ld file */
  11. static char __use_excpt_boot = 1;
  12. static char __flush_to_zero = 1;
  13. /*
  14. * Write a string, a formatted number, then a string.
  15. */
  16. static void putsnds (const char *pre, reg_t value, int digits, const char *post){
  17. char buf[digits];
  18. int shift;
  19. int idx = 0;
  20. if (pre != NULL){
  21. Hv_Chip_DebugUartPuts(pre);
  22. }
  23. for (shift = ((digits - 1) * 4) ; shift >= 0 ; shift -= 4)
  24. buf[idx++] = "0123456789ABCDEF"[(value >> shift) & 0xf];
  25. Hv_Chip_DebugUartWrite (1, buf, digits);
  26. if (post != NULL){
  27. Hv_Chip_DebugUartPuts(post);
  28. }
  29. }
  30. static void putsns (const char *pre, reg_t value, const char *post) {
  31. putsnds (pre, value, sizeof (reg_t) * 2, post);
  32. }
  33. # define WRITE(MSG) Hv_Chip_DebugUartWrite(1, (MSG), strlen (MSG))
  34. # define PUTSNDS(PRE, VALUE, DIGITS, POST) \
  35. putsnds ((PRE), (VALUE), (DIGITS), (POST))
  36. # define PUTSNS(PRE, VALUE, POST) \
  37. putsns ((PRE), (VALUE), (POST))
  38. int32_t _MIPS_HAL_NOMIPS16
  39. __uhi_exception (struct gpctx *ctx, int32_t abi)
  40. {
  41. register struct gpctx *arg1 asm ("$4") = ctx;
  42. register int32_t arg2 asm ("$5") = abi;
  43. register int32_t op asm ("$25") = __MIPS_UHI_EXCEPTION;
  44. register int32_t ret asm ("$2") = __MIPS_UHI_SYSCALL_NUM;
  45. __asm__ __volatile__ (" # %0 = __uhi_exception(%1, %2) op=%3\n"
  46. SYSCALL (__MIPS_UHI_SYSCALL_NUM)
  47. : "+r" (ret)
  48. : "r" (arg1), "r" (arg2), "r" (op)
  49. : "$3");
  50. return ret;
  51. }
  52. /* Forward a UHI SYSCALL operation to SDBPP interface. */
  53. int _MIPS_HAL_NOMIPS16
  54. __uhi_indirect (struct gpctx *ctx)
  55. {
  56. register reg_t arg1 asm ("$4") = ctx->r[C_CTX_REGNO(4)];
  57. register reg_t arg2 asm ("$5") = ctx->r[C_CTX_REGNO(5)];
  58. register reg_t arg3 asm ("$6") = ctx->r[C_CTX_REGNO(6)];
  59. register reg_t arg4 asm ("$7") = ctx->r[C_CTX_REGNO(7)];
  60. register reg_t op asm ("$25") = ctx->r[C_CTX_REGNO(25)];
  61. register reg_t ret1 asm ("$2") = 1;
  62. register reg_t ret2 asm ("$3");
  63. __asm__ __volatile__(" # UHI indirect\n"
  64. "\tsdbbp 1"
  65. : "+r" (ret1), "=r" (ret2), "+r" (arg1), "+r" (arg2)
  66. : "r" (arg3), "r" (arg4), "r" (op));
  67. ctx->r[C_CTX_REGNO(2)] = ret1;
  68. ctx->r[C_CTX_REGNO(3)] = ret2;
  69. ctx->r[C_CTX_REGNO(4)] = arg1;
  70. ctx->r[C_CTX_REGNO(5)] = arg2;
  71. /* Handled, move on. SYSCALL is 4-bytes in all ISAs. */
  72. ctx->epc += 4;
  73. return 1; /* exception handled */
  74. }
  75. #define BT_ABS(X) ((X)>=0?(X):(-(X)))
  76. #define FRAME_DEPTH 50
  77. int __backtrace_mips32(struct gpctx *ctx, int depth)
  78. {
  79. volatile unsigned int *addr;
  80. volatile unsigned int *ra;
  81. volatile unsigned int *sp;
  82. volatile unsigned int ra_offset;
  83. volatile unsigned int stack_size;
  84. volatile unsigned int bt = 0;
  85. unsigned int bt_done = 0;
  86. if(depth < 0)
  87. {
  88. return -1;
  89. }
  90. ra = (volatile unsigned int *)ctx->ra;
  91. sp = (volatile unsigned int *)ctx->sp;
  92. if(((unsigned int)ra >= 0xC0000000) || ((unsigned int)ra < 0x80000000)
  93. || ((unsigned int)sp >= 0xC0000000) || ((unsigned int)sp < 0x80000000)){
  94. return -1;
  95. }
  96. for(bt=0;bt<depth&&ra;bt++)
  97. {
  98. ra_offset = 0;
  99. stack_size = 0;
  100. for(addr=ra; ra_offset==0||stack_size==0; addr--)
  101. {
  102. switch(*addr&0xffff0000)
  103. {
  104. case 0x27bd0000:
  105. stack_size = BT_ABS((short)(*addr&0xffff));
  106. if(stack_size != 0){
  107. PUTSNS("[BackTrace]-[", (reg_t)(bt), NULL);
  108. PUTSNS("]-[0x", (reg_t)addr, NULL);
  109. WRITE ("]\n");
  110. }
  111. if(bt_done){
  112. return bt;
  113. }
  114. break;
  115. case 0xafbf0000:
  116. ra_offset = (short)(*addr&0xffff);
  117. break;
  118. default:
  119. break;
  120. }
  121. if(*addr == 0x0c0002b4){
  122. bt_done = 1;
  123. }
  124. }
  125. ra =(unsigned int *)*((unsigned int *)((unsigned int)sp + ra_offset));
  126. sp =(unsigned int *)((unsigned int)sp + stack_size);
  127. if(((unsigned int)ra >= 0xC0000000) || ((unsigned int)ra < 0x80000000)
  128. || ((unsigned int)sp >= 0xC0000000) || ((unsigned int)sp < 0x80000000)){
  129. return -1;
  130. }
  131. }
  132. return bt;
  133. }
  134. #ifdef SW_DUMMY_DEBUG
  135. int sw_dummy_debug = 0;
  136. void dumpGPR2R(struct gpctx *ctx)
  137. {
  138. if (sw_dummy_debug)
  139. {
  140. sw_dummy_debug = 1;
  141. return;
  142. }
  143. /* Dump registers */
  144. HV_WT32 (SW_DUMMY_AT, ctx->at);
  145. HV_WT32 (SW_DUMMY_V0, ctx->v[0]);
  146. HV_WT32 (SW_DUMMY_V1, ctx->v[1]);
  147. HV_WT32 (SW_DUMMY_A0, ctx->a[0]);
  148. HV_WT32 (SW_DUMMY_A1, ctx->a[1]);
  149. HV_WT32 (SW_DUMMY_A2, ctx->a[2]);
  150. HV_WT32 (SW_DUMMY_A3, ctx->a[3]);
  151. HV_WT32 (SW_DUMMY_T0, ctx->t[0]);
  152. HV_WT32 (SW_DUMMY_T1, ctx->t[1]);
  153. HV_WT32 (SW_DUMMY_T2, ctx->t[2]);
  154. HV_WT32 (SW_DUMMY_T3, ctx->t[3]);
  155. HV_WT32 (SW_DUMMY_T4, ctx->t[4]);
  156. HV_WT32 (SW_DUMMY_T5, ctx->t[5]);
  157. HV_WT32 (SW_DUMMY_T6, ctx->t[6]);
  158. HV_WT32 (SW_DUMMY_T7, ctx->t[7]);
  159. HV_WT32 (SW_DUMMY_S0, ctx->s[0]);
  160. HV_WT32 (SW_DUMMY_S1, ctx->s[1]);
  161. HV_WT32 (SW_DUMMY_S2, ctx->s[2]);
  162. HV_WT32 (SW_DUMMY_S3, ctx->s[3]);
  163. HV_WT32 (SW_DUMMY_S4, ctx->s[4]);
  164. HV_WT32 (SW_DUMMY_S5, ctx->s[5]);
  165. HV_WT32 (SW_DUMMY_S6, ctx->s[6]);
  166. HV_WT32 (SW_DUMMY_S7, ctx->s[7]);
  167. HV_WT32 (SW_DUMMY_T8, ctx->t2[0]);
  168. HV_WT32 (SW_DUMMY_T9, ctx->t2[1]);
  169. HV_WT32 (SW_DUMMY_K0, ctx->k[0]);
  170. HV_WT32 (SW_DUMMY_K1, ctx->k[1]);
  171. HV_WT32 (SW_DUMMY_GP, ctx->gp);
  172. HV_WT32 (SW_DUMMY_SP, ctx->sp);
  173. HV_WT32 (SW_DUMMY_FP, ctx->fp);
  174. HV_WT32 (SW_DUMMY_RA, ctx->ra);
  175. HV_WT32 (SW_DUMMY_EPC, ctx->epc);
  176. HV_WT32 (SW_DUMMY_BADVADDR, ctx->badvaddr);
  177. HV_WT32 (SW_DUMMY_STATUS, ctx->status);
  178. HV_WT32 (SW_DUMMY_CAUSE, ctx->cause);
  179. HV_WT32 (SW_DUMMY_BADPISTR, ctx->badinstr);
  180. HV_WT32 (SW_DUMMY_BADPISTR, ctx->badpinstr);
  181. #ifdef HV_SCALER_DEBUG_VERSION
  182. while(1);
  183. #endif
  184. }
  185. #endif
  186. void dumpGPR(struct gpctx *ctx)
  187. {
  188. /* Dump registers */
  189. PUTSNS (" 0:\t", 0, "\t");
  190. PUTSNS ("at:\t", ctx->at, "\t");
  191. PUTSNS ("v0:\t", ctx->v[0], "\t");
  192. PUTSNS ("v1:\t", ctx->v[1], "\n");
  193. PUTSNS ("a0:\t", ctx->a[0], "\t");
  194. PUTSNS ("a1:\t", ctx->a[1], "\t");
  195. PUTSNS ("a2:\t", ctx->a[2], "\t");
  196. PUTSNS ("a3:\t", ctx->a[3], "\n");
  197. PUTSNS ("t0:\t", ctx->t[0], "\t");
  198. PUTSNS ("t1:\t", ctx->t[1], "\t");
  199. PUTSNS ("t2:\t", ctx->t[2], "\t");
  200. PUTSNS ("t3:\t", ctx->t[3], "\n");
  201. PUTSNS ("t4:\t", ctx->t[4], "\t");
  202. PUTSNS ("t5:\t", ctx->t[5], "\t");
  203. PUTSNS ("t6:\t", ctx->t[6], "\t");
  204. PUTSNS ("t7:\t", ctx->t[7], "\n");
  205. PUTSNS ("s0:\t", ctx->s[0], "\t");
  206. PUTSNS ("s1:\t", ctx->s[1], "\t");
  207. PUTSNS ("s2:\t", ctx->s[2], "\t");
  208. PUTSNS ("s3:\t", ctx->s[3], "\n");
  209. PUTSNS ("s4:\t", ctx->s[4], "\t");
  210. PUTSNS ("s5:\t", ctx->s[5], "\t");
  211. PUTSNS ("s6:\t", ctx->s[6], "\t");
  212. PUTSNS ("s7:\t", ctx->s[7], "\n");
  213. PUTSNS ("t8:\t", ctx->t2[0],"\t");
  214. PUTSNS ("t9:\t", ctx->t2[1],"\t");
  215. PUTSNS ("k0:\t", ctx->k[0], "\t");
  216. PUTSNS ("k1:\t", ctx->k[1], "\n");
  217. PUTSNS ("gp:\t", ctx->gp, "\t");
  218. PUTSNS ("sp:\t", ctx->sp, "\t");
  219. PUTSNS ("fp:\t", ctx->fp, "\t");
  220. PUTSNS ("ra:\t", ctx->ra, "\n");
  221. #if __mips_isa_rev < 6
  222. PUTSNS ("hi:\t", ctx->hi, "\t");
  223. PUTSNS ("lo:\t", ctx->lo, "\n");
  224. #endif
  225. PUTSNS ("epc: \t", ctx->epc, "\n");
  226. PUTSNS ("BadVAddr: \t", ctx->badvaddr, "\n");
  227. PUTSNDS ("Status: \t", ctx->status, 8, "\n");
  228. PUTSNDS ("Cause: \t", ctx->cause, 8, "\n");
  229. PUTSNDS ("BadInstr: \t", ctx->badinstr, 8, "\n");
  230. PUTSNDS ("BadPInstr:\t", ctx->badpinstr,8, "\n");
  231. __backtrace_mips32(ctx, FRAME_DEPTH);
  232. #ifdef SW_DUMMY_DEBUG
  233. dumpGPR2R(ctx);
  234. #endif
  235. }
  236. /* Handle an exception */
  237. void __exception_handle(struct gpctx *ctx, int exception)
  238. {
  239. switch (exception)
  240. {
  241. case EXC_MOD:
  242. {
  243. Hv_Chip_DebugUartPuts ("TLB modification exception\n");
  244. break;
  245. }
  246. case EXC_TLBL:
  247. {
  248. Hv_Chip_DebugUartPuts ("TLB error on load from ");
  249. dumpGPR(ctx);
  250. //PUTSNS (" @0x", ctx->epc, "\n");
  251. break;
  252. }
  253. case EXC_TLBS:
  254. {
  255. Hv_Chip_DebugUartPuts ("TLB error on store to ");
  256. dumpGPR(ctx);
  257. //PUTSNS (" @0x", ctx->epc, "\n");
  258. break;
  259. }
  260. case EXC_ADEL:
  261. {
  262. Hv_Chip_DebugUartPuts ("Address error on load from \n");
  263. dumpGPR(ctx);
  264. //PUTSNS (" @0x", ctx->epc, "\n");
  265. break;
  266. }
  267. case EXC_ADES:
  268. {
  269. Hv_Chip_DebugUartPuts ("Address error on store to \n");
  270. dumpGPR(ctx);
  271. //PUTSNS (" @0x", ctx->epc, "\n");
  272. break;
  273. }
  274. case EXC_IBE:
  275. {
  276. Hv_Chip_DebugUartPuts ("Instruction bus error\n");
  277. break;
  278. }
  279. case EXC_DBE:
  280. {
  281. Hv_Chip_DebugUartPuts ("Data bus error\n");
  282. dumpGPR(ctx);
  283. break;
  284. }
  285. case EXC_SYS:
  286. {
  287. /* Process a UHI SYSCALL, all other SYSCALLs should have been processed
  288. by our caller. __use_excpt_boot has following values:
  289. 0 = Do not use exception handler present in boot.
  290. 1 = Use exception handler present in boot if BEV
  291. is 0 at startup.
  292. 2 = Always use exception handler present in boot. */
  293. /* Special handling for boot/low level failures. */
  294. if (ctx->t2[1] == __MIPS_UHI_BOOTFAIL)
  295. {
  296. switch (ctx->a[0])
  297. {
  298. case __MIPS_UHI_BF_CACHE:
  299. {
  300. Hv_Chip_DebugUartPuts ("L2 cache configuration error\n");
  301. break;
  302. }
  303. default:
  304. {
  305. Hv_Chip_DebugUartPuts ("Unknown boot failure error\n");
  306. break;
  307. }
  308. }
  309. /* These are unrecoverable. Abort. */
  310. ctx->epc = (sreg_t)(long)&__exit;
  311. /* Exit code of 255 */
  312. ctx->a[0] = 0xff;
  313. return;
  314. }
  315. if (((long) __use_excpt_boot == 2
  316. || ((long) __use_excpt_boot == 1
  317. && __get_startup_BEV
  318. && __get_startup_BEV () == 0))
  319. && __chain_uhi_excpt)
  320. {
  321. /* This will not return. */
  322. __chain_uhi_excpt (ctx);
  323. }
  324. else
  325. {
  326. __uhi_indirect (ctx);
  327. }
  328. return;
  329. }
  330. case EXC_BP:
  331. {
  332. /* Return from exception handler if breakpoint is handled. */
  333. if (__uhi_break && __uhi_break (ctx))
  334. return;
  335. Hv_Chip_DebugUartPuts ("Breakpoint \n");
  336. break;
  337. }
  338. case EXC_RI:
  339. {
  340. Hv_Chip_DebugUartPuts ("Illegal instruction \n");
  341. break;
  342. }
  343. case EXC_CPU:
  344. {
  345. Hv_Chip_DebugUartPuts ("Coprocessor unusable\n");
  346. break;
  347. }
  348. case EXC_OVF:
  349. {
  350. Hv_Chip_DebugUartPuts ("Overflow\n");
  351. break;
  352. }
  353. case EXC_TRAP:
  354. {
  355. dumpGPR(ctx);
  356. Hv_Chip_DebugUartPuts ("Trap\n");
  357. break;
  358. }
  359. case EXC_MSAFPE:
  360. {
  361. Hv_Chip_DebugUartPuts ("MSA Floating point error\n");
  362. break;
  363. }
  364. case EXC_FPE:
  365. {
  366. /* Turn on flush to zero the first time we hit an unimplemented
  367. operation. If we hit it again then stop. */
  368. if (__flush_to_zero
  369. && (fpa_getsr () & FPA_CSR_UNI_X)
  370. && (fpa_getsr () & FPA_CSR_FS) == 0)
  371. {
  372. unsigned int sr = fpa_getsr ();
  373. sr &= ~FPA_CSR_UNI_X;
  374. sr |= FPA_CSR_FS;
  375. fpa_setsr (sr);
  376. return;
  377. }
  378. //Hv_Chip_DebugUartPuts ("Floating point error\n");
  379. break;
  380. }
  381. case EXC_IS1:
  382. {
  383. //Hv_Chip_DebugUartPuts ("Implementation specific exception (16)\n");
  384. break;
  385. }
  386. case EXC_IS2:
  387. {
  388. //Hv_Chip_DebugUartPuts ("Implementation specific exception (17)\n");
  389. break;
  390. }
  391. case EXC_C2E:
  392. {
  393. Hv_Chip_DebugUartPuts ("Precise Coprocessor 2 exception\n");
  394. break;
  395. }
  396. case EXC_TLBRI:
  397. {
  398. Hv_Chip_DebugUartPuts ("TLB read inhibit exception\n");
  399. break;
  400. }
  401. case EXC_TLBXI:
  402. {
  403. Hv_Chip_DebugUartPuts ("TLB execute inhibit exception\n");
  404. break;
  405. }
  406. case EXC_MSAU:
  407. {
  408. //Hv_Chip_DebugUartPuts ("MSA unusable \n");
  409. break;
  410. }
  411. case EXC_MDMX:
  412. {
  413. //Hv_Chip_DebugUartPuts ("MDMX exception \n");
  414. break;
  415. }
  416. case EXC_WATCH:
  417. {
  418. //Hv_Chip_DebugUartPuts ("Watchpoint\n");
  419. break;
  420. }
  421. case EXC_MCHECK:
  422. {
  423. //Hv_Chip_DebugUartPuts ("Machine check error\n");
  424. break;
  425. }
  426. case EXC_THREAD:
  427. {
  428. Hv_Chip_DebugUartPuts ("Thread exception\n");
  429. break;
  430. }
  431. case EXC_DSPU:
  432. {
  433. //Hv_Chip_DebugUartPuts ("DSP unusable\n");
  434. break;
  435. }
  436. case EXC_RES30:
  437. {
  438. //Hv_Chip_DebugUartPuts ("Cache error\n");
  439. break;
  440. }
  441. default:
  442. {
  443. //Hv_Chip_DebugUartPuts ("Unhandled exception " "\n");
  444. break;
  445. }
  446. }
  447. /* Raise UHI exception which may or may not return. */
  448. if (__uhi_exception (ctx, UHI_ABI) != 0)
  449. {
  450. /* The exception was acknowledged but not handled. Abort. */
  451. ctx->epc = (sreg_t)(long)&__exit;
  452. /* Exit code of 255 */
  453. ctx->a[0] = 0xff;
  454. }
  455. return;
  456. }