aboutsummaryrefslogtreecommitdiffstats
path: root/include/osmocom/core/stat_item.h
blob: 003c9e07ab3aff52998fca5d1958339c2b3886a9 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
#pragma once

/*! \defgroup stat_item Statistics value item
 *  @{
 */

/*! \file stat_item.h */

#include <stdint.h>

#include <osmocom/core/linuxlist.h>

struct stat_item_desc;

#define STAT_ITEM_NOVALUE_ID 0

struct stat_item_value {
	int32_t id;
	int32_t value;
};

/*! \brief data we keep for each actual value */
struct stat_item {
	const struct stat_item_desc *desc;
	/*! \brief the index of the freshest value */
	int32_t last_value_index;
	/*! \brief offset to the freshest value in the value fifo */
	int16_t last_offs;
	/*! \brief value fifo */
	struct stat_item_value values[0];
};

/*! \brief statistics value description */
struct stat_item_desc {
	const char *name;	/*!< \brief name of the item */
	const char *description;/*!< \brief description of the item */
	const char *unit;	/*!< \brief unit of a value */
	unsigned int num_values;/*!< \brief number of values to store */
	int32_t default_value;
};

/*! \brief description of a statistics value group */
struct stat_item_group_desc {
	/*! \brief The prefix to the name of all values in this group */
	const char *group_name_prefix;
	/*! \brief The human-readable description of the group */
	const char *group_description;
	/*! \brief The number of values in this group */
	const unsigned int num_items;
	/*! \brief Pointer to array of value names */
	const struct stat_item_desc *item_desc;
};

/*! \brief One instance of a counter group class */
struct stat_item_group {
	/*! \brief Linked list of all value groups in the system */
	struct llist_head list;
	/*! \brief Pointer to the counter group class */
	const struct stat_item_group_desc *desc;
	/*! \brief The index of this value group within its class */
	unsigned int idx;
	/*! \brief Actual counter structures below */
	struct stat_item *items[0];
};

struct stat_item_group *stat_item_group_alloc(
	void *ctx,
	const struct stat_item_group_desc *desc,
	unsigned int idx);

void stat_item_group_free(struct stat_item_group *statg);

void stat_item_set(struct stat_item *item, int32_t value);

int stat_item_init(void *tall_ctx);

struct stat_item_group *stat_item_get_group_by_name_idx(
	const char *name, const unsigned int idx);

const struct stat_item *stat_item_get_by_name(
	const struct stat_item_group *statg, const char *name);

/*! \brief Retrieve the next value from the stat_item object.
 * If a new value has been set, it is returned. The idx is used to decide
 * which value to return.
 * On success, *idx is updated to refer to the next unread value. If
 * values have been missed due to FIFO overflow, *idx is incremented by
 * (1 + num_lost).
 * This way, the stat_item object can be kept stateless from the reader's
 * perspective and therefore be used by several backends simultaneously.
 *
 * \param val	the stat_item object
 * \param idx	identifies the next value to be read
 * \param value	a pointer to store the value
 * \returns  the increment of the index (0: no value has been read,
 *           1: one value has been taken,
 *           (1+n): n values have been skipped, one has been taken)
 */
int stat_item_get_next(const struct stat_item *item, int32_t *idx, int32_t *value);

/*! \brief Get the last (freshest) value */
static int32_t stat_item_get_last(const struct stat_item *item);

/*! \brief Skip all values of the item and update idx accordingly */
int stat_item_discard(const struct stat_item *item, int32_t *idx);

/*! \brief Skip all values of all items and update idx accordingly */
int stat_item_discard_all(int32_t *idx);

static inline int32_t stat_item_get_last(const struct stat_item *item)
{
	return item->values[item->last_offs].value;
}
/*! @} */