]> git.proxmox.com Git - mirror_frr.git/blob - isisd/isis_routemap.c
release: FRR 3.0-rc1
[mirror_frr.git] / isisd / isis_routemap.c
1 /*
2 * IS-IS Rout(e)ing protocol - isis_routemap.c
3 *
4 * Copyright (C) 2013-2015 Christian Franke <chris@opensourcerouting.org>
5 *
6 * This program is free software; you can redistribute it and/or modify it
7 * under the terms of the GNU General Public License as published by the Free
8 * Software Foundation; either version 2 of the License, or (at your option)
9 * any later version.
10 *
11 * This program is distributed in the hope that it will be useful,but WITHOUT
12 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
13 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
14 * more details.
15
16 * You should have received a copy of the GNU General Public License along
17 * with this program; if not, write to the Free Software Foundation, Inc.,
18 * 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
19 */
20
21 #include <zebra.h>
22
23 #include "command.h"
24 #include "filter.h"
25 #include "hash.h"
26 #include "if.h"
27 #include "linklist.h"
28 #include "log.h"
29 #include "memory.h"
30 #include "prefix.h"
31 #include "plist.h"
32 #include "routemap.h"
33 #include "table.h"
34 #include "thread.h"
35 #include "vty.h"
36
37 #include "isis_constants.h"
38 #include "isis_common.h"
39 #include "isis_flags.h"
40 #include "dict.h"
41 #include "isisd.h"
42 #include "isis_misc.h"
43 #include "isis_adjacency.h"
44 #include "isis_circuit.h"
45 #include "isis_tlv.h"
46 #include "isis_pdu.h"
47 #include "isis_lsp.h"
48 #include "isis_spf.h"
49 #include "isis_route.h"
50 #include "isis_zebra.h"
51 #include "isis_routemap.h"
52
53 static route_map_result_t route_match_ip_address(void *rule,
54 struct prefix *prefix,
55 route_map_object_t type,
56 void *object)
57 {
58 struct access_list *alist;
59
60 if (type != RMAP_ISIS)
61 return RMAP_NOMATCH;
62
63 alist = access_list_lookup(AFI_IP, (char *)rule);
64 if (access_list_apply(alist, prefix) != FILTER_DENY)
65 return RMAP_MATCH;
66
67 return RMAP_NOMATCH;
68 }
69
70 static void *route_match_ip_address_compile(const char *arg)
71 {
72 return XSTRDUP(MTYPE_ROUTE_MAP_COMPILED, arg);
73 }
74
75 static void route_match_ip_address_free(void *rule)
76 {
77 XFREE(MTYPE_ROUTE_MAP_COMPILED, rule);
78 }
79
80 static struct route_map_rule_cmd route_match_ip_address_cmd = {
81 "ip address", route_match_ip_address, route_match_ip_address_compile,
82 route_match_ip_address_free};
83
84 /* ------------------------------------------------------------*/
85
86 static route_map_result_t
87 route_match_ip_address_prefix_list(void *rule, struct prefix *prefix,
88 route_map_object_t type, void *object)
89 {
90 struct prefix_list *plist;
91
92 if (type != RMAP_ISIS)
93 return RMAP_NOMATCH;
94
95 plist = prefix_list_lookup(AFI_IP, (char *)rule);
96 if (prefix_list_apply(plist, prefix) != PREFIX_DENY)
97 return RMAP_MATCH;
98
99 return RMAP_NOMATCH;
100 }
101
102 static void *route_match_ip_address_prefix_list_compile(const char *arg)
103 {
104 return XSTRDUP(MTYPE_ROUTE_MAP_COMPILED, arg);
105 }
106
107 static void route_match_ip_address_prefix_list_free(void *rule)
108 {
109 XFREE(MTYPE_ROUTE_MAP_COMPILED, rule);
110 }
111
112 struct route_map_rule_cmd route_match_ip_address_prefix_list_cmd = {
113 "ip address prefix-list", route_match_ip_address_prefix_list,
114 route_match_ip_address_prefix_list_compile,
115 route_match_ip_address_prefix_list_free};
116
117 /* ------------------------------------------------------------*/
118
119 static route_map_result_t route_match_ipv6_address(void *rule,
120 struct prefix *prefix,
121 route_map_object_t type,
122 void *object)
123 {
124 struct access_list *alist;
125
126 if (type != RMAP_ISIS)
127 return RMAP_NOMATCH;
128
129 alist = access_list_lookup(AFI_IP6, (char *)rule);
130 if (access_list_apply(alist, prefix) != FILTER_DENY)
131 return RMAP_MATCH;
132
133 return RMAP_NOMATCH;
134 }
135
136 static void *route_match_ipv6_address_compile(const char *arg)
137 {
138 return XSTRDUP(MTYPE_ROUTE_MAP_COMPILED, arg);
139 }
140
141 static void route_match_ipv6_address_free(void *rule)
142 {
143 XFREE(MTYPE_ROUTE_MAP_COMPILED, rule);
144 }
145
146 static struct route_map_rule_cmd route_match_ipv6_address_cmd = {
147 "ipv6 address", route_match_ipv6_address,
148 route_match_ipv6_address_compile, route_match_ipv6_address_free};
149
150 /* ------------------------------------------------------------*/
151
152 static route_map_result_t
153 route_match_ipv6_address_prefix_list(void *rule, struct prefix *prefix,
154 route_map_object_t type, void *object)
155 {
156 struct prefix_list *plist;
157
158 if (type != RMAP_ISIS)
159 return RMAP_NOMATCH;
160
161 plist = prefix_list_lookup(AFI_IP6, (char *)rule);
162 if (prefix_list_apply(plist, prefix) != PREFIX_DENY)
163 return RMAP_MATCH;
164
165 return RMAP_NOMATCH;
166 }
167
168 static void *route_match_ipv6_address_prefix_list_compile(const char *arg)
169 {
170 return XSTRDUP(MTYPE_ROUTE_MAP_COMPILED, arg);
171 }
172
173 static void route_match_ipv6_address_prefix_list_free(void *rule)
174 {
175 XFREE(MTYPE_ROUTE_MAP_COMPILED, rule);
176 }
177
178 struct route_map_rule_cmd route_match_ipv6_address_prefix_list_cmd = {
179 "ipv6 address prefix-list", route_match_ipv6_address_prefix_list,
180 route_match_ipv6_address_prefix_list_compile,
181 route_match_ipv6_address_prefix_list_free};
182
183 /* ------------------------------------------------------------*/
184
185 static route_map_result_t route_set_metric(void *rule, struct prefix *prefix,
186 route_map_object_t type,
187 void *object)
188 {
189 uint32_t *metric;
190 struct isis_ext_info *info;
191
192 if (type == RMAP_ISIS) {
193 metric = rule;
194 info = object;
195
196 info->metric = *metric;
197 }
198 return RMAP_OKAY;
199 }
200
201 static void *route_set_metric_compile(const char *arg)
202 {
203 unsigned long metric;
204 char *endp;
205 uint32_t *ret;
206
207 metric = strtoul(arg, &endp, 10);
208 if (arg[0] == '\0' || *endp != '\0' || metric > MAX_WIDE_PATH_METRIC)
209 return NULL;
210
211 ret = XCALLOC(MTYPE_ROUTE_MAP_COMPILED, sizeof(*ret));
212 *ret = metric;
213
214 return ret;
215 }
216
217 static void route_set_metric_free(void *rule)
218 {
219 XFREE(MTYPE_ROUTE_MAP_COMPILED, rule);
220 }
221
222 static struct route_map_rule_cmd route_set_metric_cmd = {
223 "metric", route_set_metric, route_set_metric_compile,
224 route_set_metric_free};
225
226 void isis_route_map_init(void)
227 {
228 route_map_init();
229
230 route_map_match_ip_address_hook(generic_match_add);
231 route_map_no_match_ip_address_hook(generic_match_delete);
232
233 route_map_match_ip_address_prefix_list_hook(generic_match_add);
234 route_map_no_match_ip_address_prefix_list_hook(generic_match_delete);
235
236 route_map_match_ipv6_address_hook(generic_match_add);
237 route_map_no_match_ipv6_address_hook(generic_match_delete);
238
239 route_map_match_ipv6_address_prefix_list_hook(generic_match_add);
240 route_map_no_match_ipv6_address_prefix_list_hook(generic_match_delete);
241
242 route_map_set_metric_hook(generic_set_add);
243 route_map_no_set_metric_hook(generic_set_delete);
244
245 route_map_install_match(&route_match_ip_address_cmd);
246 route_map_install_match(&route_match_ip_address_prefix_list_cmd);
247 route_map_install_match(&route_match_ipv6_address_cmd);
248 route_map_install_match(&route_match_ipv6_address_prefix_list_cmd);
249 route_map_install_set(&route_set_metric_cmd);
250 }