| 
									
										
										
										
											2012-06-29 21:11:24 +00:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Copyright (c) 2012 Ed Maste. All rights reserved. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Licensed under the Apache License, Version 2.0 (the "License"); | 
					
						
							|  |  |  |  * you may not use this file except in compliance with the License. | 
					
						
							|  |  |  |  * You may obtain a copy of the License at: | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *     http://www.apache.org/licenses/LICENSE-2.0
 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Unless required by applicable law or agreed to in writing, software | 
					
						
							|  |  |  |  * distributed under the License is distributed on an "AS IS" BASIS, | 
					
						
							|  |  |  |  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
					
						
							|  |  |  |  * See the License for the specific language governing permissions and | 
					
						
							|  |  |  |  * limitations under the License. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <config.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "route-table.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <sys/socket.h>
 | 
					
						
							|  |  |  | #include <sys/types.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <net/if.h>
 | 
					
						
							|  |  |  | #include <net/route.h>
 | 
					
						
							|  |  |  | #include <net/if_dl.h>
 | 
					
						
							|  |  |  | #include <netinet/in.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <string.h>
 | 
					
						
							|  |  |  | #include <unistd.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-04-30 13:34:55 +09:00
										 |  |  | #include "util.h"
 | 
					
						
							| 
									
										
										
										
											2012-06-29 21:11:24 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static int pid; | 
					
						
							|  |  |  | static unsigned int register_count = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool | 
					
						
							| 
									
										
										
										
											2014-10-16 11:38:12 -07:00
										 |  |  | ovs_router_lookup(ovs_be32 ip, char name[], ovs_be32 *gw) | 
					
						
							| 
									
										
										
										
											2012-06-29 21:11:24 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     struct { | 
					
						
							|  |  |  |         struct rt_msghdr rtm; | 
					
						
							|  |  |  |         char space[512]; | 
					
						
							|  |  |  |     } rtmsg; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     struct rt_msghdr *rtm = &rtmsg.rtm; | 
					
						
							|  |  |  |     struct sockaddr_dl *ifp = NULL; | 
					
						
							|  |  |  |     struct sockaddr_in *sin; | 
					
						
							|  |  |  |     struct sockaddr *sa; | 
					
						
							|  |  |  |     static int seq; | 
					
						
							|  |  |  |     int i, len, namelen, rtsock; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     rtsock = socket(PF_ROUTE, SOCK_RAW, 0); | 
					
						
							|  |  |  |     if (rtsock < 0) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     memset(&rtmsg, 0, sizeof(rtmsg)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     rtm->rtm_msglen = sizeof(struct rt_msghdr) + sizeof(struct sockaddr_in); | 
					
						
							|  |  |  |     rtm->rtm_version = RTM_VERSION; | 
					
						
							|  |  |  |     rtm->rtm_type = RTM_GET; | 
					
						
							|  |  |  |     rtm->rtm_addrs = RTA_DST | RTA_IFP; | 
					
						
							|  |  |  |     rtm->rtm_seq = ++seq; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     sin = (struct sockaddr_in *)(rtm + 1); | 
					
						
							|  |  |  |     sin->sin_len = len = sizeof(struct sockaddr_in); | 
					
						
							|  |  |  |     sin->sin_family = AF_INET; | 
					
						
							|  |  |  |     sin->sin_addr.s_addr = ip; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if ((write(rtsock, (char *)&rtmsg, rtm->rtm_msglen)) < 0) { | 
					
						
							|  |  |  |         close(rtsock); | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     do { | 
					
						
							|  |  |  |         len = read(rtsock, (char *)&rtmsg, sizeof(rtmsg)); | 
					
						
							|  |  |  |     } while (len > 0 && (rtmsg.rtm.rtm_seq != seq || | 
					
						
							|  |  |  |         rtmsg.rtm.rtm_pid != pid)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     close(rtsock); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (len < 0) { | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     sa = (struct sockaddr *)(rtm + 1); | 
					
						
							|  |  |  |     for (i = 1; i; i <<= 1) { | 
					
						
							|  |  |  |         if (rtm->rtm_addrs & i) { | 
					
						
							|  |  |  |             if (i == RTA_IFP && sa->sa_family == AF_LINK && | 
					
						
							| 
									
										
										
										
											2014-04-30 13:34:55 +09:00
										 |  |  |               ALIGNED_CAST(struct sockaddr_dl *, sa)->sdl_nlen) { | 
					
						
							|  |  |  |                 ifp = ALIGNED_CAST(struct sockaddr_dl *, sa); | 
					
						
							| 
									
										
										
										
											2012-06-29 21:11:24 +00:00
										 |  |  |                 namelen = ifp->sdl_nlen; | 
					
						
							|  |  |  |                 if (namelen > IFNAMSIZ - 1) | 
					
						
							|  |  |  |                     namelen = IFNAMSIZ - 1; | 
					
						
							|  |  |  |                 memcpy(name, ifp->sdl_data, namelen); | 
					
						
							|  |  |  |                 name[namelen] = '\0'; | 
					
						
							| 
									
										
										
										
											2014-10-16 11:38:12 -07:00
										 |  |  |                 *gw = 0; | 
					
						
							| 
									
										
										
										
											2012-06-29 21:11:24 +00:00
										 |  |  |                 return true; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2013-04-22 22:20:12 +09:00
										 |  |  | #if defined(__FreeBSD__)
 | 
					
						
							| 
									
										
										
										
											2012-06-29 21:11:24 +00:00
										 |  |  |             sa = (struct sockaddr *)((char *)sa + SA_SIZE(sa)); | 
					
						
							| 
									
										
										
										
											2013-04-22 22:20:12 +09:00
										 |  |  | #elif defined(__NetBSD__)
 | 
					
						
							|  |  |  |             sa = (struct sockaddr *)((char *)sa + RT_ROUNDUP(sa->sa_len)); | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | #error unimplemented
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2012-06-29 21:11:24 +00:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-01 10:53:48 -07:00
										 |  |  | uint64_t | 
					
						
							|  |  |  | route_table_get_change_seq(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-29 21:11:24 +00:00
										 |  |  | void | 
					
						
							| 
									
										
										
										
											2012-08-08 18:29:32 +00:00
										 |  |  | route_table_register(void) | 
					
						
							| 
									
										
										
										
											2012-06-29 21:11:24 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     if (!register_count) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         pid = getpid(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     register_count++; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void | 
					
						
							| 
									
										
										
										
											2012-08-08 18:29:32 +00:00
										 |  |  | route_table_unregister(void) | 
					
						
							| 
									
										
										
										
											2012-06-29 21:11:24 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     register_count--; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void | 
					
						
							|  |  |  | route_table_run(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void | 
					
						
							|  |  |  | route_table_wait(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } |