953499af63f534b66d5d2ae869d9a996d9d50935
[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         char *sep = strchr(value, ' ');
124         if(sep)
125                 *sep++ = 0;
126         if(!str2net(&subnet, value) || !net2str(subnet_str, sizeof subnet_str, &subnet) || subnet.type == SUBNET_MAC) {
127                 fprintf(stderr, "Could not parse subnet in Route statement\n");
128                 return;
129         }
130         if(sep) {
131                 if(!str2net(&gateway, sep) || !net2str(gateway_str, sizeof gateway_str, &gateway) || gateway.type != subnet.type) {
132                         fprintf(stderr, "Could not parse gateway in Route statement\n");
133                         return;
134                 }
135                 char *slash = strchr(gateway_str, '/'); if(slash) *slash = 0;
136         }
137 #if defined(HAVE_LINUX)
138         if(*gateway_str) {
139                 switch(subnet.type) {
140                         case SUBNET_IPV4: fprintf(out, "ip route add %s via %s dev \"$INTERFACE\"\n", subnet_str, gateway_str); break;
141                         case SUBNET_IPV6: fprintf(out, "ip route add %s via %s dev \"$INTERFACE\"\n", subnet_str, gateway_str); break;
142                 }
143         } else {
144                 switch(subnet.type) {
145                         case SUBNET_IPV4: fprintf(out, "ip route add %s dev \"$INTERFACE\"\n", subnet_str); break;
146                         case SUBNET_IPV6: fprintf(out, "ip route add %s dev \"$INTERFACE\"\n", subnet_str); break;
147                 }
148         }
149 #elif defined(HAVE_BSD)
150         // BSD route command is silly and doesn't accept an interface name as a destination.
151         if(!*gateway_str) {
152                 switch(subnet.type) {
153                         case SUBNET_IPV4:
154                                 if(!ipv4.type) {
155                                         fprintf(stderr, "Route requested but no Ifconfig\n");
156                                         return;
157                                 }
158                                 net2str(gateway_str, sizeof gateway_str, &ipv4);
159                                 break;
160                         case SUBNET_IPV6:
161                                 if(!ipv6.type) {
162                                         fprintf(stderr, "Route requested but no Ifconfig\n");
163                                         return;
164                                 }
165                                 net2str(gateway_str, sizeof gateway_str, &ipv6);
166                                 break;
167                 }
168                 char *slash = strchr(gateway_str, '/'); if(slash) *slash = 0;
169         }
170
171         switch(subnet.type) {
172                 case SUBNET_IPV4: fprintf(out, "route add %s %s\n", subnet_str, gateway_str); break;
173                 case SUBNET_IPV6: fprintf(out, "route add -inet6 %s %s\n", subnet_str, gateway_str); break;
174         }
175 #elif defined(HAVE_MINGW) || defined(HAVE_CYGWIN)
176         if(*gateway_str) {
177                 switch(subnet.type) {
178                         case SUBNET_IPV4: fprintf(out, "netsh inetface ipv4 add route %s \"%%INTERFACE%%\" %s\n", subnet_str, gateway_str); break;
179                         case SUBNET_IPV6: fprintf(out, "netsh inetface ipv6 add route %s \"%%INTERFACE%%\" %s\n", subnet_str, gateway_str); break;
180                 }
181         } else {
182                 switch(subnet.type) {
183                         case SUBNET_IPV4: fprintf(out, "netsh inetface ipv4 add route %s \"%%INTERFACE%%\"\n", subnet_str); break;
184                         case SUBNET_IPV6: fprintf(out, "netsh inetface ipv6 add route %s \"%%INTERFACE%%\"\n", subnet_str); break;
185                 }
186         }
187 #endif
188 }