e182920032d8d65cd47015a87a35ecbbddb801dc
[olsrd.git] / src / linux / kernel_routes_ioctl.c
1 /*
2  * The olsr.org Optimized Link-State Routing daemon(olsrd)
3  * Copyright (c) 2004, Andreas Tonnesen(andreto@olsr.org)
4  * Copyright (c) 2007, Sven-Ola for the policy routing stuff
5  * All rights reserved.
6  *
7  * Redistribution and use in source and binary forms, with or without
8  * modification, are permitted provided that the following conditions
9  * are met:
10  *
11  * * Redistributions of source code must retain the above copyright
12  *   notice, this list of conditions and the following disclaimer.
13  * * Redistributions in binary form must reproduce the above copyright
14  *   notice, this list of conditions and the following disclaimer in
15  *   the documentation and/or other materials provided with the
16  *   distribution.
17  * * Neither the name of olsr.org, olsrd nor the names of its
18  *   contributors may be used to endorse or promote products derived
19  *   from this software without specific prior written permission.
20  *
21  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
22  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
23  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
24  * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
25  * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
26  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
27  * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
28  * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
29  * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
30  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
31  * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
32  * POSSIBILITY OF SUCH DAMAGE.
33  *
34  * Visit http://www.olsr.org for more information.
35  *
36  * If you find this software useful feel free to make a donation
37  * to the project. For more information see the website or contact
38  * the copyright holders.
39  *
40  */
41
42 #include "defs.h"
43 #include "kernel_routes.h"
44 #include "ipc_frontend.h"
45 #include "log.h"
46 #include "net_os.h"
47 #include "process_routes.h"
48
49 #include <assert.h>
50 #include <linux/types.h>
51 #include <linux/rtnetlink.h>
52
53 /*
54  * This file contains the old ioctl version of the linux routing code.
55  * You will find the current netlink version in kernel_routes_nl.c
56  *
57  * You can deactivate this code (and activating the netlink one)
58  * by adding a -DLINUX_NETLINK_ROUTING to CPPFLAGS in make/Makefile.linux
59  */
60 #ifndef LINUX_NETLINK_ROUTING
61
62 static int
63 delete_all_inet_gws(void)
64 {
65   int s;
66   char buf[BUFSIZ], *cp, *cplim;
67   struct ifconf ifc;
68   struct ifreq *ifr;
69
70   OLSR_PRINTF(1, "Internet gateway detected...\nTrying to delete default gateways\n");
71
72   /* Get a socket */
73   if ((s = socket(AF_INET, SOCK_DGRAM, 0)) < 0) {
74     olsr_syslog(OLSR_LOG_ERR, "socket: %m");
75     return -1;
76   }
77
78   ifc.ifc_len = sizeof(buf);
79   ifc.ifc_buf = buf;
80   if (ioctl(s, SIOCGIFCONF, (char *)&ifc) < 0) {
81     olsr_syslog(OLSR_LOG_ERR, "ioctl (get interface configuration)");
82     close(s);
83     return -1;
84   }
85
86   ifr = ifc.ifc_req;
87   cplim = buf + ifc.ifc_len;    /*skip over if's with big ifr_addr's */
88   for (cp = buf; cp < cplim; cp += sizeof(ifr->ifr_name) + sizeof(ifr->ifr_addr)) {
89     struct rtentry kernel_route;
90     ifr = (struct ifreq *)cp;
91
92     if (strcmp(ifr->ifr_ifrn.ifrn_name, "lo") == 0) {
93       OLSR_PRINTF(1, "Skipping loopback...\n");
94       continue;
95     }
96
97     OLSR_PRINTF(1, "Trying 0.0.0.0/0 %s...", ifr->ifr_ifrn.ifrn_name);
98
99     memset(&kernel_route, 0, sizeof(struct rtentry));
100
101     ((struct sockaddr_in *)&kernel_route.rt_dst)->sin_addr.s_addr = 0;
102     ((struct sockaddr_in *)&kernel_route.rt_dst)->sin_family = AF_INET;
103     ((struct sockaddr_in *)&kernel_route.rt_genmask)->sin_addr.s_addr = 0;
104     ((struct sockaddr_in *)&kernel_route.rt_genmask)->sin_family = AF_INET;
105
106     ((struct sockaddr_in *)&kernel_route.rt_gateway)->sin_addr.s_addr = INADDR_ANY;
107     ((struct sockaddr_in *)&kernel_route.rt_gateway)->sin_family = AF_INET;
108
109     kernel_route.rt_flags = RTF_UP | RTF_GATEWAY;
110
111     kernel_route.rt_dev = ifr->ifr_ifrn.ifrn_name;
112
113     if ((ioctl(s, SIOCDELRT, &kernel_route)) < 0)
114       OLSR_PRINTF(1, "NO\n");
115     else
116       OLSR_PRINTF(1, "YES\n");
117   }
118   close(s);
119   return 0;
120 }
121
122 /**
123  * Insert a route in the kernel routing table
124  *
125  * @param destination the route to add
126  *
127  * @return negative on error
128  */
129 int
130 olsr_ioctl_add_route(const struct rt_entry *rt)
131 {
132   char if_name[IFNAMSIZ];
133   struct rtentry kernel_route;
134   union olsr_ip_addr mask;
135   int rslt;
136
137   OLSR_PRINTF(2, "KERN: Adding %s\n", olsr_rtp_to_string(rt->rt_best));
138
139   memset(&kernel_route, 0, sizeof(struct rtentry));
140
141   ((struct sockaddr_in *)&kernel_route.rt_dst)->sin_family = AF_INET;
142   ((struct sockaddr_in *)&kernel_route.rt_gateway)->sin_family = AF_INET;
143   ((struct sockaddr_in *)&kernel_route.rt_genmask)->sin_family = AF_INET;
144
145   ((struct sockaddr_in *)&kernel_route.rt_dst)->sin_addr = rt->rt_dst.prefix.v4;
146
147   if (!olsr_prefix_to_netmask(&mask, rt->rt_dst.prefix_len)) {
148     return -1;
149   }
150   ((struct sockaddr_in *)&kernel_route.rt_genmask)->sin_addr = mask.v4;
151
152   if (rt->rt_dst.prefix.v4.s_addr != rt->rt_best->rtp_nexthop.gateway.v4.s_addr) {
153     ((struct sockaddr_in *)&kernel_route.rt_gateway)->sin_addr = rt->rt_best->rtp_nexthop.gateway.v4;
154   }
155
156   kernel_route.rt_flags = olsr_rt_flags(rt);
157   kernel_route.rt_metric = olsr_fib_metric(&rt->rt_best->rtp_metric);
158
159   /*
160    * Set interface
161    */
162   strcpy(if_name, if_ifwithindex_name(rt->rt_best->rtp_nexthop.iif_index));
163   kernel_route.rt_dev = if_name;
164
165   /* delete existing default route before ? */
166   if ((olsr_cnf->del_gws) && (rt->rt_dst.prefix.v4.s_addr == INADDR_ANY) && (rt->rt_dst.prefix_len == INADDR_ANY)) {
167     delete_all_inet_gws();
168     olsr_cnf->del_gws = false;
169   }
170
171   if ((rslt = ioctl(olsr_cnf->ioctl_s, SIOCADDRT, &kernel_route)) >= 0) {
172
173     /*
174      * Send IPC route update message
175      */
176     ipc_route_send_rtentry(&rt->rt_dst.prefix, &rt->rt_best->rtp_nexthop.gateway, rt->rt_best->rtp_metric.hops, 1,
177                            if_ifwithindex_name(rt->rt_best->rtp_nexthop.iif_index));
178   }
179
180   return rslt;
181 }
182
183 /**
184  *Insert a route in the kernel routing table
185  *
186  *@param destination the route to add
187  *
188  *@return negative on error
189  */
190 int
191 olsr_ioctl_add_route6(const struct rt_entry *rt)
192 {
193   struct in6_rtmsg kernel_route;
194   int rslt;
195
196   OLSR_PRINTF(2, "KERN: Adding %s\n", olsr_rtp_to_string(rt->rt_best));
197
198   memset(&kernel_route, 0, sizeof(struct in6_rtmsg));
199
200   kernel_route.rtmsg_dst = rt->rt_dst.prefix.v6;
201   kernel_route.rtmsg_dst_len = rt->rt_dst.prefix_len;
202
203   kernel_route.rtmsg_gateway = rt->rt_best->rtp_nexthop.gateway.v6;
204
205   kernel_route.rtmsg_flags = olsr_rt_flags(rt);
206   kernel_route.rtmsg_metric = olsr_fib_metric(&rt->rt_best->rtp_metric);
207
208   /*
209    * set interface
210    */
211   kernel_route.rtmsg_ifindex = rt->rt_best->rtp_nexthop.iif_index;
212
213   /* XXX delete 0/0 route before ? */
214
215   if ((rslt = ioctl(olsr_cnf->ioctl_s, SIOCADDRT, &kernel_route)) >= 0) {
216
217     /*
218      * Send IPC route update message
219      */
220     ipc_route_send_rtentry(&rt->rt_dst.prefix, &rt->rt_best->rtp_nexthop.gateway, rt->rt_best->rtp_metric.hops, 1,
221                            if_ifwithindex_name(rt->rt_best->rtp_nexthop.iif_index));
222   }
223   return rslt;
224 }
225
226 /**
227  *Remove a route from the kernel
228  *
229  *@param destination the route to remove
230  *
231  *@return negative on error
232  */
233 int
234 olsr_ioctl_del_route(const struct rt_entry *rt)
235 {
236   struct rtentry kernel_route;
237   union olsr_ip_addr mask;
238   int rslt;
239
240   OLSR_PRINTF(2, "KERN: Deleting %s\n", olsr_rt_to_string(rt));
241
242   memset(&kernel_route, 0, sizeof(struct rtentry));
243
244   ((struct sockaddr_in *)&kernel_route.rt_dst)->sin_family = AF_INET;
245   ((struct sockaddr_in *)&kernel_route.rt_gateway)->sin_family = AF_INET;
246   ((struct sockaddr_in *)&kernel_route.rt_genmask)->sin_family = AF_INET;
247
248   ((struct sockaddr_in *)&kernel_route.rt_dst)->sin_addr = rt->rt_dst.prefix.v4;
249
250   if (rt->rt_dst.prefix.v4.s_addr != rt->rt_nexthop.gateway.v4.s_addr) {
251     ((struct sockaddr_in *)&kernel_route.rt_gateway)->sin_addr = rt->rt_nexthop.gateway.v4;
252   }
253
254   if (!olsr_prefix_to_netmask(&mask, rt->rt_dst.prefix_len)) {
255     return -1;
256   } else {
257     ((struct sockaddr_in *)&kernel_route.rt_genmask)->sin_addr = mask.v4;
258   }
259
260   kernel_route.rt_flags = olsr_rt_flags(rt);
261   kernel_route.rt_metric = olsr_fib_metric(&rt->rt_metric);
262
263   /*
264    * Set interface
265    */
266   kernel_route.rt_dev = NULL;
267
268   if ((rslt = ioctl(olsr_cnf->ioctl_s, SIOCDELRT, &kernel_route)) >= 0) {
269
270     /*
271      * Send IPC route update message
272      */
273     ipc_route_send_rtentry(&rt->rt_dst.prefix, NULL, 0, 0, NULL);
274   }
275
276   return rslt;
277 }
278
279 /**
280  *Remove a route from the kernel
281  *
282  *@param destination the route to remove
283  *
284  *@return negative on error
285  */
286 int
287 olsr_ioctl_del_route6(const struct rt_entry *rt)
288 {
289   struct in6_rtmsg kernel_route;
290   int rslt;
291
292   OLSR_PRINTF(2, "KERN: Deleting %s\n", olsr_rt_to_string(rt));
293
294   memset(&kernel_route, 0, sizeof(struct in6_rtmsg));
295
296   kernel_route.rtmsg_dst = rt->rt_dst.prefix.v6;
297   kernel_route.rtmsg_dst_len = rt->rt_dst.prefix_len;
298
299   kernel_route.rtmsg_gateway = rt->rt_best->rtp_nexthop.gateway.v6;
300
301   kernel_route.rtmsg_flags = olsr_rt_flags(rt);
302   kernel_route.rtmsg_metric = olsr_fib_metric(&rt->rt_best->rtp_metric);
303
304   if ((rslt = ioctl(olsr_cnf->ioctl_s, SIOCDELRT, &kernel_route) >= 0)) {
305
306     /*
307      * Send IPC route update message
308      */
309     ipc_route_send_rtentry(&rt->rt_dst.prefix, NULL, 0, 0, NULL);
310   }
311
312   return rslt;
313 }
314
315 #endif /* LINUX_NETLINK_ROUTING */