024179d8ee
POSIX defines PRId64/PRIu64/PRIx64 for printing 64-bit values in a portable way. Replace a few references to %ld to remove the assumption about the size of a long. Similarly, use %z with size_t arguments. Where the value being printed is an unsigned 64-bit value, use PRIu64 instead of %ld. Signed-off-by: Nick Connolly <nick.connolly@mayadata.io> Change-Id: I0ec4c0cb5e7a8f4b94a1ee6272c8df9ee7f06aa6 Reviewed-on: https://review.spdk.io/gerrit/c/spdk/spdk/+/5141 Community-CI: Mellanox Build Bot Tested-by: SPDK CI Jenkins <sys_sgci@intel.com> Reviewed-by: Aleksey Marchuk <alexeymar@mellanox.com> Reviewed-by: Jim Harris <james.r.harris@intel.com> Reviewed-by: Shuhei Matsumoto <shuhei.matsumoto.xt@hitachi.com>
331 lines
9.0 KiB
C
331 lines
9.0 KiB
C
/*-
|
|
* 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 "spdk/stdinc.h"
|
|
#include "spdk/blobfs.h"
|
|
#include "spdk/bdev.h"
|
|
#include "spdk/event.h"
|
|
#include "spdk/blob_bdev.h"
|
|
#include "spdk/blobfs_bdev.h"
|
|
#include "spdk/log.h"
|
|
#include "spdk/string.h"
|
|
#include "spdk/rpc.h"
|
|
#include "spdk/util.h"
|
|
|
|
#ifndef PAGE_SIZE
|
|
#define PAGE_SIZE 4096
|
|
#endif
|
|
|
|
#define MIN_CLUSTER_SZ (1024 * 1024)
|
|
|
|
struct rpc_blobfs_set_cache_size {
|
|
uint64_t size_in_mb;
|
|
};
|
|
|
|
static const struct spdk_json_object_decoder rpc_blobfs_set_cache_size_decoders[] = {
|
|
{"size_in_mb", offsetof(struct rpc_blobfs_set_cache_size, size_in_mb), spdk_json_decode_uint64},
|
|
};
|
|
|
|
static void
|
|
rpc_blobfs_set_cache_size(struct spdk_jsonrpc_request *request,
|
|
const struct spdk_json_val *params)
|
|
{
|
|
struct rpc_blobfs_set_cache_size req;
|
|
int rc;
|
|
|
|
if (spdk_json_decode_object(params, rpc_blobfs_set_cache_size_decoders,
|
|
SPDK_COUNTOF(rpc_blobfs_set_cache_size_decoders),
|
|
&req)) {
|
|
SPDK_ERRLOG("spdk_json_decode_object failed\n");
|
|
spdk_jsonrpc_send_error_response(request, SPDK_JSONRPC_ERROR_INVALID_PARAMS,
|
|
"spdk_json_decode_object failed");
|
|
|
|
return;
|
|
}
|
|
|
|
if (req.size_in_mb == 0) {
|
|
spdk_jsonrpc_send_error_response(request, SPDK_JSONRPC_ERROR_INVALID_PARAMS,
|
|
"spdk_json_decode_object failed");
|
|
|
|
return;
|
|
}
|
|
|
|
rc = spdk_fs_set_cache_size(req.size_in_mb);
|
|
|
|
spdk_jsonrpc_send_bool_response(request, rc == 0);
|
|
}
|
|
|
|
SPDK_RPC_REGISTER("blobfs_set_cache_size", rpc_blobfs_set_cache_size,
|
|
SPDK_RPC_STARTUP | SPDK_RPC_RUNTIME)
|
|
|
|
struct rpc_blobfs_detect {
|
|
char *bdev_name;
|
|
|
|
struct spdk_jsonrpc_request *request;
|
|
};
|
|
|
|
static void
|
|
free_rpc_blobfs_detect(struct rpc_blobfs_detect *req)
|
|
{
|
|
free(req->bdev_name);
|
|
free(req);
|
|
}
|
|
|
|
static const struct spdk_json_object_decoder rpc_blobfs_detect_decoders[] = {
|
|
{"bdev_name", offsetof(struct rpc_blobfs_detect, bdev_name), spdk_json_decode_string},
|
|
};
|
|
|
|
static void
|
|
_rpc_blobfs_detect_done(void *cb_arg, int fserrno)
|
|
{
|
|
struct rpc_blobfs_detect *req = cb_arg;
|
|
bool existed = true;
|
|
|
|
if (fserrno == -EILSEQ) {
|
|
/* There is no blobfs existing on bdev */
|
|
existed = false;
|
|
} else if (fserrno != 0) {
|
|
spdk_jsonrpc_send_error_response(req->request, SPDK_JSONRPC_ERROR_INTERNAL_ERROR,
|
|
spdk_strerror(-fserrno));
|
|
|
|
return;
|
|
}
|
|
|
|
spdk_jsonrpc_send_bool_response(req->request, existed);
|
|
|
|
free_rpc_blobfs_detect(req);
|
|
}
|
|
|
|
static void
|
|
rpc_blobfs_detect(struct spdk_jsonrpc_request *request,
|
|
const struct spdk_json_val *params)
|
|
{
|
|
struct rpc_blobfs_detect *req;
|
|
|
|
req = calloc(1, sizeof(*req));
|
|
if (req == NULL) {
|
|
SPDK_ERRLOG("could not allocate rpc_blobfs_detect request.\n");
|
|
spdk_jsonrpc_send_error_response(request, SPDK_JSONRPC_ERROR_INTERNAL_ERROR, "Out of memory");
|
|
return;
|
|
}
|
|
|
|
if (spdk_json_decode_object(params, rpc_blobfs_detect_decoders,
|
|
SPDK_COUNTOF(rpc_blobfs_detect_decoders),
|
|
req)) {
|
|
SPDK_ERRLOG("spdk_json_decode_object failed\n");
|
|
spdk_jsonrpc_send_error_response(request, SPDK_JSONRPC_ERROR_INVALID_PARAMS,
|
|
"spdk_json_decode_object failed");
|
|
|
|
free_rpc_blobfs_detect(req);
|
|
|
|
return;
|
|
}
|
|
|
|
req->request = request;
|
|
spdk_blobfs_bdev_detect(req->bdev_name, _rpc_blobfs_detect_done, req);
|
|
}
|
|
|
|
SPDK_RPC_REGISTER("blobfs_detect", rpc_blobfs_detect, SPDK_RPC_RUNTIME)
|
|
|
|
struct rpc_blobfs_create {
|
|
char *bdev_name;
|
|
uint64_t cluster_sz;
|
|
|
|
struct spdk_jsonrpc_request *request;
|
|
};
|
|
|
|
static void
|
|
free_rpc_blobfs_create(struct rpc_blobfs_create *req)
|
|
{
|
|
free(req->bdev_name);
|
|
free(req);
|
|
}
|
|
|
|
static int
|
|
rpc_decode_cluster_sz(const struct spdk_json_val *val, void *out)
|
|
{
|
|
uint64_t *cluster_sz = out;
|
|
char *sz_str = NULL;
|
|
bool has_prefix;
|
|
int rc;
|
|
|
|
rc = spdk_json_decode_string(val, &sz_str);
|
|
if (rc) {
|
|
SPDK_NOTICELOG("Invalid parameter value: cluster_sz\n");
|
|
return -EINVAL;
|
|
}
|
|
|
|
rc = spdk_parse_capacity(sz_str, cluster_sz, &has_prefix);
|
|
free(sz_str);
|
|
|
|
if (rc || *cluster_sz % PAGE_SIZE != 0 || *cluster_sz < MIN_CLUSTER_SZ) {
|
|
SPDK_NOTICELOG("Invalid parameter value: cluster_sz\n");
|
|
return -EINVAL;
|
|
}
|
|
|
|
SPDK_DEBUGLOG(blobfs_bdev_rpc, "cluster_sz of blobfs: %" PRId64 "\n", *cluster_sz);
|
|
return 0;
|
|
}
|
|
|
|
static const struct spdk_json_object_decoder rpc_blobfs_create_decoders[] = {
|
|
{"bdev_name", offsetof(struct rpc_blobfs_create, bdev_name), spdk_json_decode_string},
|
|
{"cluster_sz", offsetof(struct rpc_blobfs_create, cluster_sz), rpc_decode_cluster_sz, true},
|
|
};
|
|
|
|
static void
|
|
_rpc_blobfs_create_done(void *cb_arg, int fserrno)
|
|
{
|
|
struct rpc_blobfs_create *req = cb_arg;
|
|
|
|
if (fserrno != 0) {
|
|
spdk_jsonrpc_send_error_response(req->request, SPDK_JSONRPC_ERROR_INTERNAL_ERROR,
|
|
spdk_strerror(-fserrno));
|
|
|
|
return;
|
|
}
|
|
|
|
spdk_jsonrpc_send_bool_response(req->request, true);
|
|
|
|
free_rpc_blobfs_create(req);
|
|
}
|
|
|
|
static void
|
|
rpc_blobfs_create(struct spdk_jsonrpc_request *request,
|
|
const struct spdk_json_val *params)
|
|
{
|
|
struct rpc_blobfs_create *req;
|
|
|
|
req = calloc(1, sizeof(*req));
|
|
if (req == NULL) {
|
|
SPDK_ERRLOG("could not allocate rpc_blobfs_create request.\n");
|
|
spdk_jsonrpc_send_error_response(request, SPDK_JSONRPC_ERROR_INTERNAL_ERROR, "Out of memory");
|
|
return;
|
|
}
|
|
|
|
if (spdk_json_decode_object(params, rpc_blobfs_create_decoders,
|
|
SPDK_COUNTOF(rpc_blobfs_create_decoders),
|
|
req)) {
|
|
SPDK_ERRLOG("spdk_json_decode_object failed\n");
|
|
spdk_jsonrpc_send_error_response(request, SPDK_JSONRPC_ERROR_INVALID_PARAMS,
|
|
"spdk_json_decode_object failed");
|
|
|
|
free_rpc_blobfs_create(req);
|
|
|
|
return;
|
|
}
|
|
|
|
req->request = request;
|
|
spdk_blobfs_bdev_create(req->bdev_name, req->cluster_sz, _rpc_blobfs_create_done, req);
|
|
}
|
|
|
|
SPDK_RPC_REGISTER("blobfs_create", rpc_blobfs_create, SPDK_RPC_RUNTIME)
|
|
|
|
SPDK_LOG_REGISTER_COMPONENT(blobfs_bdev_rpc)
|
|
#ifdef SPDK_CONFIG_FUSE
|
|
|
|
struct rpc_blobfs_mount {
|
|
char *bdev_name;
|
|
char *mountpoint;
|
|
|
|
struct spdk_jsonrpc_request *request;
|
|
};
|
|
|
|
static void
|
|
free_rpc_blobfs_mount(struct rpc_blobfs_mount *req)
|
|
{
|
|
free(req->bdev_name);
|
|
free(req->mountpoint);
|
|
free(req);
|
|
}
|
|
|
|
static const struct spdk_json_object_decoder rpc_blobfs_mount_decoders[] = {
|
|
{"bdev_name", offsetof(struct rpc_blobfs_mount, bdev_name), spdk_json_decode_string},
|
|
{"mountpoint", offsetof(struct rpc_blobfs_mount, mountpoint), spdk_json_decode_string},
|
|
};
|
|
|
|
static void
|
|
_rpc_blobfs_mount_done(void *cb_arg, int fserrno)
|
|
{
|
|
struct rpc_blobfs_mount *req = cb_arg;
|
|
|
|
if (fserrno == -EILSEQ) {
|
|
/* There is no blobfs existing on bdev */
|
|
spdk_jsonrpc_send_error_response(req->request, SPDK_JSONRPC_ERROR_INVALID_PARAMS,
|
|
"No blobfs detected on given bdev");
|
|
|
|
return;
|
|
} else if (fserrno != 0) {
|
|
spdk_jsonrpc_send_error_response(req->request, SPDK_JSONRPC_ERROR_INTERNAL_ERROR,
|
|
spdk_strerror(-fserrno));
|
|
|
|
return;
|
|
}
|
|
|
|
spdk_jsonrpc_send_bool_response(req->request, true);
|
|
|
|
free_rpc_blobfs_mount(req);
|
|
}
|
|
|
|
static void
|
|
rpc_blobfs_mount(struct spdk_jsonrpc_request *request,
|
|
const struct spdk_json_val *params)
|
|
{
|
|
struct rpc_blobfs_mount *req;
|
|
|
|
req = calloc(1, sizeof(*req));
|
|
if (req == NULL) {
|
|
SPDK_ERRLOG("could not allocate rpc_blobfs_mount request.\n");
|
|
spdk_jsonrpc_send_error_response(request, SPDK_JSONRPC_ERROR_INTERNAL_ERROR, "Out of memory");
|
|
return;
|
|
}
|
|
|
|
if (spdk_json_decode_object(params, rpc_blobfs_mount_decoders,
|
|
SPDK_COUNTOF(rpc_blobfs_mount_decoders),
|
|
req)) {
|
|
SPDK_ERRLOG("spdk_json_decode_object failed\n");
|
|
spdk_jsonrpc_send_error_response(request, SPDK_JSONRPC_ERROR_INVALID_PARAMS,
|
|
"spdk_json_decode_object failed");
|
|
|
|
free_rpc_blobfs_mount(req);
|
|
|
|
return;
|
|
}
|
|
|
|
req->request = request;
|
|
spdk_blobfs_bdev_mount(req->bdev_name, req->mountpoint, _rpc_blobfs_mount_done, req);
|
|
}
|
|
|
|
SPDK_RPC_REGISTER("blobfs_mount", rpc_blobfs_mount, SPDK_RPC_RUNTIME)
|
|
|
|
#endif
|