blob: 187ca5d86578e54fe25d7276d193e8204e3bb7c1 [file] [log] [blame]
Wichert Akkerman76baf7c1999-02-19 00:21:36 +00001/*
2 * Copyright (c) 1993, 1994, 1995, 1996 Rick Sladkey <jrs@world.std.com>
3 * All rights reserved.
4 *
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions
7 * are met:
8 * 1. Redistributions of source code must retain the above copyright
9 * notice, this list of conditions and the following disclaimer.
10 * 2. Redistributions in binary form must reproduce the above copyright
11 * notice, this list of conditions and the following disclaimer in the
12 * documentation and/or other materials provided with the distribution.
13 * 3. The name of the author may not be used to endorse or promote products
14 * derived from this software without specific prior written permission.
15 *
16 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
17 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
18 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
19 * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
20 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
21 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
22 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
23 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
24 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
25 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
26 *
27 * $Id$
28 */
29
30#include "defs.h"
31
Roland McGrath561c7992003-04-02 01:10:44 +000032#ifdef LINUX
Wichert Akkerman76baf7c1999-02-19 00:21:36 +000033#include <sys/socket.h>
Roland McGrath5687ff12004-07-12 07:13:06 +000034#include <linux/sockios.h>
Wichert Akkerman76baf7c1999-02-19 00:21:36 +000035#else
Roland McGrath8988d1e2004-10-19 23:33:50 +000036#include <sys/socket.h>
Wichert Akkerman76baf7c1999-02-19 00:21:36 +000037#include <sys/sockio.h>
38#endif
Roland McGrath5687ff12004-07-12 07:13:06 +000039#include <arpa/inet.h>
Wichert Akkerman76baf7c1999-02-19 00:21:36 +000040
Roland McGrathf5a47772003-06-26 22:40:42 +000041#if defined (ALPHA) || defined(SH) || defined(SH64)
Wichert Akkerman2e2553a1999-05-09 00:29:58 +000042#ifdef HAVE_SYS_IOCTL_H
43#include <sys/ioctl.h>
44#elif defined(HAVE_IOCTLS_H)
Wichert Akkerman76baf7c1999-02-19 00:21:36 +000045#include <ioctls.h>
46#endif
Wichert Akkerman2e2553a1999-05-09 00:29:58 +000047#endif
Roland McGrath5687ff12004-07-12 07:13:06 +000048#include <net/if.h>
49
Roland McGrathd9f816f2004-09-04 03:39:20 +000050extern const struct xlat addrfams[];
Wichert Akkerman76baf7c1999-02-19 00:21:36 +000051
Roland McGrathe6a432d2005-02-02 20:25:17 +000052static const struct xlat iffflags[] = {
53 { IFF_UP, "IFF_UP" },
54 { IFF_BROADCAST, "IFF_BROADCAST" },
55 { IFF_DEBUG, "IFF_DEBUG" },
56 { IFF_LOOPBACK, "IFF_LOOPBACK" },
57 { IFF_POINTOPOINT, "IFF_POINTOPOINT" },
58 { IFF_NOTRAILERS, "IFF_NOTRAILERS" },
59 { IFF_RUNNING, "IFF_RUNNING" },
60 { IFF_NOARP, "IFF_NOARP" },
61 { IFF_PROMISC, "IFF_PROMISC" },
62 { IFF_ALLMULTI, "IFF_ALLMULTI" },
63 { IFF_MASTER, "IFF_MASTER" },
64 { IFF_SLAVE, "IFF_SLAVE" },
65 { IFF_MULTICAST, "IFF_MULTICAST" },
66 { IFF_PORTSEL, "IFF_PORTSEL" },
67 { IFF_AUTOMEDIA, "IFF_AUTOMEDIA" },
68 { 0, NULL }
69};
70
71
72static void
73print_addr(tcp, addr, ifr)
74struct tcb *tcp;
75long addr;
76struct ifreq *ifr;
77{
78 if (ifr->ifr_addr.sa_family == AF_INET) {
79 struct sockaddr_in *sinp;
80 sinp = (struct sockaddr_in *) &ifr->ifr_addr;
81 tprintf("inet_addr(\"%s\")", inet_ntoa(sinp->sin_addr));
82 } else
83 printstr(tcp, addr, sizeof(ifr->ifr_addr.sa_data));
84}
85
Wichert Akkerman76baf7c1999-02-19 00:21:36 +000086int
Dmitry V. Levin40284242007-03-21 13:52:14 +000087sock_ioctl(struct tcb *tcp, long code, long arg)
Wichert Akkerman76baf7c1999-02-19 00:21:36 +000088{
Roland McGrath5687ff12004-07-12 07:13:06 +000089 struct ifreq ifr;
90 struct ifconf ifc;
Roland McGrathe6a432d2005-02-02 20:25:17 +000091 const char *str = NULL;
92 unsigned char *bytes;
Roland McGrath5687ff12004-07-12 07:13:06 +000093
94 if (entering(tcp)) {
95 if (code == SIOCGIFCONF) {
Dmitry V. Levin652e4482007-03-21 14:18:17 +000096 if (umove(tcp, tcp->u_arg[2], &ifc) >= 0
97 && ifc.ifc_buf == NULL)
Roland McGrath5687ff12004-07-12 07:13:06 +000098 tprintf(", {%d -> ", ifc.ifc_len);
99 else
100 tprintf(", {");
101 }
Wichert Akkerman76baf7c1999-02-19 00:21:36 +0000102 return 0;
Roland McGrath5687ff12004-07-12 07:13:06 +0000103 }
Wichert Akkerman76baf7c1999-02-19 00:21:36 +0000104
105 switch (code) {
106#ifdef SIOCSHIWAT
107 case SIOCSHIWAT:
108#endif
109#ifdef SIOCGHIWAT
110 case SIOCGHIWAT:
111#endif
112#ifdef SIOCSLOWAT
113 case SIOCSLOWAT:
114#endif
115#ifdef SIOCGLOWAT
116 case SIOCGLOWAT:
117#endif
118#ifdef FIOSETOWN
119 case FIOSETOWN:
120#endif
121#ifdef FIOGETOWN
122 case FIOGETOWN:
123#endif
124#ifdef SIOCSPGRP
125 case SIOCSPGRP:
126#endif
127#ifdef SIOCGPGRP
128 case SIOCGPGRP:
129#endif
130#ifdef SIOCATMARK
131 case SIOCATMARK:
132#endif
133 printnum(tcp, arg, ", %#d");
134 return 1;
Roland McGrath5687ff12004-07-12 07:13:06 +0000135#ifdef LINUX
136 case SIOCGIFNAME:
137 case SIOCGIFINDEX:
Roland McGrathe6a432d2005-02-02 20:25:17 +0000138 case SIOCGIFADDR:
139 case SIOCGIFDSTADDR:
140 case SIOCGIFBRDADDR:
141 case SIOCGIFNETMASK:
142 case SIOCGIFFLAGS:
143 case SIOCGIFMETRIC:
144 case SIOCGIFMTU:
145 case SIOCGIFSLAVE:
146 case SIOCGIFHWADDR:
Dmitry V. Levin40284242007-03-21 13:52:14 +0000147 case SIOCGIFTXQLEN:
Dmitry V. Levinecdd0bb2007-03-21 13:57:50 +0000148 case SIOCGIFMAP:
Dmitry V. Levin652e4482007-03-21 14:18:17 +0000149 if (umove(tcp, tcp->u_arg[2], &ifr) < 0)
150 tprintf(", %#lx", tcp->u_arg[2]);
151 else if (syserror(tcp)) {
Roland McGrath5687ff12004-07-12 07:13:06 +0000152 if (code == SIOCGIFNAME)
Roland McGrathe6a432d2005-02-02 20:25:17 +0000153 tprintf(", {ifr_index=%d, ifr_name=???}", ifr.ifr_ifindex);
154 else
155 tprintf(", {ifr_name=\"%s\", ???}", ifr.ifr_name);
156 } else if (code == SIOCGIFNAME)
157 tprintf(", {ifr_index=%d, ifr_name=\"%s\"}",
Roland McGrath5687ff12004-07-12 07:13:06 +0000158 ifr.ifr_ifindex, ifr.ifr_name);
Roland McGrathe6a432d2005-02-02 20:25:17 +0000159 else {
160 tprintf(", {ifr_name=\"%s\", ", ifr.ifr_name);
161 switch (code) {
162 case SIOCGIFINDEX:
163 tprintf("ifr_index=%d", ifr.ifr_ifindex);
164 break;
165 case SIOCGIFADDR:
166 str = "ifr_addr";
167 case SIOCGIFDSTADDR:
168 if (str == NULL)
169 str = "ifr_dstaddr";
170 case SIOCGIFBRDADDR:
171 if (str == NULL)
172 str = "ifr_broadaddr";
173 case SIOCGIFNETMASK:
174 if (str == NULL)
175 str = "ifr_netmask";
176 tprintf("%s={", str);
177 printxval(addrfams,
178 ifr.ifr_addr.sa_family,
179 "AF_???");
180 tprintf(", ");
181 print_addr(tcp, ((long) tcp->u_arg[2]
182 + offsetof (struct ifreq,
183 ifr_addr.sa_data)),
184 &ifr);
185 tprintf("}");
186 break;
187 case SIOCGIFHWADDR:
188 /* XXX Are there other hardware addresses
189 than 6-byte MACs? */
190 bytes = (unsigned char *) &ifr.ifr_hwaddr.sa_data;
191 tprintf("ifr_hwaddr=%02x:%02x:%02x:%02x:%02x:%02x",
192 bytes[0], bytes[1], bytes[2],
193 bytes[3], bytes[4], bytes[5]);
194 break;
195 case SIOCGIFFLAGS:
196 tprintf("ifr_flags=");
Roland McGrathb2dee132005-06-01 19:02:36 +0000197 printflags(iffflags, ifr.ifr_flags, "IFF_???");
Roland McGrathe6a432d2005-02-02 20:25:17 +0000198 break;
199 case SIOCGIFMETRIC:
200 tprintf("ifr_metric=%d", ifr.ifr_metric);
201 break;
202 case SIOCGIFMTU:
203 tprintf("ifr_mtu=%d", ifr.ifr_mtu);
204 break;
205 case SIOCGIFSLAVE:
206 tprintf("ifr_slave=\"%s\"", ifr.ifr_slave);
207 break;
Dmitry V. Levin40284242007-03-21 13:52:14 +0000208 case SIOCGIFTXQLEN:
209 tprintf("ifr_qlen=%d", ifr.ifr_qlen);
210 break;
Dmitry V. Levinecdd0bb2007-03-21 13:57:50 +0000211 case SIOCGIFMAP:
212 tprintf("ifr_map={mem_start=%#lx, "
213 "mem_end=%#lx, base_addr=%#x, "
214 "irq=%u, dma=%u, port=%u}",
215 ifr.ifr_map.mem_start,
216 ifr.ifr_map.mem_end,
217 (unsigned) ifr.ifr_map.base_addr,
218 (unsigned) ifr.ifr_map.irq,
219 (unsigned) ifr.ifr_map.dma,
220 (unsigned) ifr.ifr_map.port);
221 break;
Roland McGrathe6a432d2005-02-02 20:25:17 +0000222 }
223 tprintf("}");
224 }
Roland McGrath5687ff12004-07-12 07:13:06 +0000225 return 1;
226 case SIOCGIFCONF:
Dmitry V. Levin652e4482007-03-21 14:18:17 +0000227 if (umove(tcp, tcp->u_arg[2], &ifc) < 0) {
228 tprintf("???}");
229 return 1;
230 }
Roland McGrath5687ff12004-07-12 07:13:06 +0000231 tprintf("%d, ", ifc.ifc_len);
232 if (syserror(tcp)) {
233 tprintf("%lx", (unsigned long) ifc.ifc_buf);
234 } else if (ifc.ifc_buf == NULL) {
235 tprintf("NULL");
236 } else {
237 int i;
238 unsigned nifra = ifc.ifc_len / sizeof(struct ifreq);
239 struct ifreq ifra[nifra];
240 umoven(tcp, (unsigned long) ifc.ifc_buf, sizeof(ifra),
241 (char *) ifra);
242 tprintf("{");
243 for (i = 0; i < nifra; ++i ) {
244 if (i > 0)
245 tprintf(", ");
246 tprintf("{\"%s\", {",
247 ifra[i].ifr_name);
248 if (verbose(tcp)) {
249 printxval(addrfams,
250 ifra[i].ifr_addr.sa_family,
251 "AF_???");
252 tprintf(", ");
Roland McGrathe6a432d2005-02-02 20:25:17 +0000253 print_addr(tcp, ((long) tcp->u_arg[2]
254 + offsetof (struct ifreq,
255 ifr_addr.sa_data)
256 + ((char *) &ifra[i]
257 - (char *) &ifra[0])),
258 &ifra[i]);
Roland McGrath5687ff12004-07-12 07:13:06 +0000259 } else
260 tprintf("...");
261 tprintf("}}");
262 }
263 tprintf("}");
264 }
265 tprintf("}");
266 return 1;
267#endif
Wichert Akkerman76baf7c1999-02-19 00:21:36 +0000268 default:
269 return 0;
270 }
271}