xref: /spdk/module/bdev/ocf/vbdev_ocf_rpc.c (revision 927f1fd57bd004df581518466ec4c1b8083e5d23)
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 "utils.h"
37 #include "spdk/log.h"
38 #include "spdk/rpc.h"
39 #include "spdk/string.h"
40 
41 /* Structure to hold the parameters for this RPC method. */
42 struct rpc_bdev_ocf_create {
43 	char *name;			/* main vbdev */
44 	char *mode;			/* OCF mode (choose one) */
45 	uint64_t cache_line_size;	/* OCF cache line size */
46 	char *cache_bdev_name;		/* sub bdev */
47 	char *core_bdev_name;		/* sub bdev */
48 };
49 
50 static void
51 free_rpc_bdev_ocf_create(struct rpc_bdev_ocf_create *r)
52 {
53 	free(r->name);
54 	free(r->core_bdev_name);
55 	free(r->cache_bdev_name);
56 	free(r->mode);
57 }
58 
59 /* Structure to decode the input parameters for this RPC method. */
60 static const struct spdk_json_object_decoder rpc_bdev_ocf_create_decoders[] = {
61 	{"name", offsetof(struct rpc_bdev_ocf_create, name), spdk_json_decode_string},
62 	{"mode", offsetof(struct rpc_bdev_ocf_create, mode), spdk_json_decode_string},
63 	{"cache_line_size", offsetof(struct rpc_bdev_ocf_create, cache_line_size), spdk_json_decode_uint64, true},
64 	{"cache_bdev_name", offsetof(struct rpc_bdev_ocf_create, cache_bdev_name), spdk_json_decode_string},
65 	{"core_bdev_name", offsetof(struct rpc_bdev_ocf_create, core_bdev_name), spdk_json_decode_string},
66 };
67 
68 static void
69 construct_cb(int status, struct vbdev_ocf *vbdev, void *cb_arg)
70 {
71 	struct spdk_jsonrpc_request *request = cb_arg;
72 	struct spdk_json_write_ctx *w;
73 
74 	if (status) {
75 		spdk_jsonrpc_send_error_response_fmt(request, SPDK_JSONRPC_ERROR_INTERNAL_ERROR,
76 						     "Could not create OCF vbdev: %d",
77 						     status);
78 	} else {
79 		w = spdk_jsonrpc_begin_result(request);
80 		spdk_json_write_string(w, vbdev->name);
81 		spdk_jsonrpc_end_result(request, w);
82 	}
83 }
84 
85 static void
86 rpc_bdev_ocf_create(struct spdk_jsonrpc_request *request,
87 		    const struct spdk_json_val *params)
88 {
89 	struct rpc_bdev_ocf_create req = {NULL};
90 	int ret;
91 
92 	ret = spdk_json_decode_object(params, rpc_bdev_ocf_create_decoders,
93 				      SPDK_COUNTOF(rpc_bdev_ocf_create_decoders),
94 				      &req);
95 	if (ret) {
96 		spdk_jsonrpc_send_error_response(request, SPDK_JSONRPC_ERROR_INVALID_PARAMS,
97 						 "Invalid parameters");
98 		free_rpc_bdev_ocf_create(&req);
99 		return;
100 	}
101 
102 	vbdev_ocf_construct(req.name, req.mode, req.cache_line_size, req.cache_bdev_name,
103 			    req.core_bdev_name, false, construct_cb, request);
104 	free_rpc_bdev_ocf_create(&req);
105 }
106 SPDK_RPC_REGISTER("bdev_ocf_create", rpc_bdev_ocf_create, SPDK_RPC_RUNTIME)
107 SPDK_RPC_REGISTER_ALIAS_DEPRECATED(bdev_ocf_create, construct_ocf_bdev)
108 
109 /* Structure to hold the parameters for this RPC method. */
110 struct rpc_bdev_ocf_delete {
111 	char *name;             /* main vbdev name */
112 };
113 
114 static void
115 free_rpc_bdev_ocf_delete(struct rpc_bdev_ocf_delete *r)
116 {
117 	free(r->name);
118 }
119 
120 /* Structure to decode the input parameters for this RPC method. */
121 static const struct spdk_json_object_decoder rpc_bdev_ocf_delete_decoders[] = {
122 	{"name", offsetof(struct rpc_bdev_ocf_delete, name), spdk_json_decode_string},
123 };
124 
125 static void
126 delete_cb(void *cb_arg, int status)
127 {
128 	struct spdk_jsonrpc_request *request = cb_arg;
129 
130 	if (status) {
131 		spdk_jsonrpc_send_error_response_fmt(request, SPDK_JSONRPC_ERROR_INTERNAL_ERROR,
132 						     "Could not delete OCF vbdev: %d",
133 						     status);
134 	} else {
135 		spdk_jsonrpc_send_bool_response(request, true);
136 	}
137 }
138 
139 static void
140 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", 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;             /* main 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 	char *core_name;
196 };
197 
198 static void
199 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_name, &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 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_name = vbdev->core.name;
263 	ctx->request = request;
264 	ocf_mngt_cache_read_lock(vbdev->ocf_cache, 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", 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 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", rpc_bdev_ocf_get_bdevs, SPDK_RPC_RUNTIME)
362 SPDK_RPC_REGISTER_ALIAS_DEPRECATED(bdev_ocf_get_bdevs, get_ocf_bdevs)
363 
364 /* Structure to hold the parameters for this RPC method. */
365 struct rpc_bdev_ocf_set_cache_mode {
366 	char *name;			/* main vbdev name */
367 	char *mode;			/* OCF cache mode to switch to */
368 };
369 
370 static void
371 free_rpc_bdev_ocf_set_cache_mode(struct rpc_bdev_ocf_set_cache_mode *r)
372 {
373 	free(r->name);
374 	free(r->mode);
375 }
376 
377 /* Structure to decode the input parameters for this RPC method. */
378 static const struct spdk_json_object_decoder rpc_bdev_ocf_set_cache_mode_decoders[] = {
379 	{"name", offsetof(struct rpc_bdev_ocf_set_cache_mode, name), spdk_json_decode_string},
380 	{"mode", offsetof(struct rpc_bdev_ocf_set_cache_mode, mode), spdk_json_decode_string},
381 };
382 
383 static void
384 cache_mode_cb(int status, struct vbdev_ocf *vbdev, void *cb_arg)
385 {
386 	struct spdk_jsonrpc_request *request = cb_arg;
387 	struct spdk_json_write_ctx *w;
388 
389 	if (status) {
390 		spdk_jsonrpc_send_error_response_fmt(request, SPDK_JSONRPC_ERROR_INTERNAL_ERROR,
391 						     "Could not change OCF vbdev cache mode: %d",
392 						     status);
393 	} else {
394 		w = spdk_jsonrpc_begin_result(request);
395 		spdk_json_write_string(w, ocf_get_cache_modename(
396 					       ocf_cache_get_mode(vbdev->ocf_cache)));
397 		spdk_jsonrpc_end_result(request, w);
398 	}
399 }
400 
401 static void
402 rpc_bdev_ocf_set_cache_mode(struct spdk_jsonrpc_request *request,
403 			    const struct spdk_json_val *params)
404 {
405 	struct rpc_bdev_ocf_set_cache_mode req = {NULL};
406 	struct vbdev_ocf *vbdev;
407 	int status;
408 
409 	status = spdk_json_decode_object(params, rpc_bdev_ocf_set_cache_mode_decoders,
410 					 SPDK_COUNTOF(rpc_bdev_ocf_set_cache_mode_decoders),
411 					 &req);
412 	if (status) {
413 		spdk_jsonrpc_send_error_response(request, SPDK_JSONRPC_ERROR_INVALID_PARAMS,
414 						 "Invalid parameters");
415 		goto end;
416 	}
417 
418 	vbdev = vbdev_ocf_get_by_name(req.name);
419 	if (vbdev == NULL) {
420 		spdk_jsonrpc_send_error_response(request, SPDK_JSONRPC_ERROR_INVALID_PARAMS,
421 						 spdk_strerror(ENODEV));
422 		goto end;
423 	}
424 
425 	vbdev_ocf_set_cache_mode(vbdev, req.mode, cache_mode_cb, request);
426 
427 end:
428 	free_rpc_bdev_ocf_set_cache_mode(&req);
429 }
430 SPDK_RPC_REGISTER("bdev_ocf_set_cache_mode", rpc_bdev_ocf_set_cache_mode, SPDK_RPC_RUNTIME)
431 
432 static void
433 seqcutoff_cb(int status, void *cb_arg)
434 {
435 	struct spdk_jsonrpc_request *request = cb_arg;
436 
437 	if (status) {
438 		spdk_jsonrpc_send_error_response_fmt(request, SPDK_JSONRPC_ERROR_INTERNAL_ERROR,
439 						     "OCF could not set sequential cutoff parameters: %d", status);
440 	} else {
441 		spdk_jsonrpc_send_bool_response(request, true);
442 	}
443 }
444 
445 /* Structure to hold the parameters for this RPC method. */
446 struct rpc_bdev_ocf_set_seqcutoff {
447 	char *name;		/* main vbdev name */
448 	char *policy;
449 	uint32_t threshold;
450 	uint32_t promotion_count;
451 };
452 
453 static void
454 free_rpc_bdev_ocf_set_seqcutoff(struct rpc_bdev_ocf_set_seqcutoff *r)
455 {
456 	free(r->name);
457 	free(r->policy);
458 }
459 
460 /* Structure to decode the input parameters for this RPC method. */
461 static const struct spdk_json_object_decoder rpc_bdev_ocf_set_seqcutoff_decoders[] = {
462 	{"name", offsetof(struct rpc_bdev_ocf_set_seqcutoff, name), spdk_json_decode_string},
463 	{"policy", offsetof(struct rpc_bdev_ocf_set_seqcutoff, policy), spdk_json_decode_string},
464 	{"threshold", offsetof(struct rpc_bdev_ocf_set_seqcutoff, threshold), spdk_json_decode_uint32, true},
465 	{"promotion_count", offsetof(struct rpc_bdev_ocf_set_seqcutoff, promotion_count), spdk_json_decode_uint32, true},
466 };
467 
468 static void
469 rpc_bdev_ocf_set_seqcutoff(struct spdk_jsonrpc_request *request,
470 			   const struct spdk_json_val *params)
471 {
472 	struct rpc_bdev_ocf_set_seqcutoff req = {NULL};
473 	struct vbdev_ocf *vbdev;
474 	int ret;
475 
476 	ret = spdk_json_decode_object(params, rpc_bdev_ocf_set_seqcutoff_decoders,
477 				      SPDK_COUNTOF(rpc_bdev_ocf_set_seqcutoff_decoders), &req);
478 	if (ret) {
479 		spdk_jsonrpc_send_error_response(request, SPDK_JSONRPC_ERROR_INVALID_PARAMS,
480 						 "Invalid parameters");
481 		goto end;
482 	}
483 
484 	vbdev = vbdev_ocf_get_by_name(req.name);
485 	if (vbdev == NULL) {
486 		spdk_jsonrpc_send_error_response(request, SPDK_JSONRPC_ERROR_INVALID_PARAMS,
487 						 spdk_strerror(ENODEV));
488 		goto end;
489 	}
490 
491 	vbdev_ocf_set_seqcutoff(vbdev, req.policy, req.threshold, req.promotion_count, seqcutoff_cb,
492 				request);
493 
494 end:
495 	free_rpc_bdev_ocf_set_seqcutoff(&req);
496 }
497 SPDK_RPC_REGISTER("bdev_ocf_set_seqcutoff", rpc_bdev_ocf_set_seqcutoff, SPDK_RPC_RUNTIME)
498