/* * QMP commands related to stats * * This work is licensed under the terms of the GNU GPL, version 2 or * (at your option) any later version. */ #include "qemu/osdep.h" #include "system/stats.h" #include "qapi/qapi-commands-stats.h" #include "qemu/queue.h" #include "qapi/error.h" typedef struct StatsCallbacks { StatsProvider provider; StatRetrieveFunc *stats_cb; SchemaRetrieveFunc *schemas_cb; QTAILQ_ENTRY(StatsCallbacks) next; } StatsCallbacks; static QTAILQ_HEAD(, StatsCallbacks) stats_callbacks = QTAILQ_HEAD_INITIALIZER(stats_callbacks); void add_stats_callbacks(StatsProvider provider, StatRetrieveFunc *stats_fn, SchemaRetrieveFunc *schemas_fn) { StatsCallbacks *entry = g_new(StatsCallbacks, 1); entry->provider = provider; entry->stats_cb = stats_fn; entry->schemas_cb = schemas_fn; QTAILQ_INSERT_TAIL(&stats_callbacks, entry, next); } static bool invoke_stats_cb(StatsCallbacks *entry, StatsResultList **stats_results, StatsFilter *filter, StatsRequest *request, Error **errp) { ERRP_GUARD(); strList *targets = NULL; strList *names = NULL; if (request) { if (request->provider != entry->provider) { return true; } if (request->has_names && !request->names) { return true; } names = request->has_names ? request->names : NULL; } switch (filter->target) { case STATS_TARGET_VM: break; case STATS_TARGET_VCPU: if (filter->u.vcpu.has_vcpus) { if (!filter->u.vcpu.vcpus) { /* No targets allowed? Return no statistics. */ return true; } targets = filter->u.vcpu.vcpus; } break; case STATS_TARGET_CRYPTODEV: break; default: abort(); } entry->stats_cb(stats_results, filter->target, names, targets, errp); if (*errp) { qapi_free_StatsResultList(*stats_results); *stats_results = NULL; return false; } return true; } StatsResultList *qmp_query_stats(StatsFilter *filter, Error **errp) { StatsResultList *stats_results = NULL; StatsCallbacks *entry; StatsRequestList *request; QTAILQ_FOREACH(entry, &stats_callbacks, next) { if (filter->has_providers) { for (request = filter->providers; request; request = request->next) { if (!invoke_stats_cb(entry, &stats_results, filter, request->value, errp)) { break; } } } else { if (!invoke_stats_cb(entry, &stats_results, filter, NULL, errp)) { break; } } } return stats_results; } StatsSchemaList *qmp_query_stats_schemas(bool has_provider, StatsProvider provider, Error **errp) { ERRP_GUARD(); StatsSchemaList *stats_results = NULL; StatsCallbacks *entry; QTAILQ_FOREACH(entry, &stats_callbacks, next) { if (!has_provider || provider == entry->provider) { entry->schemas_cb(&stats_results, errp); if (*errp) { qapi_free_StatsSchemaList(stats_results); return NULL; } } } return stats_results; } void add_stats_entry(StatsResultList **stats_results, StatsProvider provider, const char *qom_path, StatsList *stats_list) { StatsResult *entry = g_new0(StatsResult, 1); entry->provider = provider; entry->qom_path = g_strdup(qom_path); entry->stats = stats_list; QAPI_LIST_PREPEND(*stats_results, entry); } void add_stats_schema(StatsSchemaList **schema_results, StatsProvider provider, StatsTarget target, StatsSchemaValueList *stats_list) { StatsSchema *entry = g_new0(StatsSchema, 1); entry->provider = provider; entry->target = target; entry->stats = stats_list; QAPI_LIST_PREPEND(*schema_results, entry); } bool apply_str_list_filter(const char *string, strList *list) { strList *str_list = NULL; if (!list) { return true; } for (str_list = list; str_list; str_list = str_list->next) { if (g_str_equal(string, str_list->value)) { return true; } } return false; }