blob: 398ebe4a7d09f9eabe0a11e3cff7f52d60016702 [file] [log] [blame]
Anthony Liguoria9b7b2a2012-06-25 10:03:47 -05001/*
2 * QEMU Random Number Generator Backend
3 *
4 * Copyright IBM, Corp. 2012
5 *
6 * Authors:
7 * Anthony Liguori <aliguori@us.ibm.com>
8 *
9 * This work is licensed under the terms of the GNU GPL, version 2 or later.
10 * See the COPYING file in the top-level directory.
11 */
12
Peter Maydell9c058332016-01-29 17:49:54 +000013#include "qemu/osdep.h"
Paolo Bonzinidccfcd02013-04-08 16:55:25 +020014#include "sysemu/rng.h"
Markus Armbrusterda34e652016-03-14 09:01:28 +010015#include "qapi/error.h"
Paolo Bonzini7b1b5d12012-12-17 18:19:43 +010016#include "qapi/qmp/qerror.h"
Igor Mammedov269e09f2014-01-16 17:34:38 +010017#include "qom/object_interfaces.h"
Anthony Liguoria9b7b2a2012-06-25 10:03:47 -050018
19void rng_backend_request_entropy(RngBackend *s, size_t size,
20 EntropyReceiveFunc *receive_entropy,
21 void *opaque)
22{
23 RngBackendClass *k = RNG_BACKEND_GET_CLASS(s);
Ladi Prosek60253ed2016-03-03 09:37:18 +010024 RngRequest *req;
Anthony Liguoria9b7b2a2012-06-25 10:03:47 -050025
26 if (k->request_entropy) {
Ladi Prosek60253ed2016-03-03 09:37:18 +010027 req = g_malloc(sizeof(*req));
28
29 req->offset = 0;
30 req->size = size;
31 req->receive_entropy = receive_entropy;
32 req->opaque = opaque;
33 req->data = g_malloc(req->size);
34
35 k->request_entropy(s, req);
36
Ladi Prosek443590c2016-03-03 14:16:11 +010037 QSIMPLEQ_INSERT_TAIL(&s->requests, req, next);
Anthony Liguoria9b7b2a2012-06-25 10:03:47 -050038 }
39}
40
Anthony Liguoria9b7b2a2012-06-25 10:03:47 -050041static bool rng_backend_prop_get_opened(Object *obj, Error **errp)
42{
43 RngBackend *s = RNG_BACKEND(obj);
44
45 return s->opened;
46}
47
Igor Mammedov57d3e1b2014-01-16 17:34:39 +010048static void rng_backend_complete(UserCreatable *uc, Error **errp)
Anthony Liguoria9b7b2a2012-06-25 10:03:47 -050049{
Igor Mammedov57d3e1b2014-01-16 17:34:39 +010050 object_property_set_bool(OBJECT(uc), true, "opened", errp);
Anthony Liguoria9b7b2a2012-06-25 10:03:47 -050051}
52
53static void rng_backend_prop_set_opened(Object *obj, bool value, Error **errp)
54{
55 RngBackend *s = RNG_BACKEND(obj);
56 RngBackendClass *k = RNG_BACKEND_GET_CLASS(s);
Markus Armbruster65cd9062014-04-25 12:44:22 +020057 Error *local_err = NULL;
Anthony Liguoria9b7b2a2012-06-25 10:03:47 -050058
59 if (value == s->opened) {
60 return;
61 }
62
63 if (!value && s->opened) {
Markus Armbrusterc6bd8c72015-03-17 11:54:50 +010064 error_setg(errp, QERR_PERMISSION_DENIED);
Anthony Liguoria9b7b2a2012-06-25 10:03:47 -050065 return;
66 }
67
68 if (k->opened) {
Markus Armbruster65cd9062014-04-25 12:44:22 +020069 k->opened(s, &local_err);
70 if (local_err) {
71 error_propagate(errp, local_err);
72 return;
73 }
Anthony Liguoria9b7b2a2012-06-25 10:03:47 -050074 }
75
Markus Armbruster65cd9062014-04-25 12:44:22 +020076 s->opened = true;
Anthony Liguoria9b7b2a2012-06-25 10:03:47 -050077}
78
Ladi Prosek9f14b0a2016-03-03 09:37:17 +010079static void rng_backend_free_request(RngRequest *req)
80{
81 g_free(req->data);
82 g_free(req);
83}
84
85static void rng_backend_free_requests(RngBackend *s)
86{
Ladi Prosek443590c2016-03-03 14:16:11 +010087 RngRequest *req, *next;
Ladi Prosek9f14b0a2016-03-03 09:37:17 +010088
Ladi Prosek443590c2016-03-03 14:16:11 +010089 QSIMPLEQ_FOREACH_SAFE(req, &s->requests, next, next) {
90 rng_backend_free_request(req);
Ladi Prosek9f14b0a2016-03-03 09:37:17 +010091 }
92
Ladi Prosek443590c2016-03-03 14:16:11 +010093 QSIMPLEQ_INIT(&s->requests);
Ladi Prosek9f14b0a2016-03-03 09:37:17 +010094}
95
96void rng_backend_finalize_request(RngBackend *s, RngRequest *req)
97{
Ladi Prosek443590c2016-03-03 14:16:11 +010098 QSIMPLEQ_REMOVE(&s->requests, req, RngRequest, next);
Ladi Prosek9f14b0a2016-03-03 09:37:17 +010099 rng_backend_free_request(req);
100}
101
Anthony Liguoria9b7b2a2012-06-25 10:03:47 -0500102static void rng_backend_init(Object *obj)
103{
Ladi Prosek443590c2016-03-03 14:16:11 +0100104 RngBackend *s = RNG_BACKEND(obj);
105
106 QSIMPLEQ_INIT(&s->requests);
107
Anthony Liguoria9b7b2a2012-06-25 10:03:47 -0500108 object_property_add_bool(obj, "opened",
109 rng_backend_prop_get_opened,
110 rng_backend_prop_set_opened,
111 NULL);
112}
113
Ladi Prosek9f14b0a2016-03-03 09:37:17 +0100114static void rng_backend_finalize(Object *obj)
115{
116 RngBackend *s = RNG_BACKEND(obj);
117
118 rng_backend_free_requests(s);
119}
120
Igor Mammedov57d3e1b2014-01-16 17:34:39 +0100121static void rng_backend_class_init(ObjectClass *oc, void *data)
122{
123 UserCreatableClass *ucc = USER_CREATABLE_CLASS(oc);
124
125 ucc->complete = rng_backend_complete;
126}
127
Andreas Färber8c43a6f2013-01-10 16:19:07 +0100128static const TypeInfo rng_backend_info = {
Anthony Liguoria9b7b2a2012-06-25 10:03:47 -0500129 .name = TYPE_RNG_BACKEND,
130 .parent = TYPE_OBJECT,
131 .instance_size = sizeof(RngBackend),
132 .instance_init = rng_backend_init,
Ladi Prosek9f14b0a2016-03-03 09:37:17 +0100133 .instance_finalize = rng_backend_finalize,
Anthony Liguoria9b7b2a2012-06-25 10:03:47 -0500134 .class_size = sizeof(RngBackendClass),
Igor Mammedov57d3e1b2014-01-16 17:34:39 +0100135 .class_init = rng_backend_class_init,
Anthony Liguoria9b7b2a2012-06-25 10:03:47 -0500136 .abstract = true,
Igor Mammedov269e09f2014-01-16 17:34:38 +0100137 .interfaces = (InterfaceInfo[]) {
138 { TYPE_USER_CREATABLE },
139 { }
140 }
Anthony Liguoria9b7b2a2012-06-25 10:03:47 -0500141};
142
143static void register_types(void)
144{
145 type_register_static(&rng_backend_info);
146}
147
148type_init(register_types);