Allow gateways to be specified for routes.
[tinc] / src / ifconfig.c
1 /*
2     ifconfig.c -- Generate platform specific interface configuration commands
3     Copyright (C) 2016 Guus Sliepen <guus@tinc-vpn.org>
4
5     This program is free software; you can redistribute it and/or modify
6     it under the terms of the GNU General Public License as published by
7     the Free Software Foundation; either version 2 of the License, or
8     (at your option) any later version.
9
10     This program is distributed in the hope that it will be useful,
11     but WITHOUT ANY WARRANTY; without even the implied warranty of
12     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
13     GNU General Public License for more details.
14
15     You should have received a copy of the GNU General Public License along
16     with this program; if not, write to the Free Software Foundation, Inc.,
17     51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
18 */
19
20 #include "system.h"
21
22 #include "conf.h"
23 #include "ifconfig.h"
24 #include "subnet.h"
25
26 static long start;
27
28 #ifndef HAVE_MINGW
29 void ifconfig_header(FILE *out) {
30         fprintf(out, "#!/bin/sh\n");
31         start = ftell(out);
32 }
33
34 void ifconfig_dhcp(FILE *out) {
35         fprintf(out, "dhclient -nw \"$INTERFACE\"\n");
36 }
37
38 void ifconfig_dhcp6(FILE *out) {
39         fprintf(out, "dhclient -6 -nw \"$INTERFACE\"\n");
40 }
41
42 void ifconfig_slaac(FILE *out) {
43 #ifdef HAVE_LINUX
44         fprintf(out, "echo 1 >\"/proc/sys/net/ipv6/conf/$INTERFACE/accept_ra\"\n");
45         fprintf(out, "echo 1 >\"/proc/sys/net/ipv6/conf/$INTERFACE/autoconf\"\n");
46 #else
47         fprintf(out, "rtsol \"$INTERFACE\" &\n");
48 #endif
49 }
50
51 bool ifconfig_footer(FILE *out) {
52         if(ftell(out) == start) {
53                 fprintf(out, "echo 'Unconfigured tinc-up script, please edit '$0'!'\n\n#ifconfig $INTERFACE <your vpn IP address> netmask <netmask of whole VPN>\n");
54                 return false;
55         } else {
56 #ifdef HAVE_LINUX
57                 fprintf(out, "ip link set \"$INTERFACE\" up\n");
58 #else
59                 fprintf(out, "ifconfig \"$INTERFACE\" up\n");
60 #endif
61                 return true;
62         }
63 }
64 #else
65 void ifconfig_header(FILE *out) {
66         start = ftell(out);
67 }
68
69 void ifconfig_dhcp(FILE *out) {
70         fprintf(out, "netsh interface ipv4 set address \"%%INTERFACE%%\" dhcp\n");
71 }
72
73 void ifconfig_dhcp6(FILE *out) {
74         fprintf(stderr, "DHCPv6 requested, but not supported by tinc on this platform\n");
75 }
76
77 void ifconfig_slaac(FILE *out) {
78         // It's the default?
79 }
80
81 bool ifconfig_footer(FILE *out) {
82         return ftell(out) != start;
83 }
84 #endif
85
86 static subnet_t ipv4, ipv6;
87
88 void ifconfig_address(FILE *out, const char *value) {
89         subnet_t address = {};
90         char address_str[MAXNETSTR];
91         if(!str2net(&address, value) || !net2str(address_str, sizeof address_str, &address)) {
92                 fprintf(stderr, "Could not parse Ifconfig statement\n");
93                 return;
94         }
95         switch(address.type) {
96                 case SUBNET_IPV4: ipv4 = address; break;
97                 case SUBNET_IPV6: ipv6 = address; break;
98         }
99 #if defined(HAVE_LINUX)
100         switch(address.type) {
101                 case SUBNET_MAC:  fprintf(out, "ip link set \"$INTERFACE\" address %s\n", address_str); break;
102                 case SUBNET_IPV4: fprintf(out, "ip addr replace %s dev \"$INTERFACE\"\n", address_str); break;
103                 case SUBNET_IPV6: fprintf(out, "ip addr replace %s dev \"$INTERFACE\"\n", address_str); break;
104         }
105 #elif defined(HAVE_BSD)
106         switch(address.type) {
107                 case SUBNET_MAC:  fprintf(out, "ifconfig \"$INTERFACE\" link %s\n", address_str); break;
108                 case SUBNET_IPV4: fprintf(out, "ifconfig \"$INTERFACE\" %s\n", address_str); break;
109                 case SUBNET_IPV6: fprintf(out, "ifconfig \"$INTERFACE\" inet6 %s\n", address_str); break;
110         }
111 #elif defined(HAVE_MINGW) || defined(HAVE_CYGWIN)
112         switch(address.type) {
113                 case SUBNET_MAC:  fprintf(out, "ip link set \"$INTERFACE\" address %s\n", address_str); break;
114                 case SUBNET_IPV4: fprintf(out, "netsh inetface ipv4 set address \"$INTERFACE\" static %s\n", address_str); break;
115                 case SUBNET_IPV6: fprintf(out, "netsh inetface ipv6 set address \"$INTERFACE\" static %s\n", address_str); break;
116         }
117 #endif
118 }
119
120 void ifconfig_route(FILE *out, const char *value) {
121         subnet_t subnet = {}, gateway = {};
122         char subnet_str[MAXNETSTR] = "", gateway_str[MAXNETSTR] = "";
123         const char *sep = strchr(value, ' ');
124         if(!str2net(&subnet, value) || !net2str(subnet_str, sizeof subnet_str, &subnet) || subnet.type == SUBNET_MAC) {
125                 fprintf(stderr, "Could not parse Ifconfig statement\n");
126                 return;
127         }
128         if(sep) {
129                 sep++;
130                 if(!str2net(&gateway, sep) || !net2str(gateway_str, sizeof gateway_str, &gateway) || gateway.type != subnet.type) {
131                         fprintf(stderr, "Could not parse Ifconfig statement\n");
132                         return;
133                 }
134                 char *slash = strchr(gateway_str, '/'); if(slash) *slash = 0;
135         }
136 #if defined(HAVE_LINUX)
137         if(*gateway_str) {
138                 switch(subnet.type) {
139                         case SUBNET_IPV4: fprintf(out, "ip route add %s via %s dev \"$INTERFACE\"\n", subnet_str, gateway_str); break;
140                         case SUBNET_IPV6: fprintf(out, "ip route add %s via %s dev \"$INTERFACE\"\n", subnet_str, gateway_str); break;
141                 }
142         } else {
143                 switch(subnet.type) {
144                         case SUBNET_IPV4: fprintf(out, "ip route add %s dev \"$INTERFACE\"\n", subnet_str); break;
145                         case SUBNET_IPV6: fprintf(out, "ip route add %s dev \"$INTERFACE\"\n", subnet_str); break;
146                 }
147         }
148 #elif defined(HAVE_BSD)
149         // BSD route command is silly and doesn't accept an interface name as a destination.
150         if(!*gateway_str) {
151                 switch(subnet.type) {
152                         case SUBNET_IPV4:
153                                 if(!ipv4.type) {
154                                         fprintf(stderr, "Route requested but no Ifconfig\n");
155                                         return;
156                                 }
157                                 net2str(gateway_str, sizeof gateway_str, &ipv4);
158                                 break;
159                         case SUBNET_IPV6:
160                                 if(!ipv6.type) {
161                                         fprintf(stderr, "Route requested but no Ifconfig\n");
162                                         return;
163                                 }
164                                 net2str(gateway_str, sizeof gateway_str, &ipv6);
165                                 break;
166                 }
167                 char *slash = strchr(gateway_str, '/'); if(slash) *slash = 0;
168         }
169
170         switch(subnet.type) {
171                 case SUBNET_IPV4: fprintf(out, "route add %s %s\n", subnet_str, gateway_str); break;
172                 case SUBNET_IPV6: fprintf(out, "route add -inet6 %s %s\n", subnet_str, gateway_str); break;
173         }
174 #elif defined(HAVE_MINGW) || defined(HAVE_CYGWIN)
175         if(*gateway_str) {
176                 switch(subnet.type) {
177                         case SUBNET_IPV4: fprintf(out, "netsh inetface ipv4 add route %s \"%%INTERFACE%%\" %s\n", subnet_str, gateway_str); break;
178                         case SUBNET_IPV6: fprintf(out, "netsh inetface ipv6 add route %s \"%%INTERFACE%%\" %s\n", subnet_str, gateway_str); break;
179                 }
180         } else {
181                 switch(subnet.type) {
182                         case SUBNET_IPV4: fprintf(out, "netsh inetface ipv4 add route %s \"%%INTERFACE%%\"\n", subnet_str); break;
183                         case SUBNET_IPV6: fprintf(out, "netsh inetface ipv6 add route %s \"%%INTERFACE%%\"\n", subnet_str); break;
184                 }
185         }
186 #endif
187 }