xref: /spdk/module/bdev/compress/vbdev_compress_rpc.c (revision 7506a7aa53d239f533af3bc768f0d2af55e735fe)
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_compress.h"
35  #include "spdk/rpc.h"
36  #include "spdk/util.h"
37  #include "spdk/string.h"
38  #include "spdk/log.h"
39  
40  struct rpc_bdev_compress_get_orphans {
41  	char *name;
42  };
43  
44  static void
45  free_rpc_bdev_compress_get_orphans(struct rpc_bdev_compress_get_orphans *r)
46  {
47  	free(r->name);
48  }
49  
50  static const struct spdk_json_object_decoder rpc_bdev_compress_get_orphans_decoders[] = {
51  	{"name", offsetof(struct rpc_bdev_compress_get_orphans, name), spdk_json_decode_string, true},
52  };
53  
54  static void
55  rpc_bdev_compress_get_orphans(struct spdk_jsonrpc_request *request,
56  			      const struct spdk_json_val *params)
57  {
58  	struct rpc_bdev_compress_get_orphans req = {};
59  	struct spdk_json_write_ctx *w;
60  	struct vbdev_compress *comp_bdev;
61  	bool found = false;
62  
63  
64  	if (params && spdk_json_decode_object(params, rpc_bdev_compress_get_orphans_decoders,
65  					      SPDK_COUNTOF(rpc_bdev_compress_get_orphans_decoders),
66  					      &req)) {
67  		SPDK_ERRLOG("spdk_json_decode_object failed\n");
68  		spdk_jsonrpc_send_error_response(request, SPDK_JSONRPC_ERROR_INTERNAL_ERROR,
69  						 "spdk_json_decode_object failed");
70  		free_rpc_bdev_compress_get_orphans(&req);
71  		return;
72  	}
73  
74  	if (req.name) {
75  		if (compress_has_orphan(req.name) == false) {
76  			spdk_jsonrpc_send_error_response(request, -ENODEV, spdk_strerror(ENODEV));
77  			free_rpc_bdev_compress_get_orphans(&req);
78  			return;
79  		}
80  		found = true;
81  	}
82  
83  	w = spdk_jsonrpc_begin_result(request);
84  	spdk_json_write_array_begin(w);
85  	if (found) {
86  		spdk_json_write_string(w, req.name);
87  	} else {
88  		for (comp_bdev = compress_bdev_first(); comp_bdev != NULL;
89  		     comp_bdev = compress_bdev_next(comp_bdev)) {
90  			if (compress_has_orphan(compress_get_name(comp_bdev))) {
91  				spdk_json_write_string(w, compress_get_name(comp_bdev));
92  			}
93  		}
94  	}
95  	spdk_json_write_array_end(w);
96  	spdk_jsonrpc_end_result(request, w);
97  	free_rpc_bdev_compress_get_orphans(&req);
98  }
99  SPDK_RPC_REGISTER("bdev_compress_get_orphans", rpc_bdev_compress_get_orphans, SPDK_RPC_RUNTIME)
100  
101  struct rpc_compress_set_pmd {
102  	enum compress_pmd pmd;
103  };
104  
105  static const struct spdk_json_object_decoder rpc_compress_pmd_decoder[] = {
106  	{"pmd", offsetof(struct rpc_compress_set_pmd, pmd), spdk_json_decode_int32},
107  };
108  
109  static void
110  rpc_bdev_compress_set_pmd(struct spdk_jsonrpc_request *request,
111  			  const struct spdk_json_val *params)
112  {
113  	struct rpc_compress_set_pmd req;
114  	int rc = 0;
115  
116  	if (spdk_json_decode_object(params, rpc_compress_pmd_decoder,
117  				    SPDK_COUNTOF(rpc_compress_pmd_decoder),
118  				    &req)) {
119  		SPDK_ERRLOG("spdk_json_decode_object failed\n");
120  		spdk_jsonrpc_send_error_response(request, SPDK_JSONRPC_ERROR_INTERNAL_ERROR,
121  						 "spdk_json_decode_object failed");
122  		return;
123  	}
124  
125  	if (req.pmd >= COMPRESS_PMD_MAX) {
126  		spdk_jsonrpc_send_error_response_fmt(request, -EINVAL,
127  						     "PMD value %d should be less than %d", req.pmd, COMPRESS_PMD_MAX);
128  		return;
129  	}
130  
131  	rc = compress_set_pmd(&req.pmd);
132  	if (rc) {
133  		spdk_jsonrpc_send_error_response(request, rc, spdk_strerror(-rc));
134  		return;
135  	}
136  
137  	spdk_jsonrpc_send_bool_response(request, true);
138  }
139  SPDK_RPC_REGISTER("bdev_compress_set_pmd", rpc_bdev_compress_set_pmd,
140  		  SPDK_RPC_STARTUP | SPDK_RPC_RUNTIME)
141  
142  /* Structure to hold the parameters for this RPC method. */
143  struct rpc_construct_compress {
144  	char *base_bdev_name;
145  	char *pm_path;
146  	uint32_t lb_size;
147  };
148  
149  /* Free the allocated memory resource after the RPC handling. */
150  static void
151  free_rpc_construct_compress(struct rpc_construct_compress *r)
152  {
153  	free(r->base_bdev_name);
154  	free(r->pm_path);
155  }
156  
157  /* Structure to decode the input parameters for this RPC method. */
158  static const struct spdk_json_object_decoder rpc_construct_compress_decoders[] = {
159  	{"base_bdev_name", offsetof(struct rpc_construct_compress, base_bdev_name), spdk_json_decode_string},
160  	{"pm_path", offsetof(struct rpc_construct_compress, pm_path), spdk_json_decode_string},
161  	{"lb_size", offsetof(struct rpc_construct_compress, lb_size), spdk_json_decode_uint32},
162  };
163  
164  /* Decode the parameters for this RPC method and properly construct the compress
165   * device. Error status returned in the failed cases.
166   */
167  static void
168  rpc_bdev_compress_create(struct spdk_jsonrpc_request *request,
169  			 const struct spdk_json_val *params)
170  {
171  	struct rpc_construct_compress req = {NULL};
172  	struct spdk_json_write_ctx *w;
173  	char *name;
174  	int rc;
175  
176  	if (spdk_json_decode_object(params, rpc_construct_compress_decoders,
177  				    SPDK_COUNTOF(rpc_construct_compress_decoders),
178  				    &req)) {
179  		SPDK_DEBUGLOG(vbdev_compress, "spdk_json_decode_object failed\n");
180  		spdk_jsonrpc_send_error_response(request, SPDK_JSONRPC_ERROR_PARSE_ERROR,
181  						 "spdk_json_decode_object failed");
182  		goto cleanup;
183  	}
184  
185  	rc = create_compress_bdev(req.base_bdev_name, req.pm_path, req.lb_size);
186  	if (rc != 0) {
187  		if (rc == -EBUSY) {
188  			spdk_jsonrpc_send_error_response(request, rc, "Base bdev already in use for compression.");
189  		} else {
190  			spdk_jsonrpc_send_error_response(request, rc, spdk_strerror(-rc));
191  		}
192  		goto cleanup;
193  	}
194  
195  	w = spdk_jsonrpc_begin_result(request);
196  	name = spdk_sprintf_alloc("COMP_%s", req.base_bdev_name);
197  	spdk_json_write_string(w, name);
198  	spdk_jsonrpc_end_result(request, w);
199  	free(name);
200  
201  cleanup:
202  	free_rpc_construct_compress(&req);
203  }
204  SPDK_RPC_REGISTER("bdev_compress_create", rpc_bdev_compress_create, SPDK_RPC_RUNTIME)
205  
206  struct rpc_delete_compress {
207  	char *name;
208  };
209  
210  static void
211  free_rpc_delete_compress(struct rpc_delete_compress *req)
212  {
213  	free(req->name);
214  }
215  
216  static const struct spdk_json_object_decoder rpc_delete_compress_decoders[] = {
217  	{"name", offsetof(struct rpc_delete_compress, name), spdk_json_decode_string},
218  };
219  
220  static void
221  _rpc_bdev_compress_delete_cb(void *cb_arg, int bdeverrno)
222  {
223  	struct spdk_jsonrpc_request *request = cb_arg;
224  
225  	spdk_jsonrpc_send_bool_response(request, bdeverrno == 0);
226  }
227  
228  static void
229  rpc_bdev_compress_delete(struct spdk_jsonrpc_request *request,
230  			 const struct spdk_json_val *params)
231  {
232  	struct rpc_delete_compress req = {NULL};
233  
234  	if (spdk_json_decode_object(params, rpc_delete_compress_decoders,
235  				    SPDK_COUNTOF(rpc_delete_compress_decoders),
236  				    &req)) {
237  		spdk_jsonrpc_send_error_response(request, SPDK_JSONRPC_ERROR_INTERNAL_ERROR,
238  						 "spdk_json_decode_object failed");
239  	} else {
240  		bdev_compress_delete(req.name, _rpc_bdev_compress_delete_cb, request);
241  	}
242  
243  	free_rpc_delete_compress(&req);
244  }
245  SPDK_RPC_REGISTER("bdev_compress_delete", rpc_bdev_compress_delete, SPDK_RPC_RUNTIME)
246