xref: /spdk/module/bdev/ocf/vbdev_ocf_rpc.c (revision 94a84ae98590bea46939eb1dcd7a9876bd393b54)
1 /*-
2  *   BSD LICENSE
3  *
4  *   Copyright (c) Intel Corporation.
5  *   All rights reserved.
6  *
7  *   Redistribution and use in source and binary forms, with or without
8  *   modification, are permitted provided that the following conditions
9  *   are met:
10  *
11  *     * Redistributions of source code must retain the above copyright
12  *       notice, this list of conditions and the following disclaimer.
13  *     * Redistributions in binary form must reproduce the above copyright
14  *       notice, this list of conditions and the following disclaimer in
15  *       the documentation and/or other materials provided with the
16  *       distribution.
17  *     * Neither the name of Intel Corporation nor the names of its
18  *       contributors may be used to endorse or promote products derived
19  *       from this software without specific prior written permission.
20  *
21  *   THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
22  *   "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
23  *   LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
24  *   A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
25  *   OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
26  *   SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
27  *   LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
28  *   DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
29  *   THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
30  *   (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
31  *   OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
32  */
33 
34 #include "vbdev_ocf.h"
35 #include "stats.h"
36 #include "spdk/log.h"
37 #include "spdk/rpc.h"
38 #include "spdk/string.h"
39 
40 /* Structure to hold the parameters for this RPC method. */
41 struct rpc_bdev_ocf_create {
42 	char *name;             /* master vbdev */
43 	char *mode;             /* OCF mode (choose one) */
44 	char *cache_bdev_name;  /* sub bdev */
45 	char *core_bdev_name;   /* sub bdev */
46 };
47 
48 static void
49 free_rpc_bdev_ocf_create(struct rpc_bdev_ocf_create *r)
50 {
51 	free(r->name);
52 	free(r->core_bdev_name);
53 	free(r->cache_bdev_name);
54 	free(r->mode);
55 }
56 
57 /* Structure to decode the input parameters for this RPC method. */
58 static const struct spdk_json_object_decoder rpc_bdev_ocf_create_decoders[] = {
59 	{"name", offsetof(struct rpc_bdev_ocf_create, name), spdk_json_decode_string},
60 	{"mode", offsetof(struct rpc_bdev_ocf_create, mode), spdk_json_decode_string},
61 	{"cache_bdev_name", offsetof(struct rpc_bdev_ocf_create, cache_bdev_name), spdk_json_decode_string},
62 	{"core_bdev_name", offsetof(struct rpc_bdev_ocf_create, core_bdev_name), spdk_json_decode_string},
63 };
64 
65 static void
66 construct_cb(int status, struct vbdev_ocf *vbdev, void *cb_arg)
67 {
68 	struct spdk_jsonrpc_request *request = cb_arg;
69 	struct spdk_json_write_ctx *w;
70 
71 	if (status) {
72 		spdk_jsonrpc_send_error_response_fmt(request, SPDK_JSONRPC_ERROR_INTERNAL_ERROR,
73 						     "Could not create OCF vbdev: %d",
74 						     status);
75 	} else {
76 		w = spdk_jsonrpc_begin_result(request);
77 		spdk_json_write_string(w, vbdev->name);
78 		spdk_jsonrpc_end_result(request, w);
79 	}
80 }
81 
82 static void
83 spdk_rpc_bdev_ocf_create(struct spdk_jsonrpc_request *request,
84 			 const struct spdk_json_val *params)
85 {
86 	struct rpc_bdev_ocf_create req = {NULL};
87 	int ret;
88 
89 	ret = spdk_json_decode_object(params, rpc_bdev_ocf_create_decoders,
90 				      SPDK_COUNTOF(rpc_bdev_ocf_create_decoders),
91 				      &req);
92 	if (ret) {
93 		spdk_jsonrpc_send_error_response(request, SPDK_JSONRPC_ERROR_INVALID_PARAMS,
94 						 "Invalid parameters");
95 		free_rpc_bdev_ocf_create(&req);
96 		return;
97 	}
98 
99 	vbdev_ocf_construct(req.name, req.mode, req.cache_bdev_name, req.core_bdev_name, false,
100 			    construct_cb, request);
101 	free_rpc_bdev_ocf_create(&req);
102 }
103 SPDK_RPC_REGISTER("bdev_ocf_create", spdk_rpc_bdev_ocf_create, SPDK_RPC_RUNTIME)
104 SPDK_RPC_REGISTER_ALIAS_DEPRECATED(bdev_ocf_create, construct_ocf_bdev)
105 
106 /* Structure to hold the parameters for this RPC method. */
107 struct rpc_bdev_ocf_delete {
108 	char *name;             /* master vbdev name */
109 };
110 
111 static void
112 free_rpc_bdev_ocf_delete(struct rpc_bdev_ocf_delete *r)
113 {
114 	free(r->name);
115 }
116 
117 /* Structure to decode the input parameters for this RPC method. */
118 static const struct spdk_json_object_decoder rpc_bdev_ocf_delete_decoders[] = {
119 	{"name", offsetof(struct rpc_bdev_ocf_delete, name), spdk_json_decode_string},
120 };
121 
122 static void
123 delete_cb(void *cb_arg, int status)
124 {
125 	struct spdk_jsonrpc_request *request = cb_arg;
126 	struct spdk_json_write_ctx *w;
127 
128 	if (status) {
129 		spdk_jsonrpc_send_error_response_fmt(request, SPDK_JSONRPC_ERROR_INTERNAL_ERROR,
130 						     "Could not delete OCF vbdev: %d",
131 						     status);
132 	} else {
133 		w = spdk_jsonrpc_begin_result(request);
134 		spdk_json_write_bool(w, true);
135 		spdk_jsonrpc_end_result(request, w);
136 	}
137 }
138 
139 static void
140 spdk_rpc_bdev_ocf_delete(struct spdk_jsonrpc_request *request,
141 			 const struct spdk_json_val *params)
142 {
143 	struct rpc_bdev_ocf_delete req = {NULL};
144 	struct vbdev_ocf *vbdev;
145 	int status;
146 
147 	status = spdk_json_decode_object(params, rpc_bdev_ocf_delete_decoders,
148 					 SPDK_COUNTOF(rpc_bdev_ocf_delete_decoders),
149 					 &req);
150 	if (status) {
151 		spdk_jsonrpc_send_error_response(request, SPDK_JSONRPC_ERROR_INVALID_PARAMS,
152 						 "Invalid parameters");
153 		goto end;
154 	}
155 
156 	vbdev = vbdev_ocf_get_by_name(req.name);
157 	if (vbdev == NULL) {
158 		spdk_jsonrpc_send_error_response(request, SPDK_JSONRPC_ERROR_INVALID_PARAMS,
159 						 spdk_strerror(ENODEV));
160 		goto end;
161 	}
162 
163 	status = vbdev_ocf_delete_clean(vbdev, delete_cb, request);
164 	if (status) {
165 		spdk_jsonrpc_send_error_response_fmt(request, SPDK_JSONRPC_ERROR_INTERNAL_ERROR,
166 						     "Could not delete OCF vbdev: %s",
167 						     spdk_strerror(-status));
168 		goto end;
169 	}
170 
171 end:
172 	free_rpc_bdev_ocf_delete(&req);
173 }
174 SPDK_RPC_REGISTER("bdev_ocf_delete", spdk_rpc_bdev_ocf_delete, SPDK_RPC_RUNTIME)
175 SPDK_RPC_REGISTER_ALIAS_DEPRECATED(bdev_ocf_delete, delete_ocf_bdev)
176 
177 /* Structure to hold the parameters for this RPC method. */
178 struct rpc_bdev_ocf_get_stats {
179 	char *name;             /* master vbdev name */
180 };
181 
182 static void
183 free_rpc_bdev_ocf_get_stats(struct rpc_bdev_ocf_get_stats *r)
184 {
185 	free(r->name);
186 }
187 
188 /* Structure to decode the input parameters for this RPC method. */
189 static const struct spdk_json_object_decoder rpc_bdev_ocf_get_stats_decoders[] = {
190 	{"name", offsetof(struct rpc_bdev_ocf_get_stats, name), spdk_json_decode_string},
191 };
192 
193 struct get_ocf_stats_ctx {
194 	struct spdk_jsonrpc_request *request;
195 	int core_id;
196 };
197 
198 static void
199 spdk_rpc_bdev_ocf_get_stats_cmpl(ocf_cache_t cache, void *priv, int error)
200 {
201 	struct get_ocf_stats_ctx *ctx = (struct get_ocf_stats_ctx *) priv;
202 	struct spdk_json_write_ctx *w;
203 	struct vbdev_ocf_stats stats;
204 
205 	if (error) {
206 		goto end;
207 	}
208 
209 	error = vbdev_ocf_stats_get(cache, ctx->core_id, &stats);
210 
211 	ocf_mngt_cache_read_unlock(cache);
212 
213 	if (error) {
214 		goto end;
215 	}
216 
217 	w = spdk_jsonrpc_begin_result(ctx->request);
218 	vbdev_ocf_stats_write_json(w, &stats);
219 	spdk_jsonrpc_end_result(ctx->request, w);
220 
221 end:
222 	if (error) {
223 		spdk_jsonrpc_send_error_response_fmt(ctx->request, SPDK_JSONRPC_ERROR_INTERNAL_ERROR,
224 						     "Could not get stats: %s",
225 						     spdk_strerror(-error));
226 	}
227 	free(ctx);
228 }
229 
230 static void
231 spdk_rpc_bdev_ocf_get_stats(struct spdk_jsonrpc_request *request,
232 			    const struct spdk_json_val *params)
233 {
234 	struct rpc_bdev_ocf_get_stats req = {NULL};
235 	struct vbdev_ocf *vbdev;
236 	struct get_ocf_stats_ctx *ctx;
237 
238 	ctx = calloc(1, sizeof(*ctx));
239 	if (!ctx) {
240 		spdk_jsonrpc_send_error_response(request, SPDK_JSONRPC_ERROR_INVALID_PARAMS,
241 						 "Not enough memory to process request");
242 		goto end;
243 	}
244 
245 	if (spdk_json_decode_object(params, rpc_bdev_ocf_get_stats_decoders,
246 				    SPDK_COUNTOF(rpc_bdev_ocf_get_stats_decoders),
247 				    &req)) {
248 		spdk_jsonrpc_send_error_response(request, SPDK_JSONRPC_ERROR_INVALID_PARAMS,
249 						 "Invalid parameters");
250 		free(ctx);
251 		goto end;
252 	}
253 
254 	vbdev = vbdev_ocf_get_by_name(req.name);
255 	if (vbdev == NULL) {
256 		spdk_jsonrpc_send_error_response(request, SPDK_JSONRPC_ERROR_INVALID_PARAMS,
257 						 spdk_strerror(ENODEV));
258 		free(ctx);
259 		goto end;
260 	}
261 
262 	ctx->core_id = vbdev->core.id;
263 	ctx->request = request;
264 	ocf_mngt_cache_read_lock(vbdev->ocf_cache, spdk_rpc_bdev_ocf_get_stats_cmpl, ctx);
265 
266 end:
267 	free_rpc_bdev_ocf_get_stats(&req);
268 }
269 SPDK_RPC_REGISTER("bdev_ocf_get_stats", spdk_rpc_bdev_ocf_get_stats, SPDK_RPC_RUNTIME)
270 SPDK_RPC_REGISTER_ALIAS_DEPRECATED(bdev_ocf_get_stats, get_ocf_stats)
271 
272 /* Structure to hold the parameters for this RPC method. */
273 struct rpc_bdev_ocf_get_bdevs {
274 	char *name;
275 };
276 
277 static void
278 free_rpc_bdev_ocf_get_bdevs(struct rpc_bdev_ocf_get_bdevs *r)
279 {
280 	free(r->name);
281 }
282 
283 /* Structure to decode the input parameters for this RPC method. */
284 static const struct spdk_json_object_decoder rpc_bdev_ocf_get_bdevs_decoders[] = {
285 	{"name", offsetof(struct rpc_bdev_ocf_get_bdevs, name), spdk_json_decode_string, true},
286 };
287 
288 struct bdev_get_bdevs_ctx {
289 	char *name;
290 	struct spdk_json_write_ctx *w;
291 };
292 
293 static void
294 bdev_get_bdevs_fn(struct vbdev_ocf *vbdev, void *ctx)
295 {
296 	struct bdev_get_bdevs_ctx *cctx = ctx;
297 	struct spdk_json_write_ctx *w = cctx->w;
298 
299 	if (cctx->name != NULL &&
300 	    strcmp(vbdev->name, cctx->name) &&
301 	    strcmp(vbdev->cache.name, cctx->name) &&
302 	    strcmp(vbdev->core.name, cctx->name)) {
303 		return;
304 	}
305 
306 	spdk_json_write_object_begin(w);
307 	spdk_json_write_named_string(w, "name", vbdev->name);
308 	spdk_json_write_named_bool(w, "started", vbdev->state.started);
309 
310 	spdk_json_write_named_object_begin(w, "cache");
311 	spdk_json_write_named_string(w, "name", vbdev->cache.name);
312 	spdk_json_write_named_bool(w, "attached", vbdev->cache.attached);
313 	spdk_json_write_object_end(w);
314 
315 	spdk_json_write_named_object_begin(w, "core");
316 	spdk_json_write_named_string(w, "name", vbdev->core.name);
317 	spdk_json_write_named_bool(w, "attached", vbdev->core.attached);
318 	spdk_json_write_object_end(w);
319 
320 	spdk_json_write_object_end(w);
321 }
322 
323 static void
324 spdk_rpc_bdev_ocf_get_bdevs(struct spdk_jsonrpc_request *request,
325 			    const struct spdk_json_val *params)
326 {
327 	struct spdk_json_write_ctx *w;
328 	struct rpc_bdev_ocf_get_bdevs req = {NULL};
329 	struct bdev_get_bdevs_ctx cctx;
330 
331 	if (params && spdk_json_decode_object(params, rpc_bdev_ocf_get_bdevs_decoders,
332 					      SPDK_COUNTOF(rpc_bdev_ocf_get_bdevs_decoders),
333 					      &req)) {
334 		spdk_jsonrpc_send_error_response(request, SPDK_JSONRPC_ERROR_INVALID_PARAMS,
335 						 "Invalid parameters");
336 		goto end;
337 	}
338 
339 	if (req.name) {
340 		if (!(vbdev_ocf_get_by_name(req.name) || vbdev_ocf_get_base_by_name(req.name))) {
341 			spdk_jsonrpc_send_error_response(request,
342 							 SPDK_JSONRPC_ERROR_INVALID_PARAMS,
343 							 spdk_strerror(ENODEV));
344 			goto end;
345 		}
346 	}
347 
348 	w = spdk_jsonrpc_begin_result(request);
349 
350 	cctx.name    = req.name;
351 	cctx.w       = w;
352 
353 	spdk_json_write_array_begin(w);
354 	vbdev_ocf_foreach(bdev_get_bdevs_fn, &cctx);
355 	spdk_json_write_array_end(w);
356 	spdk_jsonrpc_end_result(request, w);
357 
358 end:
359 	free_rpc_bdev_ocf_get_bdevs(&req);
360 }
361 SPDK_RPC_REGISTER("bdev_ocf_get_bdevs", spdk_rpc_bdev_ocf_get_bdevs, SPDK_RPC_RUNTIME)
362 SPDK_RPC_REGISTER_ALIAS_DEPRECATED(bdev_ocf_get_bdevs, get_ocf_bdevs)
363