OpenSDN source code
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Pages
gmp_router.h
Go to the documentation of this file.
1 /* $Id: gmp_router.h 514187 2012-05-06 12:25:25Z ib-builder $
2  *
3  * gmp_router.h - External definitions for gmp_router
4  *
5  * Dave Katz, March 2008
6  *
7  * Copyright (c) 2008, Juniper Networks, Inc.
8  * All rights reserved.
9  */
10 
11 #ifndef __GMP_ROUTER_H__
12 #define __GMP_ROUTER_H__
13 
14 /*
15  * This file defines the interface between the GMP toolkit and
16  * router-side clients.
17  *
18  * The client must first create an instance by calling
19  * gmpr_create_instance(). An instance is effectively a set of
20  * interfaces; an interface must be referred to by only one instance.
21  *
22  * Each instance has at least one client. A client is an entity that
23  * will receive notifications about changes in group state. The
24  * number of clients is limited to GMPR_MAX_RTR_CLIENTS. This number
25  * is expected to be small, since brute force and arrays are used with
26  * this expectation.
27  *
28  * An interface is bound to an instance by calling gmpr_attach_intf.
29  * The interface ID is a 32 bit opaque value defined in the
30  * environment file; the GMP toolkit doesn't attempt to dereference it
31  * in any way, so the environment can be creative as to what it
32  * actually is.
33  */
34 
35 /*
36  * Notifications
37  *
38  * Notifications to clients are done on a pull model. When a
39  * notification is enqueued for a client, the client notification
40  * callback is made, passing only the client context. When the client
41  * is ready to consume notifications, it calls gmpr_get_notification()
42  * to get the next notification. No additional notification callbacks
43  * will be made until the notification queue is drained (and
44  * gmpr_get_notification() returns NULL), so the client must either
45  * drain the queue when called, or keep track of whether the queue is
46  * drained while deferring.
47  *
48  * Notifications to clients are done on a delta basis. A notification
49  * consists of a type, an interface, a group, and possibly one source.
50  * The types are:
51  *
52  * GMPR_NOTIF_GROUP_ADD_EXCL: Add a group in Exclude mode
53  * GMPR_NOTIF_GROUP_ADD_INCL: Add a group in Include mode
54  * GMPR_NOTIF_GROUP_DELETE: Delete a group
55  * GMPR_NOTIF_ALLOW_SOURCE: Allow reception of a source
56  * GMPR_NOTIF_BLOCK_SOURCE: Block reception of a source
57  *
58  * If either ADD_INCL or ADD_EXCL is received, this means all previous
59  * sources for the group must be discarded. Once the mode is
60  * established, ALLOW and BLOCK calls are used to modify the source
61  * address set.
62  *
63  * Note that an ADD_INCL notification is guaranteed to be followed by
64  * at least one ALLOW; group deletions are handled by GROUP_DELETE,
65  * and there are no null Includes. The client can create group state
66  * with confidence that an ALLOW will follow (or potentially a DELETE
67  * due to state compression.)
68  *
69  * Note that, depending on timing and the kinds of messages sent by
70  * hosts, the client may see a GROUP_DELETE without BLOCK_SOURCE
71  * messages (indicating that all sources should be deleted along with
72  * the group), or it may see a series of BLOCK_SOURCE messages that
73  * delete all sources in a group without seeing a GROUP_DELETE
74  * message. These are semantically identical, and the client must
75  * deal with both cases.
76  *
77  * If the client wishes full state instead of (or along with) deltas, a
78  * notification of GMPR_NOTIF_GROUP_STATE is passed that contains the
79  * full group state (filter mode and all sources) whenever an ADD_INCL,
80  * ADD_EXCL, ALLOW_SOURCE, or BLOCK_SOURCE notification would be passed.
81  * GROUP_DELETE is used in this case to indicate the group deletion.
82  *
83  * When a client is registered, all state is enqueued for it.
84  *
85  * If desired, a client can refresh its state by calling
86  * gmpr_refresh(). It will receive all state in the form of
87  * notifications. In addition, a special notification of type
88  * GMPR_NOTIF_REFRESH_END will be enqueued at the end of the enqueued
89  * state. This allows a client that wants to do soft refresh
90  * (updating state before deleting any old state) to clean up old
91  * state knowing that everything has been refreshed.
92  *
93  * Note that redundant notifications can be received because of state
94  * compression. For example, if a group is active, then is deleted,
95  * then becomes active again, but the client did not read
96  * notifications after the first ADD_INCL, it will receive another
97  * ADD_INCL without ever seeing the DELETE (because the deletion was
98  * overtaken by the add.) This is clean semantically because the ADD
99  * notifications cause old state to be discarded.
100  *
101  * Similar things can happen with ALLOWs and BLOCKs.
102  *
103  *
104  * Host Notifications
105  *
106  * A similar system exists for notifications of host events. Note that
107  * host notifications are only defined for receive interest in groups
108  * and sources; in particular, non-null Exclude lists are not currently
109  * expressed in this API.
110  *
111  * When the first notification is enqueued for a client, the host notification
112  * callback is made. The client then calls gmpr_get_host_notification()
113  * at its leisure to pick up each notification. The callback will not be
114  * made again until the notification queue is drained.
115  *
116  * The notifications are of the following types:
117  * GMPR_NOTIF_HOST_JOIN: Join a *,G or S,G
118  * GMPR_NOTIF_HOST_LEAVE: Leave a *,G or S,G
119  * GMPR_NOTIF_HOST_TIMEOUT: Leave due to a host timeout
120  * GMPR_NOTIF_HOST_IFDOWN: Leave due to interface down
121  *
122  * State compression is done; if a host leaves a group and then rejoins it
123  * before the client reads the first notification, it will see only a JOIN,
124  * and if it joins and then leaves quickly, the client will see only a LEAVE.
125  *
126  *
127  * OIF remapping
128  *
129  * If the OIF remapping callback is defined, it is called for each received
130  * interest in a (*,G) or (S,G). The callback is expected to return the
131  * remapped interface ID corresponding to the input interface and (S,G).
132  */
133 
134 /*
135  * Client notification callback type
136  *
137  * This defines the callback to a client when a notification is ready.
138  */
139 typedef void (*gmpr_notif_cb)(void *cli_context);
140 
141 
142 /*
143  * OIF mapping callback type
144  *
145  * This defines the callback to a client to find out how to map an
146  * interface and (S,G) to an outgoing interface.
147  *
148  * Returns TRUE if the output_if parameter is filled in with an output
149  * interface, or FALSE if there is no output interface (it is blocked
150  * by policy.)
151  */
152 typedef boolean (*gmpr_oif_map_cb)(void *inst_context,
153  gmpx_intf_id rcv_if,
154  uint8_t *group_addr,
155  uint8_t *source_addr,
156  gmpx_intf_id *output_if);
157 
158 /*
159  * Policy callback type
160  *
161  * This defines the callback to a client to perform a policy check given
162  * an interface, group and (potentially) source. It returns TRUE if the
163  * report is allowed, or FALSE if not.
164  */
165 typedef boolean (*gmpr_policy_cb)(void *inst_context,
166  gmpx_intf_id rcv_if,
167  uint8_t *group_addr,
168  uint8_t *source_addr,
169  gmpx_packet_attr attribute);
170 
171 /*
172  * SSM check callback type
173  *
174  * This defines the callback to a client to verify if a group address is
175  * allowed to be used in a (*,G) join. This is used to detect attempts to
176  * join an SSM group without a source. It returns TRUE if the join is
177  * allowed, or FALSE if not.
178  */
179 typedef boolean (*gmpr_ssm_check_cb)(void *inst_context,
180  gmpx_intf_id rcv_if,
181  uint8_t *group_addr);
182 
183 /*
184  * Querier change callback type
185  *
186  * This defines the callback to a client when the querier status changes.
187  */
188 typedef void (*gmpr_querier_cb)(void *cli_context, gmpx_intf_id intf,
189  boolean querier, uint8_t *querier_addr);
190 
191 /*
192  * Router-side instance context block
193  */
194 typedef struct gmpr_instance_context_ {
195  gmpr_oif_map_cb rctx_oif_map_cb; /* OIF mapping callback */
196  gmpr_policy_cb rctx_policy_cb; /* Multicast policy callback */
197  gmpr_ssm_check_cb rctx_ssm_check_cb; /* SSM check callback */
199 
200 /*
201  * Router-side client context block
202  *
203  * This data structure carries linkage and other context information
204  * from a client of router-side GMP. This consists of a set of callbacks
205  * for various conditions. In addition, two booleans are used to say
206  * whether the client wishes to see full notifications, deltas, or both.
207  * At least one of these must be set.
208  */
209 typedef struct gmpr_client_context_ {
210  gmpr_notif_cb rctx_notif_cb; /* Notification callback */
211  gmpr_notif_cb rctx_host_notif_cb; /* Host notification callback */
212  gmpr_querier_cb rctx_querier_cb; /* Querier change callback */
213  boolean rctx_delta_notifications; /* TRUE if client wants deltas */
214  boolean rctx_full_notifications; /* TRUE if client wants full state */
216 
217 
218 /*
219  * Router-side client notification
220  *
221  * This data structure carries a single notification for a client. A
222  * notification consists of a single (intf, group) tuple, possibly
223  * with a source address (in the case of Allow and Block.) This is
224  * used to communicated aggregated group state to the upper layers.
225  *
226  * Normally, the addition and deletion of sources is done as a set of
227  * deltas (ALLOW/BLOCK of individual sources.) However, a client may
228  * optionally ask for complete group state upon each change; this is
229  * delvered in a GROUP_STATE notification.
230  */
231 typedef enum {
232  GMPR_NOTIF_GROUP_DELETE, /* Delete group */
233  GMPR_NOTIF_GROUP_ADD_EXCL, /* Add group in Exclude mode */
234  GMPR_NOTIF_GROUP_ADD_INCL, /* Add group in Include mode */
235  GMPR_NOTIF_ALLOW_SOURCE, /* Allow traffic for source */
236  GMPR_NOTIF_BLOCK_SOURCE, /* Block traffic to source */
237  GMPR_NOTIF_REFRESH_END, /* End of refresh stream */
238  GMPR_NOTIF_GROUP_STATE, /* Complete group state */
240 
242  gmpx_intf_id notif_intf_id; /* Interface ID */
243  gmp_addr_string notif_group_addr; /* Group address */
244  gmpr_client_notification_type notif_type; /* Notification type */
245  gmp_addr_string notif_source_addr; /* Source address if applicable */
246  gmp_filter_mode notif_filter_mode; /* Current group filter mode */
247  gmp_addr_thread *notif_addr_thread; /* Address thread if GROUP_STATE */
248  boolean notif_last_sg; /* Last (s,g) for the same group? */
250 
251 
252 /*
253  * Router-side client host notification
254  *
255  * This data structure carries a single host notification for a client. A
256  * notification consists of a single (intf, host, group) tuple, possibly
257  * with a source address (in the case of Allow and Block.) This is used
258  * to communicate host request state to the upper layers if host tracking
259  * is enabled.
260  */
261 typedef enum {
262  GMPR_NOTIF_HOST_UNKNOWN, /* We don't know yet */
263  GMPR_NOTIF_HOST_JOIN, /* Join a *,G or S,G */
264  GMPR_NOTIF_HOST_LEAVE, /* Leave a *,G or S,G */
265  GMPR_NOTIF_HOST_TIMEOUT, /* Leave due to a host timeout */
266  GMPR_NOTIF_HOST_IFDOWN, /* Leave due to interface down */
268 
271  gmpx_intf_id host_notif_intf_id; /* Interface ID */
273  gmp_addr_string host_notif_source_addr; /* Source address or 0 */
274  boolean host_notif_source_present; /* TRUE if source address present */
277 
278 
279 /*
280  * Interface list
281  *
282  * This structure is returned by gmpr_get_intf_list(), and must be freed
283  * by gmpr_free_intf_list() when it is processed.
284  */
285 
286 #define GMPR_INTF_LIST_SIZE 10 /* Number of interfaces per entry */
287 
288 typedef struct gmpr_client_intf_list_ {
289  struct gmpr_client_intf_list_ *gci_next; /* Next entry */
290  uint32_t gci_intf_count; /* Number of entries */
291  gmpx_intf_id gci_intfs[GMPR_INTF_LIST_SIZE]; /* Array of entries */
293 
294 
295 /*
296  * Group entry
297  *
298  * A pointer to this structure is returned by gmpr_get_intf_groups().
299  * It represents the state of a single group, and is linked to further
300  * groups.
301  *
302  * The caller is expected to pass it back via gmpr_destroy_intf_group().
303  */
304 typedef struct gmpr_intf_group_entry_ {
305  struct gmpr_intf_group_entry_ *gig_next; /* Next entry */
306  gmp_addr_string gig_group_addr; /* Group address */
307  gmp_filter_mode gig_filter_mode; /* Filter mode */
308  gmp_addr_thread *gig_sources; /* Source list, or NULL */
310 
311 
312 /*
313  * Host entry
314  *
315  * A pointer to this structure is returned by gmpr_get_intf_hosts().
316  * It represents a single host that has INCLUDE join state present and
317  * is linked to further hosts.
318  *
319  * The caller is expected to pass it back via gmpr_destroy_intf_hosts().
320  */
321 typedef struct gmpr_intf_host_entry_ {
322  struct gmpr_intf_host_entry_ *gih_next; /* Next entry */
323  gmp_addr_string gih_host_addr; /* Host address */
325 
326 
327 /*
328  * External references
329  */
331  void *inst_context,
332  gmpr_instance_context *context);
333 extern void gmpr_destroy_instance(gmp_instance_id instance_id);
334 extern gmp_client_id gmpr_register(gmp_instance_id instance_id,
335  void *cli_context,
336  gmpr_client_context *context);
337 extern void gmpr_detach(gmp_client_id client_id);
338 extern void gmpr_refresh(gmp_client_id client_id, boolean flush);
339 extern void gmpr_refresh_intf(gmp_client_id client_id, gmpx_intf_id intf_id,
340  boolean flush);
341 extern void gmpr_refresh_host_state(gmp_client_id client_id);
342 extern int gmpr_attach_intf(gmp_instance_id instance_id, gmpx_intf_id intf_id);
343 extern int gmpr_detach_intf(gmp_instance_id instance_id,
344  gmpx_intf_id intf_id);
345 extern int gmpr_set_intf_params(gmp_instance_id instance_id,
346  gmpx_intf_id intf_id,
347  gmpr_intf_params *params);
348 extern void gmpr_chk_grp_limit(gmp_instance_id instance_id,
349  gmpx_intf_id intf_id);
350 extern int gmpr_disable_host_tracking(gmp_instance_id instance_id,
351  gmpx_intf_id intf_id);
354  gmpr_client_notification *last_notification);
355 extern void gmpr_return_notification(gmpr_client_notification *notification);
358  gmpr_client_host_notification *last_notification);
360  gmpr_client_host_notification *host_notif);
361 extern boolean
363 
366  uint8_t *group_addr,
367  uint8_t *source_addr,
369 extern void gmpr_free_intf_list(gmpr_client_intf_list *intf_list);
370 extern boolean
372  gmpx_intf_id intf_id,
373  const uint8_t *source_addr,
374  const uint8_t *group_addr, boolean exact);
375 extern void gmpr_update_intf_state(gmp_instance_id instance_id,
376  gmpx_intf_id intf_id,
377  const uint8_t *intf_addr);
379  gmpx_intf_id intf_id);
381  gmpx_intf_id intf_id,
382  const uint8_t *host_addr);
384  gmpx_intf_id intf_id);
385 extern void gmpr_destroy_intf_group(gmpr_intf_group_entry *group_list);
386 extern void gmpr_destroy_intf_host(gmpr_intf_host_entry *host_list);
387 extern boolean gmpr_is_initialized(void);
388 extern void gmpr_timeout_group_range(gmp_instance_id instance_id,
389  gmpx_intf_id intf_id,
390  const uint8_t *group_addr,
391  uint32_t pfx_len, boolean send_query);
392 extern boolean gmpr_sg_is_excluded(gmp_instance_id instance_id,
393  gmpx_intf_id intf_id,
394  const uint8_t *group_addr,
395  const uint8_t *source_addr);
396 extern void gmpr_update_trace_flags(gmp_instance_id instance_id,
397  uint32_t trace_flags);
398 extern void gmpr_force_general_queries(gmp_instance_id instance_id,
399  gmpx_intf_id intf_id);
400 extern void gmpr_request_general_queries(gmp_instance_id instance_id,
401  gmpx_intf_id intf_id);
402 extern void gmpr_force_one_general_query(gmp_instance_id instance_id,
403  gmpx_intf_id intf_id);
404 extern void gmpr_request_one_general_query(gmp_instance_id instance_id,
405  gmpx_intf_id intf_id);
406 extern void gmpr_notify_oif_map_change (gmp_proto proto, gmpx_intf_id intf_id);
407 
408 #endif /* __GMP_ROUTER_H__ */
void(* gmpr_notif_cb)(void *cli_context)
Definition: gmp_router.h:139
void gmpr_notify_oif_map_change(gmp_proto proto, gmpx_intf_id intf_id)
struct gmpr_intf_host_entry_ gmpr_intf_host_entry
gmp_filter_mode notif_filter_mode
Definition: gmp_router.h:246
gmp_addr_string host_notif_host_addr
Definition: gmp_router.h:275
int gmpr_detach_intf(gmp_instance_id instance_id, gmpx_intf_id intf_id)
gmpr_intf_host_entry * gmpr_get_intf_hosts(gmp_instance_id instance_id, gmpx_intf_id intf_id)
gmpr_client_host_notification_type host_notif_type
Definition: gmp_router.h:270
struct gmpr_client_context_ gmpr_client_context
gmpr_client_host_notification_type
Definition: gmp_router.h:261
gmpr_intf_list_match
Definition: gmp_router.h:364
struct gmpr_intf_group_entry_ * gig_next
Definition: gmp_router.h:305
gmp_addr_string gig_group_addr
Definition: gmp_router.h:306
gmp_filter_mode gig_filter_mode
Definition: gmp_router.h:307
void gmpr_return_notification(gmpr_client_notification *notification)
gmp_addr_thread * notif_addr_thread
Definition: gmp_router.h:247
void * gmp_instance_id
Definition: gmp.h:32
gmp_addr_thread * gig_sources
Definition: gmp_router.h:308
gmpr_policy_cb rctx_policy_cb
Definition: gmp_router.h:196
gmpr_client_notification * gmpr_get_notification(gmp_client_id client_id, gmpr_client_notification *last_notification)
gmp_addr_string notif_group_addr
Definition: gmp_router.h:243
struct gmpr_client_host_notification_ gmpr_client_host_notification
gmpr_notif_cb rctx_notif_cb
Definition: gmp_router.h:210
boolean(* gmpr_ssm_check_cb)(void *inst_context, gmpx_intf_id rcv_if, uint8_t *group_addr)
Definition: gmp_router.h:179
gmpr_client_intf_list * gmpr_get_intf_list(gmp_instance_id instance_id, uint8_t *group_addr, uint8_t *source_addr, gmpr_intf_list_match type)
gmp_proto
Definition: gmp.h:45
gmpr_notif_cb rctx_host_notif_cb
Definition: gmp_router.h:211
void gmpr_return_host_notification(gmpr_client_host_notification *host_notif)
gmpr_client_host_notification * gmpr_get_host_notification(gmp_client_id client_id, gmpr_client_host_notification *last_notification)
unsigned int boolean
Definition: mcast_common.h:11
#define GMPR_INTF_LIST_SIZE
Definition: gmp_router.h:286
void gmpr_free_intf_list(gmpr_client_intf_list *intf_list)
int gmpr_attach_intf(gmp_instance_id instance_id, gmpx_intf_id intf_id)
void gmpr_destroy_intf_host(gmpr_intf_host_entry *host_list)
gmpr_client_notification_type
Definition: gmp_router.h:231
uint8_t type
Definition: load_balance.h:109
gmpr_intf_group_entry * gmpr_get_intf_groups(gmp_instance_id instance_id, gmpx_intf_id intf_id)
gmpx_intf_id gci_intfs[GMPR_INTF_LIST_SIZE]
Definition: gmp_router.h:291
void gmpr_timeout_group_range(gmp_instance_id instance_id, gmpx_intf_id intf_id, const uint8_t *group_addr, uint32_t pfx_len, boolean send_query)
struct gmpr_intf_group_entry_ gmpr_intf_group_entry
boolean(* gmpr_oif_map_cb)(void *inst_context, gmpx_intf_id rcv_if, uint8_t *group_addr, uint8_t *source_addr, gmpx_intf_id *output_if)
Definition: gmp_router.h:152
gmpx_intf_id notif_intf_id
Definition: gmp_router.h:242
void gmpr_refresh_host_state(gmp_client_id client_id)
int gmpr_set_intf_params(gmp_instance_id instance_id, gmpx_intf_id intf_id, gmpr_intf_params *params)
gmp_addr_string notif_source_addr
Definition: gmp_router.h:245
void gmpr_refresh(gmp_client_id client_id, boolean flush)
void gmpr_update_trace_flags(gmp_instance_id instance_id, uint32_t trace_flags)
gmp_addr_string host_notif_source_addr
Definition: gmp_router.h:273
boolean gmpr_sg_is_excluded(gmp_instance_id instance_id, gmpx_intf_id intf_id, const uint8_t *group_addr, const uint8_t *source_addr)
boolean gmpr_is_forwarding_channel(gmp_instance_id instance_id, gmpx_intf_id intf_id, const uint8_t *source_addr, const uint8_t *group_addr, boolean exact)
boolean rctx_delta_notifications
Definition: gmp_router.h:213
boolean rctx_full_notifications
Definition: gmp_router.h:214
void gmpr_refresh_intf(gmp_client_id client_id, gmpx_intf_id intf_id, boolean flush)
boolean gmpx_packet_attr
void(* gmpr_querier_cb)(void *cli_context, gmpx_intf_id intf, boolean querier, uint8_t *querier_addr)
Definition: gmp_router.h:188
gmpr_intf_group_entry * gmpr_get_host_groups(gmp_instance_id instance_id, gmpx_intf_id intf_id, const uint8_t *host_addr)
gmpr_ssm_check_cb rctx_ssm_check_cb
Definition: gmp_router.h:197
struct gmpr_intf_host_entry_ * gih_next
Definition: gmp_router.h:322
int gmpr_disable_host_tracking(gmp_instance_id instance_id, gmpx_intf_id intf_id)
gmp_addr_string gih_host_addr
Definition: gmp_router.h:323
void gmpr_destroy_instance(gmp_instance_id instance_id)
gmp_client_id gmpr_register(gmp_instance_id instance_id, void *cli_context, gmpr_client_context *context)
gmp_addr_string host_notif_group_addr
Definition: gmp_router.h:272
struct gmpr_client_notification_ gmpr_client_notification
gmp_instance_id gmpr_create_instance(gmp_proto proto, void *inst_context, gmpr_instance_context *context)
struct gmpr_client_intf_list_ gmpr_client_intf_list
void gmpr_chk_grp_limit(gmp_instance_id instance_id, gmpx_intf_id intf_id)
gmpr_client_notification_type notif_type
Definition: gmp_router.h:244
struct gmpr_instance_context_ gmpr_instance_context
boolean(* gmpr_policy_cb)(void *inst_context, gmpx_intf_id rcv_if, uint8_t *group_addr, uint8_t *source_addr, gmpx_packet_attr attribute)
Definition: gmp_router.h:165
void gmpr_request_general_queries(gmp_instance_id instance_id, gmpx_intf_id intf_id)
gmp_filter_mode
Definition: gmp.h:66
boolean gmpr_is_initialized(void)
Definition: gmp_router.h:304
void gmpr_force_general_queries(gmp_instance_id instance_id, gmpx_intf_id intf_id)
void gmpr_destroy_intf_group(gmpr_intf_group_entry *group_list)
void gmpr_detach(gmp_client_id client_id)
void gmpr_force_one_general_query(gmp_instance_id instance_id, gmpx_intf_id intf_id)
struct gmpr_client_intf_list_ * gci_next
Definition: gmp_router.h:289
void * gmp_client_id
Definition: gmp.h:39
gmpr_querier_cb rctx_querier_cb
Definition: gmp_router.h:212
boolean gmpr_notification_last_sg(gmpr_client_notification *notification)
void gmpr_request_one_general_query(gmp_instance_id instance_id, gmpx_intf_id intf_id)
void gmpr_update_intf_state(gmp_instance_id instance_id, gmpx_intf_id intf_id, const uint8_t *intf_addr)
gmpr_oif_map_cb rctx_oif_map_cb
Definition: gmp_router.h:195
Definition: gmp_router.h:321