cfb33a8e9df8b3dfec5edc6940c83db8258035d7
[platform/upstream/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 <stdio.h>
27
28 #include <errno.h>
29 #include <unistd.h>
30 #include <limits.h>
31 #include <string.h>
32 #include <sys/ioctl.h>
33 #include <sys/inotify.h>
34 #include <sys/socket.h>
35 #include <arpa/inet.h>
36 #include <net/if.h>
37
38 #include <glib.h>
39 #include <glib/gprintf.h>
40
41
42 #define CONNMAN_API_SUBJECT_TO_CHANGE
43 #include <connman/plugin.h>
44 #include <connman/utsname.h>
45 #include <connman/log.h>
46 #include <connman/inet.h>
47
48 static in_addr_t loopback_address;
49 static in_addr_t loopback_netmask;
50
51 static char system_hostname[HOST_NAME_MAX + 1];
52
53 static void create_hostname(void)
54 {
55         const char *name = "localhost";
56
57         if (sethostname(name, strlen(name)) < 0)
58                 connman_error("Failed to set hostname to %s", name);
59
60         strncpy(system_hostname, name, HOST_NAME_MAX);
61 }
62
63 static int setup_hostname(void)
64 {
65         char name[HOST_NAME_MAX + 1];
66
67         memset(system_hostname, 0, sizeof(system_hostname));
68
69 #if defined TIZEN_EXT
70                 FILE *fp = NULL;
71
72 #if defined TIZEN_WEARABLE
73 #define BT_MAC "/csa/bluetooth/.bd_addr"
74                 {
75                         gchar* dev_id = "GearS2";
76                         char bt_mac[HOST_NAME_MAX + 1];
77                         char addr[5] = {0, };
78
79                         fp = fopen(BT_MAC, "r");
80                         if(!fp){
81                                 connman_error("Failed to get current hostname");
82                                 strncpy(system_hostname, dev_id, strlen(dev_id));
83                                 goto host_name_end;
84                         }
85
86                         // get the last line's address
87                         while (fgets(bt_mac, HOST_NAME_MAX, fp)) {}
88
89                         if (strlen(bt_mac) == 6) {
90                                 addr[0] = bt_mac[2];
91                                 addr[1] = bt_mac[3];
92                                 addr[2] = bt_mac[4];
93                                 addr[3] = bt_mac[5];
94                                 g_sprintf(system_hostname, "%s-%s", dev_id, addr);
95                         } else
96                                 strncpy(system_hostname, dev_id, strlen(dev_id));
97
98                         fclose(fp);
99                 }
100 #else
101 #define WIFI_MAC "/opt/etc/.mac.info"
102                 {
103                         char* rv = 0;
104                         gchar* dev_id = "TIZEN";
105                         char wifi_mac[HOST_NAME_MAX + 1];
106
107                         fp = fopen(WIFI_MAC, "r");
108                         if(!fp){
109                                 connman_error("Failed to get current hostname");
110                                 strncpy(system_hostname, dev_id, strlen(dev_id));
111                                 goto host_name_end;
112                         }
113
114                         rv = fgets(wifi_mac, HOST_NAME_MAX, fp);
115                         if(!rv){
116                                 connman_error("Failed to get current hostname");
117                                 strncpy(system_hostname, dev_id, strlen(dev_id));
118                                 fclose(fp);
119                                 goto host_name_end;
120                         }
121
122                         dev_id = g_base64_encode((const guchar *)wifi_mac, strlen(wifi_mac));
123                         g_sprintf(system_hostname, "TIZEN-%s", dev_id);
124                         g_free(dev_id);
125                         fclose(fp);
126                 }
127 #endif
128
129         host_name_end :
130 #else
131                 if (gethostname(system_hostname, HOST_NAME_MAX) < 0) {
132                         connman_error("Failed to get current hostname");
133                         return -EIO;
134                 }
135 #endif
136                 if (strlen(system_hostname) > 0 &&
137                                         strcmp(system_hostname, "(none)") != 0)
138                         connman_info("System hostname is %s", system_hostname);
139                 else
140                         create_hostname();
141
142         memset(name, 0, sizeof(name));
143
144         if (getdomainname(name, HOST_NAME_MAX) < 0) {
145                 connman_error("Failed to get current domainname");
146                 return -EIO;
147         }
148
149         if (strlen(name) > 0 && strcmp(name, "(none)") != 0)
150                 connman_info("System domainname is %s", name);
151
152         return 0;
153 }
154
155 static bool valid_loopback(int sk, struct ifreq *ifr)
156 {
157         struct sockaddr_in *addr;
158         int err;
159         char buf[INET_ADDRSTRLEN];
160
161         /* It is possible to end up in situations in which the
162          * loopback interface is up but has no valid address. In that
163          * case, we expect EADDRNOTAVAIL and should return FALSE.
164          */
165
166         err = ioctl(sk, SIOCGIFADDR, ifr);
167         if (err < 0) {
168                 err = -errno;
169                 connman_error("Getting address failed (%s)", strerror(-err));
170                 return err != -EADDRNOTAVAIL ? TRUE : FALSE;
171         }
172
173         addr = (struct sockaddr_in *) &ifr->ifr_addr;
174         if (addr->sin_addr.s_addr != loopback_address) {
175                 connman_warn("Invalid loopback address %s",
176                         inet_ntop(AF_INET, &addr->sin_addr, buf, sizeof(buf)));
177                 return false;
178         }
179
180         err = ioctl(sk, SIOCGIFNETMASK, ifr);
181         if (err < 0) {
182                 err = -errno;
183                 connman_error("Getting netmask failed (%s)", strerror(-err));
184                 return true;
185         }
186
187         addr = (struct sockaddr_in *) &ifr->ifr_netmask;
188         if (addr->sin_addr.s_addr != loopback_netmask) {
189                 connman_warn("Invalid loopback netmask %s",
190                         inet_ntop(AF_INET, &addr->sin_addr, buf, sizeof(buf)));
191                 return false;
192         }
193
194         return true;
195 }
196
197 static int setup_loopback(void)
198 {
199         struct ifreq ifr;
200         struct sockaddr_in addr;
201         int sk, err;
202
203         sk = socket(PF_INET, SOCK_DGRAM | SOCK_CLOEXEC, 0);
204         if (sk < 0)
205                 return -errno;
206
207         memset(&ifr, 0, sizeof(ifr));
208         strcpy(ifr.ifr_name, "lo");
209
210         if (ioctl(sk, SIOCGIFFLAGS, &ifr) < 0) {
211                 err = -errno;
212                 goto done;
213         }
214
215         if (ifr.ifr_flags & IFF_UP) {
216                 connman_info("Checking loopback interface settings");
217                 if (valid_loopback(sk, &ifr)) {
218                         err = -EALREADY;
219                         goto done;
220                 }
221
222                 connman_warn("Correcting wrong loopback settings");
223         }
224
225         memset(&addr, 0, sizeof(addr));
226         addr.sin_family = AF_INET;
227         addr.sin_addr.s_addr = loopback_address;
228         memcpy(&ifr.ifr_addr, &addr, sizeof(ifr.ifr_addr));
229
230         err = ioctl(sk, SIOCSIFADDR, &ifr);
231         if (err < 0) {
232                 err = -errno;
233                 connman_error("Setting address failed (%s)", strerror(-err));
234                 goto done;
235         }
236
237         memset(&addr, 0, sizeof(addr));
238         addr.sin_family = AF_INET;
239         addr.sin_addr.s_addr = loopback_netmask;
240         memcpy(&ifr.ifr_netmask, &addr, sizeof(ifr.ifr_netmask));
241
242         err = ioctl(sk, SIOCSIFNETMASK, &ifr);
243         if (err < 0) {
244                 err = -errno;
245                 connman_error("Setting netmask failed (%s)", strerror(-err));
246                 goto done;
247         }
248
249         if (ioctl(sk, SIOCGIFFLAGS, &ifr) < 0) {
250                 err = -errno;
251                 goto done;
252         }
253
254         ifr.ifr_flags |= IFF_UP;
255
256         if (ioctl(sk, SIOCSIFFLAGS, &ifr) < 0) {
257                 err = -errno;
258                 connman_error("Activating loopback interface failed (%s)",
259                                                         strerror(-err));
260                 goto done;
261         }
262
263 done:
264         close(sk);
265
266         return err;
267 }
268
269 static const char *loopback_get_hostname(void)
270 {
271         return system_hostname;
272 }
273
274 static int loopback_set_hostname(const char *hostname)
275 {
276         const char *ptr;
277         int err, len;
278
279         if (g_strcmp0(hostname, "<hostname>") == 0)
280                 return 0;
281
282         len = strlen(hostname);
283
284         if (!connman_inet_check_hostname(hostname, len))
285                 return -EINVAL;
286
287         if ((ptr = strstr(hostname, ".")))
288                 len = ptr - hostname;
289
290         if (sethostname(hostname, len) < 0) {
291                 err = -errno;
292                 connman_error("Failed to set hostname to %s", hostname);
293                 return err;
294         }
295
296         connman_info("Setting hostname to %s", hostname);
297
298         return 0;
299 }
300
301 static int loopback_set_domainname(const char *domainname)
302 {
303         int err, len;
304
305         len = strlen(domainname);
306
307         if (!connman_inet_check_hostname(domainname, len))
308                 return -EINVAL;
309
310         if (setdomainname(domainname, len) < 0) {
311                 err = -errno;
312                 connman_error("Failed to set domainname to %s", domainname);
313                 return err;
314         }
315
316         connman_info("Setting domainname to %s", domainname);
317
318         return 0;
319 }
320
321 static struct connman_utsname_driver loopback_driver = {
322         .name           = "loopback",
323         .get_hostname   = loopback_get_hostname,
324         .set_hostname   = loopback_set_hostname,
325         .set_domainname = loopback_set_domainname,
326 };
327
328 static int loopback_init(void)
329 {
330         loopback_address = inet_addr("127.0.0.1");
331         loopback_netmask = inet_addr("255.0.0.0");
332
333         setup_loopback();
334
335         setup_hostname();
336
337         connman_utsname_driver_register(&loopback_driver);
338
339         return 0;
340 }
341
342 static void loopback_exit(void)
343 {
344         connman_utsname_driver_unregister(&loopback_driver);
345 }
346
347 CONNMAN_PLUGIN_DEFINE(loopback, "Loopback device plugin", VERSION,
348                 CONNMAN_PLUGIN_PRIORITY_HIGH, loopback_init, loopback_exit)