]> git.proxmox.com Git - mirror_qemu.git/blame - backends/cryptodev.c
cryptodev: Account statistics
[mirror_qemu.git] / backends / cryptodev.c
CommitLineData
d0ee7a13
GA
1/*
2 * QEMU Crypto Device Implementation
3 *
4 * Copyright (c) 2016 HUAWEI TECHNOLOGIES CO., LTD.
5 *
6 * Authors:
7 * Gonglei <arei.gonglei@huawei.com>
8 *
9 * This library is free software; you can redistribute it and/or
10 * modify it under the terms of the GNU Lesser General Public
11 * License as published by the Free Software Foundation; either
0dda001b 12 * version 2.1 of the License, or (at your option) any later version.
d0ee7a13
GA
13 *
14 * This library is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
17 * Lesser General Public License for more details.
18 *
19 * You should have received a copy of the GNU Lesser General Public
20 * License along with this library; if not, see <http://www.gnu.org/licenses/>.
21 *
22 */
23
24#include "qemu/osdep.h"
25#include "sysemu/cryptodev.h"
d0ee7a13 26#include "qapi/error.h"
5dcb0198 27#include "qapi/qapi-commands-cryptodev.h"
d0ee7a13 28#include "qapi/visitor.h"
d0ee7a13 29#include "qemu/config-file.h"
2fda101d 30#include "qemu/error-report.h"
d0ee7a13 31#include "qom/object_interfaces.h"
d6634ac0
GA
32#include "hw/virtio/virtio-crypto.h"
33
d0ee7a13
GA
34
35static QTAILQ_HEAD(, CryptoDevBackendClient) crypto_clients;
36
5dcb0198
ZP
37static int qmp_query_cryptodev_foreach(Object *obj, void *data)
38{
39 CryptoDevBackend *backend;
40 QCryptodevInfoList **infolist = data;
41 uint32_t services, i;
42
43 if (!object_dynamic_cast(obj, TYPE_CRYPTODEV_BACKEND)) {
44 return 0;
45 }
46
47 QCryptodevInfo *info = g_new0(QCryptodevInfo, 1);
48 info->id = g_strdup(object_get_canonical_path_component(obj));
49
50 backend = CRYPTODEV_BACKEND(obj);
51 services = backend->conf.crypto_services;
52 for (i = 0; i < QCRYPTODEV_BACKEND_SERVICE__MAX; i++) {
53 if (services & (1 << i)) {
54 QAPI_LIST_PREPEND(info->service, i);
55 }
56 }
57
58 for (i = 0; i < backend->conf.peers.queues; i++) {
59 CryptoDevBackendClient *cc = backend->conf.peers.ccs[i];
60 QCryptodevBackendClient *client = g_new0(QCryptodevBackendClient, 1);
61
62 client->queue = cc->queue_index;
63 client->type = cc->type;
64 QAPI_LIST_PREPEND(info->client, client);
65 }
66
67 QAPI_LIST_PREPEND(*infolist, info);
68
69 return 0;
70}
71
72QCryptodevInfoList *qmp_query_cryptodev(Error **errp)
73{
74 QCryptodevInfoList *list = NULL;
75 Object *objs = container_get(object_get_root(), "/objects");
76
77 object_child_foreach(objs, qmp_query_cryptodev_foreach, &list);
78
79 return list;
80}
d0ee7a13 81
3f478371 82CryptoDevBackendClient *cryptodev_backend_new_client(void)
d0ee7a13
GA
83{
84 CryptoDevBackendClient *cc;
85
b21e2380 86 cc = g_new0(CryptoDevBackendClient, 1);
d0ee7a13
GA
87 QTAILQ_INSERT_TAIL(&crypto_clients, cc, next);
88
89 return cc;
90}
91
92void cryptodev_backend_free_client(
93 CryptoDevBackendClient *cc)
94{
95 QTAILQ_REMOVE(&crypto_clients, cc, next);
d0ee7a13
GA
96 g_free(cc->info_str);
97 g_free(cc);
98}
99
100void cryptodev_backend_cleanup(
101 CryptoDevBackend *backend,
102 Error **errp)
103{
104 CryptoDevBackendClass *bc =
105 CRYPTODEV_BACKEND_GET_CLASS(backend);
106
107 if (bc->cleanup) {
108 bc->cleanup(backend, errp);
109 }
e7a775fd
ZP
110
111 g_free(backend->sym_stat);
112 g_free(backend->asym_stat);
d0ee7a13
GA
113}
114
2fda101d 115int cryptodev_backend_create_session(
9e4f86a8 116 CryptoDevBackend *backend,
0e660a6f 117 CryptoDevBackendSessionInfo *sess_info,
2fda101d
LH
118 uint32_t queue_index,
119 CryptoDevCompletionFunc cb,
120 void *opaque)
9e4f86a8
GA
121{
122 CryptoDevBackendClass *bc =
123 CRYPTODEV_BACKEND_GET_CLASS(backend);
124
125 if (bc->create_session) {
2fda101d 126 return bc->create_session(backend, sess_info, queue_index, cb, opaque);
9e4f86a8 127 }
2fda101d 128 return -VIRTIO_CRYPTO_NOTSUPP;
9e4f86a8
GA
129}
130
0e660a6f 131int cryptodev_backend_close_session(
9e4f86a8
GA
132 CryptoDevBackend *backend,
133 uint64_t session_id,
2fda101d
LH
134 uint32_t queue_index,
135 CryptoDevCompletionFunc cb,
136 void *opaque)
9e4f86a8
GA
137{
138 CryptoDevBackendClass *bc =
139 CRYPTODEV_BACKEND_GET_CLASS(backend);
140
141 if (bc->close_session) {
2fda101d 142 return bc->close_session(backend, session_id, queue_index, cb, opaque);
9e4f86a8 143 }
2fda101d 144 return -VIRTIO_CRYPTO_NOTSUPP;
9e4f86a8
GA
145}
146
0e660a6f 147static int cryptodev_backend_operation(
9e4f86a8 148 CryptoDevBackend *backend,
2cb06927 149 CryptoDevBackendOpInfo *op_info)
9e4f86a8
GA
150{
151 CryptoDevBackendClass *bc =
152 CRYPTODEV_BACKEND_GET_CLASS(backend);
153
0e660a6f 154 if (bc->do_op) {
2cb06927 155 return bc->do_op(backend, op_info);
9e4f86a8 156 }
2fda101d 157 return -VIRTIO_CRYPTO_NOTSUPP;
d6634ac0
GA
158}
159
e7a775fd
ZP
160static int cryptodev_backend_account(CryptoDevBackend *backend,
161 CryptoDevBackendOpInfo *op_info)
162{
163 enum QCryptodevBackendAlgType algtype = op_info->algtype;
164 int len;
165
166 if (algtype == QCRYPTODEV_BACKEND_ALG_ASYM) {
167 CryptoDevBackendAsymOpInfo *asym_op_info = op_info->u.asym_op_info;
168 len = asym_op_info->src_len;
169 switch (op_info->op_code) {
170 case VIRTIO_CRYPTO_AKCIPHER_ENCRYPT:
171 CryptodevAsymStatIncEncrypt(backend, len);
172 break;
173 case VIRTIO_CRYPTO_AKCIPHER_DECRYPT:
174 CryptodevAsymStatIncDecrypt(backend, len);
175 break;
176 case VIRTIO_CRYPTO_AKCIPHER_SIGN:
177 CryptodevAsymStatIncSign(backend, len);
178 break;
179 case VIRTIO_CRYPTO_AKCIPHER_VERIFY:
180 CryptodevAsymStatIncVerify(backend, len);
181 break;
182 default:
183 return -VIRTIO_CRYPTO_NOTSUPP;
184 }
185 } else if (algtype == QCRYPTODEV_BACKEND_ALG_SYM) {
186 CryptoDevBackendSymOpInfo *sym_op_info = op_info->u.sym_op_info;
187 len = sym_op_info->src_len;
188 switch (op_info->op_code) {
189 case VIRTIO_CRYPTO_CIPHER_ENCRYPT:
190 CryptodevSymStatIncEncrypt(backend, len);
191 break;
192 case VIRTIO_CRYPTO_CIPHER_DECRYPT:
193 CryptodevSymStatIncDecrypt(backend, len);
194 break;
195 default:
196 return -VIRTIO_CRYPTO_NOTSUPP;
197 }
198 } else {
199 error_report("Unsupported cryptodev alg type: %" PRIu32 "", algtype);
200 return -VIRTIO_CRYPTO_NOTSUPP;
201 }
202
203 return len;
204}
205
d6634ac0
GA
206int cryptodev_backend_crypto_operation(
207 CryptoDevBackend *backend,
2cb06927 208 CryptoDevBackendOpInfo *op_info)
d6634ac0 209{
e7a775fd 210 int ret;
d6634ac0 211
e7a775fd
ZP
212 ret = cryptodev_backend_account(backend, op_info);
213 if (ret < 0) {
214 return ret;
d6634ac0
GA
215 }
216
2cb06927 217 return cryptodev_backend_operation(backend, op_info);
9e4f86a8
GA
218}
219
d0ee7a13
GA
220static void
221cryptodev_backend_get_queues(Object *obj, Visitor *v, const char *name,
222 void *opaque, Error **errp)
223{
224 CryptoDevBackend *backend = CRYPTODEV_BACKEND(obj);
225 uint32_t value = backend->conf.peers.queues;
226
227 visit_type_uint32(v, name, &value, errp);
228}
229
230static void
231cryptodev_backend_set_queues(Object *obj, Visitor *v, const char *name,
232 void *opaque, Error **errp)
233{
234 CryptoDevBackend *backend = CRYPTODEV_BACKEND(obj);
d0ee7a13
GA
235 uint32_t value;
236
668f62ec 237 if (!visit_type_uint32(v, name, &value, errp)) {
dcfe4805 238 return;
d0ee7a13
GA
239 }
240 if (!value) {
dcfe4805
MA
241 error_setg(errp, "Property '%s.%s' doesn't take value '%" PRIu32 "'",
242 object_get_typename(obj), name, value);
243 return;
d0ee7a13
GA
244 }
245 backend->conf.peers.queues = value;
d0ee7a13
GA
246}
247
248static void
249cryptodev_backend_complete(UserCreatable *uc, Error **errp)
250{
251 CryptoDevBackend *backend = CRYPTODEV_BACKEND(uc);
252 CryptoDevBackendClass *bc = CRYPTODEV_BACKEND_GET_CLASS(uc);
e7a775fd 253 uint32_t services;
d0ee7a13
GA
254
255 if (bc->init) {
7dc75edb 256 bc->init(backend, errp);
d0ee7a13 257 }
e7a775fd
ZP
258
259 services = backend->conf.crypto_services;
260 if (services & (1 << QCRYPTODEV_BACKEND_SERVICE_CIPHER)) {
261 backend->sym_stat = g_new0(CryptodevBackendSymStat, 1);
262 }
263
264 if (services & (1 << QCRYPTODEV_BACKEND_SERVICE_AKCIPHER)) {
265 backend->asym_stat = g_new0(CryptodevBackendAsymStat, 1);
266 }
d0ee7a13
GA
267}
268
46fd1705
GA
269void cryptodev_backend_set_used(CryptoDevBackend *backend, bool used)
270{
271 backend->is_used = used;
272}
273
274bool cryptodev_backend_is_used(CryptoDevBackend *backend)
275{
276 return backend->is_used;
277}
278
6138dbda
GA
279void cryptodev_backend_set_ready(CryptoDevBackend *backend, bool ready)
280{
281 backend->ready = ready;
282}
283
284bool cryptodev_backend_is_ready(CryptoDevBackend *backend)
285{
286 return backend->ready;
287}
288
46fd1705 289static bool
3beacfb9 290cryptodev_backend_can_be_deleted(UserCreatable *uc)
46fd1705
GA
291{
292 return !cryptodev_backend_is_used(CRYPTODEV_BACKEND(uc));
293}
294
d0ee7a13
GA
295static void cryptodev_backend_instance_init(Object *obj)
296{
d0ee7a13 297 /* Initialize devices' queues property to 1 */
5325cc34 298 object_property_set_int(obj, "queues", 1, NULL);
d0ee7a13
GA
299}
300
301static void cryptodev_backend_finalize(Object *obj)
302{
46fd1705 303 CryptoDevBackend *backend = CRYPTODEV_BACKEND(obj);
d0ee7a13 304
46fd1705 305 cryptodev_backend_cleanup(backend, NULL);
d0ee7a13
GA
306}
307
308static void
309cryptodev_backend_class_init(ObjectClass *oc, void *data)
310{
311 UserCreatableClass *ucc = USER_CREATABLE_CLASS(oc);
312
313 ucc->complete = cryptodev_backend_complete;
46fd1705 314 ucc->can_be_deleted = cryptodev_backend_can_be_deleted;
d0ee7a13
GA
315
316 QTAILQ_INIT(&crypto_clients);
1f14e388
EH
317 object_class_property_add(oc, "queues", "uint32",
318 cryptodev_backend_get_queues,
319 cryptodev_backend_set_queues,
320 NULL, NULL);
d0ee7a13
GA
321}
322
323static const TypeInfo cryptodev_backend_info = {
324 .name = TYPE_CRYPTODEV_BACKEND,
325 .parent = TYPE_OBJECT,
326 .instance_size = sizeof(CryptoDevBackend),
327 .instance_init = cryptodev_backend_instance_init,
328 .instance_finalize = cryptodev_backend_finalize,
329 .class_size = sizeof(CryptoDevBackendClass),
330 .class_init = cryptodev_backend_class_init,
331 .interfaces = (InterfaceInfo[]) {
332 { TYPE_USER_CREATABLE },
333 { }
334 }
335};
336
337static void
338cryptodev_backend_register_types(void)
339{
340 type_register_static(&cryptodev_backend_info);
341}
342
343type_init(cryptodev_backend_register_types);