blob: c2ad8cfd426d750f867fb1937d773fa5eecdcff3 [file] [log] [blame]
Jacob Erlbeck9732cb42015-10-01 20:43:53 +02001#pragma once
2
Jacob Erlbeckfc9533d2015-10-29 00:55:58 +01003/*! \defgroup osmo_stat_item Statistics value item
Jacob Erlbeck9732cb42015-10-01 20:43:53 +02004 * @{
5 */
6
7/*! \file stat_item.h */
8
9#include <stdint.h>
10
11#include <osmocom/core/linuxlist.h>
12
Jacob Erlbeckfc9533d2015-10-29 00:55:58 +010013struct osmo_stat_item_desc;
Jacob Erlbeck9732cb42015-10-01 20:43:53 +020014
Jacob Erlbeckb27b3522015-10-12 18:47:09 +020015#define STAT_ITEM_NOVALUE_ID 0
16
Jacob Erlbeckfc9533d2015-10-29 00:55:58 +010017struct osmo_stat_item_value {
Jacob Erlbeckb27b3522015-10-12 18:47:09 +020018 int32_t id;
19 int32_t value;
20};
21
Jacob Erlbeck9732cb42015-10-01 20:43:53 +020022/*! \brief data we keep for each actual value */
Jacob Erlbeckfc9533d2015-10-29 00:55:58 +010023struct osmo_stat_item {
24 const struct osmo_stat_item_desc *desc;
Jacob Erlbeck9732cb42015-10-01 20:43:53 +020025 /*! \brief the index of the freshest value */
26 int32_t last_value_index;
27 /*! \brief offset to the freshest value in the value fifo */
28 int16_t last_offs;
29 /*! \brief value fifo */
Jacob Erlbeckfc9533d2015-10-29 00:55:58 +010030 struct osmo_stat_item_value values[0];
Jacob Erlbeck9732cb42015-10-01 20:43:53 +020031};
32
33/*! \brief statistics value description */
Jacob Erlbeckfc9533d2015-10-29 00:55:58 +010034struct osmo_stat_item_desc {
Jacob Erlbeck9732cb42015-10-01 20:43:53 +020035 const char *name; /*!< \brief name of the item */
36 const char *description;/*!< \brief description of the item */
37 const char *unit; /*!< \brief unit of a value */
38 unsigned int num_values;/*!< \brief number of values to store */
39 int32_t default_value;
40};
41
42/*! \brief description of a statistics value group */
Jacob Erlbeckfc9533d2015-10-29 00:55:58 +010043struct osmo_stat_item_group_desc {
Jacob Erlbeck9732cb42015-10-01 20:43:53 +020044 /*! \brief The prefix to the name of all values in this group */
45 const char *group_name_prefix;
46 /*! \brief The human-readable description of the group */
47 const char *group_description;
Jacob Erlbeckbc9d9ac2015-11-02 14:49:35 +010048 /*! \brief The class to which this group belongs */
49 int class_id;
Jacob Erlbeck9732cb42015-10-01 20:43:53 +020050 /*! \brief The number of values in this group */
51 const unsigned int num_items;
52 /*! \brief Pointer to array of value names */
Jacob Erlbeckfc9533d2015-10-29 00:55:58 +010053 const struct osmo_stat_item_desc *item_desc;
Jacob Erlbeck9732cb42015-10-01 20:43:53 +020054};
55
56/*! \brief One instance of a counter group class */
Jacob Erlbeckfc9533d2015-10-29 00:55:58 +010057struct osmo_stat_item_group {
Jacob Erlbeck9732cb42015-10-01 20:43:53 +020058 /*! \brief Linked list of all value groups in the system */
59 struct llist_head list;
60 /*! \brief Pointer to the counter group class */
Jacob Erlbeckfc9533d2015-10-29 00:55:58 +010061 const struct osmo_stat_item_group_desc *desc;
Jacob Erlbeck9732cb42015-10-01 20:43:53 +020062 /*! \brief The index of this value group within its class */
63 unsigned int idx;
64 /*! \brief Actual counter structures below */
Jacob Erlbeckfc9533d2015-10-29 00:55:58 +010065 struct osmo_stat_item *items[0];
Jacob Erlbeck9732cb42015-10-01 20:43:53 +020066};
67
Jacob Erlbeckfc9533d2015-10-29 00:55:58 +010068struct osmo_stat_item_group *osmo_stat_item_group_alloc(
Jacob Erlbeck9732cb42015-10-01 20:43:53 +020069 void *ctx,
Jacob Erlbeckfc9533d2015-10-29 00:55:58 +010070 const struct osmo_stat_item_group_desc *desc,
Jacob Erlbeck9732cb42015-10-01 20:43:53 +020071 unsigned int idx);
72
Jacob Erlbeckfc9533d2015-10-29 00:55:58 +010073void osmo_stat_item_group_free(struct osmo_stat_item_group *statg);
Jacob Erlbeck9732cb42015-10-01 20:43:53 +020074
Jacob Erlbeckfc9533d2015-10-29 00:55:58 +010075void osmo_stat_item_set(struct osmo_stat_item *item, int32_t value);
Jacob Erlbeck9732cb42015-10-01 20:43:53 +020076
Jacob Erlbeckfc9533d2015-10-29 00:55:58 +010077int osmo_stat_item_init(void *tall_ctx);
Jacob Erlbeck9732cb42015-10-01 20:43:53 +020078
Jacob Erlbeckfc9533d2015-10-29 00:55:58 +010079struct osmo_stat_item_group *osmo_stat_item_get_group_by_name_idx(
Jacob Erlbeck9732cb42015-10-01 20:43:53 +020080 const char *name, const unsigned int idx);
81
Jacob Erlbeckfc9533d2015-10-29 00:55:58 +010082const struct osmo_stat_item *osmo_stat_item_get_by_name(
83 const struct osmo_stat_item_group *statg, const char *name);
Jacob Erlbeck9732cb42015-10-01 20:43:53 +020084
Jacob Erlbeckfc9533d2015-10-29 00:55:58 +010085/*! \brief Retrieve the next value from the osmo_stat_item object.
Jacob Erlbeck9732cb42015-10-01 20:43:53 +020086 * If a new value has been set, it is returned. The idx is used to decide
87 * which value to return.
88 * On success, *idx is updated to refer to the next unread value. If
89 * values have been missed due to FIFO overflow, *idx is incremented by
90 * (1 + num_lost).
Jacob Erlbeckfc9533d2015-10-29 00:55:58 +010091 * This way, the osmo_stat_item object can be kept stateless from the reader's
Jacob Erlbeck9732cb42015-10-01 20:43:53 +020092 * perspective and therefore be used by several backends simultaneously.
93 *
Jacob Erlbeckfc9533d2015-10-29 00:55:58 +010094 * \param val the osmo_stat_item object
Jacob Erlbeck9732cb42015-10-01 20:43:53 +020095 * \param idx identifies the next value to be read
96 * \param value a pointer to store the value
97 * \returns the increment of the index (0: no value has been read,
98 * 1: one value has been taken,
99 * (1+n): n values have been skipped, one has been taken)
100 */
Jacob Erlbeckfc9533d2015-10-29 00:55:58 +0100101int osmo_stat_item_get_next(const struct osmo_stat_item *item, int32_t *idx, int32_t *value);
Jacob Erlbeck9732cb42015-10-01 20:43:53 +0200102
103/*! \brief Get the last (freshest) value */
Jacob Erlbeckfc9533d2015-10-29 00:55:58 +0100104static int32_t osmo_stat_item_get_last(const struct osmo_stat_item *item);
Jacob Erlbeck9732cb42015-10-01 20:43:53 +0200105
106/*! \brief Skip all values of the item and update idx accordingly */
Jacob Erlbeckfc9533d2015-10-29 00:55:58 +0100107int osmo_stat_item_discard(const struct osmo_stat_item *item, int32_t *idx);
Jacob Erlbeck9732cb42015-10-01 20:43:53 +0200108
Jacob Erlbeckb27b3522015-10-12 18:47:09 +0200109/*! \brief Skip all values of all items and update idx accordingly */
Jacob Erlbeckfc9533d2015-10-29 00:55:58 +0100110int osmo_stat_item_discard_all(int32_t *idx);
Jacob Erlbeckb27b3522015-10-12 18:47:09 +0200111
Jacob Erlbeckfc9533d2015-10-29 00:55:58 +0100112typedef int (*osmo_stat_item_handler_t)(
113 struct osmo_stat_item_group *, struct osmo_stat_item *, void *);
Jacob Erlbeckc6a71082015-10-19 14:04:38 +0200114
Jacob Erlbeckfc9533d2015-10-29 00:55:58 +0100115typedef int (*osmo_stat_item_group_handler_t)(struct osmo_stat_item_group *, void *);
Jacob Erlbeckc6a71082015-10-19 14:04:38 +0200116
117/*! \brief Iteate over all items
118 * \param[in] handle_item Call-back function, aborts if rc < 0
119 * \param[in] data Private data handed through to \a handle_item
120 */
Jacob Erlbeckfc9533d2015-10-29 00:55:58 +0100121int osmo_stat_item_for_each_item(struct osmo_stat_item_group *statg,
122 osmo_stat_item_handler_t handle_item, void *data);
Jacob Erlbeckc6a71082015-10-19 14:04:38 +0200123
Jacob Erlbeckfc9533d2015-10-29 00:55:58 +0100124int osmo_stat_item_for_each_group(osmo_stat_item_group_handler_t handle_group, void *data);
Jacob Erlbeckc6a71082015-10-19 14:04:38 +0200125
Jacob Erlbeckfc9533d2015-10-29 00:55:58 +0100126static inline int32_t osmo_stat_item_get_last(const struct osmo_stat_item *item)
Jacob Erlbeck9732cb42015-10-01 20:43:53 +0200127{
Jacob Erlbeckb27b3522015-10-12 18:47:09 +0200128 return item->values[item->last_offs].value;
Jacob Erlbeck9732cb42015-10-01 20:43:53 +0200129}
130/*! @} */