f-stack/dpdk/lib/telemetry/telemetry_data.c

277 lines
6.4 KiB
C
Raw Normal View History

2021-02-05 08:48:47 +00:00
/* SPDX-License-Identifier: BSD-3-Clause
* Copyright(c) 2020 Intel Corporation
*/
2025-01-10 11:50:43 +00:00
#include <ctype.h>
2023-09-13 12:21:49 +00:00
#include <errno.h>
#include <stdlib.h>
2025-01-10 11:50:43 +00:00
#include <inttypes.h>
2023-09-13 12:21:49 +00:00
2021-02-05 08:48:47 +00:00
#undef RTE_USE_LIBBSD
2023-09-13 12:21:49 +00:00
#include <stdbool.h>
2021-02-05 08:48:47 +00:00
#include <rte_string_fns.h>
#include "telemetry_data.h"
2025-01-10 11:50:43 +00:00
#define RTE_TEL_UINT_HEX_STR_BUF_LEN 64
2021-02-05 08:48:47 +00:00
int
rte_tel_data_start_array(struct rte_tel_data *d, enum rte_tel_value_type type)
{
enum tel_container_types array_types[] = {
2025-01-10 11:50:43 +00:00
[RTE_TEL_STRING_VAL] = TEL_ARRAY_STRING,
[RTE_TEL_INT_VAL] = TEL_ARRAY_INT,
[RTE_TEL_UINT_VAL] = TEL_ARRAY_UINT,
[RTE_TEL_CONTAINER] = TEL_ARRAY_CONTAINER,
2021-02-05 08:48:47 +00:00
};
d->type = array_types[type];
d->data_len = 0;
return 0;
}
int
rte_tel_data_start_dict(struct rte_tel_data *d)
{
2025-01-10 11:50:43 +00:00
d->type = TEL_DICT;
2021-02-05 08:48:47 +00:00
d->data_len = 0;
return 0;
}
int
rte_tel_data_string(struct rte_tel_data *d, const char *str)
{
2025-01-10 11:50:43 +00:00
d->type = TEL_STRING;
2021-02-05 08:48:47 +00:00
d->data_len = strlcpy(d->data.str, str, sizeof(d->data.str));
if (d->data_len >= RTE_TEL_MAX_SINGLE_STRING_LEN) {
d->data_len = RTE_TEL_MAX_SINGLE_STRING_LEN - 1;
return E2BIG; /* not necessarily and error, just truncation */
}
return 0;
}
int
rte_tel_data_add_array_string(struct rte_tel_data *d, const char *str)
{
2025-01-10 11:50:43 +00:00
if (d->type != TEL_ARRAY_STRING)
2021-02-05 08:48:47 +00:00
return -EINVAL;
if (d->data_len >= RTE_TEL_MAX_ARRAY_ENTRIES)
return -ENOSPC;
const size_t bytes = strlcpy(d->data.array[d->data_len++].sval,
str, RTE_TEL_MAX_STRING_LEN);
return bytes < RTE_TEL_MAX_STRING_LEN ? 0 : E2BIG;
}
int
2025-01-10 11:50:43 +00:00
rte_tel_data_add_array_int(struct rte_tel_data *d, int64_t x)
2021-02-05 08:48:47 +00:00
{
2025-01-10 11:50:43 +00:00
if (d->type != TEL_ARRAY_INT)
2021-02-05 08:48:47 +00:00
return -EINVAL;
if (d->data_len >= RTE_TEL_MAX_ARRAY_ENTRIES)
return -ENOSPC;
d->data.array[d->data_len++].ival = x;
return 0;
}
int
2025-01-10 11:50:43 +00:00
rte_tel_data_add_array_uint(struct rte_tel_data *d, uint64_t x)
2021-02-05 08:48:47 +00:00
{
2025-01-10 11:50:43 +00:00
if (d->type != TEL_ARRAY_UINT)
2021-02-05 08:48:47 +00:00
return -EINVAL;
if (d->data_len >= RTE_TEL_MAX_ARRAY_ENTRIES)
return -ENOSPC;
2025-01-10 11:50:43 +00:00
d->data.array[d->data_len++].uval = x;
2021-02-05 08:48:47 +00:00
return 0;
}
2025-01-10 11:50:43 +00:00
int
rte_tel_data_add_array_u64(struct rte_tel_data *d, uint64_t x)
{
return rte_tel_data_add_array_uint(d, x);
}
2021-02-05 08:48:47 +00:00
int
rte_tel_data_add_array_container(struct rte_tel_data *d,
struct rte_tel_data *val, int keep)
{
2025-01-10 11:50:43 +00:00
if (d->type != TEL_ARRAY_CONTAINER ||
(val->type != TEL_ARRAY_UINT
&& val->type != TEL_ARRAY_INT
&& val->type != TEL_ARRAY_STRING))
2021-02-05 08:48:47 +00:00
return -EINVAL;
if (d->data_len >= RTE_TEL_MAX_ARRAY_ENTRIES)
return -ENOSPC;
d->data.array[d->data_len].container.data = val;
d->data.array[d->data_len++].container.keep = !!keep;
return 0;
}
2025-01-10 11:50:43 +00:00
static int
rte_tel_uint_to_hex_encoded_str(char *buf, size_t buf_len, uint64_t val,
uint8_t display_bitwidth)
{
int spec_hex_width = (display_bitwidth + 3) / 4;
int len;
if (display_bitwidth != 0)
len = snprintf(buf, buf_len, "0x%0*" PRIx64, spec_hex_width, val);
else
len = snprintf(buf, buf_len, "0x%" PRIx64, val);
return len < (int)buf_len ? 0 : -EINVAL;
}
int
rte_tel_data_add_array_uint_hex(struct rte_tel_data *d, uint64_t val,
uint8_t display_bitwidth)
{
char hex_str[RTE_TEL_UINT_HEX_STR_BUF_LEN];
int ret;
ret = rte_tel_uint_to_hex_encoded_str(hex_str,
RTE_TEL_UINT_HEX_STR_BUF_LEN, val, display_bitwidth);
if (ret != 0)
return ret;
return rte_tel_data_add_array_string(d, hex_str);
}
2023-09-13 12:21:49 +00:00
static bool
valid_name(const char *name)
{
2025-01-10 11:50:43 +00:00
/* non-alphanumeric characters allowed in names */
static const char allowed[128] = { ['_'] = 1, ['/'] = 1 };
for (; *name != '\0'; name++) {
if (isalnum(*name))
continue;
2023-09-13 12:21:49 +00:00
if ((size_t)*name >= RTE_DIM(allowed) || allowed[(int)*name] == 0)
return false;
}
return true;
}
2021-02-05 08:48:47 +00:00
int
rte_tel_data_add_dict_string(struct rte_tel_data *d, const char *name,
const char *val)
{
struct tel_dict_entry *e = &d->data.dict[d->data_len];
size_t nbytes, vbytes;
2025-01-10 11:50:43 +00:00
if (d->type != TEL_DICT)
2021-02-05 08:48:47 +00:00
return -EINVAL;
if (d->data_len >= RTE_TEL_MAX_DICT_ENTRIES)
return -ENOSPC;
2023-09-13 12:21:49 +00:00
if (!valid_name(name))
return -EINVAL;
2021-02-05 08:48:47 +00:00
d->data_len++;
e->type = RTE_TEL_STRING_VAL;
vbytes = strlcpy(e->value.sval, val, RTE_TEL_MAX_STRING_LEN);
nbytes = strlcpy(e->name, name, RTE_TEL_MAX_STRING_LEN);
if (vbytes >= RTE_TEL_MAX_STRING_LEN ||
nbytes >= RTE_TEL_MAX_STRING_LEN)
return E2BIG;
return 0;
}
int
2025-01-10 11:50:43 +00:00
rte_tel_data_add_dict_int(struct rte_tel_data *d, const char *name, int64_t val)
2021-02-05 08:48:47 +00:00
{
struct tel_dict_entry *e = &d->data.dict[d->data_len];
2025-01-10 11:50:43 +00:00
if (d->type != TEL_DICT)
2021-02-05 08:48:47 +00:00
return -EINVAL;
if (d->data_len >= RTE_TEL_MAX_DICT_ENTRIES)
return -ENOSPC;
2023-09-13 12:21:49 +00:00
if (!valid_name(name))
return -EINVAL;
2021-02-05 08:48:47 +00:00
d->data_len++;
e->type = RTE_TEL_INT_VAL;
e->value.ival = val;
const size_t bytes = strlcpy(e->name, name, RTE_TEL_MAX_STRING_LEN);
return bytes < RTE_TEL_MAX_STRING_LEN ? 0 : E2BIG;
}
int
2025-01-10 11:50:43 +00:00
rte_tel_data_add_dict_uint(struct rte_tel_data *d,
2021-02-05 08:48:47 +00:00
const char *name, uint64_t val)
{
struct tel_dict_entry *e = &d->data.dict[d->data_len];
2025-01-10 11:50:43 +00:00
if (d->type != TEL_DICT)
2021-02-05 08:48:47 +00:00
return -EINVAL;
if (d->data_len >= RTE_TEL_MAX_DICT_ENTRIES)
return -ENOSPC;
2023-09-13 12:21:49 +00:00
if (!valid_name(name))
return -EINVAL;
2021-02-05 08:48:47 +00:00
d->data_len++;
2025-01-10 11:50:43 +00:00
e->type = RTE_TEL_UINT_VAL;
e->value.uval = val;
2021-02-05 08:48:47 +00:00
const size_t bytes = strlcpy(e->name, name, RTE_TEL_MAX_STRING_LEN);
return bytes < RTE_TEL_MAX_STRING_LEN ? 0 : E2BIG;
}
2025-01-10 11:50:43 +00:00
int
rte_tel_data_add_dict_u64(struct rte_tel_data *d, const char *name, uint64_t val)
{
return rte_tel_data_add_dict_uint(d, name, val);
}
2021-02-05 08:48:47 +00:00
int
rte_tel_data_add_dict_container(struct rte_tel_data *d, const char *name,
struct rte_tel_data *val, int keep)
{
struct tel_dict_entry *e = &d->data.dict[d->data_len];
2025-01-10 11:50:43 +00:00
if (d->type != TEL_DICT || (val->type != TEL_ARRAY_UINT
&& val->type != TEL_ARRAY_INT
&& val->type != TEL_ARRAY_STRING
&& val->type != TEL_DICT))
2021-02-05 08:48:47 +00:00
return -EINVAL;
if (d->data_len >= RTE_TEL_MAX_DICT_ENTRIES)
return -ENOSPC;
2023-09-13 12:21:49 +00:00
if (!valid_name(name))
return -EINVAL;
2021-02-05 08:48:47 +00:00
d->data_len++;
e->type = RTE_TEL_CONTAINER;
e->value.container.data = val;
e->value.container.keep = !!keep;
const size_t bytes = strlcpy(e->name, name, RTE_TEL_MAX_STRING_LEN);
return bytes < RTE_TEL_MAX_STRING_LEN ? 0 : E2BIG;
}
2025-01-10 11:50:43 +00:00
int
rte_tel_data_add_dict_uint_hex(struct rte_tel_data *d, const char *name,
uint64_t val, uint8_t display_bitwidth)
{
char hex_str[RTE_TEL_UINT_HEX_STR_BUF_LEN];
int ret;
ret = rte_tel_uint_to_hex_encoded_str(hex_str,
RTE_TEL_UINT_HEX_STR_BUF_LEN, val, display_bitwidth);
if (ret != 0)
return ret;
return rte_tel_data_add_dict_string(d, name, hex_str);
}
2021-02-05 08:48:47 +00:00
struct rte_tel_data *
rte_tel_data_alloc(void)
{
return malloc(sizeof(struct rte_tel_data));
}
void
rte_tel_data_free(struct rte_tel_data *data)
{
free(data);
}