ptrace-tm-tar.c 3.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157
  1. // SPDX-License-Identifier: GPL-2.0-or-later
  2. /*
  3. * Ptrace test for TAR, PPR, DSCR registers in the TM context
  4. *
  5. * Copyright (C) 2015 Anshuman Khandual, IBM Corporation.
  6. */
  7. #include "ptrace.h"
  8. #include "tm.h"
  9. #include "ptrace-tar.h"
  10. int shm_id;
  11. unsigned long *cptr, *pptr;
  12. void tm_tar(void)
  13. {
  14. unsigned long result, texasr;
  15. unsigned long regs[3];
  16. int ret;
  17. cptr = (unsigned long *)shmat(shm_id, NULL, 0);
  18. trans:
  19. cptr[1] = 0;
  20. asm __volatile__(
  21. "li 4, %[tar_1];"
  22. "mtspr %[sprn_tar], 4;" /* TAR_1 */
  23. "li 4, %[dscr_1];"
  24. "mtspr %[sprn_dscr], 4;" /* DSCR_1 */
  25. "or 31,31,31;" /* PPR_1*/
  26. "1: ;"
  27. "tbegin.;"
  28. "beq 2f;"
  29. "li 4, %[tar_2];"
  30. "mtspr %[sprn_tar], 4;" /* TAR_2 */
  31. "li 4, %[dscr_2];"
  32. "mtspr %[sprn_dscr], 4;" /* DSCR_2 */
  33. "or 1,1,1;" /* PPR_2 */
  34. "tsuspend.;"
  35. "li 0, 1;"
  36. "stw 0, 0(%[cptr1]);"
  37. "tresume.;"
  38. "b .;"
  39. "tend.;"
  40. "li 0, 0;"
  41. "ori %[res], 0, 0;"
  42. "b 3f;"
  43. /* Transaction abort handler */
  44. "2: ;"
  45. "li 0, 1;"
  46. "ori %[res], 0, 0;"
  47. "mfspr %[texasr], %[sprn_texasr];"
  48. "3: ;"
  49. : [res] "=r" (result), [texasr] "=r" (texasr)
  50. : [sprn_dscr]"i"(SPRN_DSCR), [sprn_tar]"i"(SPRN_TAR),
  51. [sprn_ppr]"i"(SPRN_PPR), [sprn_texasr]"i"(SPRN_TEXASR),
  52. [tar_1]"i"(TAR_1), [dscr_1]"i"(DSCR_1), [tar_2]"i"(TAR_2),
  53. [dscr_2]"i"(DSCR_2), [cptr1] "b" (&cptr[1])
  54. : "memory", "r0", "r3", "r4", "r5", "r6"
  55. );
  56. /* TM failed, analyse */
  57. if (result) {
  58. if (!cptr[0])
  59. goto trans;
  60. regs[0] = mfspr(SPRN_TAR);
  61. regs[1] = mfspr(SPRN_PPR);
  62. regs[2] = mfspr(SPRN_DSCR);
  63. shmdt(&cptr);
  64. printf("%-30s TAR: %lu PPR: %lx DSCR: %lu\n",
  65. user_read, regs[0], regs[1], regs[2]);
  66. ret = validate_tar_registers(regs, TAR_4, PPR_4, DSCR_4);
  67. if (ret)
  68. exit(1);
  69. exit(0);
  70. }
  71. shmdt(&cptr);
  72. exit(1);
  73. }
  74. int trace_tm_tar(pid_t child)
  75. {
  76. unsigned long regs[3];
  77. FAIL_IF(start_trace(child));
  78. FAIL_IF(show_tar_registers(child, regs));
  79. printf("%-30s TAR: %lu PPR: %lx DSCR: %lu\n",
  80. ptrace_read_running, regs[0], regs[1], regs[2]);
  81. FAIL_IF(validate_tar_registers(regs, TAR_2, PPR_2, DSCR_2));
  82. FAIL_IF(show_tm_checkpointed_state(child, regs));
  83. printf("%-30s TAR: %lu PPR: %lx DSCR: %lu\n",
  84. ptrace_read_ckpt, regs[0], regs[1], regs[2]);
  85. FAIL_IF(validate_tar_registers(regs, TAR_1, PPR_1, DSCR_1));
  86. FAIL_IF(write_ckpt_tar_registers(child, TAR_4, PPR_4, DSCR_4));
  87. printf("%-30s TAR: %u PPR: %lx DSCR: %u\n",
  88. ptrace_write_ckpt, TAR_4, PPR_4, DSCR_4);
  89. pptr[0] = 1;
  90. FAIL_IF(stop_trace(child));
  91. return TEST_PASS;
  92. }
  93. int ptrace_tm_tar(void)
  94. {
  95. pid_t pid;
  96. int ret, status;
  97. SKIP_IF(!have_htm());
  98. SKIP_IF(htm_is_synthetic());
  99. shm_id = shmget(IPC_PRIVATE, sizeof(int) * 2, 0777|IPC_CREAT);
  100. pid = fork();
  101. if (pid == 0)
  102. tm_tar();
  103. pptr = (unsigned long *)shmat(shm_id, NULL, 0);
  104. pptr[0] = 0;
  105. if (pid) {
  106. while (!pptr[1])
  107. asm volatile("" : : : "memory");
  108. ret = trace_tm_tar(pid);
  109. if (ret) {
  110. kill(pid, SIGTERM);
  111. shmdt(&pptr);
  112. shmctl(shm_id, IPC_RMID, NULL);
  113. return TEST_FAIL;
  114. }
  115. shmdt(&pptr);
  116. ret = wait(&status);
  117. shmctl(shm_id, IPC_RMID, NULL);
  118. if (ret != pid) {
  119. printf("Child's exit status not captured\n");
  120. return TEST_FAIL;
  121. }
  122. return (WIFEXITED(status) && WEXITSTATUS(status)) ? TEST_FAIL :
  123. TEST_PASS;
  124. }
  125. return TEST_PASS;
  126. }
  127. int main(int argc, char *argv[])
  128. {
  129. return test_harness(ptrace_tm_tar, "ptrace_tm_tar");
  130. }