The Android Open Source Project | dd7bc33 | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 1 | |
| 2 | #include <stdio.h> |
| 3 | #include <stdlib.h> |
| 4 | #include <unistd.h> |
Chung-yih Wang | 98c297f | 2009-04-14 18:45:32 -0700 | [diff] [blame] | 5 | #include <stdarg.h> |
The Android Open Source Project | dd7bc33 | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 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 | |
Chung-yih Wang | 98c297f | 2009-04-14 18:45:32 -0700 | [diff] [blame] | 18 | static void die(const char *fmt, ...) |
The Android Open Source Project | dd7bc33 | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 19 | { |
Chung-yih Wang | 98c297f | 2009-04-14 18:45:32 -0700 | [diff] [blame] | 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); |
The Android Open Source Project | dd7bc33 | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 26 | exit(-1); |
| 27 | } |
| 28 | |
| 29 | static 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 | |
Chung-yih Wang | 98c297f | 2009-04-14 18:45:32 -0700 | [diff] [blame] | 45 | /* current support the following routing entries */ |
The Android Open Source Project | dd7bc33 | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 46 | /* route add default dev wlan0 */ |
| 47 | /* route add default gw 192.168.20.1 dev wlan0 */ |
Chung-yih Wang | 98c297f | 2009-04-14 18:45:32 -0700 | [diff] [blame] | 48 | /* route add net 192.168.1.1 netmask 255.255.255.0 gw 172.24.192.10 */ |
The Android Open Source Project | dd7bc33 | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 49 | |
| 50 | int route_main(int argc, char *argv[]) |
| 51 | { |
| 52 | struct ifreq ifr; |
| 53 | int s,i; |
San Mehat | 83ec181 | 2009-05-13 17:18:27 -0700 | [diff] [blame^] | 54 | struct rtentry rt; |
| 55 | struct sockaddr_in ina; |
The Android Open Source Project | dd7bc33 | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 56 | |
San Mehat | 83ec181 | 2009-05-13 17:18:27 -0700 | [diff] [blame^] | 57 | if (!argc) |
| 58 | return 0; |
| 59 | |
The Android Open Source Project | dd7bc33 | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 60 | strncpy(ifr.ifr_name, argv[0], IFNAMSIZ); |
| 61 | ifr.ifr_name[IFNAMSIZ-1] = 0; |
San Mehat | 83ec181 | 2009-05-13 17:18:27 -0700 | [diff] [blame^] | 62 | ADVANCE(argc, argv); |
The Android Open Source Project | dd7bc33 | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 63 | |
San Mehat | 83ec181 | 2009-05-13 17:18:27 -0700 | [diff] [blame^] | 64 | if ((s = socket(AF_INET, SOCK_DGRAM, 0)) < 0) { |
The Android Open Source Project | dd7bc33 | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 65 | die("cannot open control socket\n"); |
| 66 | } |
| 67 | |
| 68 | while(argc > 0){ |
San Mehat | 83ec181 | 2009-05-13 17:18:27 -0700 | [diff] [blame^] | 69 | if (!strcmp(argv[0], "add")) { |
| 70 | EXPECT_NEXT(argc, argv); |
| 71 | if (!strcmp(argv[0], "default")) { |
| 72 | EXPECT_NEXT(argc, argv); |
| 73 | memset((char *) &rt, 0, sizeof(struct rtentry)); |
| 74 | rt.rt_dst.sa_family = AF_INET; |
| 75 | if(!strcmp(argv[0], "dev")) { |
| 76 | EXPECT_NEXT(argc, argv); |
| 77 | rt.rt_flags = RTF_UP | RTF_HOST; |
| 78 | rt.rt_dev = argv[0]; |
| 79 | if (ioctl(s, SIOCADDRT, &rt) < 0) |
| 80 | die("SIOCADDRT\n"); |
| 81 | } else if (!strcmp(argv[0], "gw")) { |
| 82 | EXPECT_NEXT(argc, argv); |
| 83 | rt.rt_flags = RTF_UP | RTF_GATEWAY; |
| 84 | init_sockaddr_in((struct sockaddr_in *)&(rt.rt_genmask), "0.0.0.0"); |
| 85 | if(isdigit(argv[0][0])) { |
| 86 | init_sockaddr_in((struct sockaddr_in *)&(rt.rt_gateway), argv[0]); |
| 87 | } else { |
| 88 | die("expecting an IP address for parameter \"gw\"\n"); |
| 89 | } |
| 90 | EXPECT_NEXT(argc, argv); |
| 91 | if (!strcmp(argv[0], "dev")) { |
| 92 | EXPECT_NEXT(argc, argv); |
| 93 | rt.rt_dev = argv[0]; |
| 94 | if (ioctl(s, SIOCADDRT, &rt) < 0) { |
Chung-yih Wang | 98c297f | 2009-04-14 18:45:32 -0700 | [diff] [blame] | 95 | die("SIOCADDRT\n"); |
Chung-yih Wang | 98c297f | 2009-04-14 18:45:32 -0700 | [diff] [blame] | 96 | } |
San Mehat | 83ec181 | 2009-05-13 17:18:27 -0700 | [diff] [blame^] | 97 | } |
| 98 | } |
| 99 | } else { |
| 100 | char keywords[3][10] = { "-net", "netmask", "gw" }; |
| 101 | struct sockaddr_in *paddr[3] = { &rt.rt_dst, &rt.rt_genmask, &rt.rt_gateway }; |
| 102 | int k = 0; |
| 103 | |
| 104 | memset((char *) &rt, 0, sizeof(struct rtentry)); |
| 105 | rt.rt_flags = RTF_UP | RTF_GATEWAY; |
| 106 | do { |
| 107 | if (!strcmp(argv[0], keywords[k])) { |
| 108 | EXPECT_NEXT(argc, argv); |
| 109 | if (isdigit(argv[0][0])) { |
| 110 | init_sockaddr_in(paddr[k], argv[0]); |
| 111 | } else { |
| 112 | die("expecting an IP/MASK address for parameter %s\n", keywords[k]); |
| 113 | } |
| 114 | if (k < 2) |
| 115 | EXPECT_NEXT(argc, argv); |
| 116 | } else { |
| 117 | die("expecting keyword(s)\n"); |
| 118 | } |
| 119 | } while (++k < 3); |
| 120 | |
| 121 | if (ioctl(s, SIOCADDRT, &rt) < 0) { |
| 122 | die("SIOCADDRT\n"); |
| 123 | } |
| 124 | } |
The Android Open Source Project | dd7bc33 | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 125 | } |
San Mehat | 83ec181 | 2009-05-13 17:18:27 -0700 | [diff] [blame^] | 126 | ADVANCE(argc, argv); |
The Android Open Source Project | dd7bc33 | 2009-03-03 19:32:55 -0800 | [diff] [blame] | 127 | } |
| 128 | |
| 129 | return 0; |
| 130 | } |