route.c revision 98c297f2448fd8194cae16d213dfc9194180fa3f
1 2#include <stdio.h> 3#include <stdlib.h> 4#include <unistd.h> 5#include <stdarg.h> 6 7#include <errno.h> 8#include <string.h> 9#include <ctype.h> 10 11#include <sys/socket.h> 12#include <netinet/in.h> 13#include <linux/if.h> 14#include <linux/sockios.h> 15#include <arpa/inet.h> 16#include <linux/route.h> 17 18static void die(const char *fmt, ...) 19{ 20 va_list p; 21 22 va_start(p, fmt); 23 fprintf(stderr,"error(%s): ", strerror(errno)); 24 fprintf(stderr, fmt, p); 25 va_end(p); 26 exit(-1); 27} 28 29static inline void init_sockaddr_in(struct sockaddr_in *sin, const char *addr) 30{ 31 sin->sin_family = AF_INET; 32 sin->sin_port = 0; 33 sin->sin_addr.s_addr = inet_addr(addr); 34} 35 36#define ADVANCE(argc, argv) do { argc--, argv++; } while(0) 37#define EXPECT_NEXT(argc, argv) do { \ 38 ADVANCE(argc, argv); \ 39 if (0 == argc) { \ 40 errno = EINVAL; \ 41 die("expecting one more argument"); \ 42 } \ 43} while(0) 44 45/* current support the following routing entries */ 46/* route add default dev wlan0 */ 47/* route add default gw 192.168.20.1 dev wlan0 */ 48/* route add net 192.168.1.1 netmask 255.255.255.0 gw 172.24.192.10 */ 49 50int route_main(int argc, char *argv[]) 51{ 52 struct ifreq ifr; 53 int s,i; 54 struct rtentry rt; 55 struct sockaddr_in ina; 56 57 if(argc == 0) return 0; 58 59 strncpy(ifr.ifr_name, argv[0], IFNAMSIZ); 60 ifr.ifr_name[IFNAMSIZ-1] = 0; 61 ADVANCE(argc, argv); 62 63 if((s = socket(AF_INET, SOCK_DGRAM, 0)) < 0) { 64 die("cannot open control socket\n"); 65 } 66 67 while(argc > 0){ 68 if(!strcmp(argv[0], "add")) { 69 EXPECT_NEXT(argc, argv); 70 if(!strcmp(argv[0], "default")) { 71 EXPECT_NEXT(argc, argv); 72 memset((char *) &rt, 0, sizeof(struct rtentry)); 73 rt.rt_dst.sa_family = AF_INET; 74 if(!strcmp(argv[0], "dev")) { 75 EXPECT_NEXT(argc, argv); 76 rt.rt_flags = RTF_UP | RTF_HOST; 77 rt.rt_dev = argv[0]; 78 if (ioctl(s, SIOCADDRT, &rt) < 0) die("SIOCADDRT\n"); 79 }else if(!strcmp(argv[0], "gw")) { 80 EXPECT_NEXT(argc, argv); 81 rt.rt_flags = RTF_UP | RTF_GATEWAY; 82 init_sockaddr_in((struct sockaddr_in *)&(rt.rt_genmask), "0.0.0.0"); 83 if(isdigit(argv[0][0])){ 84 init_sockaddr_in((struct sockaddr_in *)&(rt.rt_gateway), argv[0]); 85 }else{ 86 die("expecting an IP address for parameter \"gw\"\n"); 87 } 88 EXPECT_NEXT(argc, argv); 89 if(!strcmp(argv[0], "dev")) { 90 EXPECT_NEXT(argc, argv); 91 rt.rt_dev = argv[0]; 92 if (ioctl(s, SIOCADDRT, &rt) < 0){ 93 die("SIOCADDRT\n"); 94 } 95 } 96 } 97 } else { 98 char keywords[3][5] = { "net", "mask", "gw" }; 99 struct sockaddr_in *paddr[3] = { &rt.rt_dst, &rt.rt_genmask, &rt.rt_gateway }; 100 int k = 0; 101 102 memset((char *) &rt, 0, sizeof(struct rtentry)); 103 rt.rt_flags = RTF_UP | RTF_GATEWAY; 104 do { 105 if(!strcmp(argv[0], keywords[k])) { 106 EXPECT_NEXT(argc, argv); 107 if(isdigit(argv[0][0])) { 108 init_sockaddr_in(paddr[k], argv[0]); 109 } else { 110 die("expecting an IP/MASK address for parameter %s\n", keywords[k]); 111 } 112 if(k < 2) EXPECT_NEXT(argc, argv); 113 } else { 114 die("expecting keyword(s)\n"); 115 } 116 } while(++k < 3); 117 if(ioctl(s, SIOCADDRT, &rt) < 0) { 118 die("SIOCADDRT\n"); 119 } 120 } 121 } 122 ADVANCE(argc, argv); 123 } 124 125 return 0; 126} 127