trace.c 9.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357
  1. // SPDX-License-Identifier: GPL-2.0
  2. /*
  3. * NVM Express 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 *nvme_trace_delete_sq(struct trace_seq *p, u8 *cdw10)
  9. {
  10. const char *ret = trace_seq_buffer_ptr(p);
  11. u16 sqid = get_unaligned_le16(cdw10);
  12. trace_seq_printf(p, "sqid=%u", sqid);
  13. trace_seq_putc(p, 0);
  14. return ret;
  15. }
  16. static const char *nvme_trace_create_sq(struct trace_seq *p, u8 *cdw10)
  17. {
  18. const char *ret = trace_seq_buffer_ptr(p);
  19. u16 sqid = get_unaligned_le16(cdw10);
  20. u16 qsize = get_unaligned_le16(cdw10 + 2);
  21. u16 sq_flags = get_unaligned_le16(cdw10 + 4);
  22. u16 cqid = get_unaligned_le16(cdw10 + 6);
  23. trace_seq_printf(p, "sqid=%u, qsize=%u, sq_flags=0x%x, cqid=%u",
  24. sqid, qsize, sq_flags, cqid);
  25. trace_seq_putc(p, 0);
  26. return ret;
  27. }
  28. static const char *nvme_trace_delete_cq(struct trace_seq *p, u8 *cdw10)
  29. {
  30. const char *ret = trace_seq_buffer_ptr(p);
  31. u16 cqid = get_unaligned_le16(cdw10);
  32. trace_seq_printf(p, "cqid=%u", cqid);
  33. trace_seq_putc(p, 0);
  34. return ret;
  35. }
  36. static const char *nvme_trace_create_cq(struct trace_seq *p, u8 *cdw10)
  37. {
  38. const char *ret = trace_seq_buffer_ptr(p);
  39. u16 cqid = get_unaligned_le16(cdw10);
  40. u16 qsize = get_unaligned_le16(cdw10 + 2);
  41. u16 cq_flags = get_unaligned_le16(cdw10 + 4);
  42. u16 irq_vector = get_unaligned_le16(cdw10 + 6);
  43. trace_seq_printf(p, "cqid=%u, qsize=%u, cq_flags=0x%x, irq_vector=%u",
  44. cqid, qsize, cq_flags, irq_vector);
  45. trace_seq_putc(p, 0);
  46. return ret;
  47. }
  48. static const char *nvme_trace_admin_identify(struct trace_seq *p, u8 *cdw10)
  49. {
  50. const char *ret = trace_seq_buffer_ptr(p);
  51. u8 cns = cdw10[0];
  52. u16 ctrlid = get_unaligned_le16(cdw10 + 2);
  53. trace_seq_printf(p, "cns=%u, ctrlid=%u", cns, ctrlid);
  54. trace_seq_putc(p, 0);
  55. return ret;
  56. }
  57. static const char *nvme_trace_admin_set_features(struct trace_seq *p,
  58. u8 *cdw10)
  59. {
  60. const char *ret = trace_seq_buffer_ptr(p);
  61. u8 fid = cdw10[0];
  62. u8 sv = cdw10[3] & 0x8;
  63. u32 cdw11 = get_unaligned_le32(cdw10 + 4);
  64. trace_seq_printf(p, "fid=0x%x, sv=0x%x, cdw11=0x%x", fid, sv, cdw11);
  65. trace_seq_putc(p, 0);
  66. return ret;
  67. }
  68. static const char *nvme_trace_admin_get_features(struct trace_seq *p,
  69. u8 *cdw10)
  70. {
  71. const char *ret = trace_seq_buffer_ptr(p);
  72. u8 fid = cdw10[0];
  73. u8 sel = cdw10[1] & 0x7;
  74. u32 cdw11 = get_unaligned_le32(cdw10 + 4);
  75. trace_seq_printf(p, "fid=0x%x, sel=0x%x, cdw11=0x%x", fid, sel, cdw11);
  76. trace_seq_putc(p, 0);
  77. return ret;
  78. }
  79. static const char *nvme_trace_get_lba_status(struct trace_seq *p,
  80. u8 *cdw10)
  81. {
  82. const char *ret = trace_seq_buffer_ptr(p);
  83. u64 slba = get_unaligned_le64(cdw10);
  84. u32 mndw = get_unaligned_le32(cdw10 + 8);
  85. u16 rl = get_unaligned_le16(cdw10 + 12);
  86. u8 atype = cdw10[15];
  87. trace_seq_printf(p, "slba=0x%llx, mndw=0x%x, rl=0x%x, atype=%u",
  88. slba, mndw, rl, atype);
  89. trace_seq_putc(p, 0);
  90. return ret;
  91. }
  92. static const char *nvme_trace_admin_format_nvm(struct trace_seq *p, u8 *cdw10)
  93. {
  94. const char *ret = trace_seq_buffer_ptr(p);
  95. u8 lbaf = cdw10[0] & 0xF;
  96. u8 mset = (cdw10[0] >> 4) & 0x1;
  97. u8 pi = (cdw10[0] >> 5) & 0x7;
  98. u8 pil = cdw10[1] & 0x1;
  99. u8 ses = (cdw10[1] >> 1) & 0x7;
  100. trace_seq_printf(p, "lbaf=%u, mset=%u, pi=%u, pil=%u, ses=%u",
  101. lbaf, mset, pi, pil, ses);
  102. trace_seq_putc(p, 0);
  103. return ret;
  104. }
  105. static const char *nvme_trace_read_write(struct trace_seq *p, u8 *cdw10)
  106. {
  107. const char *ret = trace_seq_buffer_ptr(p);
  108. u64 slba = get_unaligned_le64(cdw10);
  109. u16 length = get_unaligned_le16(cdw10 + 8);
  110. u16 control = get_unaligned_le16(cdw10 + 10);
  111. u32 dsmgmt = get_unaligned_le32(cdw10 + 12);
  112. u32 reftag = get_unaligned_le32(cdw10 + 16);
  113. trace_seq_printf(p,
  114. "slba=%llu, len=%u, ctrl=0x%x, dsmgmt=%u, reftag=%u",
  115. slba, length, control, dsmgmt, reftag);
  116. trace_seq_putc(p, 0);
  117. return ret;
  118. }
  119. static const char *nvme_trace_dsm(struct trace_seq *p, u8 *cdw10)
  120. {
  121. const char *ret = trace_seq_buffer_ptr(p);
  122. trace_seq_printf(p, "nr=%u, attributes=%u",
  123. get_unaligned_le32(cdw10),
  124. get_unaligned_le32(cdw10 + 4));
  125. trace_seq_putc(p, 0);
  126. return ret;
  127. }
  128. static const char *nvme_trace_zone_mgmt_send(struct trace_seq *p, u8 *cdw10)
  129. {
  130. const char *ret = trace_seq_buffer_ptr(p);
  131. u64 slba = get_unaligned_le64(cdw10);
  132. u8 zsa = cdw10[12];
  133. u8 all = cdw10[13];
  134. trace_seq_printf(p, "slba=%llu, zsa=%u, all=%u", slba, zsa, all);
  135. trace_seq_putc(p, 0);
  136. return ret;
  137. }
  138. static const char *nvme_trace_zone_mgmt_recv(struct trace_seq *p, u8 *cdw10)
  139. {
  140. const char *ret = trace_seq_buffer_ptr(p);
  141. u64 slba = get_unaligned_le64(cdw10);
  142. u32 numd = get_unaligned_le32(cdw10 + 8);
  143. u8 zra = cdw10[12];
  144. u8 zrasf = cdw10[13];
  145. u8 pr = cdw10[14];
  146. trace_seq_printf(p, "slba=%llu, numd=%u, zra=%u, zrasf=%u, pr=%u",
  147. slba, numd, zra, zrasf, pr);
  148. trace_seq_putc(p, 0);
  149. return ret;
  150. }
  151. static const char *nvme_trace_common(struct trace_seq *p, u8 *cdw10)
  152. {
  153. const char *ret = trace_seq_buffer_ptr(p);
  154. trace_seq_printf(p, "cdw10=%*ph", 24, cdw10);
  155. trace_seq_putc(p, 0);
  156. return ret;
  157. }
  158. const char *nvme_trace_parse_admin_cmd(struct trace_seq *p,
  159. u8 opcode, u8 *cdw10)
  160. {
  161. switch (opcode) {
  162. case nvme_admin_delete_sq:
  163. return nvme_trace_delete_sq(p, cdw10);
  164. case nvme_admin_create_sq:
  165. return nvme_trace_create_sq(p, cdw10);
  166. case nvme_admin_delete_cq:
  167. return nvme_trace_delete_cq(p, cdw10);
  168. case nvme_admin_create_cq:
  169. return nvme_trace_create_cq(p, cdw10);
  170. case nvme_admin_identify:
  171. return nvme_trace_admin_identify(p, cdw10);
  172. case nvme_admin_set_features:
  173. return nvme_trace_admin_set_features(p, cdw10);
  174. case nvme_admin_get_features:
  175. return nvme_trace_admin_get_features(p, cdw10);
  176. case nvme_admin_get_lba_status:
  177. return nvme_trace_get_lba_status(p, cdw10);
  178. case nvme_admin_format_nvm:
  179. return nvme_trace_admin_format_nvm(p, cdw10);
  180. default:
  181. return nvme_trace_common(p, cdw10);
  182. }
  183. }
  184. const char *nvme_trace_parse_nvm_cmd(struct trace_seq *p,
  185. u8 opcode, u8 *cdw10)
  186. {
  187. switch (opcode) {
  188. case nvme_cmd_read:
  189. case nvme_cmd_write:
  190. case nvme_cmd_write_zeroes:
  191. case nvme_cmd_zone_append:
  192. return nvme_trace_read_write(p, cdw10);
  193. case nvme_cmd_dsm:
  194. return nvme_trace_dsm(p, cdw10);
  195. case nvme_cmd_zone_mgmt_send:
  196. return nvme_trace_zone_mgmt_send(p, cdw10);
  197. case nvme_cmd_zone_mgmt_recv:
  198. return nvme_trace_zone_mgmt_recv(p, cdw10);
  199. default:
  200. return nvme_trace_common(p, cdw10);
  201. }
  202. }
  203. static const char *nvme_trace_fabrics_property_set(struct trace_seq *p, u8 *spc)
  204. {
  205. const char *ret = trace_seq_buffer_ptr(p);
  206. u8 attrib = spc[0];
  207. u32 ofst = get_unaligned_le32(spc + 4);
  208. u64 value = get_unaligned_le64(spc + 8);
  209. trace_seq_printf(p, "attrib=%u, ofst=0x%x, value=0x%llx",
  210. attrib, ofst, value);
  211. trace_seq_putc(p, 0);
  212. return ret;
  213. }
  214. static const char *nvme_trace_fabrics_connect(struct trace_seq *p, u8 *spc)
  215. {
  216. const char *ret = trace_seq_buffer_ptr(p);
  217. u16 recfmt = get_unaligned_le16(spc);
  218. u16 qid = get_unaligned_le16(spc + 2);
  219. u16 sqsize = get_unaligned_le16(spc + 4);
  220. u8 cattr = spc[6];
  221. u32 kato = get_unaligned_le32(spc + 8);
  222. trace_seq_printf(p, "recfmt=%u, qid=%u, sqsize=%u, cattr=%u, kato=%u",
  223. recfmt, qid, sqsize, cattr, kato);
  224. trace_seq_putc(p, 0);
  225. return ret;
  226. }
  227. static const char *nvme_trace_fabrics_property_get(struct trace_seq *p, u8 *spc)
  228. {
  229. const char *ret = trace_seq_buffer_ptr(p);
  230. u8 attrib = spc[0];
  231. u32 ofst = get_unaligned_le32(spc + 4);
  232. trace_seq_printf(p, "attrib=%u, ofst=0x%x", attrib, ofst);
  233. trace_seq_putc(p, 0);
  234. return ret;
  235. }
  236. static const char *nvme_trace_fabrics_auth_send(struct trace_seq *p, u8 *spc)
  237. {
  238. const char *ret = trace_seq_buffer_ptr(p);
  239. u8 spsp0 = spc[1];
  240. u8 spsp1 = spc[2];
  241. u8 secp = spc[3];
  242. u32 tl = get_unaligned_le32(spc + 4);
  243. trace_seq_printf(p, "spsp0=%02x, spsp1=%02x, secp=%02x, tl=%u",
  244. spsp0, spsp1, secp, tl);
  245. trace_seq_putc(p, 0);
  246. return ret;
  247. }
  248. static const char *nvme_trace_fabrics_auth_receive(struct trace_seq *p, u8 *spc)
  249. {
  250. const char *ret = trace_seq_buffer_ptr(p);
  251. u8 spsp0 = spc[1];
  252. u8 spsp1 = spc[2];
  253. u8 secp = spc[3];
  254. u32 al = get_unaligned_le32(spc + 4);
  255. trace_seq_printf(p, "spsp0=%02x, spsp1=%02x, secp=%02x, al=%u",
  256. spsp0, spsp1, secp, al);
  257. trace_seq_putc(p, 0);
  258. return ret;
  259. }
  260. static const char *nvme_trace_fabrics_common(struct trace_seq *p, u8 *spc)
  261. {
  262. const char *ret = trace_seq_buffer_ptr(p);
  263. trace_seq_printf(p, "specific=%*ph", 24, spc);
  264. trace_seq_putc(p, 0);
  265. return ret;
  266. }
  267. const char *nvme_trace_parse_fabrics_cmd(struct trace_seq *p,
  268. u8 fctype, u8 *spc)
  269. {
  270. switch (fctype) {
  271. case nvme_fabrics_type_property_set:
  272. return nvme_trace_fabrics_property_set(p, spc);
  273. case nvme_fabrics_type_connect:
  274. return nvme_trace_fabrics_connect(p, spc);
  275. case nvme_fabrics_type_property_get:
  276. return nvme_trace_fabrics_property_get(p, spc);
  277. case nvme_fabrics_type_auth_send:
  278. return nvme_trace_fabrics_auth_send(p, spc);
  279. case nvme_fabrics_type_auth_receive:
  280. return nvme_trace_fabrics_auth_receive(p, spc);
  281. default:
  282. return nvme_trace_fabrics_common(p, spc);
  283. }
  284. }
  285. const char *nvme_trace_disk_name(struct trace_seq *p, char *name)
  286. {
  287. const char *ret = trace_seq_buffer_ptr(p);
  288. if (*name)
  289. trace_seq_printf(p, "disk=%s, ", name);
  290. trace_seq_putc(p, 0);
  291. return ret;
  292. }
  293. EXPORT_TRACEPOINT_SYMBOL_GPL(nvme_sq);