fabrics-cmd.c 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265
  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. req->ns = NULL;
  72. switch (cmd->fabrics.fctype) {
  73. case nvme_fabrics_type_property_set:
  74. req->data_len = 0;
  75. req->execute = nvmet_execute_prop_set;
  76. break;
  77. case nvme_fabrics_type_property_get:
  78. req->data_len = 0;
  79. req->execute = nvmet_execute_prop_get;
  80. break;
  81. default:
  82. pr_err("received unknown capsule type 0x%x\n",
  83. cmd->fabrics.fctype);
  84. return NVME_SC_INVALID_OPCODE | NVME_SC_DNR;
  85. }
  86. return 0;
  87. }
  88. static u16 nvmet_install_queue(struct nvmet_ctrl *ctrl, struct nvmet_req *req)
  89. {
  90. struct nvmf_connect_command *c = &req->cmd->connect;
  91. u16 qid = le16_to_cpu(c->qid);
  92. u16 sqsize = le16_to_cpu(c->sqsize);
  93. struct nvmet_ctrl *old;
  94. old = cmpxchg(&req->sq->ctrl, NULL, ctrl);
  95. if (old) {
  96. pr_warn("queue already connected!\n");
  97. return NVME_SC_CONNECT_CTRL_BUSY | NVME_SC_DNR;
  98. }
  99. if (!sqsize) {
  100. pr_warn("queue size zero!\n");
  101. return NVME_SC_CONNECT_INVALID_PARAM | NVME_SC_DNR;
  102. }
  103. /* note: convert queue size from 0's-based value to 1's-based value */
  104. nvmet_cq_setup(ctrl, req->cq, qid, sqsize + 1);
  105. nvmet_sq_setup(ctrl, req->sq, qid, sqsize + 1);
  106. return 0;
  107. }
  108. static void nvmet_execute_admin_connect(struct nvmet_req *req)
  109. {
  110. struct nvmf_connect_command *c = &req->cmd->connect;
  111. struct nvmf_connect_data *d;
  112. struct nvmet_ctrl *ctrl = NULL;
  113. u16 status = 0;
  114. d = kmalloc(sizeof(*d), GFP_KERNEL);
  115. if (!d) {
  116. status = NVME_SC_INTERNAL;
  117. goto complete;
  118. }
  119. status = nvmet_copy_from_sgl(req, 0, d, sizeof(*d));
  120. if (status)
  121. goto out;
  122. /* zero out initial completion result, assign values as needed */
  123. req->rsp->result.u32 = 0;
  124. if (c->recfmt != 0) {
  125. pr_warn("invalid connect version (%d).\n",
  126. le16_to_cpu(c->recfmt));
  127. status = NVME_SC_CONNECT_FORMAT | NVME_SC_DNR;
  128. goto out;
  129. }
  130. if (unlikely(d->cntlid != cpu_to_le16(0xffff))) {
  131. pr_warn("connect attempt for invalid controller ID %#x\n",
  132. d->cntlid);
  133. status = NVME_SC_CONNECT_INVALID_PARAM | NVME_SC_DNR;
  134. req->rsp->result.u32 = IPO_IATTR_CONNECT_DATA(cntlid);
  135. goto out;
  136. }
  137. status = nvmet_alloc_ctrl(d->subsysnqn, d->hostnqn, req,
  138. le32_to_cpu(c->kato), &ctrl);
  139. if (status)
  140. goto out;
  141. uuid_copy(&ctrl->hostid, &d->hostid);
  142. status = nvmet_install_queue(ctrl, req);
  143. if (status) {
  144. nvmet_ctrl_put(ctrl);
  145. goto out;
  146. }
  147. pr_info("creating controller %d for subsystem %s for NQN %s.\n",
  148. ctrl->cntlid, ctrl->subsys->subsysnqn, ctrl->hostnqn);
  149. req->rsp->result.u16 = cpu_to_le16(ctrl->cntlid);
  150. out:
  151. kfree(d);
  152. complete:
  153. nvmet_req_complete(req, status);
  154. }
  155. static void nvmet_execute_io_connect(struct nvmet_req *req)
  156. {
  157. struct nvmf_connect_command *c = &req->cmd->connect;
  158. struct nvmf_connect_data *d;
  159. struct nvmet_ctrl *ctrl = NULL;
  160. u16 qid = le16_to_cpu(c->qid);
  161. u16 status = 0;
  162. d = kmalloc(sizeof(*d), GFP_KERNEL);
  163. if (!d) {
  164. status = NVME_SC_INTERNAL;
  165. goto complete;
  166. }
  167. status = nvmet_copy_from_sgl(req, 0, d, sizeof(*d));
  168. if (status)
  169. goto out;
  170. /* zero out initial completion result, assign values as needed */
  171. req->rsp->result.u32 = 0;
  172. if (c->recfmt != 0) {
  173. pr_warn("invalid connect version (%d).\n",
  174. le16_to_cpu(c->recfmt));
  175. status = NVME_SC_CONNECT_FORMAT | NVME_SC_DNR;
  176. goto out;
  177. }
  178. status = nvmet_ctrl_find_get(d->subsysnqn, d->hostnqn,
  179. le16_to_cpu(d->cntlid),
  180. req, &ctrl);
  181. if (status)
  182. goto out;
  183. if (unlikely(qid > ctrl->subsys->max_qid)) {
  184. pr_warn("invalid queue id (%d)\n", qid);
  185. status = NVME_SC_CONNECT_INVALID_PARAM | NVME_SC_DNR;
  186. req->rsp->result.u32 = IPO_IATTR_CONNECT_SQE(qid);
  187. goto out_ctrl_put;
  188. }
  189. status = nvmet_install_queue(ctrl, req);
  190. if (status) {
  191. /* pass back cntlid that had the issue of installing queue */
  192. req->rsp->result.u16 = cpu_to_le16(ctrl->cntlid);
  193. goto out_ctrl_put;
  194. }
  195. pr_info("adding queue %d to ctrl %d.\n", qid, ctrl->cntlid);
  196. out:
  197. kfree(d);
  198. complete:
  199. nvmet_req_complete(req, status);
  200. return;
  201. out_ctrl_put:
  202. nvmet_ctrl_put(ctrl);
  203. goto out;
  204. }
  205. u16 nvmet_parse_connect_cmd(struct nvmet_req *req)
  206. {
  207. struct nvme_command *cmd = req->cmd;
  208. req->ns = NULL;
  209. if (cmd->common.opcode != nvme_fabrics_command) {
  210. pr_err("invalid command 0x%x on unconnected queue.\n",
  211. cmd->fabrics.opcode);
  212. return NVME_SC_INVALID_OPCODE | NVME_SC_DNR;
  213. }
  214. if (cmd->fabrics.fctype != nvme_fabrics_type_connect) {
  215. pr_err("invalid capsule type 0x%x on unconnected queue.\n",
  216. cmd->fabrics.fctype);
  217. return NVME_SC_INVALID_OPCODE | NVME_SC_DNR;
  218. }
  219. req->data_len = sizeof(struct nvmf_connect_data);
  220. if (cmd->connect.qid == 0)
  221. req->execute = nvmet_execute_admin_connect;
  222. else
  223. req->execute = nvmet_execute_io_connect;
  224. return 0;
  225. }