rng.c 3.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149
  1. /*
  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. #include "qemu/osdep.h"
  13. #include "sysemu/rng.h"
  14. #include "qapi/error.h"
  15. #include "qapi/qmp/qerror.h"
  16. #include "qemu/module.h"
  17. #include "qom/object_interfaces.h"
  18. void rng_backend_request_entropy(RngBackend *s, size_t size,
  19. EntropyReceiveFunc *receive_entropy,
  20. void *opaque)
  21. {
  22. RngBackendClass *k = RNG_BACKEND_GET_CLASS(s);
  23. RngRequest *req;
  24. if (k->request_entropy) {
  25. req = g_malloc(sizeof(*req));
  26. req->offset = 0;
  27. req->size = size;
  28. req->receive_entropy = receive_entropy;
  29. req->opaque = opaque;
  30. req->data = g_malloc(req->size);
  31. k->request_entropy(s, req);
  32. QSIMPLEQ_INSERT_TAIL(&s->requests, req, next);
  33. }
  34. }
  35. static bool rng_backend_prop_get_opened(Object *obj, Error **errp)
  36. {
  37. RngBackend *s = RNG_BACKEND(obj);
  38. return s->opened;
  39. }
  40. static void rng_backend_complete(UserCreatable *uc, Error **errp)
  41. {
  42. object_property_set_bool(OBJECT(uc), true, "opened", errp);
  43. }
  44. static void rng_backend_prop_set_opened(Object *obj, bool value, Error **errp)
  45. {
  46. RngBackend *s = RNG_BACKEND(obj);
  47. RngBackendClass *k = RNG_BACKEND_GET_CLASS(s);
  48. Error *local_err = NULL;
  49. if (value == s->opened) {
  50. return;
  51. }
  52. if (!value && s->opened) {
  53. error_setg(errp, QERR_PERMISSION_DENIED);
  54. return;
  55. }
  56. if (k->opened) {
  57. k->opened(s, &local_err);
  58. if (local_err) {
  59. error_propagate(errp, local_err);
  60. return;
  61. }
  62. }
  63. s->opened = true;
  64. }
  65. static void rng_backend_free_request(RngRequest *req)
  66. {
  67. g_free(req->data);
  68. g_free(req);
  69. }
  70. static void rng_backend_free_requests(RngBackend *s)
  71. {
  72. RngRequest *req, *next;
  73. QSIMPLEQ_FOREACH_SAFE(req, &s->requests, next, next) {
  74. rng_backend_free_request(req);
  75. }
  76. QSIMPLEQ_INIT(&s->requests);
  77. }
  78. void rng_backend_finalize_request(RngBackend *s, RngRequest *req)
  79. {
  80. QSIMPLEQ_REMOVE(&s->requests, req, RngRequest, next);
  81. rng_backend_free_request(req);
  82. }
  83. static void rng_backend_init(Object *obj)
  84. {
  85. RngBackend *s = RNG_BACKEND(obj);
  86. QSIMPLEQ_INIT(&s->requests);
  87. object_property_add_bool(obj, "opened",
  88. rng_backend_prop_get_opened,
  89. rng_backend_prop_set_opened,
  90. NULL);
  91. }
  92. static void rng_backend_finalize(Object *obj)
  93. {
  94. RngBackend *s = RNG_BACKEND(obj);
  95. rng_backend_free_requests(s);
  96. }
  97. static void rng_backend_class_init(ObjectClass *oc, void *data)
  98. {
  99. UserCreatableClass *ucc = USER_CREATABLE_CLASS(oc);
  100. ucc->complete = rng_backend_complete;
  101. }
  102. static const TypeInfo rng_backend_info = {
  103. .name = TYPE_RNG_BACKEND,
  104. .parent = TYPE_OBJECT,
  105. .instance_size = sizeof(RngBackend),
  106. .instance_init = rng_backend_init,
  107. .instance_finalize = rng_backend_finalize,
  108. .class_size = sizeof(RngBackendClass),
  109. .class_init = rng_backend_class_init,
  110. .abstract = true,
  111. .interfaces = (InterfaceInfo[]) {
  112. { TYPE_USER_CREATABLE },
  113. { }
  114. }
  115. };
  116. static void register_types(void)
  117. {
  118. type_register_static(&rng_backend_info);
  119. }
  120. type_init(register_types);