trace.c 5.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235
  1. // SPDX-License-Identifier: GPL-2.0
  2. /*
  3. * NVM Express target device driver tracepoints
  4. * Copyright (c) 2018 Johannes Thumshirn, SUSE Linux GmbH
  5. */
  6. #include <asm/unaligned.h>
  7. #include "trace.h"
  8. static const char *nvmet_trace_admin_identify(struct trace_seq *p, u8 *cdw10)
  9. {
  10. const char *ret = trace_seq_buffer_ptr(p);
  11. u8 cns = cdw10[0];
  12. u16 ctrlid = get_unaligned_le16(cdw10 + 2);
  13. trace_seq_printf(p, "cns=%u, ctrlid=%u", cns, ctrlid);
  14. trace_seq_putc(p, 0);
  15. return ret;
  16. }
  17. static const char *nvmet_trace_admin_get_features(struct trace_seq *p,
  18. u8 *cdw10)
  19. {
  20. const char *ret = trace_seq_buffer_ptr(p);
  21. u8 fid = cdw10[0];
  22. u8 sel = cdw10[1] & 0x7;
  23. u32 cdw11 = get_unaligned_le32(cdw10 + 4);
  24. trace_seq_printf(p, "fid=0x%x, sel=0x%x, cdw11=0x%x", fid, sel, cdw11);
  25. trace_seq_putc(p, 0);
  26. return ret;
  27. }
  28. static const char *nvmet_trace_get_lba_status(struct trace_seq *p,
  29. u8 *cdw10)
  30. {
  31. const char *ret = trace_seq_buffer_ptr(p);
  32. u64 slba = get_unaligned_le64(cdw10);
  33. u32 mndw = get_unaligned_le32(cdw10 + 8);
  34. u16 rl = get_unaligned_le16(cdw10 + 12);
  35. u8 atype = cdw10[15];
  36. trace_seq_printf(p, "slba=0x%llx, mndw=0x%x, rl=0x%x, atype=%u",
  37. slba, mndw, rl, atype);
  38. trace_seq_putc(p, 0);
  39. return ret;
  40. }
  41. static const char *nvmet_trace_admin_set_features(struct trace_seq *p,
  42. u8 *cdw10)
  43. {
  44. const char *ret = trace_seq_buffer_ptr(p);
  45. u8 fid = cdw10[0];
  46. u8 sv = cdw10[3] & 0x8;
  47. u32 cdw11 = get_unaligned_le32(cdw10 + 4);
  48. trace_seq_printf(p, "fid=0x%x, sv=0x%x, cdw11=0x%x", fid, sv, cdw11);
  49. trace_seq_putc(p, 0);
  50. return ret;
  51. }
  52. static const char *nvmet_trace_read_write(struct trace_seq *p, u8 *cdw10)
  53. {
  54. const char *ret = trace_seq_buffer_ptr(p);
  55. u64 slba = get_unaligned_le64(cdw10);
  56. u16 length = get_unaligned_le16(cdw10 + 8);
  57. u16 control = get_unaligned_le16(cdw10 + 10);
  58. u32 dsmgmt = get_unaligned_le32(cdw10 + 12);
  59. u32 reftag = get_unaligned_le32(cdw10 + 16);
  60. trace_seq_printf(p,
  61. "slba=%llu, len=%u, ctrl=0x%x, dsmgmt=%u, reftag=%u",
  62. slba, length, control, dsmgmt, reftag);
  63. trace_seq_putc(p, 0);
  64. return ret;
  65. }
  66. static const char *nvmet_trace_dsm(struct trace_seq *p, u8 *cdw10)
  67. {
  68. const char *ret = trace_seq_buffer_ptr(p);
  69. trace_seq_printf(p, "nr=%u, attributes=%u",
  70. get_unaligned_le32(cdw10),
  71. get_unaligned_le32(cdw10 + 4));
  72. trace_seq_putc(p, 0);
  73. return ret;
  74. }
  75. static const char *nvmet_trace_common(struct trace_seq *p, u8 *cdw10)
  76. {
  77. const char *ret = trace_seq_buffer_ptr(p);
  78. trace_seq_printf(p, "cdw10=%*ph", 24, cdw10);
  79. trace_seq_putc(p, 0);
  80. return ret;
  81. }
  82. const char *nvmet_trace_parse_admin_cmd(struct trace_seq *p,
  83. u8 opcode, u8 *cdw10)
  84. {
  85. switch (opcode) {
  86. case nvme_admin_identify:
  87. return nvmet_trace_admin_identify(p, cdw10);
  88. case nvme_admin_set_features:
  89. return nvmet_trace_admin_set_features(p, cdw10);
  90. case nvme_admin_get_features:
  91. return nvmet_trace_admin_get_features(p, cdw10);
  92. case nvme_admin_get_lba_status:
  93. return nvmet_trace_get_lba_status(p, cdw10);
  94. default:
  95. return nvmet_trace_common(p, cdw10);
  96. }
  97. }
  98. const char *nvmet_trace_parse_nvm_cmd(struct trace_seq *p,
  99. u8 opcode, u8 *cdw10)
  100. {
  101. switch (opcode) {
  102. case nvme_cmd_read:
  103. case nvme_cmd_write:
  104. case nvme_cmd_write_zeroes:
  105. return nvmet_trace_read_write(p, cdw10);
  106. case nvme_cmd_dsm:
  107. return nvmet_trace_dsm(p, cdw10);
  108. default:
  109. return nvmet_trace_common(p, cdw10);
  110. }
  111. }
  112. static const char *nvmet_trace_fabrics_property_set(struct trace_seq *p,
  113. u8 *spc)
  114. {
  115. const char *ret = trace_seq_buffer_ptr(p);
  116. u8 attrib = spc[0];
  117. u32 ofst = get_unaligned_le32(spc + 4);
  118. u64 value = get_unaligned_le64(spc + 8);
  119. trace_seq_printf(p, "attrib=%u, ofst=0x%x, value=0x%llx",
  120. attrib, ofst, value);
  121. trace_seq_putc(p, 0);
  122. return ret;
  123. }
  124. static const char *nvmet_trace_fabrics_connect(struct trace_seq *p,
  125. u8 *spc)
  126. {
  127. const char *ret = trace_seq_buffer_ptr(p);
  128. u16 recfmt = get_unaligned_le16(spc);
  129. u16 qid = get_unaligned_le16(spc + 2);
  130. u16 sqsize = get_unaligned_le16(spc + 4);
  131. u8 cattr = spc[6];
  132. u32 kato = get_unaligned_le32(spc + 8);
  133. trace_seq_printf(p, "recfmt=%u, qid=%u, sqsize=%u, cattr=%u, kato=%u",
  134. recfmt, qid, sqsize, cattr, kato);
  135. trace_seq_putc(p, 0);
  136. return ret;
  137. }
  138. static const char *nvmet_trace_fabrics_property_get(struct trace_seq *p,
  139. u8 *spc)
  140. {
  141. const char *ret = trace_seq_buffer_ptr(p);
  142. u8 attrib = spc[0];
  143. u32 ofst = get_unaligned_le32(spc + 4);
  144. trace_seq_printf(p, "attrib=%u, ofst=0x%x", attrib, ofst);
  145. trace_seq_putc(p, 0);
  146. return ret;
  147. }
  148. static const char *nvmet_trace_fabrics_common(struct trace_seq *p, u8 *spc)
  149. {
  150. const char *ret = trace_seq_buffer_ptr(p);
  151. trace_seq_printf(p, "specific=%*ph", 24, spc);
  152. trace_seq_putc(p, 0);
  153. return ret;
  154. }
  155. const char *nvmet_trace_parse_fabrics_cmd(struct trace_seq *p,
  156. u8 fctype, u8 *spc)
  157. {
  158. switch (fctype) {
  159. case nvme_fabrics_type_property_set:
  160. return nvmet_trace_fabrics_property_set(p, spc);
  161. case nvme_fabrics_type_connect:
  162. return nvmet_trace_fabrics_connect(p, spc);
  163. case nvme_fabrics_type_property_get:
  164. return nvmet_trace_fabrics_property_get(p, spc);
  165. default:
  166. return nvmet_trace_fabrics_common(p, spc);
  167. }
  168. }
  169. const char *nvmet_trace_disk_name(struct trace_seq *p, char *name)
  170. {
  171. const char *ret = trace_seq_buffer_ptr(p);
  172. if (*name)
  173. trace_seq_printf(p, "disk=%s, ", name);
  174. trace_seq_putc(p, 0);
  175. return ret;
  176. }
  177. const char *nvmet_trace_ctrl_name(struct trace_seq *p, struct nvmet_ctrl *ctrl)
  178. {
  179. const char *ret = trace_seq_buffer_ptr(p);
  180. /*
  181. * XXX: We don't know the controller instance before executing the
  182. * connect command itself because the connect command for the admin
  183. * queue will not provide the cntlid which will be allocated in this
  184. * command. In case of io queues, the controller instance will be
  185. * mapped by the extra data of the connect command.
  186. * If we can know the extra data of the connect command in this stage,
  187. * we can update this print statement later.
  188. */
  189. if (ctrl)
  190. trace_seq_printf(p, "%d", ctrl->cntlid);
  191. else
  192. trace_seq_printf(p, "_");
  193. trace_seq_putc(p, 0);
  194. return ret;
  195. }