2 * EIGRP Filter Functions.
3 * Copyright (C) 2013-2015
16 * This file is part of GNU Zebra.
18 * GNU Zebra is free software; you can redistribute it and/or modify it
19 * under the terms of the GNU General Public License as published by the
20 * Free Software Foundation; either version 2, or (at your option) any
23 * GNU Zebra is distributed in the hope that it will be useful, but
24 * WITHOUT ANY WARRANTY; without even the implied warranty of
25 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
26 * General Public License for more details.
28 * You should have received a copy of the GNU General Public License along
29 * with this program; see the file COPYING; if not, write to the Free Software
30 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
44 #include "sockunion.h"
49 #include "distribute.h"
55 #include "eigrpd/eigrp_structs.h"
56 #include "eigrpd/eigrpd.h"
57 #include "eigrpd/eigrp_const.h"
58 #include "eigrpd/eigrp_filter.h"
59 #include "eigrpd/eigrp_packet.h"
60 #include "eigrpd/eigrp_memory.h"
63 * Distribute-list update functions.
65 void eigrp_distribute_update(struct distribute_ctx
*ctx
,
66 struct distribute
*dist
)
68 struct eigrp
*e
= eigrp_lookup(ctx
->vrf
->vrf_id
);
69 struct interface
*ifp
;
70 struct eigrp_interface
*ei
= NULL
;
71 struct access_list
*alist
;
72 struct prefix_list
*plist
;
73 // struct route_map *routemap;
75 /* if no interface address is present, set list to eigrp process struct
78 /* Check if distribute-list was set for process or interface */
80 /* access list IN for whole process */
81 if (dist
->list
[DISTRIBUTE_V4_IN
]) {
82 alist
= access_list_lookup(
83 AFI_IP
, dist
->list
[DISTRIBUTE_V4_IN
]);
85 e
->list
[EIGRP_FILTER_IN
] = alist
;
87 e
->list
[EIGRP_FILTER_IN
] = NULL
;
89 e
->list
[EIGRP_FILTER_IN
] = NULL
;
92 /* access list OUT for whole process */
93 if (dist
->list
[DISTRIBUTE_V4_OUT
]) {
94 alist
= access_list_lookup(
95 AFI_IP
, dist
->list
[DISTRIBUTE_V4_OUT
]);
97 e
->list
[EIGRP_FILTER_OUT
] = alist
;
99 e
->list
[EIGRP_FILTER_OUT
] = NULL
;
101 e
->list
[EIGRP_FILTER_OUT
] = NULL
;
104 /* PREFIX_LIST IN for process */
105 if (dist
->prefix
[DISTRIBUTE_V4_IN
]) {
106 plist
= prefix_list_lookup(
107 AFI_IP
, dist
->prefix
[DISTRIBUTE_V4_IN
]);
109 e
->prefix
[EIGRP_FILTER_IN
] = plist
;
111 e
->prefix
[EIGRP_FILTER_IN
] = NULL
;
113 e
->prefix
[EIGRP_FILTER_IN
] = NULL
;
115 /* PREFIX_LIST OUT for process */
116 if (dist
->prefix
[DISTRIBUTE_V4_OUT
]) {
117 plist
= prefix_list_lookup(
118 AFI_IP
, dist
->prefix
[DISTRIBUTE_V4_OUT
]);
120 e
->prefix
[EIGRP_FILTER_OUT
] = plist
;
123 e
->prefix
[EIGRP_FILTER_OUT
] = NULL
;
125 e
->prefix
[EIGRP_FILTER_OUT
] = NULL
;
127 // This is commented out, because the distribute.[ch] code
128 // changes looked poorly written from first glance
129 // commit was 133bdf2d
132 /* route-map IN for whole process */
133 if (dist
->route
[DISTRIBUTE_V4_IN
])
135 routemap
= route_map_lookup_by_name (dist
->route
[DISTRIBUTE_V4_IN
]);
137 e
->routemap
[EIGRP_FILTER_IN
] = routemap
;
139 e
->routemap
[EIGRP_FILTER_IN
] = NULL
;
143 e
->routemap
[EIGRP_FILTER_IN
] = NULL
;
146 /* route-map OUT for whole process */
147 if (dist
->route
[DISTRIBUTE_V4_OUT
])
149 routemap
= route_map_lookup_by_name (dist
->route
[DISTRIBUTE_V4_OUT
]);
151 e
->routemap
[EIGRP_FILTER_OUT
] = routemap
;
153 e
->routemap
[EIGRP_FILTER_OUT
] = NULL
;
157 e
->routemap
[EIGRP_FILTER_OUT
] = NULL
;
160 // TODO: check Graceful restart after 10sec
162 /* check if there is already GR scheduled */
163 if (e
->t_distribute
!= NULL
) {
164 /* if is, cancel schedule */
165 thread_cancel(e
->t_distribute
);
167 /* schedule Graceful restart for whole process in 10sec */
168 e
->t_distribute
= NULL
;
169 thread_add_timer(master
, eigrp_distribute_timer_process
, e
,
170 (10), &e
->t_distribute
);
175 ifp
= if_lookup_by_name(dist
->ifname
, e
->vrf_id
);
179 /*struct eigrp_if_info * info = ifp->info;
180 ei = info->eigrp_interface;*/
181 struct listnode
*node
, *nnode
;
182 struct eigrp_interface
*ei2
;
183 /* Find proper interface */
184 for (ALL_LIST_ELEMENTS(e
->eiflist
, node
, nnode
, ei2
)) {
185 if (strcmp(ei2
->ifp
->name
, ifp
->name
) == 0) {
192 /* Access-list for interface in */
193 if (dist
->list
[DISTRIBUTE_V4_IN
]) {
194 alist
= access_list_lookup(AFI_IP
,
195 dist
->list
[DISTRIBUTE_V4_IN
]);
197 ei
->list
[EIGRP_FILTER_IN
] = alist
;
199 ei
->list
[EIGRP_FILTER_IN
] = NULL
;
201 ei
->list
[EIGRP_FILTER_IN
] = NULL
;
204 /* Access-list for interface in */
205 if (dist
->list
[DISTRIBUTE_V4_OUT
]) {
206 alist
= access_list_lookup(AFI_IP
,
207 dist
->list
[DISTRIBUTE_V4_OUT
]);
209 ei
->list
[EIGRP_FILTER_OUT
] = alist
;
211 ei
->list
[EIGRP_FILTER_OUT
] = NULL
;
214 ei
->list
[EIGRP_FILTER_OUT
] = NULL
;
216 /* Prefix-list for interface in */
217 if (dist
->prefix
[DISTRIBUTE_V4_IN
]) {
218 plist
= prefix_list_lookup(AFI_IP
,
219 dist
->prefix
[DISTRIBUTE_V4_IN
]);
221 ei
->prefix
[EIGRP_FILTER_IN
] = plist
;
223 ei
->prefix
[EIGRP_FILTER_IN
] = NULL
;
225 ei
->prefix
[EIGRP_FILTER_IN
] = NULL
;
227 /* Prefix-list for interface out */
228 if (dist
->prefix
[DISTRIBUTE_V4_OUT
]) {
229 plist
= prefix_list_lookup(AFI_IP
,
230 dist
->prefix
[DISTRIBUTE_V4_OUT
]);
232 ei
->prefix
[EIGRP_FILTER_OUT
] = plist
;
234 ei
->prefix
[EIGRP_FILTER_OUT
] = NULL
;
236 ei
->prefix
[EIGRP_FILTER_OUT
] = NULL
;
239 /* route-map IN for whole process */
240 if (dist
->route
[DISTRIBUTE_V4_IN
])
242 zlog_info("<DEBUG ACL ALL in");
243 routemap
= route_map_lookup_by_name (dist
->route
[DISTRIBUTE_V4_IN
]);
245 ei
->routemap
[EIGRP_FILTER_IN
] = routemap
;
247 ei
->routemap
[EIGRP_FILTER_IN
] = NULL
;
251 ei
->routemap
[EIGRP_FILTER_IN
] = NULL
;
254 /* route-map OUT for whole process */
255 if (dist
->route
[DISTRIBUTE_V4_OUT
])
257 routemap
= route_map_lookup_by_name (dist
->route
[DISTRIBUTE_V4_OUT
]);
259 ei
->routemap
[EIGRP_FILTER_OUT
] = routemap
;
261 ei
->routemap
[EIGRP_FILTER_OUT
] = NULL
;
265 ei
->routemap
[EIGRP_FILTER_OUT
] = NULL
;
268 // TODO: check Graceful restart after 10sec
270 /* check if there is already GR scheduled */
271 if (ei
->t_distribute
!= NULL
) {
272 /* if is, cancel schedule */
273 thread_cancel(ei
->t_distribute
);
275 /* schedule Graceful restart for interface in 10sec */
276 e
->t_distribute
= NULL
;
277 thread_add_timer(master
, eigrp_distribute_timer_interface
, ei
, 10,
282 * Function called by prefix-list and access-list update
284 void eigrp_distribute_update_interface(struct interface
*ifp
)
286 struct distribute
*dist
;
289 eigrp
= eigrp_lookup(ifp
->vrf_id
);
292 dist
= distribute_lookup(eigrp
->distribute_ctx
, ifp
->name
);
294 eigrp_distribute_update(eigrp
->distribute_ctx
,
298 /* Update all interface's distribute list.
299 * Function used in hook for prefix-list
301 void eigrp_distribute_update_all(struct prefix_list
*notused
)
304 struct interface
*ifp
;
306 RB_FOREACH (vrf
, vrf_name_head
, &vrfs_by_name
) {
307 FOR_ALL_INTERFACES (vrf
, ifp
)
308 eigrp_distribute_update_interface(ifp
);
313 * Function used in hook for acces-list
315 void eigrp_distribute_update_all_wrapper(struct access_list
*notused
)
317 eigrp_distribute_update_all(NULL
);
321 * @fn eigrp_distribute_timer_process
323 * @param[in] thread current execution thread timer is associated with
325 * @return int always returns 0
328 * Called when 10sec waiting time expire and
329 * executes Graceful restart for whole process
331 int eigrp_distribute_timer_process(struct thread
*thread
)
335 eigrp
= THREAD_ARG(thread
);
336 eigrp
->t_distribute
= NULL
;
338 /* execute GR for whole process */
339 eigrp_update_send_process_GR(eigrp
, EIGRP_GR_FILTER
, NULL
);
345 * @fn eigrp_distribute_timer_interface
347 * @param[in] thread current execution thread timer is associated with
349 * @return int always returns 0
352 * Called when 10sec waiting time expire and
353 * executes Graceful restart for interface
355 int eigrp_distribute_timer_interface(struct thread
*thread
)
357 struct eigrp_interface
*ei
;
359 ei
= THREAD_ARG(thread
);
360 ei
->t_distribute
= NULL
;
362 /* execute GR for interface */
363 eigrp_update_send_interface_GR(ei
, EIGRP_GR_FILTER
, NULL
);