238 lines
6.5 KiB
C
238 lines
6.5 KiB
C
/* SPDX-License-Identifier: GPL-2.0 */
|
|
#ifndef __TASK_LOCAL_DATA_BPF_H
|
|
#define __TASK_LOCAL_DATA_BPF_H
|
|
|
|
/*
|
|
* Task local data is a library that facilitates sharing per-task data
|
|
* between user space and bpf programs.
|
|
*
|
|
*
|
|
* USAGE
|
|
*
|
|
* A TLD, an entry of data in task local data, first needs to be created by the
|
|
* user space. This is done by calling user space API, TLD_DEFINE_KEY() or
|
|
* tld_create_key(), with the name of the TLD and the size.
|
|
*
|
|
* TLD_DEFINE_KEY(prio, "priority", sizeof(int));
|
|
*
|
|
* or
|
|
*
|
|
* void func_call(...) {
|
|
* tld_key_t prio, in_cs;
|
|
*
|
|
* prio = tld_create_key("priority", sizeof(int));
|
|
* in_cs = tld_create_key("in_critical_section", sizeof(bool));
|
|
* ...
|
|
*
|
|
* A key associated with the TLD, which has an opaque type tld_key_t, will be
|
|
* initialized or returned. It can be used to get a pointer to the TLD in the
|
|
* user space by calling tld_get_data().
|
|
*
|
|
* In a bpf program, tld_object_init() first needs to be called to initialized a
|
|
* tld_object on the stack. Then, TLDs can be accessed by calling tld_get_data().
|
|
* The API will try to fetch the key by the name and use it to locate the data.
|
|
* A pointer to the TLD will be returned. It also caches the key in a task local
|
|
* storage map, tld_key_map, whose value type, struct tld_keys, must be defined
|
|
* by the developer.
|
|
*
|
|
* struct tld_keys {
|
|
* tld_key_t prio;
|
|
* tld_key_t in_cs;
|
|
* };
|
|
*
|
|
* SEC("struct_ops")
|
|
* void prog(struct task_struct task, ...)
|
|
* {
|
|
* struct tld_object tld_obj;
|
|
* int err, *p;
|
|
*
|
|
* err = tld_object_init(task, &tld_obj);
|
|
* if (err)
|
|
* return;
|
|
*
|
|
* p = tld_get_data(&tld_obj, prio, "priority", sizeof(int));
|
|
* if (p)
|
|
* // do something depending on *p
|
|
*/
|
|
#include <errno.h>
|
|
#include <bpf/bpf_helpers.h>
|
|
|
|
#define TLD_ROUND_MASK(x, y) ((__typeof__(x))((y) - 1))
|
|
#define TLD_ROUND_UP(x, y) ((((x) - 1) | TLD_ROUND_MASK(x, y)) + 1)
|
|
|
|
#define TLD_MAX_DATA_CNT (__PAGE_SIZE / sizeof(struct tld_metadata) - 1)
|
|
|
|
#ifndef TLD_NAME_LEN
|
|
#define TLD_NAME_LEN 62
|
|
#endif
|
|
|
|
#ifndef TLD_KEY_MAP_CREATE_RETRY
|
|
#define TLD_KEY_MAP_CREATE_RETRY 10
|
|
#endif
|
|
|
|
typedef struct {
|
|
__s16 off;
|
|
} tld_key_t;
|
|
|
|
struct tld_metadata {
|
|
char name[TLD_NAME_LEN];
|
|
__u16 size;
|
|
};
|
|
|
|
struct tld_meta_u {
|
|
__u8 cnt;
|
|
__u16 size;
|
|
struct tld_metadata metadata[TLD_MAX_DATA_CNT];
|
|
};
|
|
|
|
struct tld_data_u {
|
|
__u64 start; /* offset of tld_data_u->data in a page */
|
|
char data[__PAGE_SIZE - sizeof(__u64)];
|
|
};
|
|
|
|
struct tld_map_value {
|
|
struct tld_data_u __uptr *data;
|
|
struct tld_meta_u __uptr *meta;
|
|
};
|
|
|
|
typedef struct tld_uptr_dummy {
|
|
struct tld_data_u data[0];
|
|
struct tld_meta_u meta[0];
|
|
} *tld_uptr_dummy_t;
|
|
|
|
struct tld_object {
|
|
struct tld_map_value *data_map;
|
|
struct tld_keys *key_map;
|
|
/*
|
|
* Force the compiler to generate the actual definition of tld_meta_u
|
|
* and tld_data_u in BTF. Without it, tld_meta_u and u_tld_data will
|
|
* be BTF_KIND_FWD.
|
|
*/
|
|
tld_uptr_dummy_t dummy[0];
|
|
};
|
|
|
|
/*
|
|
* Map value of tld_key_map for caching keys. Must be defined by the developer.
|
|
* Members should be tld_key_t and passed to the 3rd argument of tld_fetch_key().
|
|
*/
|
|
struct tld_keys;
|
|
|
|
struct {
|
|
__uint(type, BPF_MAP_TYPE_TASK_STORAGE);
|
|
__uint(map_flags, BPF_F_NO_PREALLOC);
|
|
__type(key, int);
|
|
__type(value, struct tld_map_value);
|
|
} tld_data_map SEC(".maps");
|
|
|
|
struct {
|
|
__uint(type, BPF_MAP_TYPE_TASK_STORAGE);
|
|
__uint(map_flags, BPF_F_NO_PREALLOC);
|
|
__type(key, int);
|
|
__type(value, struct tld_keys);
|
|
} tld_key_map SEC(".maps");
|
|
|
|
/**
|
|
* tld_object_init() - Initialize a tld_object.
|
|
*
|
|
* @task: The task_struct of the target task
|
|
* @tld_obj: A pointer to a tld_object to be initialized
|
|
*
|
|
* Return 0 on success; -ENODATA if the user space did not initialize task local data
|
|
* for the current task through tld_get_data(); -ENOMEM if the creation of tld_key_map
|
|
* fails
|
|
*/
|
|
__attribute__((unused))
|
|
static int tld_object_init(struct task_struct *task, struct tld_object *tld_obj)
|
|
{
|
|
int i;
|
|
|
|
tld_obj->data_map = bpf_task_storage_get(&tld_data_map, task, 0, 0);
|
|
if (!tld_obj->data_map)
|
|
return -ENODATA;
|
|
|
|
bpf_for(i, 0, TLD_KEY_MAP_CREATE_RETRY) {
|
|
tld_obj->key_map = bpf_task_storage_get(&tld_key_map, task, 0,
|
|
BPF_LOCAL_STORAGE_GET_F_CREATE);
|
|
if (likely(tld_obj->key_map))
|
|
break;
|
|
}
|
|
if (!tld_obj->key_map)
|
|
return -ENOMEM;
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Return the offset of TLD if @name is found. Otherwise, return the current TLD count
|
|
* using the nonpositive range so that the next tld_get_data() can skip fetching key if
|
|
* no new TLD is added or start comparing name from the first newly added TLD.
|
|
*/
|
|
__attribute__((unused))
|
|
static int __tld_fetch_key(struct tld_object *tld_obj, const char *name, int i_start)
|
|
{
|
|
struct tld_metadata *metadata;
|
|
int i, cnt, start, off = 0;
|
|
|
|
if (!tld_obj->data_map || !tld_obj->data_map->data || !tld_obj->data_map->meta)
|
|
return 0;
|
|
|
|
start = tld_obj->data_map->data->start;
|
|
cnt = tld_obj->data_map->meta->cnt;
|
|
metadata = tld_obj->data_map->meta->metadata;
|
|
|
|
bpf_for(i, 0, cnt) {
|
|
if (i >= TLD_MAX_DATA_CNT)
|
|
break;
|
|
|
|
if (i >= i_start && !bpf_strncmp(metadata[i].name, TLD_NAME_LEN, name))
|
|
return start + off;
|
|
|
|
off += TLD_ROUND_UP(metadata[i].size, 8);
|
|
}
|
|
|
|
return -cnt;
|
|
}
|
|
|
|
/**
|
|
* tld_get_data() - Retrieve a pointer to the TLD associated with the name.
|
|
*
|
|
* @tld_obj: A pointer to a valid tld_object initialized by tld_object_init()
|
|
* @key: The cached key of the TLD in tld_key_map
|
|
* @name: The name of the key associated with a TLD
|
|
* @size: The size of the TLD. Must be a known constant value
|
|
*
|
|
* Return a pointer to the TLD associated with @name; NULL if not found or @size is too
|
|
* big. @key is used to cache the key if the TLD is found to speed up subsequent calls.
|
|
* It should be defined as an member of tld_keys of tld_key_t type by the developer.
|
|
*/
|
|
#define tld_get_data(tld_obj, key, name, size) \
|
|
({ \
|
|
void *data = NULL, *_data = (tld_obj)->data_map->data; \
|
|
long off = (tld_obj)->key_map->key.off; \
|
|
int cnt; \
|
|
\
|
|
if (likely(_data)) { \
|
|
if (likely(off > 0)) { \
|
|
barrier_var(off); \
|
|
if (likely(off < __PAGE_SIZE - size)) \
|
|
data = _data + off; \
|
|
} else { \
|
|
cnt = -(off); \
|
|
if (likely((tld_obj)->data_map->meta) && \
|
|
cnt < (tld_obj)->data_map->meta->cnt) { \
|
|
off = __tld_fetch_key(tld_obj, name, cnt); \
|
|
(tld_obj)->key_map->key.off = off; \
|
|
\
|
|
if (likely(off < __PAGE_SIZE - size)) { \
|
|
barrier_var(off); \
|
|
if (off > 0) \
|
|
data = _data + off; \
|
|
} \
|
|
} \
|
|
} \
|
|
} \
|
|
data; \
|
|
})
|
|
|
|
#endif
|