📄 netlink.h
字号:
#ifndef __NET_NETLINK_H#define __NET_NETLINK_H#include <linux/types.h>#include <linux/netlink.h>#include <linux/jiffies.h>/* ======================================================================== * Netlink Messages and Attributes Interface (As Seen On TV) * ------------------------------------------------------------------------ * Messages Interface * ------------------------------------------------------------------------ * * Message Format: * <--- nlmsg_total_size(payload) ---> * <-- nlmsg_msg_size(payload) -> * +----------+- - -+-------------+- - -+-------- - - * | nlmsghdr | Pad | Payload | Pad | nlmsghdr * +----------+- - -+-------------+- - -+-------- - - * nlmsg_data(nlh)---^ ^ * nlmsg_next(nlh)-----------------------+ * * Payload Format: * <---------------------- nlmsg_len(nlh) ---------------------> * <------ hdrlen ------> <- nlmsg_attrlen(nlh, hdrlen) -> * +----------------------+- - -+--------------------------------+ * | Family Header | Pad | Attributes | * +----------------------+- - -+--------------------------------+ * nlmsg_attrdata(nlh, hdrlen)---^ * * Data Structures: * struct nlmsghdr netlink message header * * Message Construction: * nlmsg_new() create a new netlink message * nlmsg_put() add a netlink message to an skb * nlmsg_put_answer() callback based nlmsg_put() * nlmsg_end() finanlize netlink message * nlmsg_get_pos() return current position in message * nlmsg_trim() trim part of message * nlmsg_cancel() cancel message construction * nlmsg_free() free a netlink message * * Message Sending: * nlmsg_multicast() multicast message to several groups * nlmsg_unicast() unicast a message to a single socket * nlmsg_notify() send notification message * * Message Length Calculations: * nlmsg_msg_size(payload) length of message w/o padding * nlmsg_total_size(payload) length of message w/ padding * nlmsg_padlen(payload) length of padding at tail * * Message Payload Access: * nlmsg_data(nlh) head of message payload * nlmsg_len(nlh) length of message payload * nlmsg_attrdata(nlh, hdrlen) head of attributes data * nlmsg_attrlen(nlh, hdrlen) length of attributes data * * Message Parsing: * nlmsg_ok(nlh, remaining) does nlh fit into remaining bytes? * nlmsg_next(nlh, remaining) get next netlink message * nlmsg_parse() parse attributes of a message * nlmsg_find_attr() find an attribute in a message * nlmsg_for_each_msg() loop over all messages * nlmsg_validate() validate netlink message incl. attrs * nlmsg_for_each_attr() loop over all attributes * * Misc: * nlmsg_report() report back to application? * * ------------------------------------------------------------------------ * Attributes Interface * ------------------------------------------------------------------------ * * Attribute Format: * <------- nla_total_size(payload) -------> * <---- nla_attr_size(payload) -----> * +----------+- - -+- - - - - - - - - +- - -+-------- - - * | Header | Pad | Payload | Pad | Header * +----------+- - -+- - - - - - - - - +- - -+-------- - - * <- nla_len(nla) -> ^ * nla_data(nla)----^ | * nla_next(nla)-----------------------------' * * Data Structures: * struct nlattr netlink attribute header * * Attribute Construction: * nla_reserve(skb, type, len) reserve room for an attribute * nla_reserve_nohdr(skb, len) reserve room for an attribute w/o hdr * nla_put(skb, type, len, data) add attribute to skb * nla_put_nohdr(skb, len, data) add attribute w/o hdr * * Attribute Construction for Basic Types: * nla_put_u8(skb, type, value) add u8 attribute to skb * nla_put_u16(skb, type, value) add u16 attribute to skb * nla_put_u32(skb, type, value) add u32 attribute to skb * nla_put_u64(skb, type, value) add u64 attribute to skb * nla_put_string(skb, type, str) add string attribute to skb * nla_put_flag(skb, type) add flag attribute to skb * nla_put_msecs(skb, type, jiffies) add msecs attribute to skb * * Exceptions Based Attribute Construction: * NLA_PUT(skb, type, len, data) add attribute to skb * NLA_PUT_U8(skb, type, value) add u8 attribute to skb * NLA_PUT_U16(skb, type, value) add u16 attribute to skb * NLA_PUT_U32(skb, type, value) add u32 attribute to skb * NLA_PUT_U64(skb, type, value) add u64 attribute to skb * NLA_PUT_STRING(skb, type, str) add string attribute to skb * NLA_PUT_FLAG(skb, type) add flag attribute to skb * NLA_PUT_MSECS(skb, type, jiffies) add msecs attribute to skb * * The meaning of these functions is equal to their lower case * variants but they jump to the label nla_put_failure in case * of a failure. * * Nested Attributes Construction: * nla_nest_start(skb, type) start a nested attribute * nla_nest_end(skb, nla) finalize a nested attribute * nla_nest_compat_start(skb, type, start a nested compat attribute * len, data) * nla_nest_compat_end(skb, type) finalize a nested compat attribute * nla_nest_cancel(skb, nla) cancel nested attribute construction * * Attribute Length Calculations: * nla_attr_size(payload) length of attribute w/o padding * nla_total_size(payload) length of attribute w/ padding * nla_padlen(payload) length of padding * * Attribute Payload Access: * nla_data(nla) head of attribute payload * nla_len(nla) length of attribute payload * * Attribute Payload Access for Basic Types: * nla_get_u8(nla) get payload for a u8 attribute * nla_get_u16(nla) get payload for a u16 attribute * nla_get_u32(nla) get payload for a u32 attribute * nla_get_u64(nla) get payload for a u64 attribute * nla_get_flag(nla) return 1 if flag is true * nla_get_msecs(nla) get payload for a msecs attribute * * Attribute Misc: * nla_memcpy(dest, nla, count) copy attribute into memory * nla_memcmp(nla, data, size) compare attribute with memory area * nla_strlcpy(dst, nla, size) copy attribute to a sized string * nla_strcmp(nla, str) compare attribute with string * * Attribute Parsing: * nla_ok(nla, remaining) does nla fit into remaining bytes? * nla_next(nla, remaining) get next netlink attribute * nla_validate() validate a stream of attributes * nla_validate_nested() validate a stream of nested attributes * nla_find() find attribute in stream of attributes * nla_find_nested() find attribute in nested attributes * nla_parse() parse and validate stream of attrs * nla_parse_nested() parse nested attribuets * nla_parse_nested_compat() parse nested compat attributes * nla_for_each_attr() loop over all attributes * nla_for_each_nested() loop over the nested attributes *========================================================================= */ /** * Standard attribute types to specify validation policy */enum { NLA_UNSPEC, NLA_U8, NLA_U16, NLA_U32, NLA_U64, NLA_STRING, NLA_FLAG, NLA_MSECS, NLA_NESTED, NLA_NESTED_COMPAT, NLA_NUL_STRING, NLA_BINARY, __NLA_TYPE_MAX,};#define NLA_TYPE_MAX (__NLA_TYPE_MAX - 1)/** * struct nla_policy - attribute validation policy * @type: Type of attribute or NLA_UNSPEC * @len: Type specific length of payload * * Policies are defined as arrays of this struct, the array must be * accessible by attribute type up to the highest identifier to be expected. * * Meaning of `len' field: * NLA_STRING Maximum length of string * NLA_NUL_STRING Maximum length of string (excluding NUL) * NLA_FLAG Unused * NLA_BINARY Maximum length of attribute payload * NLA_NESTED_COMPAT Exact length of structure payload * All other Exact length of attribute payload * * Example: * static struct nla_policy my_policy[ATTR_MAX+1] __read_mostly = { * [ATTR_FOO] = { .type = NLA_U16 }, * [ATTR_BAR] = { .type = NLA_STRING, .len = BARSIZ }, * [ATTR_BAZ] = { .len = sizeof(struct mystruct) }, * }; */struct nla_policy { u16 type; u16 len;};/** * struct nl_info - netlink source information * @nlh: Netlink message header of original request * @pid: Netlink PID of requesting application */struct nl_info { struct nlmsghdr *nlh; u32 pid;};extern int netlink_rcv_skb(struct sk_buff *skb, int (*cb)(struct sk_buff *, struct nlmsghdr *));extern int nlmsg_notify(struct sock *sk, struct sk_buff *skb, u32 pid, unsigned int group, int report, gfp_t flags);extern int nla_validate(struct nlattr *head, int len, int maxtype, const struct nla_policy *policy);extern int nla_parse(struct nlattr *tb[], int maxtype, struct nlattr *head, int len, const struct nla_policy *policy);extern struct nlattr * nla_find(struct nlattr *head, int len, int attrtype);extern size_t nla_strlcpy(char *dst, const struct nlattr *nla, size_t dstsize);extern int nla_memcpy(void *dest, struct nlattr *src, int count);extern int nla_memcmp(const struct nlattr *nla, const void *data, size_t size);extern int nla_strcmp(const struct nlattr *nla, const char *str);extern struct nlattr * __nla_reserve(struct sk_buff *skb, int attrtype, int attrlen);extern void * __nla_reserve_nohdr(struct sk_buff *skb, int attrlen);extern struct nlattr * nla_reserve(struct sk_buff *skb, int attrtype, int attrlen);extern void * nla_reserve_nohdr(struct sk_buff *skb, int attrlen);extern void __nla_put(struct sk_buff *skb, int attrtype, int attrlen, const void *data);extern void __nla_put_nohdr(struct sk_buff *skb, int attrlen, const void *data);extern int nla_put(struct sk_buff *skb, int attrtype, int attrlen, const void *data);extern int nla_put_nohdr(struct sk_buff *skb, int attrlen, const void *data);/************************************************************************** * Netlink Messages **************************************************************************//** * nlmsg_msg_size - length of netlink message not including padding * @payload: length of message payload */static inline int nlmsg_msg_size(int payload){ return NLMSG_HDRLEN + payload;}/** * nlmsg_total_size - length of netlink message including padding * @payload: length of message payload */static inline int nlmsg_total_size(int payload){ return NLMSG_ALIGN(nlmsg_msg_size(payload));}/** * nlmsg_padlen - length of padding at the message's tail * @payload: length of message payload */static inline int nlmsg_padlen(int payload){ return nlmsg_total_size(payload) - nlmsg_msg_size(payload);}/** * nlmsg_data - head of message payload * @nlh: netlink messsage header */static inline void *nlmsg_data(const struct nlmsghdr *nlh){ return (unsigned char *) nlh + NLMSG_HDRLEN;}/** * nlmsg_len - length of message payload * @nlh: netlink message header */static inline int nlmsg_len(const struct nlmsghdr *nlh){ return nlh->nlmsg_len - NLMSG_HDRLEN;}/** * nlmsg_attrdata - head of attributes data * @nlh: netlink message header * @hdrlen: length of family specific header */static inline struct nlattr *nlmsg_attrdata(const struct nlmsghdr *nlh, int hdrlen){ unsigned char *data = nlmsg_data(nlh); return (struct nlattr *) (data + NLMSG_ALIGN(hdrlen));}/** * nlmsg_attrlen - length of attributes data * @nlh: netlink message header * @hdrlen: length of family specific header */static inline int nlmsg_attrlen(const struct nlmsghdr *nlh, int hdrlen){ return nlmsg_len(nlh) - NLMSG_ALIGN(hdrlen);}/** * nlmsg_ok - check if the netlink message fits into the remaining bytes * @nlh: netlink message header * @remaining: number of bytes remaining in message stream */static inline int nlmsg_ok(const struct nlmsghdr *nlh, int remaining){ return (remaining >= sizeof(struct nlmsghdr) && nlh->nlmsg_len >= sizeof(struct nlmsghdr) && nlh->nlmsg_len <= remaining);}/** * nlmsg_next - next netlink message in message stream * @nlh: netlink message header * @remaining: number of bytes remaining in message stream * * Returns the next netlink message in the message stream and * decrements remaining by the size of the current message. */static inline struct nlmsghdr *nlmsg_next(struct nlmsghdr *nlh, int *remaining){ int totlen = NLMSG_ALIGN(nlh->nlmsg_len); *remaining -= totlen; return (struct nlmsghdr *) ((unsigned char *) nlh + totlen);}/** * nlmsg_parse - parse attributes of a netlink message * @nlh: netlink message header * @hdrlen: length of family specific header * @tb: destination array with maxtype+1 elements * @maxtype: maximum attribute type to be expected * @policy: validation policy * * See nla_parse() */static inline int nlmsg_parse(struct nlmsghdr *nlh, int hdrlen, struct nlattr *tb[], int maxtype, const struct nla_policy *policy){ if (nlh->nlmsg_len < nlmsg_msg_size(hdrlen))
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -