Add helpers for ifup and ifdown
[connman] / plugins / inet.c
1 /*
2  *
3  *  Connection Manager
4  *
5  *  Copyright (C) 2007-2008  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 #include <errno.h>
28 #include <unistd.h>
29 #include <stdlib.h>
30 #include <string.h>
31 #include <sys/ioctl.h>
32 #include <sys/socket.h>
33 #include <net/if.h>
34 #include <net/ethernet.h>
35
36 #include "inet.h"
37
38 char *inet_index2name(int index)
39 {
40         struct ifreq ifr;
41         int sk, err;
42
43         if (index < 0)
44                 return NULL;
45
46         sk = socket(PF_INET, SOCK_DGRAM, 0);
47         if (sk < 0)
48                 return NULL;
49
50         memset(&ifr, 0, sizeof(ifr));
51         ifr.ifr_ifindex = index;
52
53         err = ioctl(sk, SIOCGIFNAME, &ifr);
54
55         close(sk);
56
57         if (err < 0)
58                 return NULL;
59
60         return strdup(ifr.ifr_name);
61 }
62
63 char *inet_index2ident(int index, const char *prefix)
64 {
65         struct ifreq ifr;
66         struct ether_addr *eth;
67         char *str;
68         int sk, err, len;
69
70         if (index < 0)
71                 return NULL;
72
73         sk = socket(PF_INET, SOCK_DGRAM, 0);
74         if (sk < 0)
75                 return NULL;
76
77         memset(&ifr, 0, sizeof(ifr));
78         ifr.ifr_ifindex = index;
79
80         err = ioctl(sk, SIOCGIFNAME, &ifr);
81
82         if (err == 0)
83                 err = ioctl(sk, SIOCGIFHWADDR, &ifr);
84
85         close(sk);
86
87         if (err < 0)
88                 return NULL;
89
90         len = prefix ? strlen(prefix) + 18 : 18;
91
92         str = malloc(len);
93         if (!str)
94                 return NULL;
95
96         eth = (void *) &ifr.ifr_hwaddr.sa_data;
97         snprintf(str, len, "%s%02X_%02X_%02X_%02X_%02X_%02X",
98                                                 prefix ? prefix : "",
99                                                 eth->ether_addr_octet[0],
100                                                 eth->ether_addr_octet[1],
101                                                 eth->ether_addr_octet[2],
102                                                 eth->ether_addr_octet[3],
103                                                 eth->ether_addr_octet[4],
104                                                 eth->ether_addr_octet[5]);
105
106         return str;
107 }
108
109 int inet_ifup(int index)
110 {
111         struct ifreq ifr;
112         int sk, err;
113
114         sk = socket(PF_INET, SOCK_DGRAM, 0);
115         if (sk < 0)
116                 return -errno;
117
118         memset(&ifr, 0, sizeof(ifr));
119         ifr.ifr_ifindex = index;
120
121         if (ioctl(sk, SIOCGIFNAME, &ifr) < 0) {
122                 err = -errno;
123                 goto done;
124         }
125
126         if (ioctl(sk, SIOCGIFFLAGS, &ifr) < 0) {
127                 err = -errno;
128                 goto done;
129         }
130
131         if (ifr.ifr_flags & IFF_UP) {
132                 err = -EALREADY;
133                 goto done;
134         }
135
136         ifr.ifr_flags |= IFF_UP;
137
138         if (ioctl(sk, SIOCSIFFLAGS, &ifr) < 0) {
139                 err = -errno;
140                 goto done;
141         }
142
143         err = 0;
144
145 done:
146         close(sk);
147
148         return err;
149 }
150
151 int inet_ifdown(int index)
152 {
153         struct ifreq ifr;
154         int sk, err;
155
156         sk = socket(PF_INET, SOCK_DGRAM, 0);
157         if (sk < 0)
158                 return -errno;
159
160         memset(&ifr, 0, sizeof(ifr));
161         ifr.ifr_ifindex = index;
162
163         if (ioctl(sk, SIOCGIFNAME, &ifr) < 0) {
164                 err = -errno;
165                 goto done;
166         }
167
168         if (ioctl(sk, SIOCGIFFLAGS, &ifr) < 0) {
169                 err = -errno;
170                 goto done;
171         }
172
173         if (!(ifr.ifr_flags & IFF_UP)) {
174                 err = -EALREADY;
175                 goto done;
176         }
177
178         ifr.ifr_flags &= ~IFF_UP;
179
180         if (ioctl(sk, SIOCSIFFLAGS, &ifr) < 0)
181                 err = -errno;
182         else
183                 err = 0;
184
185 done:
186         close(sk);
187
188         return err;
189 }