blob: ba7b0bc7705ad1b5665ac75ed37b83043ab4de66 [file] [log] [blame]
Gongleid0ee7a12016-10-28 16:33:20 +08001/*
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
Chetan Pant0dda0012020-10-14 13:37:22 +000012 * version 2.1 of the License, or (at your option) any later version.
Gongleid0ee7a12016-10-28 16:33:20 +080013 *
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"
Gongleid0ee7a12016-10-28 16:33:20 +080026#include "qapi/error.h"
zhenwei pi5dcb0192023-03-01 18:58:40 +080027#include "qapi/qapi-commands-cryptodev.h"
Gongleid0ee7a12016-10-28 16:33:20 +080028#include "qapi/visitor.h"
Gongleid0ee7a12016-10-28 16:33:20 +080029#include "qemu/config-file.h"
Lei He2fda1012022-10-08 16:50:27 +080030#include "qemu/error-report.h"
Gongleid0ee7a12016-10-28 16:33:20 +080031#include "qom/object_interfaces.h"
Gongleid6634ac2016-10-28 16:33:29 +080032#include "hw/virtio/virtio-crypto.h"
33
Gongleid0ee7a12016-10-28 16:33:20 +080034
35static QTAILQ_HEAD(, CryptoDevBackendClient) crypto_clients;
36
zhenwei pi5dcb0192023-03-01 18:58:40 +080037static 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}
Gongleid0ee7a12016-10-28 16:33:20 +080081
zhenwei pi3f478372023-03-01 18:58:37 +080082CryptoDevBackendClient *cryptodev_backend_new_client(void)
Gongleid0ee7a12016-10-28 16:33:20 +080083{
84 CryptoDevBackendClient *cc;
85
Markus Armbrusterb21e2382022-03-15 15:41:56 +010086 cc = g_new0(CryptoDevBackendClient, 1);
Gongleid0ee7a12016-10-28 16:33:20 +080087 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);
Gongleid0ee7a12016-10-28 16:33:20 +080096 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 }
Gongleid0ee7a12016-10-28 16:33:20 +0800110}
111
Lei He2fda1012022-10-08 16:50:27 +0800112int cryptodev_backend_create_session(
Gonglei9e4f86a2016-10-28 16:33:21 +0800113 CryptoDevBackend *backend,
zhenwei pi0e660a62022-06-11 14:42:43 +0800114 CryptoDevBackendSessionInfo *sess_info,
Lei He2fda1012022-10-08 16:50:27 +0800115 uint32_t queue_index,
116 CryptoDevCompletionFunc cb,
117 void *opaque)
Gonglei9e4f86a2016-10-28 16:33:21 +0800118{
119 CryptoDevBackendClass *bc =
120 CRYPTODEV_BACKEND_GET_CLASS(backend);
121
122 if (bc->create_session) {
Lei He2fda1012022-10-08 16:50:27 +0800123 return bc->create_session(backend, sess_info, queue_index, cb, opaque);
Gonglei9e4f86a2016-10-28 16:33:21 +0800124 }
Lei He2fda1012022-10-08 16:50:27 +0800125 return -VIRTIO_CRYPTO_NOTSUPP;
Gonglei9e4f86a2016-10-28 16:33:21 +0800126}
127
zhenwei pi0e660a62022-06-11 14:42:43 +0800128int cryptodev_backend_close_session(
Gonglei9e4f86a2016-10-28 16:33:21 +0800129 CryptoDevBackend *backend,
130 uint64_t session_id,
Lei He2fda1012022-10-08 16:50:27 +0800131 uint32_t queue_index,
132 CryptoDevCompletionFunc cb,
133 void *opaque)
Gonglei9e4f86a2016-10-28 16:33:21 +0800134{
135 CryptoDevBackendClass *bc =
136 CRYPTODEV_BACKEND_GET_CLASS(backend);
137
138 if (bc->close_session) {
Lei He2fda1012022-10-08 16:50:27 +0800139 return bc->close_session(backend, session_id, queue_index, cb, opaque);
Gonglei9e4f86a2016-10-28 16:33:21 +0800140 }
Lei He2fda1012022-10-08 16:50:27 +0800141 return -VIRTIO_CRYPTO_NOTSUPP;
Gonglei9e4f86a2016-10-28 16:33:21 +0800142}
143
zhenwei pi0e660a62022-06-11 14:42:43 +0800144static int cryptodev_backend_operation(
Gonglei9e4f86a2016-10-28 16:33:21 +0800145 CryptoDevBackend *backend,
zhenwei pi2cb06922023-03-01 18:58:43 +0800146 CryptoDevBackendOpInfo *op_info)
Gonglei9e4f86a2016-10-28 16:33:21 +0800147{
148 CryptoDevBackendClass *bc =
149 CRYPTODEV_BACKEND_GET_CLASS(backend);
150
zhenwei pi0e660a62022-06-11 14:42:43 +0800151 if (bc->do_op) {
zhenwei pi2cb06922023-03-01 18:58:43 +0800152 return bc->do_op(backend, op_info);
Gonglei9e4f86a2016-10-28 16:33:21 +0800153 }
Lei He2fda1012022-10-08 16:50:27 +0800154 return -VIRTIO_CRYPTO_NOTSUPP;
Gongleid6634ac2016-10-28 16:33:29 +0800155}
156
157int cryptodev_backend_crypto_operation(
158 CryptoDevBackend *backend,
zhenwei pi2cb06922023-03-01 18:58:43 +0800159 CryptoDevBackendOpInfo *op_info)
Gongleid6634ac2016-10-28 16:33:29 +0800160{
zhenwei pi2cb06922023-03-01 18:58:43 +0800161 QCryptodevBackendAlgType algtype = op_info->algtype;
Gongleid6634ac2016-10-28 16:33:29 +0800162
zhenwei pi999c7892023-03-01 18:58:38 +0800163 if ((algtype != QCRYPTODEV_BACKEND_ALG_SYM)
164 && (algtype != QCRYPTODEV_BACKEND_ALG_ASYM)) {
Lei He2fda1012022-10-08 16:50:27 +0800165 error_report("Unsupported cryptodev alg type: %" PRIu32 "", algtype);
zhenwei pi0e660a62022-06-11 14:42:43 +0800166 return -VIRTIO_CRYPTO_NOTSUPP;
Gongleid6634ac2016-10-28 16:33:29 +0800167 }
168
zhenwei pi2cb06922023-03-01 18:58:43 +0800169 return cryptodev_backend_operation(backend, op_info);
Gonglei9e4f86a2016-10-28 16:33:21 +0800170}
171
Gongleid0ee7a12016-10-28 16:33:20 +0800172static void
173cryptodev_backend_get_queues(Object *obj, Visitor *v, const char *name,
174 void *opaque, Error **errp)
175{
176 CryptoDevBackend *backend = CRYPTODEV_BACKEND(obj);
177 uint32_t value = backend->conf.peers.queues;
178
179 visit_type_uint32(v, name, &value, errp);
180}
181
182static void
183cryptodev_backend_set_queues(Object *obj, Visitor *v, const char *name,
184 void *opaque, Error **errp)
185{
186 CryptoDevBackend *backend = CRYPTODEV_BACKEND(obj);
Gongleid0ee7a12016-10-28 16:33:20 +0800187 uint32_t value;
188
Markus Armbruster668f62e2020-07-07 18:06:02 +0200189 if (!visit_type_uint32(v, name, &value, errp)) {
Markus Armbrusterdcfe4802020-07-07 18:06:01 +0200190 return;
Gongleid0ee7a12016-10-28 16:33:20 +0800191 }
192 if (!value) {
Markus Armbrusterdcfe4802020-07-07 18:06:01 +0200193 error_setg(errp, "Property '%s.%s' doesn't take value '%" PRIu32 "'",
194 object_get_typename(obj), name, value);
195 return;
Gongleid0ee7a12016-10-28 16:33:20 +0800196 }
197 backend->conf.peers.queues = value;
Gongleid0ee7a12016-10-28 16:33:20 +0800198}
199
200static void
201cryptodev_backend_complete(UserCreatable *uc, Error **errp)
202{
203 CryptoDevBackend *backend = CRYPTODEV_BACKEND(uc);
204 CryptoDevBackendClass *bc = CRYPTODEV_BACKEND_GET_CLASS(uc);
Gongleid0ee7a12016-10-28 16:33:20 +0800205
206 if (bc->init) {
Vladimir Sementsov-Ogievskiy7dc75ed2019-12-05 20:46:33 +0300207 bc->init(backend, errp);
Gongleid0ee7a12016-10-28 16:33:20 +0800208 }
Gongleid0ee7a12016-10-28 16:33:20 +0800209}
210
Gonglei46fd1702016-12-22 11:12:38 +0800211void cryptodev_backend_set_used(CryptoDevBackend *backend, bool used)
212{
213 backend->is_used = used;
214}
215
216bool cryptodev_backend_is_used(CryptoDevBackend *backend)
217{
218 return backend->is_used;
219}
220
Gonglei6138dbd2016-12-22 11:12:39 +0800221void cryptodev_backend_set_ready(CryptoDevBackend *backend, bool ready)
222{
223 backend->ready = ready;
224}
225
226bool cryptodev_backend_is_ready(CryptoDevBackend *backend)
227{
228 return backend->ready;
229}
230
Gonglei46fd1702016-12-22 11:12:38 +0800231static bool
Eduardo Habkost3beacfb2017-08-29 19:03:37 -0300232cryptodev_backend_can_be_deleted(UserCreatable *uc)
Gonglei46fd1702016-12-22 11:12:38 +0800233{
234 return !cryptodev_backend_is_used(CRYPTODEV_BACKEND(uc));
235}
236
Gongleid0ee7a12016-10-28 16:33:20 +0800237static void cryptodev_backend_instance_init(Object *obj)
238{
Gongleid0ee7a12016-10-28 16:33:20 +0800239 /* Initialize devices' queues property to 1 */
Markus Armbruster5325cc32020-07-07 18:05:54 +0200240 object_property_set_int(obj, "queues", 1, NULL);
Gongleid0ee7a12016-10-28 16:33:20 +0800241}
242
243static void cryptodev_backend_finalize(Object *obj)
244{
Gonglei46fd1702016-12-22 11:12:38 +0800245 CryptoDevBackend *backend = CRYPTODEV_BACKEND(obj);
Gongleid0ee7a12016-10-28 16:33:20 +0800246
Gonglei46fd1702016-12-22 11:12:38 +0800247 cryptodev_backend_cleanup(backend, NULL);
Gongleid0ee7a12016-10-28 16:33:20 +0800248}
249
250static void
251cryptodev_backend_class_init(ObjectClass *oc, void *data)
252{
253 UserCreatableClass *ucc = USER_CREATABLE_CLASS(oc);
254
255 ucc->complete = cryptodev_backend_complete;
Gonglei46fd1702016-12-22 11:12:38 +0800256 ucc->can_be_deleted = cryptodev_backend_can_be_deleted;
Gongleid0ee7a12016-10-28 16:33:20 +0800257
258 QTAILQ_INIT(&crypto_clients);
Eduardo Habkost1f14e382020-09-21 18:10:23 -0400259 object_class_property_add(oc, "queues", "uint32",
260 cryptodev_backend_get_queues,
261 cryptodev_backend_set_queues,
262 NULL, NULL);
Gongleid0ee7a12016-10-28 16:33:20 +0800263}
264
265static const TypeInfo cryptodev_backend_info = {
266 .name = TYPE_CRYPTODEV_BACKEND,
267 .parent = TYPE_OBJECT,
268 .instance_size = sizeof(CryptoDevBackend),
269 .instance_init = cryptodev_backend_instance_init,
270 .instance_finalize = cryptodev_backend_finalize,
271 .class_size = sizeof(CryptoDevBackendClass),
272 .class_init = cryptodev_backend_class_init,
273 .interfaces = (InterfaceInfo[]) {
274 { TYPE_USER_CREATABLE },
275 { }
276 }
277};
278
279static void
280cryptodev_backend_register_types(void)
281{
282 type_register_static(&cryptodev_backend_info);
283}
284
285type_init(cryptodev_backend_register_types);