2016-09-13 02:52:21 +00:00
|
|
|
/*-
|
|
|
|
* BSD LICENSE
|
|
|
|
*
|
|
|
|
* Copyright (c) Intel Corporation.
|
|
|
|
* All rights reserved.
|
|
|
|
*
|
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted provided that the following conditions
|
|
|
|
* are met:
|
|
|
|
*
|
|
|
|
* * Redistributions of source code must retain the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer.
|
|
|
|
* * Redistributions in binary form must reproduce the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer in
|
|
|
|
* the documentation and/or other materials provided with the
|
|
|
|
* distribution.
|
|
|
|
* * Neither the name of Intel Corporation nor the names of its
|
|
|
|
* contributors may be used to endorse or promote products derived
|
|
|
|
* from this software without specific prior written permission.
|
|
|
|
*
|
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
|
|
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
|
|
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
|
|
|
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
|
|
|
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
|
|
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
|
|
|
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
|
|
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
|
|
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
|
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
|
|
|
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
*/
|
|
|
|
|
2017-05-02 18:18:25 +00:00
|
|
|
#include "spdk/stdinc.h"
|
2016-08-08 22:57:49 +00:00
|
|
|
|
2016-09-19 17:01:52 +00:00
|
|
|
#include "spdk/bdev.h"
|
2016-09-13 02:52:21 +00:00
|
|
|
#include "spdk/log.h"
|
|
|
|
#include "spdk/rpc.h"
|
2016-08-10 17:41:12 +00:00
|
|
|
#include "spdk/env.h"
|
2016-09-13 02:52:21 +00:00
|
|
|
#include "spdk/nvme.h"
|
2016-09-19 17:01:52 +00:00
|
|
|
#include "spdk/nvmf.h"
|
2017-03-03 20:44:04 +00:00
|
|
|
#include "spdk/util.h"
|
2016-09-19 17:01:52 +00:00
|
|
|
|
|
|
|
#include "nvmf_tgt.h"
|
2016-09-13 02:52:21 +00:00
|
|
|
|
|
|
|
static void
|
2016-09-19 17:01:52 +00:00
|
|
|
dump_nvmf_subsystem(struct spdk_json_write_ctx *w, struct nvmf_tgt_subsystem *tgt_subsystem)
|
2016-09-13 02:52:21 +00:00
|
|
|
{
|
2016-09-19 17:01:52 +00:00
|
|
|
struct spdk_nvmf_listen_addr *listen_addr;
|
2016-09-13 02:52:21 +00:00
|
|
|
struct spdk_nvmf_host *host;
|
2016-09-19 17:01:52 +00:00
|
|
|
struct spdk_nvmf_subsystem *subsystem = tgt_subsystem->subsystem;
|
2017-02-20 04:49:39 +00:00
|
|
|
struct spdk_nvmf_subsystem_allowed_listener *allowed_listener;
|
2016-09-13 02:52:21 +00:00
|
|
|
|
|
|
|
spdk_json_write_object_begin(w);
|
|
|
|
|
|
|
|
spdk_json_write_name(w, "core");
|
2016-09-19 17:01:52 +00:00
|
|
|
spdk_json_write_int32(w, tgt_subsystem->lcore);
|
2016-09-13 02:52:21 +00:00
|
|
|
|
|
|
|
spdk_json_write_name(w, "nqn");
|
2016-10-10 17:25:01 +00:00
|
|
|
spdk_json_write_string(w, spdk_nvmf_subsystem_get_nqn(subsystem));
|
2016-09-13 02:52:21 +00:00
|
|
|
spdk_json_write_name(w, "subtype");
|
2016-10-10 17:25:01 +00:00
|
|
|
if (spdk_nvmf_subsystem_get_type(subsystem) == SPDK_NVMF_SUBTYPE_NVME) {
|
2016-09-13 02:52:21 +00:00
|
|
|
spdk_json_write_string(w, "NVMe");
|
|
|
|
} else {
|
|
|
|
spdk_json_write_string(w, "Discovery");
|
|
|
|
}
|
|
|
|
|
2016-10-10 22:10:07 +00:00
|
|
|
spdk_json_write_name(w, "listen_addresses");
|
2016-09-13 02:52:21 +00:00
|
|
|
spdk_json_write_array_begin(w);
|
|
|
|
|
2017-02-20 04:49:39 +00:00
|
|
|
TAILQ_FOREACH(allowed_listener, &subsystem->allowed_listeners, link) {
|
|
|
|
listen_addr = allowed_listener->listen_addr;
|
2017-06-28 20:13:11 +00:00
|
|
|
|
2016-09-13 02:52:21 +00:00
|
|
|
spdk_json_write_object_begin(w);
|
2017-06-28 18:07:41 +00:00
|
|
|
/* NOTE: "transport" is kept for compatibility; new code should use "trtype" */
|
2016-09-13 02:52:21 +00:00
|
|
|
spdk_json_write_name(w, "transport");
|
2017-07-13 20:13:05 +00:00
|
|
|
spdk_json_write_string(w, spdk_nvme_transport_id_trtype_str(listen_addr->trid.trtype));
|
2017-06-28 18:07:41 +00:00
|
|
|
spdk_json_write_name(w, "trtype");
|
2017-07-13 20:13:05 +00:00
|
|
|
spdk_json_write_string(w, spdk_nvme_transport_id_trtype_str(listen_addr->trid.trtype));
|
|
|
|
spdk_json_write_name(w, "adrfam");
|
|
|
|
spdk_json_write_string(w, spdk_nvme_transport_id_adrfam_str(listen_addr->trid.adrfam));
|
2016-09-13 02:52:21 +00:00
|
|
|
spdk_json_write_name(w, "traddr");
|
2017-07-13 20:13:05 +00:00
|
|
|
spdk_json_write_string(w, listen_addr->trid.traddr);
|
2016-09-13 02:52:21 +00:00
|
|
|
spdk_json_write_name(w, "trsvcid");
|
2017-07-13 20:13:05 +00:00
|
|
|
spdk_json_write_string(w, listen_addr->trid.trsvcid);
|
2016-09-13 02:52:21 +00:00
|
|
|
spdk_json_write_object_end(w);
|
|
|
|
}
|
|
|
|
spdk_json_write_array_end(w);
|
|
|
|
|
|
|
|
spdk_json_write_name(w, "hosts");
|
|
|
|
spdk_json_write_array_begin(w);
|
|
|
|
|
2017-08-18 21:08:29 +00:00
|
|
|
for (host = spdk_nvmf_subsystem_get_first_host(subsystem); host != NULL;
|
|
|
|
host = spdk_nvmf_subsystem_get_next_host(subsystem, host)) {
|
2016-09-13 02:52:21 +00:00
|
|
|
spdk_json_write_object_begin(w);
|
|
|
|
spdk_json_write_name(w, "nqn");
|
2017-08-18 21:08:29 +00:00
|
|
|
spdk_json_write_string(w, spdk_nvmf_host_get_nqn(host));
|
2016-09-13 02:52:21 +00:00
|
|
|
spdk_json_write_object_end(w);
|
|
|
|
}
|
|
|
|
spdk_json_write_array_end(w);
|
|
|
|
|
2016-10-10 17:25:01 +00:00
|
|
|
if (spdk_nvmf_subsystem_get_type(subsystem) == SPDK_NVMF_SUBTYPE_NVME) {
|
2017-08-18 01:20:49 +00:00
|
|
|
struct spdk_nvmf_ns *ns;
|
2017-06-27 18:26:19 +00:00
|
|
|
|
|
|
|
spdk_json_write_name(w, "serial_number");
|
|
|
|
spdk_json_write_string(w, spdk_nvmf_subsystem_get_sn(subsystem));
|
|
|
|
spdk_json_write_name(w, "namespaces");
|
|
|
|
spdk_json_write_array_begin(w);
|
2017-08-18 01:20:49 +00:00
|
|
|
for (ns = spdk_nvmf_subsystem_get_first_ns(subsystem); ns != NULL;
|
|
|
|
ns = spdk_nvmf_subsystem_get_next_ns(subsystem, ns)) {
|
2017-06-27 18:26:19 +00:00
|
|
|
spdk_json_write_object_begin(w);
|
|
|
|
spdk_json_write_name(w, "nsid");
|
2017-08-18 01:20:49 +00:00
|
|
|
spdk_json_write_int32(w, spdk_nvmf_ns_get_id(ns));
|
2017-06-27 18:26:19 +00:00
|
|
|
spdk_json_write_name(w, "name");
|
2017-08-18 01:20:49 +00:00
|
|
|
spdk_json_write_string(w, spdk_bdev_get_name(spdk_nvmf_ns_get_bdev(ns)));
|
2017-06-27 18:26:19 +00:00
|
|
|
spdk_json_write_object_end(w);
|
2016-09-13 02:52:21 +00:00
|
|
|
}
|
2017-06-27 18:26:19 +00:00
|
|
|
spdk_json_write_array_end(w);
|
2016-09-13 02:52:21 +00:00
|
|
|
}
|
|
|
|
spdk_json_write_object_end(w);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2017-06-29 20:37:39 +00:00
|
|
|
spdk_rpc_get_nvmf_subsystems(struct spdk_jsonrpc_request *request,
|
|
|
|
const struct spdk_json_val *params)
|
2016-09-13 02:52:21 +00:00
|
|
|
{
|
|
|
|
struct spdk_json_write_ctx *w;
|
|
|
|
struct nvmf_tgt_subsystem *tgt_subsystem;
|
|
|
|
|
|
|
|
if (params != NULL) {
|
2017-06-29 20:37:39 +00:00
|
|
|
spdk_jsonrpc_send_error_response(request, SPDK_JSONRPC_ERROR_INVALID_PARAMS,
|
2016-09-13 02:52:21 +00:00
|
|
|
"get_nvmf_subsystems requires no parameters");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-06-29 20:37:39 +00:00
|
|
|
w = spdk_jsonrpc_begin_result(request);
|
|
|
|
if (w == NULL) {
|
2016-09-13 02:52:21 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
spdk_json_write_array_begin(w);
|
|
|
|
tgt_subsystem = nvmf_tgt_subsystem_first();
|
|
|
|
while (tgt_subsystem) {
|
2016-09-19 17:01:52 +00:00
|
|
|
dump_nvmf_subsystem(w, tgt_subsystem);
|
2016-09-13 02:52:21 +00:00
|
|
|
tgt_subsystem = nvmf_tgt_subsystem_next(tgt_subsystem);
|
|
|
|
}
|
|
|
|
spdk_json_write_array_end(w);
|
2017-06-29 20:37:39 +00:00
|
|
|
spdk_jsonrpc_end_result(request, w);
|
2016-09-13 02:52:21 +00:00
|
|
|
}
|
|
|
|
SPDK_RPC_REGISTER("get_nvmf_subsystems", spdk_rpc_get_nvmf_subsystems)
|
|
|
|
|
|
|
|
#define RPC_MAX_LISTEN_ADDRESSES 255
|
|
|
|
#define RPC_MAX_HOSTS 255
|
2017-06-01 00:26:16 +00:00
|
|
|
#define RPC_MAX_NAMESPACES 255
|
2016-09-13 02:52:21 +00:00
|
|
|
|
|
|
|
struct rpc_listen_addresses {
|
|
|
|
size_t num_listen_address;
|
|
|
|
struct rpc_listen_address addresses[RPC_MAX_LISTEN_ADDRESSES];
|
|
|
|
};
|
|
|
|
|
|
|
|
static const struct spdk_json_object_decoder rpc_listen_address_decoders[] = {
|
2017-06-28 18:07:41 +00:00
|
|
|
/* NOTE: "transport" is kept for compatibility; new code should use "trtype" */
|
|
|
|
{"transport", offsetof(struct rpc_listen_address, transport), spdk_json_decode_string, true},
|
|
|
|
{"trtype", offsetof(struct rpc_listen_address, transport), spdk_json_decode_string, true},
|
2017-06-28 20:13:11 +00:00
|
|
|
{"adrfam", offsetof(struct rpc_listen_address, adrfam), spdk_json_decode_string, true},
|
2016-09-13 02:52:21 +00:00
|
|
|
{"traddr", offsetof(struct rpc_listen_address, traddr), spdk_json_decode_string},
|
|
|
|
{"trsvcid", offsetof(struct rpc_listen_address, trsvcid), spdk_json_decode_string},
|
|
|
|
};
|
|
|
|
|
|
|
|
static int
|
|
|
|
decode_rpc_listen_address(const struct spdk_json_val *val, void *out)
|
|
|
|
{
|
|
|
|
struct rpc_listen_address *req = (struct rpc_listen_address *)out;
|
|
|
|
if (spdk_json_decode_object(val, rpc_listen_address_decoders,
|
2017-03-03 20:44:04 +00:00
|
|
|
SPDK_COUNTOF(rpc_listen_address_decoders),
|
2016-09-13 02:52:21 +00:00
|
|
|
req)) {
|
2016-11-07 21:58:55 +00:00
|
|
|
SPDK_ERRLOG("spdk_json_decode_object failed\n");
|
2016-09-13 02:52:21 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
decode_rpc_listen_addresses(const struct spdk_json_val *val, void *out)
|
|
|
|
{
|
|
|
|
struct rpc_listen_addresses *listen_addresses = out;
|
|
|
|
return spdk_json_decode_array(val, decode_rpc_listen_address, &listen_addresses->addresses,
|
|
|
|
RPC_MAX_LISTEN_ADDRESSES,
|
|
|
|
&listen_addresses->num_listen_address, sizeof(struct rpc_listen_address));
|
|
|
|
}
|
|
|
|
|
|
|
|
struct rpc_hosts {
|
|
|
|
size_t num_hosts;
|
|
|
|
char *hosts[RPC_MAX_HOSTS];
|
|
|
|
};
|
|
|
|
|
|
|
|
static int
|
|
|
|
decode_rpc_hosts(const struct spdk_json_val *val, void *out)
|
|
|
|
{
|
|
|
|
struct rpc_hosts *rpc_hosts = out;
|
|
|
|
|
|
|
|
return spdk_json_decode_array(val, spdk_json_decode_string, rpc_hosts->hosts, RPC_MAX_HOSTS,
|
|
|
|
&rpc_hosts->num_hosts, sizeof(char *));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
struct rpc_dev_names {
|
|
|
|
size_t num_names;
|
2017-06-01 00:26:16 +00:00
|
|
|
char *names[RPC_MAX_NAMESPACES];
|
2016-09-13 02:52:21 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
static int
|
|
|
|
decode_rpc_dev_names(const struct spdk_json_val *val, void *out)
|
|
|
|
{
|
|
|
|
struct rpc_dev_names *dev_names = out;
|
|
|
|
|
|
|
|
return spdk_json_decode_array(val, spdk_json_decode_string, dev_names->names,
|
2017-06-01 00:26:16 +00:00
|
|
|
SPDK_COUNTOF(dev_names->names),
|
2016-09-13 02:52:21 +00:00
|
|
|
&dev_names->num_names, sizeof(char *));
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
free_rpc_dev_names(struct rpc_dev_names *r)
|
|
|
|
{
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
for (i = 0; i < r->num_names; i++) {
|
|
|
|
free(r->names[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
free_rpc_listen_addresses(struct rpc_listen_addresses *r)
|
|
|
|
{
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
for (i = 0; i < r->num_listen_address; i++) {
|
|
|
|
free(r->addresses[i].transport);
|
2017-06-28 20:13:11 +00:00
|
|
|
free(r->addresses[i].adrfam);
|
2016-09-13 02:52:21 +00:00
|
|
|
free(r->addresses[i].traddr);
|
|
|
|
free(r->addresses[i].trsvcid);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
free_rpc_hosts(struct rpc_hosts *r)
|
|
|
|
{
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
for (i = 0; i < r->num_hosts; i++) {
|
|
|
|
free(r->hosts[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
struct rpc_subsystem {
|
|
|
|
int32_t core;
|
|
|
|
char *mode;
|
|
|
|
char *nqn;
|
|
|
|
struct rpc_listen_addresses listen_addresses;
|
|
|
|
struct rpc_hosts hosts;
|
|
|
|
char *pci_address;
|
|
|
|
char *serial_number;
|
|
|
|
struct rpc_dev_names namespaces;
|
|
|
|
};
|
|
|
|
|
|
|
|
static void
|
|
|
|
free_rpc_subsystem(struct rpc_subsystem *req)
|
|
|
|
{
|
2017-08-08 23:20:34 +00:00
|
|
|
free(req->mode);
|
|
|
|
free(req->nqn);
|
|
|
|
free(req->serial_number);
|
2016-09-13 02:52:21 +00:00
|
|
|
free_rpc_dev_names(&req->namespaces);
|
|
|
|
free_rpc_listen_addresses(&req->listen_addresses);
|
|
|
|
free_rpc_hosts(&req->hosts);
|
|
|
|
}
|
|
|
|
|
|
|
|
static const struct spdk_json_object_decoder rpc_subsystem_decoders[] = {
|
|
|
|
{"core", offsetof(struct rpc_subsystem, core), spdk_json_decode_int32, true},
|
2017-06-27 18:26:19 +00:00
|
|
|
{"mode", offsetof(struct rpc_subsystem, mode), spdk_json_decode_string, true},
|
2016-09-13 02:52:21 +00:00
|
|
|
{"nqn", offsetof(struct rpc_subsystem, nqn), spdk_json_decode_string},
|
|
|
|
{"listen_addresses", offsetof(struct rpc_subsystem, listen_addresses), decode_rpc_listen_addresses},
|
2016-10-10 22:58:29 +00:00
|
|
|
{"hosts", offsetof(struct rpc_subsystem, hosts), decode_rpc_hosts, true},
|
2016-09-13 02:52:21 +00:00
|
|
|
{"serial_number", offsetof(struct rpc_subsystem, serial_number), spdk_json_decode_string, true},
|
|
|
|
{"namespaces", offsetof(struct rpc_subsystem, namespaces), decode_rpc_dev_names, true},
|
|
|
|
};
|
|
|
|
|
|
|
|
static void
|
2017-06-29 20:37:39 +00:00
|
|
|
spdk_rpc_construct_nvmf_subsystem(struct spdk_jsonrpc_request *request,
|
|
|
|
const struct spdk_json_val *params)
|
2016-09-13 02:52:21 +00:00
|
|
|
{
|
|
|
|
struct rpc_subsystem req = {};
|
|
|
|
struct spdk_json_write_ctx *w;
|
|
|
|
int ret;
|
2017-01-24 04:52:23 +00:00
|
|
|
req.core = -1; /* Explicitly set the core as the uninitialized value */
|
2016-09-13 02:52:21 +00:00
|
|
|
|
|
|
|
if (spdk_json_decode_object(params, rpc_subsystem_decoders,
|
2017-03-03 20:44:04 +00:00
|
|
|
SPDK_COUNTOF(rpc_subsystem_decoders),
|
2016-09-13 02:52:21 +00:00
|
|
|
&req)) {
|
2016-11-07 21:58:55 +00:00
|
|
|
SPDK_ERRLOG("spdk_json_decode_object failed\n");
|
2016-09-13 02:52:21 +00:00
|
|
|
goto invalid;
|
|
|
|
}
|
|
|
|
|
2017-06-27 18:26:19 +00:00
|
|
|
/* Mode is no longer a valid parameter, but print out a nice
|
|
|
|
* message if it exists to inform users.
|
|
|
|
*/
|
|
|
|
if (req.mode) {
|
|
|
|
SPDK_NOTICELOG("Mode present in the construct NVMe-oF subsystem RPC.\n"
|
|
|
|
"Mode was removed as a valid parameter.\n");
|
2017-08-08 21:08:45 +00:00
|
|
|
if (strcasecmp(req.mode, "Virtual") == 0) {
|
2017-06-27 18:26:19 +00:00
|
|
|
SPDK_NOTICELOG("Your mode value is 'Virtual' which is now the only possible mode.\n"
|
|
|
|
"Your RPC will work as expected.\n");
|
|
|
|
} else {
|
|
|
|
SPDK_NOTICELOG("Please remove 'mode' from the RPC.\n");
|
|
|
|
goto invalid;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = spdk_nvmf_construct_subsystem(req.nqn, req.core,
|
2017-02-21 04:49:59 +00:00
|
|
|
req.listen_addresses.num_listen_address,
|
|
|
|
req.listen_addresses.addresses,
|
2017-06-27 18:26:19 +00:00
|
|
|
req.hosts.num_hosts, req.hosts.hosts,
|
2017-02-21 04:49:59 +00:00
|
|
|
req.serial_number,
|
|
|
|
req.namespaces.num_names, req.namespaces.names);
|
2016-09-13 02:52:21 +00:00
|
|
|
if (ret) {
|
|
|
|
goto invalid;
|
|
|
|
}
|
|
|
|
|
|
|
|
free_rpc_subsystem(&req);
|
|
|
|
|
2017-06-29 20:37:39 +00:00
|
|
|
w = spdk_jsonrpc_begin_result(request);
|
|
|
|
if (w == NULL) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-09-13 02:52:21 +00:00
|
|
|
spdk_json_write_bool(w, true);
|
2017-06-29 20:37:39 +00:00
|
|
|
spdk_jsonrpc_end_result(request, w);
|
2016-09-13 02:52:21 +00:00
|
|
|
return;
|
|
|
|
|
|
|
|
invalid:
|
2017-06-29 20:37:39 +00:00
|
|
|
spdk_jsonrpc_send_error_response(request, SPDK_JSONRPC_ERROR_INVALID_PARAMS, "Invalid parameters");
|
2016-09-13 02:52:21 +00:00
|
|
|
free_rpc_subsystem(&req);
|
|
|
|
}
|
|
|
|
SPDK_RPC_REGISTER("construct_nvmf_subsystem", spdk_rpc_construct_nvmf_subsystem)
|
|
|
|
|
|
|
|
struct rpc_delete_subsystem {
|
|
|
|
char *nqn;
|
|
|
|
};
|
|
|
|
|
|
|
|
static void
|
|
|
|
free_rpc_delete_subsystem(struct rpc_delete_subsystem *r)
|
|
|
|
{
|
|
|
|
free(r->nqn);
|
|
|
|
}
|
|
|
|
|
|
|
|
static const struct spdk_json_object_decoder rpc_delete_subsystem_decoders[] = {
|
|
|
|
{"nqn", offsetof(struct rpc_delete_subsystem, nqn), spdk_json_decode_string},
|
|
|
|
};
|
|
|
|
|
|
|
|
static void
|
2017-06-29 20:37:39 +00:00
|
|
|
spdk_rpc_delete_nvmf_subsystem(struct spdk_jsonrpc_request *request,
|
|
|
|
const struct spdk_json_val *params)
|
2016-09-13 02:52:21 +00:00
|
|
|
{
|
|
|
|
struct rpc_delete_subsystem req = {};
|
|
|
|
struct spdk_json_write_ctx *w;
|
|
|
|
|
|
|
|
if (spdk_json_decode_object(params, rpc_delete_subsystem_decoders,
|
2017-03-03 20:44:04 +00:00
|
|
|
SPDK_COUNTOF(rpc_delete_subsystem_decoders),
|
2016-09-13 02:52:21 +00:00
|
|
|
&req)) {
|
2016-11-07 21:58:55 +00:00
|
|
|
SPDK_ERRLOG("spdk_json_decode_object failed\n");
|
2016-09-13 02:52:21 +00:00
|
|
|
goto invalid;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (req.nqn == NULL) {
|
2016-11-07 21:58:55 +00:00
|
|
|
SPDK_ERRLOG("missing name param\n");
|
2016-09-13 02:52:21 +00:00
|
|
|
goto invalid;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (nvmf_tgt_shutdown_subsystem_by_nqn(req.nqn)) {
|
2016-11-07 21:58:55 +00:00
|
|
|
SPDK_ERRLOG("shutdown_subsystem failed\n");
|
2016-09-13 02:52:21 +00:00
|
|
|
goto invalid;
|
|
|
|
}
|
|
|
|
|
|
|
|
free_rpc_delete_subsystem(&req);
|
|
|
|
|
2017-06-29 20:37:39 +00:00
|
|
|
w = spdk_jsonrpc_begin_result(request);
|
|
|
|
if (w == NULL) {
|
2016-09-13 02:52:21 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
spdk_json_write_bool(w, true);
|
2017-06-29 20:37:39 +00:00
|
|
|
spdk_jsonrpc_end_result(request, w);
|
2016-09-13 02:52:21 +00:00
|
|
|
return;
|
|
|
|
|
|
|
|
invalid:
|
2017-06-29 20:37:39 +00:00
|
|
|
spdk_jsonrpc_send_error_response(request, SPDK_JSONRPC_ERROR_INVALID_PARAMS, "Invalid parameters");
|
2016-09-13 02:52:21 +00:00
|
|
|
free_rpc_delete_subsystem(&req);
|
|
|
|
}
|
|
|
|
SPDK_RPC_REGISTER("delete_nvmf_subsystem", spdk_rpc_delete_nvmf_subsystem)
|