|  | /* | 
|  | * QEMU event-loop base | 
|  | * | 
|  | * Copyright (C) 2022 Red Hat Inc | 
|  | * | 
|  | * Authors: | 
|  | *  Stefan Hajnoczi <stefanha@redhat.com> | 
|  | *  Nicolas Saenz Julienne <nsaenzju@redhat.com> | 
|  | * | 
|  | * This work is licensed under the terms of the GNU GPL, version 2 or later. | 
|  | * See the COPYING file in the top-level directory. | 
|  | */ | 
|  |  | 
|  | #include "qemu/osdep.h" | 
|  | #include "qom/object_interfaces.h" | 
|  | #include "qapi/error.h" | 
|  | #include "block/thread-pool.h" | 
|  | #include "sysemu/event-loop-base.h" | 
|  |  | 
|  | typedef struct { | 
|  | const char *name; | 
|  | ptrdiff_t offset; /* field's byte offset in EventLoopBase struct */ | 
|  | } EventLoopBaseParamInfo; | 
|  |  | 
|  | static void event_loop_base_instance_init(Object *obj) | 
|  | { | 
|  | EventLoopBase *base = EVENT_LOOP_BASE(obj); | 
|  |  | 
|  | base->thread_pool_max = THREAD_POOL_MAX_THREADS_DEFAULT; | 
|  | } | 
|  |  | 
|  | static EventLoopBaseParamInfo aio_max_batch_info = { | 
|  | "aio-max-batch", offsetof(EventLoopBase, aio_max_batch), | 
|  | }; | 
|  | static EventLoopBaseParamInfo thread_pool_min_info = { | 
|  | "thread-pool-min", offsetof(EventLoopBase, thread_pool_min), | 
|  | }; | 
|  | static EventLoopBaseParamInfo thread_pool_max_info = { | 
|  | "thread-pool-max", offsetof(EventLoopBase, thread_pool_max), | 
|  | }; | 
|  |  | 
|  | static void event_loop_base_get_param(Object *obj, Visitor *v, | 
|  | const char *name, void *opaque, Error **errp) | 
|  | { | 
|  | EventLoopBase *event_loop_base = EVENT_LOOP_BASE(obj); | 
|  | EventLoopBaseParamInfo *info = opaque; | 
|  | int64_t *field = (void *)event_loop_base + info->offset; | 
|  |  | 
|  | visit_type_int64(v, name, field, errp); | 
|  | } | 
|  |  | 
|  | static void event_loop_base_set_param(Object *obj, Visitor *v, | 
|  | const char *name, void *opaque, Error **errp) | 
|  | { | 
|  | EventLoopBaseClass *bc = EVENT_LOOP_BASE_GET_CLASS(obj); | 
|  | EventLoopBase *base = EVENT_LOOP_BASE(obj); | 
|  | EventLoopBaseParamInfo *info = opaque; | 
|  | int64_t *field = (void *)base + info->offset; | 
|  | int64_t value; | 
|  |  | 
|  | if (!visit_type_int64(v, name, &value, errp)) { | 
|  | return; | 
|  | } | 
|  |  | 
|  | if (value < 0) { | 
|  | error_setg(errp, "%s value must be in range [0, %" PRId64 "]", | 
|  | info->name, INT64_MAX); | 
|  | return; | 
|  | } | 
|  |  | 
|  | *field = value; | 
|  |  | 
|  | if (bc->update_params) { | 
|  | bc->update_params(base, errp); | 
|  | } | 
|  |  | 
|  | return; | 
|  | } | 
|  |  | 
|  | static void event_loop_base_complete(UserCreatable *uc, Error **errp) | 
|  | { | 
|  | EventLoopBaseClass *bc = EVENT_LOOP_BASE_GET_CLASS(uc); | 
|  | EventLoopBase *base = EVENT_LOOP_BASE(uc); | 
|  |  | 
|  | if (bc->init) { | 
|  | bc->init(base, errp); | 
|  | } | 
|  | } | 
|  |  | 
|  | static bool event_loop_base_can_be_deleted(UserCreatable *uc) | 
|  | { | 
|  | EventLoopBaseClass *bc = EVENT_LOOP_BASE_GET_CLASS(uc); | 
|  | EventLoopBase *backend = EVENT_LOOP_BASE(uc); | 
|  |  | 
|  | if (bc->can_be_deleted) { | 
|  | return bc->can_be_deleted(backend); | 
|  | } | 
|  |  | 
|  | return true; | 
|  | } | 
|  |  | 
|  | static void event_loop_base_class_init(ObjectClass *klass, void *class_data) | 
|  | { | 
|  | UserCreatableClass *ucc = USER_CREATABLE_CLASS(klass); | 
|  | ucc->complete = event_loop_base_complete; | 
|  | ucc->can_be_deleted = event_loop_base_can_be_deleted; | 
|  |  | 
|  | object_class_property_add(klass, "aio-max-batch", "int", | 
|  | event_loop_base_get_param, | 
|  | event_loop_base_set_param, | 
|  | NULL, &aio_max_batch_info); | 
|  | object_class_property_add(klass, "thread-pool-min", "int", | 
|  | event_loop_base_get_param, | 
|  | event_loop_base_set_param, | 
|  | NULL, &thread_pool_min_info); | 
|  | object_class_property_add(klass, "thread-pool-max", "int", | 
|  | event_loop_base_get_param, | 
|  | event_loop_base_set_param, | 
|  | NULL, &thread_pool_max_info); | 
|  | } | 
|  |  | 
|  | static const TypeInfo event_loop_base_info = { | 
|  | .name = TYPE_EVENT_LOOP_BASE, | 
|  | .parent = TYPE_OBJECT, | 
|  | .instance_size = sizeof(EventLoopBase), | 
|  | .instance_init = event_loop_base_instance_init, | 
|  | .class_size = sizeof(EventLoopBaseClass), | 
|  | .class_init = event_loop_base_class_init, | 
|  | .abstract = true, | 
|  | .interfaces = (InterfaceInfo[]) { | 
|  | { TYPE_USER_CREATABLE }, | 
|  | { } | 
|  | } | 
|  | }; | 
|  |  | 
|  | static void register_types(void) | 
|  | { | 
|  | type_register_static(&event_loop_base_info); | 
|  | } | 
|  | type_init(register_types); |