|  | /* SPDX-License-Identifier: (LGPL-2.1 OR BSD-2-Clause) */ | 
|  |  | 
|  | /* | 
|  | * NETLINK      Netlink attributes | 
|  | * | 
|  | * Copyright (c) 2003-2013 Thomas Graf <tgraf@suug.ch> | 
|  | */ | 
|  |  | 
|  | #ifndef __LIBBPF_NLATTR_H | 
|  | #define __LIBBPF_NLATTR_H | 
|  |  | 
|  | #include <stdint.h> | 
|  | #include <string.h> | 
|  | #include <errno.h> | 
|  | #include <linux/netlink.h> | 
|  | #include <linux/rtnetlink.h> | 
|  | #include <linux/genetlink.h> | 
|  |  | 
|  | /* avoid multiple definition of netlink features */ | 
|  | #define __LINUX_NETLINK_H | 
|  |  | 
|  | /** | 
|  | * Standard attribute types to specify validation policy | 
|  | */ | 
|  | enum { | 
|  | LIBBPF_NLA_UNSPEC,	/**< Unspecified type, binary data chunk */ | 
|  | LIBBPF_NLA_U8,		/**< 8 bit integer */ | 
|  | LIBBPF_NLA_U16,		/**< 16 bit integer */ | 
|  | LIBBPF_NLA_U32,		/**< 32 bit integer */ | 
|  | LIBBPF_NLA_U64,		/**< 64 bit integer */ | 
|  | LIBBPF_NLA_STRING,	/**< NUL terminated character string */ | 
|  | LIBBPF_NLA_FLAG,	/**< Flag */ | 
|  | LIBBPF_NLA_MSECS,	/**< Micro seconds (64bit) */ | 
|  | LIBBPF_NLA_NESTED,	/**< Nested attributes */ | 
|  | __LIBBPF_NLA_TYPE_MAX, | 
|  | }; | 
|  |  | 
|  | #define LIBBPF_NLA_TYPE_MAX (__LIBBPF_NLA_TYPE_MAX - 1) | 
|  |  | 
|  | /** | 
|  | * @ingroup attr | 
|  | * Attribute validation policy. | 
|  | * | 
|  | * See section @core_doc{core_attr_parse,Attribute Parsing} for more details. | 
|  | */ | 
|  | struct libbpf_nla_policy { | 
|  | /** Type of attribute or LIBBPF_NLA_UNSPEC */ | 
|  | uint16_t	type; | 
|  |  | 
|  | /** Minimal length of payload required */ | 
|  | uint16_t	minlen; | 
|  |  | 
|  | /** Maximal length of payload allowed */ | 
|  | uint16_t	maxlen; | 
|  | }; | 
|  |  | 
|  | struct libbpf_nla_req { | 
|  | struct nlmsghdr nh; | 
|  | union { | 
|  | struct ifinfomsg ifinfo; | 
|  | struct tcmsg tc; | 
|  | struct genlmsghdr gnl; | 
|  | }; | 
|  | char buf[128]; | 
|  | }; | 
|  |  | 
|  | /** | 
|  | * @ingroup attr | 
|  | * Iterate over a stream of attributes | 
|  | * @arg pos	loop counter, set to current attribute | 
|  | * @arg head	head of attribute stream | 
|  | * @arg len	length of attribute stream | 
|  | * @arg rem	initialized to len, holds bytes currently remaining in stream | 
|  | */ | 
|  | #define libbpf_nla_for_each_attr(pos, head, len, rem) \ | 
|  | for (pos = head, rem = len; \ | 
|  | nla_ok(pos, rem); \ | 
|  | pos = nla_next(pos, &(rem))) | 
|  |  | 
|  | /** | 
|  | * libbpf_nla_data - head of payload | 
|  | * @nla: netlink attribute | 
|  | */ | 
|  | static inline void *libbpf_nla_data(const struct nlattr *nla) | 
|  | { | 
|  | return (void *)nla + NLA_HDRLEN; | 
|  | } | 
|  |  | 
|  | static inline uint8_t libbpf_nla_getattr_u8(const struct nlattr *nla) | 
|  | { | 
|  | return *(uint8_t *)libbpf_nla_data(nla); | 
|  | } | 
|  |  | 
|  | static inline uint16_t libbpf_nla_getattr_u16(const struct nlattr *nla) | 
|  | { | 
|  | return *(uint16_t *)libbpf_nla_data(nla); | 
|  | } | 
|  |  | 
|  | static inline uint32_t libbpf_nla_getattr_u32(const struct nlattr *nla) | 
|  | { | 
|  | return *(uint32_t *)libbpf_nla_data(nla); | 
|  | } | 
|  |  | 
|  | static inline uint64_t libbpf_nla_getattr_u64(const struct nlattr *nla) | 
|  | { | 
|  | return *(uint64_t *)libbpf_nla_data(nla); | 
|  | } | 
|  |  | 
|  | static inline const char *libbpf_nla_getattr_str(const struct nlattr *nla) | 
|  | { | 
|  | return (const char *)libbpf_nla_data(nla); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * libbpf_nla_len - length of payload | 
|  | * @nla: netlink attribute | 
|  | */ | 
|  | static inline int libbpf_nla_len(const struct nlattr *nla) | 
|  | { | 
|  | return nla->nla_len - NLA_HDRLEN; | 
|  | } | 
|  |  | 
|  | int libbpf_nla_parse(struct nlattr *tb[], int maxtype, struct nlattr *head, | 
|  | int len, struct libbpf_nla_policy *policy); | 
|  | int libbpf_nla_parse_nested(struct nlattr *tb[], int maxtype, | 
|  | struct nlattr *nla, | 
|  | struct libbpf_nla_policy *policy); | 
|  |  | 
|  | int libbpf_nla_dump_errormsg(struct nlmsghdr *nlh); | 
|  |  | 
|  | static inline struct nlattr *nla_data(struct nlattr *nla) | 
|  | { | 
|  | return (struct nlattr *)((void *)nla + NLA_HDRLEN); | 
|  | } | 
|  |  | 
|  | static inline struct nlattr *req_tail(struct libbpf_nla_req *req) | 
|  | { | 
|  | return (struct nlattr *)((void *)req + NLMSG_ALIGN(req->nh.nlmsg_len)); | 
|  | } | 
|  |  | 
|  | static inline int nlattr_add(struct libbpf_nla_req *req, int type, | 
|  | const void *data, int len) | 
|  | { | 
|  | struct nlattr *nla; | 
|  |  | 
|  | if (NLMSG_ALIGN(req->nh.nlmsg_len) + NLA_ALIGN(NLA_HDRLEN + len) > sizeof(*req)) | 
|  | return -EMSGSIZE; | 
|  | if (!!data != !!len) | 
|  | return -EINVAL; | 
|  |  | 
|  | nla = req_tail(req); | 
|  | nla->nla_type = type; | 
|  | nla->nla_len = NLA_HDRLEN + len; | 
|  | if (data) | 
|  | memcpy(nla_data(nla), data, len); | 
|  | req->nh.nlmsg_len = NLMSG_ALIGN(req->nh.nlmsg_len) + NLA_ALIGN(nla->nla_len); | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static inline struct nlattr *nlattr_begin_nested(struct libbpf_nla_req *req, int type) | 
|  | { | 
|  | struct nlattr *tail; | 
|  |  | 
|  | tail = req_tail(req); | 
|  | if (nlattr_add(req, type | NLA_F_NESTED, NULL, 0)) | 
|  | return NULL; | 
|  | return tail; | 
|  | } | 
|  |  | 
|  | static inline void nlattr_end_nested(struct libbpf_nla_req *req, | 
|  | struct nlattr *tail) | 
|  | { | 
|  | tail->nla_len = (void *)req_tail(req) - (void *)tail; | 
|  | } | 
|  |  | 
|  | #endif /* __LIBBPF_NLATTR_H */ |