inet: Cleanup router solicitation callback properly after error
[framework/connectivity/connman.git] / plugins / loopback.c
1 /*
2  *
3  *  Connection Manager
4  *
5  *  Copyright (C) 2007-2010  Intel Corporation. All rights reserved.
6  *
7  *  This program is free software; you can redistribute it and/or modify
8  *  it under the terms of the GNU General Public License version 2 as
9  *  published by the Free Software Foundation.
10  *
11  *  This program is distributed in the hope that it will be useful,
12  *  but WITHOUT ANY WARRANTY; without even the implied warranty of
13  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14  *  GNU General Public License for more details.
15  *
16  *  You should have received a copy of the GNU General Public License
17  *  along with this program; if not, write to the Free Software
18  *  Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
19  *
20  */
21
22 #ifdef HAVE_CONFIG_H
23 #include <config.h>
24 #endif
25
26 #include <errno.h>
27 #include <unistd.h>
28 #include <limits.h>
29 #include <string.h>
30 #include <sys/ioctl.h>
31 #include <sys/inotify.h>
32 #include <sys/socket.h>
33 #include <arpa/inet.h>
34 #include <net/if.h>
35
36 #include <glib.h>
37
38 #define CONNMAN_API_SUBJECT_TO_CHANGE
39 #include <connman/plugin.h>
40 #include <connman/utsname.h>
41 #include <connman/log.h>
42
43 static in_addr_t loopback_address;
44 static in_addr_t loopback_netmask;
45
46 static char system_hostname[HOST_NAME_MAX + 1];
47
48 static void create_hostname(void)
49 {
50         const char *name = "localhost";
51
52         if (sethostname(name, strlen(name)) < 0)
53                 connman_error("Failed to set hostname to %s", name);
54
55         strncpy(system_hostname, name, HOST_NAME_MAX);
56 }
57
58 static int setup_hostname(void)
59 {
60         char name[HOST_NAME_MAX + 1];
61
62         memset(system_hostname, 0, sizeof(system_hostname));
63
64         if (gethostname(system_hostname, HOST_NAME_MAX) < 0) {
65                 connman_error("Failed to get current hostname");
66                 return -EIO;
67         }
68
69         if (strlen(system_hostname) > 0 &&
70                                 strcmp(system_hostname, "(none)") != 0)
71                 connman_info("System hostname is %s", system_hostname);
72         else
73                 create_hostname();
74
75         memset(name, 0, sizeof(name));
76
77         if (getdomainname(name, HOST_NAME_MAX) < 0) {
78                 connman_error("Failed to get current domainname");
79                 return -EIO;
80         }
81
82         if (strlen(name) > 0 && strcmp(name, "(none)") != 0)
83                 connman_info("System domainname is %s", name);
84
85         return 0;
86 }
87
88 static gboolean valid_loopback(int sk, struct ifreq *ifr)
89 {
90         struct sockaddr_in *addr;
91         int err;
92         char buf[INET_ADDRSTRLEN];
93
94         /* It is possible to end up in situations in which the
95          * loopback interface is up but has no valid address. In that
96          * case, we expect EADDRNOTAVAIL and should return FALSE.
97          */
98
99         err = ioctl(sk, SIOCGIFADDR, ifr);
100         if (err < 0) {
101                 err = -errno;
102                 connman_error("Getting address failed (%s)", strerror(-err));
103                 return err != -EADDRNOTAVAIL ? TRUE : FALSE;
104         }
105
106         addr = (struct sockaddr_in *) &ifr->ifr_addr;
107         if (addr->sin_addr.s_addr != loopback_address) {
108                 connman_warn("Invalid loopback address %s",
109                         inet_ntop(AF_INET, &addr->sin_addr, buf, sizeof(buf)));
110                 return FALSE;
111         }
112
113         err = ioctl(sk, SIOCGIFNETMASK, ifr);
114         if (err < 0) {
115                 err = -errno;
116                 connman_error("Getting netmask failed (%s)", strerror(-err));
117                 return TRUE;
118         }
119
120         addr = (struct sockaddr_in *) &ifr->ifr_netmask;
121         if (addr->sin_addr.s_addr != loopback_netmask) {
122                 connman_warn("Invalid loopback netmask %s",
123                         inet_ntop(AF_INET, &addr->sin_addr, buf, sizeof(buf)));
124                 return FALSE;
125         }
126
127         return TRUE;
128 }
129
130 static int setup_loopback(void)
131 {
132         struct ifreq ifr;
133         struct sockaddr_in addr;
134         int sk, err;
135
136         sk = socket(PF_INET, SOCK_DGRAM, 0);
137         if (sk < 0)
138                 return -errno;
139
140         memset(&ifr, 0, sizeof(ifr));
141         strcpy(ifr.ifr_name, "lo");
142
143         if (ioctl(sk, SIOCGIFFLAGS, &ifr) < 0) {
144                 err = -errno;
145                 goto done;
146         }
147
148         if (ifr.ifr_flags & IFF_UP) {
149                 connman_info("Checking loopback interface settings");
150                 if (valid_loopback(sk, &ifr) == TRUE) {
151                         err = -EALREADY;
152                         goto done;
153                 }
154
155                 connman_warn("Correcting wrong lookback settings");
156         }
157
158         memset(&addr, 0, sizeof(addr));
159         addr.sin_family = AF_INET;
160         addr.sin_addr.s_addr = loopback_address;
161         memcpy(&ifr.ifr_addr, &addr, sizeof(ifr.ifr_addr));
162
163         err = ioctl(sk, SIOCSIFADDR, &ifr);
164         if (err < 0) {
165                 err = -errno;
166                 connman_error("Setting address failed (%s)", strerror(-err));
167                 goto done;
168         }
169
170         memset(&addr, 0, sizeof(addr));
171         addr.sin_family = AF_INET;
172         addr.sin_addr.s_addr = loopback_netmask;
173         memcpy(&ifr.ifr_netmask, &addr, sizeof(ifr.ifr_netmask));
174
175         err = ioctl(sk, SIOCSIFNETMASK, &ifr);
176         if (err < 0) {
177                 err = -errno;
178                 connman_error("Setting netmask failed (%s)", strerror(-err));
179                 goto done;
180         }
181
182         if (ioctl(sk, SIOCGIFFLAGS, &ifr) < 0) {
183                 err = -errno;
184                 goto done;
185         }
186
187         ifr.ifr_flags |= IFF_UP;
188
189         if (ioctl(sk, SIOCSIFFLAGS, &ifr) < 0) {
190                 err = -errno;
191                 connman_error("Activating loopback interface failed (%s)",
192                                                         strerror(-err));
193                 goto done;
194         }
195
196 done:
197         close(sk);
198
199         return err;
200 }
201
202 static const char *loopback_get_hostname(void)
203 {
204         return system_hostname;
205 }
206
207 static int loopback_set_hostname(const char *hostname)
208 {
209         int err;
210
211         if (g_strcmp0(hostname, "<hostname>") == 0)
212                 return 0;
213
214         if (sethostname(hostname, strlen(hostname)) < 0) {
215                 err = -errno;
216                 connman_error("Failed to set hostname to %s", hostname);
217                 return err;
218         }
219
220         connman_info("Setting hostname to %s", hostname);
221
222         return 0;
223 }
224
225 static int loopback_set_domainname(const char *domainname)
226 {
227         int err;
228
229         if (setdomainname(domainname, strlen(domainname)) < 0) {
230                 err = -errno;
231                 connman_error("Failed to set domainname to %s", domainname);
232                 return err;
233         }
234
235         connman_info("Setting domainname to %s", domainname);
236
237         return 0;
238 }
239
240 static struct connman_utsname_driver loopback_driver = {
241         .name           = "loopback",
242         .get_hostname   = loopback_get_hostname,
243         .set_hostname   = loopback_set_hostname,
244         .set_domainname = loopback_set_domainname,
245 };
246
247 static int loopback_init(void)
248 {
249         loopback_address = inet_addr("127.0.0.1");
250         loopback_netmask = inet_addr("255.0.0.0");
251
252         setup_loopback();
253
254         setup_hostname();
255
256         connman_utsname_driver_register(&loopback_driver);
257
258         return 0;
259 }
260
261 static void loopback_exit(void)
262 {
263         connman_utsname_driver_unregister(&loopback_driver);
264 }
265
266 CONNMAN_PLUGIN_DEFINE(loopback, "Loopback device plugin", VERSION,
267                 CONNMAN_PLUGIN_PRIORITY_HIGH, loopback_init, loopback_exit)