bdev/iscsi: add RPC support and cofnig dump

Also, as we are here, switch to new spdk_json_write_named_* API in
bdev_iscsi_dump_info_json()

Change-Id: Ib29466eb5c6d6496dd1d8efc1f2064577bf56272
Signed-off-by: Pawel Wodkowski <pawelx.wodkowski@intel.com>
Reviewed-on: https://review.gerrithub.io/406635
Tested-by: SPDK Automated Test System <sys_sgsw@intel.com>
Reviewed-by: Jim Harris <james.r.harris@intel.com>
Reviewed-by: Daniel Verkamp <daniel.verkamp@intel.com>
This commit is contained in:
Pawel Wodkowski 2018-04-05 23:41:19 +02:00 committed by Jim Harris
parent 06fdb62b3b
commit fba68d61e1
6 changed files with 268 additions and 26 deletions

View File

@ -40,7 +40,7 @@ CFLAGS += -I$(SPDK_ROOT_DIR)/lib/bdev/
# this warning so just make sure the warning isn't treated as
# an error.
CFLAGS += -Wno-error
C_SRCS = bdev_iscsi.c
C_SRCS = bdev_iscsi.c bdev_iscsi_rpc.c
LIBNAME = bdev_iscsi
include $(SPDK_ROOT_DIR)/mk/spdk.lib.mk

View File

@ -49,6 +49,8 @@
#include "iscsi/iscsi.h"
#include "iscsi/scsi-lowlevel.h"
#include "bdev_iscsi.h"
struct bdev_iscsi_lun;
#define BDEV_ISCSI_CONNECTION_POLL_US 500
@ -81,8 +83,6 @@ struct bdev_iscsi_lun {
TAILQ_ENTRY(bdev_iscsi_lun) link;
};
typedef void (*spdk_bdev_iscsi_create_cb)(void);
struct bdev_iscsi_io_channel {
struct spdk_poller *poller;
struct bdev_iscsi_lun *lun;
@ -94,7 +94,9 @@ struct bdev_iscsi_conn_req {
char *initiator_iqn;
struct iscsi_context *context;
spdk_bdev_iscsi_create_cb create_cb;
spdk_bdev_iscsi_create_cb create_cb_arg;
TAILQ_ENTRY(bdev_iscsi_conn_req) link;
bool deleted;
};
static int
@ -436,17 +438,38 @@ bdev_iscsi_dump_info_json(void *ctx, struct spdk_json_write_ctx *w)
return 0;
}
static void
bdev_iscsi_write_config_json(struct spdk_bdev *bdev, struct spdk_json_write_ctx *w)
{
struct bdev_iscsi_lun *lun = bdev->ctxt;
pthread_mutex_lock(&lun->mutex);
spdk_json_write_object_begin(w);
spdk_json_write_named_string(w, "method", "construct_iscsi_bdev");
spdk_json_write_named_object_begin(w, "params");
spdk_json_write_named_string(w, "name", bdev->name);
spdk_json_write_named_string(w, "initiator_iqn", lun->initiator_iqn);
spdk_json_write_named_string(w, "url", lun->url);
spdk_json_write_object_end(w);
spdk_json_write_object_end(w);
pthread_mutex_unlock(&lun->mutex);
}
static const struct spdk_bdev_fn_table iscsi_fn_table = {
.destruct = bdev_iscsi_destruct,
.submit_request = bdev_iscsi_submit_request,
.io_type_supported = bdev_iscsi_io_type_supported,
.get_io_channel = bdev_iscsi_get_io_channel,
.dump_info_json = bdev_iscsi_dump_info_json,
.write_config_json = bdev_iscsi_write_config_json,
};
static struct spdk_bdev *
create_iscsi_lun(struct iscsi_context *context, char *url, char *initiator_iqn,
const char *name, uint64_t num_blocks, uint32_t block_size)
static int
create_iscsi_lun(struct iscsi_context *context, char *url, char *initiator_iqn, char *name,
uint64_t num_blocks, uint32_t block_size, struct spdk_bdev **bdev)
{
struct bdev_iscsi_lun *lun;
int rc;
@ -454,7 +477,7 @@ create_iscsi_lun(struct iscsi_context *context, char *url, char *initiator_iqn,
lun = calloc(sizeof(*lun), 1);
if (!lun) {
SPDK_ERRLOG("Unable to allocate enough memory for iscsi backend\n");
return NULL;
return -ENOMEM;
}
lun->context = context;
@ -463,10 +486,7 @@ create_iscsi_lun(struct iscsi_context *context, char *url, char *initiator_iqn,
pthread_mutex_init(&lun->mutex, NULL);
lun->bdev.name = strdup(name);
if (!lun->bdev.name) {
goto error_return;
}
lun->bdev.name = name;
lun->bdev.product_name = "iSCSI LUN";
lun->bdev.module = &g_iscsi_bdev_module;
lun->bdev.blocklen = block_size;
@ -484,11 +504,12 @@ create_iscsi_lun(struct iscsi_context *context, char *url, char *initiator_iqn,
}
TAILQ_INSERT_TAIL(&g_iscsi_lun_head, lun, link);
return &lun->bdev;
*bdev = &lun->bdev;
return 0;
error_return:
iscsi_free_lun(lun);
return NULL;
return rc;
}
static void
@ -497,25 +518,31 @@ iscsi_readcapacity16_cb(struct iscsi_context *iscsi, int status,
{
struct bdev_iscsi_conn_req *req = private_data;
struct scsi_readcapacity16 *readcap16;
struct spdk_bdev *bdev;
struct spdk_bdev *bdev = NULL;
struct scsi_task *task = command_data;
if (status != SPDK_SCSI_STATUS_GOOD) {
SPDK_ERRLOG("iSCSI error: %s\n", iscsi_get_error(iscsi));
goto ret;
}
readcap16 = scsi_datain_unmarshall(task);
bdev = create_iscsi_lun(req->context, req->url, req->initiator_iqn, req->bdev_name,
readcap16->returned_lba + 1, readcap16->block_length);
if (!bdev) {
SPDK_ERRLOG("Unable to create iscsi bdev\n");
if (!readcap16) {
status = -ENOMEM;
goto ret;
}
status = create_iscsi_lun(req->context, req->url, req->initiator_iqn, req->bdev_name,
readcap16->returned_lba + 1, readcap16->block_length, &bdev);
if (status) {
SPDK_ERRLOG("Unable to create iscsi bdev: %s (%d)\n", spdk_strerror(-status), status);
}
ret:
TAILQ_REMOVE(&g_iscsi_conn_req, req, link);
req->create_cb();
req->create_cb(req->create_cb_arg, bdev, status);
scsi_free_scsi_task(task);
free(req);
req->deleted = true;
}
static void
@ -537,8 +564,8 @@ iscsi_connect_cb(struct iscsi_context *iscsi, int status,
ret:
SPDK_ERRLOG("iSCSI error: %s\n", iscsi_get_error(req->context));
TAILQ_REMOVE(&g_iscsi_conn_req, req, link);
req->create_cb();
free(req);
req->create_cb(req->create_cb_arg, NULL, status);
req->deleted = true;
}
static int
@ -561,14 +588,18 @@ iscsi_bdev_conn_poll(void *arg)
SPDK_ERRLOG("iscsi_service failed: %s\n", iscsi_get_error(req->context));
}
}
if (req->deleted) {
free(req);
}
}
return 0;
}
static int
int
create_iscsi_disk(const char *bdev_name, const char *url, const char *initiator_iqn,
spdk_bdev_iscsi_create_cb cb_fn)
spdk_bdev_iscsi_create_cb cb_fn, void *cb_arg)
{
struct bdev_iscsi_conn_req *req;
struct iscsi_url *iscsi_url = NULL;
@ -595,6 +626,7 @@ create_iscsi_disk(const char *bdev_name, const char *url, const char *initiator_
}
req->create_cb = cb_fn;
req->create_cb_arg = cb_arg;
iscsi_url = iscsi_parse_full_url(req->context, url);
if (iscsi_url == NULL) {
@ -643,7 +675,7 @@ err:
}
static void
bdev_iscsi_initialize_cb(void)
bdev_iscsi_initialize_cb(void *cb_arg, struct spdk_bdev *bdev, int status)
{
if (TAILQ_EMPTY(&g_iscsi_conn_req)) {
spdk_bdev_module_init_done(&g_iscsi_bdev_module);
@ -678,7 +710,7 @@ bdev_iscsi_initialize(void)
break;
}
rc = create_iscsi_disk(bdev_name, url, initiator_iqn, bdev_iscsi_initialize_cb);
rc = create_iscsi_disk(bdev_name, url, initiator_iqn, bdev_iscsi_initialize_cb, NULL);
if (rc) {
break;
}

View File

@ -0,0 +1,64 @@
/*-
* 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.
*/
#ifndef SPDK_BDEV_ISCSI_H
#define SPDK_BDEV_ISCSI_H
#include "spdk/bdev.h"
/**
* SPDK bdev iSCSI callback type.
*
* \param cb_arg Completion callback custom arguments
* \param bdev created bdev
* \param status operation status. Zero on success.
*/
typedef void (*spdk_bdev_iscsi_create_cb)(void *cb_arg, struct spdk_bdev *bdev, int status);
/**
* Create new iSCSI bdev.
*
* \warning iSCSI URL allow providing login and password. Be careful because
* they will show up in configuration dump.
*
* \param name name for new bdev.
* \param initiator_iqn connection iqn name we identify to target as
* \param url iSCSI URL string.
* \param cb_fn Completion callback
* \param cb_arg Completion callback custom arguments
* \return 0 on success or negative error code. If success bdev with provided name was created.
*/
int create_iscsi_disk(const char *bdev_name, const char *initiator_iqn, const char *url,
spdk_bdev_iscsi_create_cb cb_fn, void *cb_arg);
#endif // SPDK_BDEV_ISCSI_H

View File

@ -0,0 +1,113 @@
/*-
* 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.
*/
#include "bdev_iscsi.h"
#include "spdk/rpc.h"
#include "spdk/util.h"
#include "spdk/string.h"
#include "spdk_internal/log.h"
struct rpc_construct_iscsi_bdev {
char *name;
char *initiator_iqn;
char *url;
};
static const struct spdk_json_object_decoder rpc_construct_iscsi_bdev_decoders[] = {
{"name", offsetof(struct rpc_construct_iscsi_bdev, name), spdk_json_decode_string},
{"initiator_iqn", offsetof(struct rpc_construct_iscsi_bdev, initiator_iqn), spdk_json_decode_string},
{"url", offsetof(struct rpc_construct_iscsi_bdev, url), spdk_json_decode_string},
};
static void
free_rpc_construct_iscsi_bdev(struct rpc_construct_iscsi_bdev *req)
{
free(req->name);
free(req->initiator_iqn);
free(req->url);
}
static void
construct_iscsi_bdev_cb(void *cb_arg, struct spdk_bdev *bdev, int status)
{
struct spdk_jsonrpc_request *request = cb_arg;
struct spdk_json_write_ctx *w;
if (status > 0) {
spdk_jsonrpc_send_error_response_fmt(request, SPDK_JSONRPC_ERROR_INVALID_PARAMS,
"iSCSI error (%d).", status);
} else if (status < 0) {
spdk_jsonrpc_send_error_response(request, SPDK_JSONRPC_ERROR_INVALID_PARAMS,
spdk_strerror(-status));
} else {
w = spdk_jsonrpc_begin_result(request);
if (w == NULL) {
return;
}
spdk_json_write_array_begin(w);
spdk_json_write_string(w, spdk_bdev_get_name(bdev));
spdk_json_write_array_end(w);
spdk_jsonrpc_end_result(request, w);
}
}
static void
spdk_rpc_construct_iscsi_bdev(struct spdk_jsonrpc_request *request,
const struct spdk_json_val *params)
{
struct rpc_construct_iscsi_bdev req = {};
int rc = 0;
if (spdk_json_decode_object(params, rpc_construct_iscsi_bdev_decoders,
SPDK_COUNTOF(rpc_construct_iscsi_bdev_decoders),
&req)) {
SPDK_ERRLOG("spdk_json_decode_object failed\n");
rc = -EINVAL;
goto invalid;
}
rc = create_iscsi_disk(req.name, req.url, req.initiator_iqn, construct_iscsi_bdev_cb, request);
if (rc) {
goto invalid;
}
free_rpc_construct_iscsi_bdev(&req);
return;
invalid:
free_rpc_construct_iscsi_bdev(&req);
spdk_jsonrpc_send_error_response(request, SPDK_JSONRPC_ERROR_INVALID_PARAMS, spdk_strerror(-rc));
}
SPDK_RPC_REGISTER("construct_iscsi_bdev", spdk_rpc_construct_iscsi_bdev, SPDK_RPC_RUNTIME)

View File

@ -210,6 +210,20 @@ if __name__ == "__main__":
p.add_argument('base_name', help='base bdev name')
p.set_defaults(func=construct_error_bdev)
@call_cmd
def construct_iscsi_bdev(args):
rpc.bdev.construct_iscsi_bdev(args.client,
name=args.name,
url=args.url,
initiator_iqn=args.initiator_iqn)
p = subparsers.add_parser('construct_iscsi_bdev',
help='Add bdev with iSCSI initiator backend')
p.add_argument('-b', '--name', help="Name of the bdev", required=True)
p.add_argument('-i', '--initiator-iqn', help="Initiator IQN", required=True)
p.add_argument('--url', help="iSCSI Lun URL", required=True)
p.set_defaults(func=construct_iscsi_bdev)
@call_cmd
def construct_pmem_bdev(args):
print_array(rpc.bdev.construct_pmem_bdev(args.client,

View File

@ -121,6 +121,25 @@ def construct_error_bdev(client, base_name):
return client.call('construct_error_bdev', params)
def construct_iscsi_bdev(client, name, url, initiator_iqn):
"""Construct a iSCSI block device.
Args:
name: name of block device
url: iSCSI URL
initiator_iqn: IQN name to be used by initiator
Returns:
List of created block devices.
"""
params = {
'name': name,
'url': url,
'initiator_iqn': initiator_iqn,
}
return client.call('construct_iscsi_bdev', params)
def construct_pmem_bdev(client, pmem_file, name):
"""Construct a libpmemblk block device.