fabrics-cmd.c 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260
  1. /*
  2. * NVMe Fabrics command implementation.
  3. * Copyright (c) 2015-2016 HGST, a Western Digital Company.
  4. *
  5. * This program is free software; you can redistribute it and/or modify it
  6. * under the terms and conditions of the GNU General Public License,
  7. * version 2, as published by the Free Software Foundation.
  8. *
  9. * This program is distributed in the hope it will be useful, but WITHOUT
  10. * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
  11. * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
  12. * more details.
  13. */
  14. #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
  15. #include <linux/blkdev.h>
  16. #include "nvmet.h"
  17. static void nvmet_execute_prop_set(struct nvmet_req *req)
  18. {
  19. u16 status = 0;
  20. if (!(req->cmd->prop_set.attrib & 1)) {
  21. u64 val = le64_to_cpu(req->cmd->prop_set.value);
  22. switch (le32_to_cpu(req->cmd->prop_set.offset)) {
  23. case NVME_REG_CC:
  24. nvmet_update_cc(req->sq->ctrl, val);
  25. break;
  26. default:
  27. status = NVME_SC_INVALID_FIELD | NVME_SC_DNR;
  28. break;
  29. }
  30. } else {
  31. status = NVME_SC_INVALID_FIELD | NVME_SC_DNR;
  32. }
  33. nvmet_req_complete(req, status);
  34. }
  35. static void nvmet_execute_prop_get(struct nvmet_req *req)
  36. {
  37. struct nvmet_ctrl *ctrl = req->sq->ctrl;
  38. u16 status = 0;
  39. u64 val = 0;
  40. if (req->cmd->prop_get.attrib & 1) {
  41. switch (le32_to_cpu(req->cmd->prop_get.offset)) {
  42. case NVME_REG_CAP:
  43. val = ctrl->cap;
  44. break;
  45. default:
  46. status = NVME_SC_INVALID_FIELD | NVME_SC_DNR;
  47. break;
  48. }
  49. } else {
  50. switch (le32_to_cpu(req->cmd->prop_get.offset)) {
  51. case NVME_REG_VS:
  52. val = ctrl->subsys->ver;
  53. break;
  54. case NVME_REG_CC:
  55. val = ctrl->cc;
  56. break;
  57. case NVME_REG_CSTS:
  58. val = ctrl->csts;
  59. break;
  60. default:
  61. status = NVME_SC_INVALID_FIELD | NVME_SC_DNR;
  62. break;
  63. }
  64. }
  65. req->rsp->result.u64 = cpu_to_le64(val);
  66. nvmet_req_complete(req, status);
  67. }
  68. u16 nvmet_parse_fabrics_cmd(struct nvmet_req *req)
  69. {
  70. struct nvme_command *cmd = req->cmd;
  71. switch (cmd->fabrics.fctype) {
  72. case nvme_fabrics_type_property_set:
  73. req->data_len = 0;
  74. req->execute = nvmet_execute_prop_set;
  75. break;
  76. case nvme_fabrics_type_property_get:
  77. req->data_len = 0;
  78. req->execute = nvmet_execute_prop_get;
  79. break;
  80. default:
  81. pr_err("received unknown capsule type 0x%x\n",
  82. cmd->fabrics.fctype);
  83. return NVME_SC_INVALID_OPCODE | NVME_SC_DNR;
  84. }
  85. return 0;
  86. }
  87. static u16 nvmet_install_queue(struct nvmet_ctrl *ctrl, struct nvmet_req *req)
  88. {
  89. struct nvmf_connect_command *c = &req->cmd->connect;
  90. u16 qid = le16_to_cpu(c->qid);
  91. u16 sqsize = le16_to_cpu(c->sqsize);
  92. struct nvmet_ctrl *old;
  93. old = cmpxchg(&req->sq->ctrl, NULL, ctrl);
  94. if (old) {
  95. pr_warn("queue already connected!\n");
  96. return NVME_SC_CONNECT_CTRL_BUSY | NVME_SC_DNR;
  97. }
  98. if (!sqsize) {
  99. pr_warn("queue size zero!\n");
  100. return NVME_SC_CONNECT_INVALID_PARAM | NVME_SC_DNR;
  101. }
  102. /* note: convert queue size from 0's-based value to 1's-based value */
  103. nvmet_cq_setup(ctrl, req->cq, qid, sqsize + 1);
  104. nvmet_sq_setup(ctrl, req->sq, qid, sqsize + 1);
  105. return 0;
  106. }
  107. static void nvmet_execute_admin_connect(struct nvmet_req *req)
  108. {
  109. struct nvmf_connect_command *c = &req->cmd->connect;
  110. struct nvmf_connect_data *d;
  111. struct nvmet_ctrl *ctrl = NULL;
  112. u16 status = 0;
  113. d = kmalloc(sizeof(*d), GFP_KERNEL);
  114. if (!d) {
  115. status = NVME_SC_INTERNAL;
  116. goto complete;
  117. }
  118. status = nvmet_copy_from_sgl(req, 0, d, sizeof(*d));
  119. if (status)
  120. goto out;
  121. /* zero out initial completion result, assign values as needed */
  122. req->rsp->result.u32 = 0;
  123. if (c->recfmt != 0) {
  124. pr_warn("invalid connect version (%d).\n",
  125. le16_to_cpu(c->recfmt));
  126. status = NVME_SC_CONNECT_FORMAT | NVME_SC_DNR;
  127. goto out;
  128. }
  129. if (unlikely(d->cntlid != cpu_to_le16(0xffff))) {
  130. pr_warn("connect attempt for invalid controller ID %#x\n",
  131. d->cntlid);
  132. status = NVME_SC_CONNECT_INVALID_PARAM | NVME_SC_DNR;
  133. req->rsp->result.u32 = IPO_IATTR_CONNECT_DATA(cntlid);
  134. goto out;
  135. }
  136. status = nvmet_alloc_ctrl(d->subsysnqn, d->hostnqn, req,
  137. le32_to_cpu(c->kato), &ctrl);
  138. if (status)
  139. goto out;
  140. uuid_copy(&ctrl->hostid, &d->hostid);
  141. status = nvmet_install_queue(ctrl, req);
  142. if (status) {
  143. nvmet_ctrl_put(ctrl);
  144. goto out;
  145. }
  146. pr_info("creating controller %d for subsystem %s for NQN %s.\n",
  147. ctrl->cntlid, ctrl->subsys->subsysnqn, ctrl->hostnqn);
  148. req->rsp->result.u16 = cpu_to_le16(ctrl->cntlid);
  149. out:
  150. kfree(d);
  151. complete:
  152. nvmet_req_complete(req, status);
  153. }
  154. static void nvmet_execute_io_connect(struct nvmet_req *req)
  155. {
  156. struct nvmf_connect_command *c = &req->cmd->connect;
  157. struct nvmf_connect_data *d;
  158. struct nvmet_ctrl *ctrl = NULL;
  159. u16 qid = le16_to_cpu(c->qid);
  160. u16 status = 0;
  161. d = kmalloc(sizeof(*d), GFP_KERNEL);
  162. if (!d) {
  163. status = NVME_SC_INTERNAL;
  164. goto complete;
  165. }
  166. status = nvmet_copy_from_sgl(req, 0, d, sizeof(*d));
  167. if (status)
  168. goto out;
  169. /* zero out initial completion result, assign values as needed */
  170. req->rsp->result.u32 = 0;
  171. if (c->recfmt != 0) {
  172. pr_warn("invalid connect version (%d).\n",
  173. le16_to_cpu(c->recfmt));
  174. status = NVME_SC_CONNECT_FORMAT | NVME_SC_DNR;
  175. goto out;
  176. }
  177. status = nvmet_ctrl_find_get(d->subsysnqn, d->hostnqn,
  178. le16_to_cpu(d->cntlid),
  179. req, &ctrl);
  180. if (status)
  181. goto out;
  182. if (unlikely(qid > ctrl->subsys->max_qid)) {
  183. pr_warn("invalid queue id (%d)\n", qid);
  184. status = NVME_SC_CONNECT_INVALID_PARAM | NVME_SC_DNR;
  185. req->rsp->result.u32 = IPO_IATTR_CONNECT_SQE(qid);
  186. goto out_ctrl_put;
  187. }
  188. status = nvmet_install_queue(ctrl, req);
  189. if (status) {
  190. /* pass back cntlid that had the issue of installing queue */
  191. req->rsp->result.u16 = cpu_to_le16(ctrl->cntlid);
  192. goto out_ctrl_put;
  193. }
  194. pr_debug("adding queue %d to ctrl %d.\n", qid, ctrl->cntlid);
  195. out:
  196. kfree(d);
  197. complete:
  198. nvmet_req_complete(req, status);
  199. return;
  200. out_ctrl_put:
  201. nvmet_ctrl_put(ctrl);
  202. goto out;
  203. }
  204. u16 nvmet_parse_connect_cmd(struct nvmet_req *req)
  205. {
  206. struct nvme_command *cmd = req->cmd;
  207. if (cmd->common.opcode != nvme_fabrics_command) {
  208. pr_err("invalid command 0x%x on unconnected queue.\n",
  209. cmd->fabrics.opcode);
  210. return NVME_SC_INVALID_OPCODE | NVME_SC_DNR;
  211. }
  212. if (cmd->fabrics.fctype != nvme_fabrics_type_connect) {
  213. pr_err("invalid capsule type 0x%x on unconnected queue.\n",
  214. cmd->fabrics.fctype);
  215. return NVME_SC_INVALID_OPCODE | NVME_SC_DNR;
  216. }
  217. req->data_len = sizeof(struct nvmf_connect_data);
  218. if (cmd->connect.qid == 0)
  219. req->execute = nvmet_execute_admin_connect;
  220. else
  221. req->execute = nvmet_execute_io_connect;
  222. return 0;
  223. }