wifi: Add support for autoscan request
[framework/connectivity/connman.git] / plugins / loopback.c
1 /*
2  *
3  *  Connection Manager
4  *
5  *  Copyright (C) 2007-2012  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 #include <connman/inet.h>
43
44 static in_addr_t loopback_address;
45 static in_addr_t loopback_netmask;
46
47 static char system_hostname[HOST_NAME_MAX + 1];
48
49 static void create_hostname(void)
50 {
51         const char *name = "localhost";
52
53         if (sethostname(name, strlen(name)) < 0)
54                 connman_error("Failed to set hostname to %s", name);
55
56         strncpy(system_hostname, name, HOST_NAME_MAX);
57 }
58
59 static int setup_hostname(void)
60 {
61         char name[HOST_NAME_MAX + 1];
62
63         memset(system_hostname, 0, sizeof(system_hostname));
64
65         if (gethostname(system_hostname, HOST_NAME_MAX) < 0) {
66                 connman_error("Failed to get current hostname");
67                 return -EIO;
68         }
69
70         if (strlen(system_hostname) > 0 &&
71                                 strcmp(system_hostname, "(none)") != 0)
72                 connman_info("System hostname is %s", system_hostname);
73         else
74                 create_hostname();
75
76         memset(name, 0, sizeof(name));
77
78         if (getdomainname(name, HOST_NAME_MAX) < 0) {
79                 connman_error("Failed to get current domainname");
80                 return -EIO;
81         }
82
83         if (strlen(name) > 0 && strcmp(name, "(none)") != 0)
84                 connman_info("System domainname is %s", name);
85
86         return 0;
87 }
88
89 static gboolean valid_loopback(int sk, struct ifreq *ifr)
90 {
91         struct sockaddr_in *addr;
92         int err;
93         char buf[INET_ADDRSTRLEN];
94
95         /* It is possible to end up in situations in which the
96          * loopback interface is up but has no valid address. In that
97          * case, we expect EADDRNOTAVAIL and should return FALSE.
98          */
99
100         err = ioctl(sk, SIOCGIFADDR, ifr);
101         if (err < 0) {
102                 err = -errno;
103                 connman_error("Getting address failed (%s)", strerror(-err));
104                 return err != -EADDRNOTAVAIL ? TRUE : FALSE;
105         }
106
107         addr = (struct sockaddr_in *) &ifr->ifr_addr;
108         if (addr->sin_addr.s_addr != loopback_address) {
109                 connman_warn("Invalid loopback address %s",
110                         inet_ntop(AF_INET, &addr->sin_addr, buf, sizeof(buf)));
111                 return FALSE;
112         }
113
114         err = ioctl(sk, SIOCGIFNETMASK, ifr);
115         if (err < 0) {
116                 err = -errno;
117                 connman_error("Getting netmask failed (%s)", strerror(-err));
118                 return TRUE;
119         }
120
121         addr = (struct sockaddr_in *) &ifr->ifr_netmask;
122         if (addr->sin_addr.s_addr != loopback_netmask) {
123                 connman_warn("Invalid loopback netmask %s",
124                         inet_ntop(AF_INET, &addr->sin_addr, buf, sizeof(buf)));
125                 return FALSE;
126         }
127
128         return TRUE;
129 }
130
131 static int setup_loopback(void)
132 {
133         struct ifreq ifr;
134         struct sockaddr_in addr;
135         int sk, err;
136
137         sk = socket(PF_INET, SOCK_DGRAM | SOCK_CLOEXEC, 0);
138         if (sk < 0)
139                 return -errno;
140
141         memset(&ifr, 0, sizeof(ifr));
142         strcpy(ifr.ifr_name, "lo");
143
144         if (ioctl(sk, SIOCGIFFLAGS, &ifr) < 0) {
145                 err = -errno;
146                 goto done;
147         }
148
149         if (ifr.ifr_flags & IFF_UP) {
150                 connman_info("Checking loopback interface settings");
151                 if (valid_loopback(sk, &ifr) == TRUE) {
152                         err = -EALREADY;
153                         goto done;
154                 }
155
156                 connman_warn("Correcting wrong lookback settings");
157         }
158
159         memset(&addr, 0, sizeof(addr));
160         addr.sin_family = AF_INET;
161         addr.sin_addr.s_addr = loopback_address;
162         memcpy(&ifr.ifr_addr, &addr, sizeof(ifr.ifr_addr));
163
164         err = ioctl(sk, SIOCSIFADDR, &ifr);
165         if (err < 0) {
166                 err = -errno;
167                 connman_error("Setting address failed (%s)", strerror(-err));
168                 goto done;
169         }
170
171         memset(&addr, 0, sizeof(addr));
172         addr.sin_family = AF_INET;
173         addr.sin_addr.s_addr = loopback_netmask;
174         memcpy(&ifr.ifr_netmask, &addr, sizeof(ifr.ifr_netmask));
175
176         err = ioctl(sk, SIOCSIFNETMASK, &ifr);
177         if (err < 0) {
178                 err = -errno;
179                 connman_error("Setting netmask failed (%s)", strerror(-err));
180                 goto done;
181         }
182
183         if (ioctl(sk, SIOCGIFFLAGS, &ifr) < 0) {
184                 err = -errno;
185                 goto done;
186         }
187
188         ifr.ifr_flags |= IFF_UP;
189
190         if (ioctl(sk, SIOCSIFFLAGS, &ifr) < 0) {
191                 err = -errno;
192                 connman_error("Activating loopback interface failed (%s)",
193                                                         strerror(-err));
194                 goto done;
195         }
196
197 done:
198         close(sk);
199
200         return err;
201 }
202
203 static const char *loopback_get_hostname(void)
204 {
205         return system_hostname;
206 }
207
208 static int loopback_set_hostname(const char *hostname)
209 {
210         const char *ptr;
211         int err, len;
212
213         if (g_strcmp0(hostname, "<hostname>") == 0)
214                 return 0;
215
216         len = strlen(hostname);
217
218         if (connman_inet_check_hostname(hostname, len) == FALSE)
219                 return -EINVAL;
220
221         if ((ptr = strstr(hostname, ".")) != NULL)
222                 len = ptr - hostname;
223
224         if (sethostname(hostname, len) < 0) {
225                 err = -errno;
226                 connman_error("Failed to set hostname to %s", hostname);
227                 return err;
228         }
229
230         connman_info("Setting hostname to %s", hostname);
231
232         return 0;
233 }
234
235 static int loopback_set_domainname(const char *domainname)
236 {
237         int err, len;
238
239         len = strlen(domainname);
240
241         if (connman_inet_check_hostname(domainname, len) == FALSE)
242                 return -EINVAL;
243
244         if (setdomainname(domainname, len) < 0) {
245                 err = -errno;
246                 connman_error("Failed to set domainname to %s", domainname);
247                 return err;
248         }
249
250         connman_info("Setting domainname to %s", domainname);
251
252         return 0;
253 }
254
255 static struct connman_utsname_driver loopback_driver = {
256         .name           = "loopback",
257         .get_hostname   = loopback_get_hostname,
258         .set_hostname   = loopback_set_hostname,
259         .set_domainname = loopback_set_domainname,
260 };
261
262 static int loopback_init(void)
263 {
264         loopback_address = inet_addr("127.0.0.1");
265         loopback_netmask = inet_addr("255.0.0.0");
266
267         setup_loopback();
268
269         setup_hostname();
270
271         connman_utsname_driver_register(&loopback_driver);
272
273         return 0;
274 }
275
276 static void loopback_exit(void)
277 {
278         connman_utsname_driver_unregister(&loopback_driver);
279 }
280
281 CONNMAN_PLUGIN_DEFINE(loopback, "Loopback device plugin", VERSION,
282                 CONNMAN_PLUGIN_PRIORITY_HIGH, loopback_init, loopback_exit)