xref: /spdk/module/blobfs/bdev/blobfs_bdev_rpc.c (revision 841cd8b939e1798ee8bd1fa9efdf7e344f53491d)
1488570ebSJim Harris /*   SPDX-License-Identifier: BSD-3-Clause
2a6dbe372Spaul luse  *   Copyright (C) 2019 Intel Corporation.
3*841cd8b9SShuhei Matsumoto  *   Copyright (c) 2023 NVIDIA CORPORATION & AFFILIATES. All rights reserved.
43ce759a1SXiaodong Liu  *   All rights reserved.
53ce759a1SXiaodong Liu  */
63ce759a1SXiaodong Liu 
73ce759a1SXiaodong Liu #include "spdk/stdinc.h"
83ce759a1SXiaodong Liu #include "spdk/blobfs.h"
93ce759a1SXiaodong Liu #include "spdk/bdev.h"
103ce759a1SXiaodong Liu #include "spdk/event.h"
113ce759a1SXiaodong Liu #include "spdk/blob_bdev.h"
123ce759a1SXiaodong Liu #include "spdk/blobfs_bdev.h"
133ce759a1SXiaodong Liu #include "spdk/log.h"
143ce759a1SXiaodong Liu #include "spdk/string.h"
153ce759a1SXiaodong Liu #include "spdk/rpc.h"
163ce759a1SXiaodong Liu #include "spdk/util.h"
173ce759a1SXiaodong Liu 
186433c103SXiaodong Liu #ifndef PAGE_SIZE
196433c103SXiaodong Liu #define PAGE_SIZE 4096
206433c103SXiaodong Liu #endif
216433c103SXiaodong Liu 
226433c103SXiaodong Liu #define MIN_CLUSTER_SZ (1024 * 1024)
236433c103SXiaodong Liu 
24138e6daeSXiaodong Liu struct rpc_blobfs_set_cache_size {
25138e6daeSXiaodong Liu 	uint64_t size_in_mb;
26138e6daeSXiaodong Liu };
27138e6daeSXiaodong Liu 
28138e6daeSXiaodong Liu static const struct spdk_json_object_decoder rpc_blobfs_set_cache_size_decoders[] = {
29138e6daeSXiaodong Liu 	{"size_in_mb", offsetof(struct rpc_blobfs_set_cache_size, size_in_mb), spdk_json_decode_uint64},
30138e6daeSXiaodong Liu };
31138e6daeSXiaodong Liu 
32138e6daeSXiaodong Liu static void
rpc_blobfs_set_cache_size(struct spdk_jsonrpc_request * request,const struct spdk_json_val * params)3330e3f4d9SSeth Howell rpc_blobfs_set_cache_size(struct spdk_jsonrpc_request *request,
34138e6daeSXiaodong Liu 			  const struct spdk_json_val *params)
35138e6daeSXiaodong Liu {
36138e6daeSXiaodong Liu 	struct rpc_blobfs_set_cache_size req;
37138e6daeSXiaodong Liu 	int rc;
38138e6daeSXiaodong Liu 
39138e6daeSXiaodong Liu 	if (spdk_json_decode_object(params, rpc_blobfs_set_cache_size_decoders,
40138e6daeSXiaodong Liu 				    SPDK_COUNTOF(rpc_blobfs_set_cache_size_decoders),
41138e6daeSXiaodong Liu 				    &req)) {
42138e6daeSXiaodong Liu 		SPDK_ERRLOG("spdk_json_decode_object failed\n");
43138e6daeSXiaodong Liu 		spdk_jsonrpc_send_error_response(request, SPDK_JSONRPC_ERROR_INVALID_PARAMS,
44138e6daeSXiaodong Liu 						 "spdk_json_decode_object failed");
45138e6daeSXiaodong Liu 
46138e6daeSXiaodong Liu 		return;
47138e6daeSXiaodong Liu 	}
48138e6daeSXiaodong Liu 
49138e6daeSXiaodong Liu 	if (req.size_in_mb == 0) {
50138e6daeSXiaodong Liu 		spdk_jsonrpc_send_error_response(request, SPDK_JSONRPC_ERROR_INVALID_PARAMS,
51138e6daeSXiaodong Liu 						 "spdk_json_decode_object failed");
52138e6daeSXiaodong Liu 
53138e6daeSXiaodong Liu 		return;
54138e6daeSXiaodong Liu 	}
55138e6daeSXiaodong Liu 
56138e6daeSXiaodong Liu 	rc = spdk_fs_set_cache_size(req.size_in_mb);
57138e6daeSXiaodong Liu 
58*841cd8b9SShuhei Matsumoto 	if (rc == 0) {
59*841cd8b9SShuhei Matsumoto 		spdk_jsonrpc_send_bool_response(request, true);
60*841cd8b9SShuhei Matsumoto 	} else {
61*841cd8b9SShuhei Matsumoto 		spdk_jsonrpc_send_error_response(request, rc, spdk_strerror(-rc));
62*841cd8b9SShuhei Matsumoto 	}
63138e6daeSXiaodong Liu }
64138e6daeSXiaodong Liu 
6530e3f4d9SSeth Howell SPDK_RPC_REGISTER("blobfs_set_cache_size", rpc_blobfs_set_cache_size,
66138e6daeSXiaodong Liu 		  SPDK_RPC_STARTUP | SPDK_RPC_RUNTIME)
67138e6daeSXiaodong Liu 
683ce759a1SXiaodong Liu struct rpc_blobfs_detect {
693ce759a1SXiaodong Liu 	char *bdev_name;
703ce759a1SXiaodong Liu 
713ce759a1SXiaodong Liu 	struct spdk_jsonrpc_request *request;
723ce759a1SXiaodong Liu };
733ce759a1SXiaodong Liu 
743ce759a1SXiaodong Liu static void
free_rpc_blobfs_detect(struct rpc_blobfs_detect * req)753ce759a1SXiaodong Liu free_rpc_blobfs_detect(struct rpc_blobfs_detect *req)
763ce759a1SXiaodong Liu {
773ce759a1SXiaodong Liu 	free(req->bdev_name);
783ce759a1SXiaodong Liu 	free(req);
793ce759a1SXiaodong Liu }
803ce759a1SXiaodong Liu 
813ce759a1SXiaodong Liu static const struct spdk_json_object_decoder rpc_blobfs_detect_decoders[] = {
823ce759a1SXiaodong Liu 	{"bdev_name", offsetof(struct rpc_blobfs_detect, bdev_name), spdk_json_decode_string},
833ce759a1SXiaodong Liu };
843ce759a1SXiaodong Liu 
853ce759a1SXiaodong Liu static void
_rpc_blobfs_detect_done(void * cb_arg,int fserrno)863ce759a1SXiaodong Liu _rpc_blobfs_detect_done(void *cb_arg, int fserrno)
873ce759a1SXiaodong Liu {
883ce759a1SXiaodong Liu 	struct rpc_blobfs_detect *req = cb_arg;
893ce759a1SXiaodong Liu 	bool existed = true;
903ce759a1SXiaodong Liu 
913ce759a1SXiaodong Liu 	if (fserrno == -EILSEQ) {
923ce759a1SXiaodong Liu 		/* There is no blobfs existing on bdev */
933ce759a1SXiaodong Liu 		existed = false;
943ce759a1SXiaodong Liu 	} else if (fserrno != 0) {
953ce759a1SXiaodong Liu 		spdk_jsonrpc_send_error_response(req->request, SPDK_JSONRPC_ERROR_INTERNAL_ERROR,
963ce759a1SXiaodong Liu 						 spdk_strerror(-fserrno));
97*841cd8b9SShuhei Matsumoto 		free_rpc_blobfs_detect(req);
983ce759a1SXiaodong Liu 		return;
993ce759a1SXiaodong Liu 	}
1003ce759a1SXiaodong Liu 
101d73077b8Syidong0635 	spdk_jsonrpc_send_bool_response(req->request, existed);
1023ce759a1SXiaodong Liu 
1033ce759a1SXiaodong Liu 	free_rpc_blobfs_detect(req);
1043ce759a1SXiaodong Liu }
1053ce759a1SXiaodong Liu 
1063ce759a1SXiaodong Liu static void
rpc_blobfs_detect(struct spdk_jsonrpc_request * request,const struct spdk_json_val * params)10730e3f4d9SSeth Howell rpc_blobfs_detect(struct spdk_jsonrpc_request *request,
1083ce759a1SXiaodong Liu 		  const struct spdk_json_val *params)
1093ce759a1SXiaodong Liu {
1103ce759a1SXiaodong Liu 	struct rpc_blobfs_detect *req;
1113ce759a1SXiaodong Liu 
1123ce759a1SXiaodong Liu 	req = calloc(1, sizeof(*req));
1133ce759a1SXiaodong Liu 	if (req == NULL) {
1143ce759a1SXiaodong Liu 		SPDK_ERRLOG("could not allocate rpc_blobfs_detect request.\n");
1153ce759a1SXiaodong Liu 		spdk_jsonrpc_send_error_response(request, SPDK_JSONRPC_ERROR_INTERNAL_ERROR, "Out of memory");
1163ce759a1SXiaodong Liu 		return;
1173ce759a1SXiaodong Liu 	}
1183ce759a1SXiaodong Liu 
1193ce759a1SXiaodong Liu 	if (spdk_json_decode_object(params, rpc_blobfs_detect_decoders,
1203ce759a1SXiaodong Liu 				    SPDK_COUNTOF(rpc_blobfs_detect_decoders),
1213ce759a1SXiaodong Liu 				    req)) {
1223ce759a1SXiaodong Liu 		SPDK_ERRLOG("spdk_json_decode_object failed\n");
1233ce759a1SXiaodong Liu 		spdk_jsonrpc_send_error_response(request, SPDK_JSONRPC_ERROR_INVALID_PARAMS,
1243ce759a1SXiaodong Liu 						 "spdk_json_decode_object failed");
1253ce759a1SXiaodong Liu 
1263ce759a1SXiaodong Liu 		free_rpc_blobfs_detect(req);
1273ce759a1SXiaodong Liu 
1283ce759a1SXiaodong Liu 		return;
1293ce759a1SXiaodong Liu 	}
1303ce759a1SXiaodong Liu 
1313ce759a1SXiaodong Liu 	req->request = request;
1323ce759a1SXiaodong Liu 	spdk_blobfs_bdev_detect(req->bdev_name, _rpc_blobfs_detect_done, req);
1333ce759a1SXiaodong Liu }
1343ce759a1SXiaodong Liu 
13530e3f4d9SSeth Howell SPDK_RPC_REGISTER("blobfs_detect", rpc_blobfs_detect, SPDK_RPC_RUNTIME)
1366433c103SXiaodong Liu 
1376433c103SXiaodong Liu struct rpc_blobfs_create {
1386433c103SXiaodong Liu 	char *bdev_name;
1396433c103SXiaodong Liu 	uint64_t cluster_sz;
1406433c103SXiaodong Liu 
1416433c103SXiaodong Liu 	struct spdk_jsonrpc_request *request;
1426433c103SXiaodong Liu };
1436433c103SXiaodong Liu 
1446433c103SXiaodong Liu static void
free_rpc_blobfs_create(struct rpc_blobfs_create * req)1456433c103SXiaodong Liu free_rpc_blobfs_create(struct rpc_blobfs_create *req)
1466433c103SXiaodong Liu {
1476433c103SXiaodong Liu 	free(req->bdev_name);
1486433c103SXiaodong Liu 	free(req);
1496433c103SXiaodong Liu }
1506433c103SXiaodong Liu 
1516433c103SXiaodong Liu static int
rpc_decode_cluster_sz(const struct spdk_json_val * val,void * out)1526433c103SXiaodong Liu rpc_decode_cluster_sz(const struct spdk_json_val *val, void *out)
1536433c103SXiaodong Liu {
1546433c103SXiaodong Liu 	uint64_t *cluster_sz = out;
1556433c103SXiaodong Liu 	char *sz_str = NULL;
1566433c103SXiaodong Liu 	bool has_prefix;
1576433c103SXiaodong Liu 	int rc;
1586433c103SXiaodong Liu 
1596433c103SXiaodong Liu 	rc = spdk_json_decode_string(val, &sz_str);
1606433c103SXiaodong Liu 	if (rc) {
1616433c103SXiaodong Liu 		SPDK_NOTICELOG("Invalid parameter value: cluster_sz\n");
1626433c103SXiaodong Liu 		return -EINVAL;
1636433c103SXiaodong Liu 	}
1646433c103SXiaodong Liu 
1656433c103SXiaodong Liu 	rc = spdk_parse_capacity(sz_str, cluster_sz, &has_prefix);
1666433c103SXiaodong Liu 	free(sz_str);
1676433c103SXiaodong Liu 
1686433c103SXiaodong Liu 	if (rc || *cluster_sz % PAGE_SIZE != 0 || *cluster_sz < MIN_CLUSTER_SZ) {
1696433c103SXiaodong Liu 		SPDK_NOTICELOG("Invalid parameter value: cluster_sz\n");
1706433c103SXiaodong Liu 		return -EINVAL;
1716433c103SXiaodong Liu 	}
1726433c103SXiaodong Liu 
173024179d8SNick Connolly 	SPDK_DEBUGLOG(blobfs_bdev_rpc, "cluster_sz of blobfs: %" PRId64 "\n", *cluster_sz);
1746433c103SXiaodong Liu 	return 0;
1756433c103SXiaodong Liu }
1766433c103SXiaodong Liu 
1776433c103SXiaodong Liu static const struct spdk_json_object_decoder rpc_blobfs_create_decoders[] = {
1786433c103SXiaodong Liu 	{"bdev_name", offsetof(struct rpc_blobfs_create, bdev_name), spdk_json_decode_string},
1796433c103SXiaodong Liu 	{"cluster_sz", offsetof(struct rpc_blobfs_create, cluster_sz), rpc_decode_cluster_sz, true},
1806433c103SXiaodong Liu };
1816433c103SXiaodong Liu 
1826433c103SXiaodong Liu static void
_rpc_blobfs_create_done(void * cb_arg,int fserrno)1836433c103SXiaodong Liu _rpc_blobfs_create_done(void *cb_arg, int fserrno)
1846433c103SXiaodong Liu {
1856433c103SXiaodong Liu 	struct rpc_blobfs_create *req = cb_arg;
1866433c103SXiaodong Liu 
1876433c103SXiaodong Liu 	if (fserrno != 0) {
1886433c103SXiaodong Liu 		spdk_jsonrpc_send_error_response(req->request, SPDK_JSONRPC_ERROR_INTERNAL_ERROR,
1896433c103SXiaodong Liu 						 spdk_strerror(-fserrno));
1906433c103SXiaodong Liu 
1916433c103SXiaodong Liu 		return;
1926433c103SXiaodong Liu 	}
1936433c103SXiaodong Liu 
194d73077b8Syidong0635 	spdk_jsonrpc_send_bool_response(req->request, true);
1956433c103SXiaodong Liu 
1966433c103SXiaodong Liu 	free_rpc_blobfs_create(req);
1976433c103SXiaodong Liu }
1986433c103SXiaodong Liu 
1996433c103SXiaodong Liu static void
rpc_blobfs_create(struct spdk_jsonrpc_request * request,const struct spdk_json_val * params)20030e3f4d9SSeth Howell rpc_blobfs_create(struct spdk_jsonrpc_request *request,
2016433c103SXiaodong Liu 		  const struct spdk_json_val *params)
2026433c103SXiaodong Liu {
2036433c103SXiaodong Liu 	struct rpc_blobfs_create *req;
2046433c103SXiaodong Liu 
2056433c103SXiaodong Liu 	req = calloc(1, sizeof(*req));
2066433c103SXiaodong Liu 	if (req == NULL) {
2076433c103SXiaodong Liu 		SPDK_ERRLOG("could not allocate rpc_blobfs_create request.\n");
2086433c103SXiaodong Liu 		spdk_jsonrpc_send_error_response(request, SPDK_JSONRPC_ERROR_INTERNAL_ERROR, "Out of memory");
2096433c103SXiaodong Liu 		return;
2106433c103SXiaodong Liu 	}
2116433c103SXiaodong Liu 
2126433c103SXiaodong Liu 	if (spdk_json_decode_object(params, rpc_blobfs_create_decoders,
2136433c103SXiaodong Liu 				    SPDK_COUNTOF(rpc_blobfs_create_decoders),
2146433c103SXiaodong Liu 				    req)) {
2156433c103SXiaodong Liu 		SPDK_ERRLOG("spdk_json_decode_object failed\n");
2166433c103SXiaodong Liu 		spdk_jsonrpc_send_error_response(request, SPDK_JSONRPC_ERROR_INVALID_PARAMS,
2176433c103SXiaodong Liu 						 "spdk_json_decode_object failed");
2186433c103SXiaodong Liu 
2196433c103SXiaodong Liu 		free_rpc_blobfs_create(req);
2206433c103SXiaodong Liu 
2216433c103SXiaodong Liu 		return;
2226433c103SXiaodong Liu 	}
2236433c103SXiaodong Liu 
2246433c103SXiaodong Liu 	req->request = request;
2256433c103SXiaodong Liu 	spdk_blobfs_bdev_create(req->bdev_name, req->cluster_sz, _rpc_blobfs_create_done, req);
2266433c103SXiaodong Liu }
2276433c103SXiaodong Liu 
22830e3f4d9SSeth Howell SPDK_RPC_REGISTER("blobfs_create", rpc_blobfs_create, SPDK_RPC_RUNTIME)
2299c6e7426SXiaodong Liu 
2302172c432STomasz Zawadzki SPDK_LOG_REGISTER_COMPONENT(blobfs_bdev_rpc)
2319c6e7426SXiaodong Liu #ifdef SPDK_CONFIG_FUSE
2329c6e7426SXiaodong Liu 
2339c6e7426SXiaodong Liu struct rpc_blobfs_mount {
2349c6e7426SXiaodong Liu 	char *bdev_name;
2359c6e7426SXiaodong Liu 	char *mountpoint;
2369c6e7426SXiaodong Liu 
2379c6e7426SXiaodong Liu 	struct spdk_jsonrpc_request *request;
2389c6e7426SXiaodong Liu };
2399c6e7426SXiaodong Liu 
2409c6e7426SXiaodong Liu static void
free_rpc_blobfs_mount(struct rpc_blobfs_mount * req)2419c6e7426SXiaodong Liu free_rpc_blobfs_mount(struct rpc_blobfs_mount *req)
2429c6e7426SXiaodong Liu {
2439c6e7426SXiaodong Liu 	free(req->bdev_name);
2449c6e7426SXiaodong Liu 	free(req->mountpoint);
2459c6e7426SXiaodong Liu 	free(req);
2469c6e7426SXiaodong Liu }
2479c6e7426SXiaodong Liu 
2489c6e7426SXiaodong Liu static const struct spdk_json_object_decoder rpc_blobfs_mount_decoders[] = {
2499c6e7426SXiaodong Liu 	{"bdev_name", offsetof(struct rpc_blobfs_mount, bdev_name), spdk_json_decode_string},
2509c6e7426SXiaodong Liu 	{"mountpoint", offsetof(struct rpc_blobfs_mount, mountpoint), spdk_json_decode_string},
2519c6e7426SXiaodong Liu };
2529c6e7426SXiaodong Liu 
2539c6e7426SXiaodong Liu static void
_rpc_blobfs_mount_done(void * cb_arg,int fserrno)2549c6e7426SXiaodong Liu _rpc_blobfs_mount_done(void *cb_arg, int fserrno)
2559c6e7426SXiaodong Liu {
2569c6e7426SXiaodong Liu 	struct rpc_blobfs_mount *req = cb_arg;
2579c6e7426SXiaodong Liu 
2589c6e7426SXiaodong Liu 	if (fserrno == -EILSEQ) {
2599c6e7426SXiaodong Liu 		/* There is no blobfs existing on bdev */
2609c6e7426SXiaodong Liu 		spdk_jsonrpc_send_error_response(req->request, SPDK_JSONRPC_ERROR_INVALID_PARAMS,
2619c6e7426SXiaodong Liu 						 "No blobfs detected on given bdev");
2629c6e7426SXiaodong Liu 
2639c6e7426SXiaodong Liu 		return;
2649c6e7426SXiaodong Liu 	} else if (fserrno != 0) {
2659c6e7426SXiaodong Liu 		spdk_jsonrpc_send_error_response(req->request, SPDK_JSONRPC_ERROR_INTERNAL_ERROR,
2669c6e7426SXiaodong Liu 						 spdk_strerror(-fserrno));
2679c6e7426SXiaodong Liu 
2689c6e7426SXiaodong Liu 		return;
2699c6e7426SXiaodong Liu 	}
2709c6e7426SXiaodong Liu 
271d73077b8Syidong0635 	spdk_jsonrpc_send_bool_response(req->request, true);
2729c6e7426SXiaodong Liu 
2739c6e7426SXiaodong Liu 	free_rpc_blobfs_mount(req);
2749c6e7426SXiaodong Liu }
2759c6e7426SXiaodong Liu 
2769c6e7426SXiaodong Liu static void
rpc_blobfs_mount(struct spdk_jsonrpc_request * request,const struct spdk_json_val * params)27730e3f4d9SSeth Howell rpc_blobfs_mount(struct spdk_jsonrpc_request *request,
2789c6e7426SXiaodong Liu 		 const struct spdk_json_val *params)
2799c6e7426SXiaodong Liu {
2809c6e7426SXiaodong Liu 	struct rpc_blobfs_mount *req;
2819c6e7426SXiaodong Liu 
2829c6e7426SXiaodong Liu 	req = calloc(1, sizeof(*req));
2839c6e7426SXiaodong Liu 	if (req == NULL) {
2849c6e7426SXiaodong Liu 		SPDK_ERRLOG("could not allocate rpc_blobfs_mount request.\n");
2859c6e7426SXiaodong Liu 		spdk_jsonrpc_send_error_response(request, SPDK_JSONRPC_ERROR_INTERNAL_ERROR, "Out of memory");
2869c6e7426SXiaodong Liu 		return;
2879c6e7426SXiaodong Liu 	}
2889c6e7426SXiaodong Liu 
2899c6e7426SXiaodong Liu 	if (spdk_json_decode_object(params, rpc_blobfs_mount_decoders,
2909c6e7426SXiaodong Liu 				    SPDK_COUNTOF(rpc_blobfs_mount_decoders),
2919c6e7426SXiaodong Liu 				    req)) {
2929c6e7426SXiaodong Liu 		SPDK_ERRLOG("spdk_json_decode_object failed\n");
2939c6e7426SXiaodong Liu 		spdk_jsonrpc_send_error_response(request, SPDK_JSONRPC_ERROR_INVALID_PARAMS,
2949c6e7426SXiaodong Liu 						 "spdk_json_decode_object failed");
2959c6e7426SXiaodong Liu 
2969c6e7426SXiaodong Liu 		free_rpc_blobfs_mount(req);
2979c6e7426SXiaodong Liu 
2989c6e7426SXiaodong Liu 		return;
2999c6e7426SXiaodong Liu 	}
3009c6e7426SXiaodong Liu 
3019c6e7426SXiaodong Liu 	req->request = request;
3029c6e7426SXiaodong Liu 	spdk_blobfs_bdev_mount(req->bdev_name, req->mountpoint, _rpc_blobfs_mount_done, req);
3039c6e7426SXiaodong Liu }
3049c6e7426SXiaodong Liu 
30530e3f4d9SSeth Howell SPDK_RPC_REGISTER("blobfs_mount", rpc_blobfs_mount, SPDK_RPC_RUNTIME)
3069c6e7426SXiaodong Liu 
3079c6e7426SXiaodong Liu #endif
308