aboutsummaryrefslogtreecommitdiffstats
path: root/package/libs/libnl-tiny/src/include/netlink/cache-api.h
blob: 22fc449d1b91401c1902c664018ae07155544fa0 (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
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
/*
 * netlink/cache-api.h		Caching API
 *
 *	This library is free software; you can redistribute it and/or
 *	modify it under the terms of the GNU Lesser General Public
 *	License as published by the Free Software Foundation version 2.1
 *	of the License.
 *
 * Copyright (c) 2003-2006 Thomas Graf <tgraf@suug.ch>
 */

#ifndef NETLINK_CACHE_API_H_
#define NETLINK_CACHE_API_H_

#include <netlink/netlink.h>

#ifdef __cplusplus
extern "C" {
#endif

/**
 * @ingroup cache
 * @defgroup cache_api Cache Implementation
 * @brief
 *
 * @par 1) Cache Definition
 * @code
 * struct nl_cache_ops my_cache_ops = {
 * 	.co_name		= "route/link",
 * 	.co_protocol		= NETLINK_ROUTE,
 * 	.co_hdrsize		= sizeof(struct ifinfomsg),
 * 	.co_obj_ops		= &my_obj_ops,
 * };
 * @endcode
 *
 * @par 2) 
 * @code
 * // The simplest way to fill a cache is by providing a request-update
 * // function which must trigger a complete dump on the kernel-side of
 * // whatever the cache covers.
 * static int my_request_update(struct nl_cache *cache,
 * 				struct nl_sock *socket)
 * {
 * 	// In this example, we request a full dump of the interface table
 * 	return nl_rtgen_request(socket, RTM_GETLINK, AF_UNSPEC, NLM_F_DUMP);
 * }
 *
 * // The resulting netlink messages sent back will be fed into a message
 * // parser one at a time. The message parser has to extract all relevant
 * // information from the message and create an object reflecting the
 * // contents of the message and pass it on to the parser callback function
 * // provide which will add the object to the cache.
 * static int my_msg_parser(struct nl_cache_ops *ops, struct sockaddr_nl *who,
 * 			    struct nlmsghdr *nlh, struct nl_parser_param *pp)
 * {
 * 	struct my_obj *obj;
 *
 * 	obj = my_obj_alloc();
 * 	obj->ce_msgtype = nlh->nlmsg_type;
 *
 * 	// Parse the netlink message and continue creating the object.
 *
 * 	err = pp->pp_cb((struct nl_object *) obj, pp);
 * 	if (err < 0)
 * 		goto errout;
 * }
 *
 * struct nl_cache_ops my_cache_ops = {
 * 	...
 * 	.co_request_update	= my_request_update,
 * 	.co_msg_parser		= my_msg_parser,
 * };
 * @endcode
 *
 * @par 3) Notification based Updates
 * @code
 * // Caches can be kept up-to-date based on notifications if the kernel
 * // sends out notifications whenever an object is added/removed/changed.
 * //
 * // It is trivial to support this, first a list of groups needs to be
 * // defined which are required to join in order to receive all necessary
 * // notifications. The groups are separated by address family to support
 * // the common situation where a separate group is used for each address
 * // family. If there is only one group, simply specify AF_UNSPEC.
 * static struct nl_af_group addr_groups[] = {
 * 	{ AF_INET,	RTNLGRP_IPV4_IFADDR },
 * 	{ AF_INET6,	RTNLGRP_IPV6_IFADDR },
 * 	{ END_OF_GROUP_LIST },
 * };
 *
 * // In order for the caching system to know the meaning of each message
 * // type it requires a table which maps each supported message type to
 * // a cache action, e.g. RTM_NEWADDR means address has been added or
 * // updated, RTM_DELADDR means address has been removed.
 * static struct nl_cache_ops rtnl_addr_ops = {
 * 	...
 * 	.co_msgtypes		= {
 * 					{ RTM_NEWADDR, NL_ACT_NEW, "new" },
 * 					{ RTM_DELADDR, NL_ACT_DEL, "del" },
 * 					{ RTM_GETADDR, NL_ACT_GET, "get" },
 * 					END_OF_MSGTYPES_LIST,
 * 				},
 * 	.co_groups		= addr_groups,
 * };
 *
 * // It is now possible to keep the cache up-to-date using the cache manager.
 * @endcode
 * @{
 */

enum {
	NL_ACT_UNSPEC,
	NL_ACT_NEW,
	NL_ACT_DEL,
	NL_ACT_GET,
	NL_ACT_SET,
	NL_ACT_CHANGE,
	__NL_ACT_MAX,
};

#define NL_ACT_MAX (__NL_ACT_MAX - 1)

#define END_OF_MSGTYPES_LIST	{ -1, -1, NULL }

/**
 * Message type to cache action association
 */
struct nl_msgtype
{
	/** Netlink message type */
	int			mt_id;

	/** Cache action to take */
	int			mt_act;

	/** Name of operation for human-readable printing */
	char *			mt_name;
};

/**
 * Address family to netlink group association
 */
struct nl_af_group
{
	/** Address family */
	int			ag_family;

	/** Netlink group identifier */
	int			ag_group;
};

#define END_OF_GROUP_LIST AF_UNSPEC, 0

struct nl_parser_param
{
	int             (*pp_cb)(struct nl_object *, struct nl_parser_param *);
	void *            pp_arg;
};

/**
 * Cache Operations
 */
struct nl_cache_ops
{
	char  *			co_name;

	int			co_hdrsize;
	int			co_protocol;
	struct nl_af_group *	co_groups;
	
	/**
	 * Called whenever an update of the cache is required. Must send
	 * a request message to the kernel requesting a complete dump.
	 */
	int   (*co_request_update)(struct nl_cache *, struct nl_sock *);

	/**
	 * Called whenever a message was received that needs to be parsed.
	 * Must parse the message and call the paser callback function
	 * (nl_parser_param) provided via the argument.
	 */
	int   (*co_msg_parser)(struct nl_cache_ops *, struct sockaddr_nl *,
			       struct nlmsghdr *, struct nl_parser_param *);

	struct nl_object_ops *	co_obj_ops;

	struct nl_cache_ops *co_next;
	struct nl_cache *co_major_cache;
	struct genl_ops *	co_genl;
	struct nl_msgtype	co_msgtypes[];
};

/** @} */

#ifdef __cplusplus
}
#endif

#endif
ue objects by use of unified comparison functions. * // In order for it to work, your object implementation must provide * // a comparison function and define a list of attributes which * // combined together make an object unique. * * static int my_obj_compare(struct nl_object *_a, struct nl_object *_b, * uint32_t attrs, int flags) * { * struct my_obj *a = nl_object_priv(_a): * struct my_obj *b = nl_object_priv(_b): * int diff = 0; * * // We help ourselves in defining our own DIFF macro which will * // call ATTR_DIFF() on both objects which will make sure to only * // compare the attributes if required. * #define MY_DIFF(ATTR, EXPR) ATTR_DIFF(attrs, MY_ATTR_##ATTR, a, b, EXPR) * * // Call our own diff macro for each attribute to build a bitmask * // representing the attributes which mismatch. * diff |= MY_DIFF(FOO, a->foo != b->foo) * diff |= MY_DIFF(BAR, strcmp(a->bar, b->bar)) * * return diff; * } * * // In order to identify identical objects with differing attributes * // you must specify the attributes required to uniquely identify * // your object. Make sure to not include too many attributes, this * // list is used when caches look for an old version of an object. * struct nl_object_ops my_ops = { * ... * .oo_id_attrs = MY_ATTR_FOO, * .oo_compare = my_obj_compare, * }; * @endcode * @{ */ /** * Common Object Header * * This macro must be included as first member in every object * definition to allow objects to be cached. */ #define NLHDR_COMMON \ int ce_refcnt; \ struct nl_object_ops * ce_ops; \ struct nl_cache * ce_cache; \ struct nl_list_head ce_list; \ int ce_msgtype; \ int ce_flags; \ uint32_t ce_mask; /** * Return true if attribute is available in both objects * @arg A an object * @arg B another object * @arg ATTR attribute bit * * @return True if the attribute is available, otherwise false is returned. */ #define AVAILABLE(A, B, ATTR) (((A)->ce_mask & (B)->ce_mask) & (ATTR)) /** * Return true if attributes mismatch * @arg A an object * @arg B another object * @arg ATTR attribute bit * @arg EXPR Comparison expression * * This function will check if the attribute in question is available * in both objects, if not this will count as a mismatch. * * If available the function will execute the expression which must * return true if the attributes mismatch. * * @return True if the attribute mismatch, or false if they match. */ #define ATTR_MISMATCH(A, B, ATTR, EXPR) (!AVAILABLE(A, B, ATTR) || (EXPR)) /** * Return attribute bit if attribute does not match * @arg LIST list of attributes to be compared * @arg ATTR attribute bit * @arg A an object * @arg B another object * @arg EXPR Comparison expression * * This function will check if the attribute in question is available * in both objects, if not this will count as a mismatch. * * If available the function will execute the expression which must * return true if the attributes mismatch. * * In case the attributes mismatch, the attribute is returned, otherwise * 0 is returned. * * @code * diff |= ATTR_DIFF(attrs, MY_ATTR_FOO, a, b, a->foo != b->foo); * @endcode */ #define ATTR_DIFF(LIST, ATTR, A, B, EXPR) \ ({ int diff = 0; \ if (((LIST) & (ATTR)) && ATTR_MISMATCH(A, B, ATTR, EXPR)) \ diff = ATTR; \ diff; }) /** * Object Operations */ struct nl_object; struct nl_object_ops { /** * Unique name of object type * * Must be in the form family/name, e.g. "route/addr" */ char * oo_name; /** Size of object including its header */ size_t oo_size; /* List of attributes needed to uniquely identify the object */ uint32_t oo_id_attrs; /** * Constructor function * * Will be called when a new object of this type is allocated. * Can be used to initialize members such as lists etc. */ void (*oo_constructor)(struct nl_object *); /** * Destructor function * * Will be called when an object is freed. Must free all * resources which may have been allocated as part of this * object. */ void (*oo_free_data)(struct nl_object *); /** * Cloning function * * Will be called when an object needs to be cloned. Please * note that the generic object code will make an exact * copy of the object first, therefore you only need to take * care of members which require reference counting etc. * * May return a negative error code to abort cloning. */ int (*oo_clone)(struct nl_object *, struct nl_object *); /** * Dumping functions * * Will be called when an object is dumped. The implementations * have to use nl_dump(), nl_dump_line(), and nl_new_line() to * dump objects. * * The functions must return the number of lines printed. */ void (*oo_dump[NL_DUMP_MAX+1])(struct nl_object *, struct nl_dump_params *); /** * Comparison function * * Will be called when two objects of the same type are * compared. It takes the two objects in question, an object * specific bitmask defining which attributes should be * compared and flags to control the behaviour. * * The function must return a bitmask with the relevant bit * set for each attribute that mismatches. */ int (*oo_compare)(struct nl_object *, struct nl_object *, uint32_t, int); char *(*oo_attrs2str)(int, char *, size_t); }; /** @} */ #ifdef __cplusplus } #endif #endif