Blame view
net/atm/ioctl.c
9.27 KB
b24413180 License cleanup: ... |
1 |
// SPDX-License-Identifier: GPL-2.0 |
1da177e4c Linux-2.6.12-rc2 |
2 3 4 5 |
/* ATM ioctl handling */ /* Written 1995-2000 by Werner Almesberger, EPFL LRC/ICA */ /* 2003 John Levon <levon@movementarian.org> */ |
99824461e net/atm: Convert ... |
6 |
#define pr_fmt(fmt) KBUILD_MODNAME ":%s: " fmt, __func__ |
1da177e4c Linux-2.6.12-rc2 |
7 |
|
1da177e4c Linux-2.6.12-rc2 |
8 9 10 11 12 13 14 |
#include <linux/module.h> #include <linux/kmod.h> #include <linux/net.h> /* struct socket, struct proto_ops */ #include <linux/atm.h> /* ATM stuff */ #include <linux/atmdev.h> #include <linux/atmclip.h> /* CLIP_*ENCAP */ #include <linux/atmarp.h> /* manifest constants */ |
4fc268d24 [PATCH] capable/c... |
15 |
#include <linux/capability.h> |
1da177e4c Linux-2.6.12-rc2 |
16 17 18 19 20 |
#include <linux/sonet.h> /* for ioctls */ #include <linux/atmsvc.h> #include <linux/atmmpc.h> #include <net/atmclip.h> #include <linux/atmlec.h> |
4a3e2f711 [NET] sem2mutex: ... |
21 |
#include <linux/mutex.h> |
1da177e4c Linux-2.6.12-rc2 |
22 |
#include <asm/ioctls.h> |
8865c418c atm: 32-bit ioctl... |
23 |
#include <net/compat.h> |
1da177e4c Linux-2.6.12-rc2 |
24 25 26 |
#include "resources.h" #include "signaling.h" /* for WAITING and sigd_attach */ |
295098e9f [ATM]: net/atm/io... |
27 |
#include "common.h" |
1da177e4c Linux-2.6.12-rc2 |
28 |
|
4a3e2f711 [NET] sem2mutex: ... |
29 |
static DEFINE_MUTEX(ioctl_mutex); |
1da177e4c Linux-2.6.12-rc2 |
30 31 32 33 34 |
static LIST_HEAD(ioctl_list); void register_atm_ioctl(struct atm_ioctl *ioctl) { |
4a3e2f711 [NET] sem2mutex: ... |
35 |
mutex_lock(&ioctl_mutex); |
1da177e4c Linux-2.6.12-rc2 |
36 |
list_add_tail(&ioctl->list, &ioctl_list); |
4a3e2f711 [NET] sem2mutex: ... |
37 |
mutex_unlock(&ioctl_mutex); |
1da177e4c Linux-2.6.12-rc2 |
38 |
} |
5ff7ef791 net/atm/ioctl.c: ... |
39 |
EXPORT_SYMBOL(register_atm_ioctl); |
1da177e4c Linux-2.6.12-rc2 |
40 41 42 |
void deregister_atm_ioctl(struct atm_ioctl *ioctl) { |
4a3e2f711 [NET] sem2mutex: ... |
43 |
mutex_lock(&ioctl_mutex); |
1da177e4c Linux-2.6.12-rc2 |
44 |
list_del(&ioctl->list); |
4a3e2f711 [NET] sem2mutex: ... |
45 |
mutex_unlock(&ioctl_mutex); |
1da177e4c Linux-2.6.12-rc2 |
46 |
} |
1da177e4c Linux-2.6.12-rc2 |
47 |
EXPORT_SYMBOL(deregister_atm_ioctl); |
5ff7ef791 net/atm/ioctl.c: ... |
48 49 |
static int do_vcc_ioctl(struct socket *sock, unsigned int cmd, unsigned long arg, int compat) |
1da177e4c Linux-2.6.12-rc2 |
50 51 52 53 |
{ struct sock *sk = sock->sk; struct atm_vcc *vcc; int error; |
5ff7ef791 net/atm/ioctl.c: ... |
54 |
struct list_head *pos; |
1da177e4c Linux-2.6.12-rc2 |
55 56 57 58 |
void __user *argp = (void __user *)arg; vcc = ATM_SD(sock); switch (cmd) { |
5ff7ef791 net/atm/ioctl.c: ... |
59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 |
case SIOCOUTQ: if (sock->state != SS_CONNECTED || !test_bit(ATM_VF_READY, &vcc->flags)) { error = -EINVAL; goto done; } error = put_user(sk->sk_sndbuf - sk_wmem_alloc_get(sk), (int __user *)argp) ? -EFAULT : 0; goto done; case SIOCINQ: { struct sk_buff *skb; if (sock->state != SS_CONNECTED) { error = -EINVAL; |
1da177e4c Linux-2.6.12-rc2 |
74 |
goto done; |
5ff7ef791 net/atm/ioctl.c: ... |
75 76 77 78 79 80 81 |
} skb = skb_peek(&sk->sk_receive_queue); error = put_user(skb ? skb->len : 0, (int __user *)argp) ? -EFAULT : 0; goto done; } case SIOCGSTAMP: /* borrowed from IP */ |
8865c418c atm: 32-bit ioctl... |
82 |
#ifdef CONFIG_COMPAT |
5ff7ef791 net/atm/ioctl.c: ... |
83 84 85 |
if (compat) error = compat_sock_get_timestamp(sk, argp); else |
8865c418c atm: 32-bit ioctl... |
86 |
#endif |
5ff7ef791 net/atm/ioctl.c: ... |
87 88 89 |
error = sock_get_timestamp(sk, argp); goto done; case SIOCGSTAMPNS: /* borrowed from IP */ |
8865c418c atm: 32-bit ioctl... |
90 |
#ifdef CONFIG_COMPAT |
5ff7ef791 net/atm/ioctl.c: ... |
91 92 93 |
if (compat) error = compat_sock_get_timestampns(sk, argp); else |
8865c418c atm: 32-bit ioctl... |
94 |
#endif |
5ff7ef791 net/atm/ioctl.c: ... |
95 96 97 |
error = sock_get_timestampns(sk, argp); goto done; case ATM_SETSC: |
e87cc4728 net: Convert net_... |
98 99 100 |
net_warn_ratelimited("ATM_SETSC is obsolete; used by %s:%d ", current->comm, task_pid_nr(current)); |
5ff7ef791 net/atm/ioctl.c: ... |
101 102 103 104 105 |
error = 0; goto done; case ATMSIGD_CTRL: if (!capable(CAP_NET_ADMIN)) { error = -EPERM; |
ae40eb1ef [NET]: Introduce ... |
106 |
goto done; |
5ff7ef791 net/atm/ioctl.c: ... |
107 108 109 110 111 112 113 114 115 116 |
} /* * The user/kernel protocol for exchanging signalling * info uses kernel pointers as opaque references, * so the holder of the file descriptor can scribble * on the kernel... so we should make sure that we * have the same privileges that /proc/kcore needs */ if (!capable(CAP_SYS_RAWIO)) { error = -EPERM; |
1da177e4c Linux-2.6.12-rc2 |
117 |
goto done; |
5ff7ef791 net/atm/ioctl.c: ... |
118 |
} |
8865c418c atm: 32-bit ioctl... |
119 |
#ifdef CONFIG_COMPAT |
5ff7ef791 net/atm/ioctl.c: ... |
120 121 122 123 |
/* WTF? I don't even want to _think_ about making this work for 32-bit userspace. TBH I don't really want to think about it at all. dwmw2. */ if (compat) { |
e87cc4728 net: Convert net_... |
124 125 |
net_warn_ratelimited("32-bit task cannot be atmsigd "); |
5ff7ef791 net/atm/ioctl.c: ... |
126 127 128 |
error = -EINVAL; goto done; } |
8865c418c atm: 32-bit ioctl... |
129 |
#endif |
5ff7ef791 net/atm/ioctl.c: ... |
130 131 132 133 134 135 136 137 138 139 |
error = sigd_attach(vcc); if (!error) sock->state = SS_CONNECTED; goto done; case ATM_SETBACKEND: case ATM_NEWBACKENDIF: { atm_backend_t backend; error = get_user(backend, (atm_backend_t __user *)argp); if (error) |
1da177e4c Linux-2.6.12-rc2 |
140 |
goto done; |
5ff7ef791 net/atm/ioctl.c: ... |
141 142 143 |
switch (backend) { case ATM_BACKEND_PPP: request_module("pppoatm"); |
e2c4b7215 [ATM]: net/atm/io... |
144 |
break; |
5ff7ef791 net/atm/ioctl.c: ... |
145 146 |
case ATM_BACKEND_BR2684: request_module("br2684"); |
1da177e4c Linux-2.6.12-rc2 |
147 |
break; |
5ff7ef791 net/atm/ioctl.c: ... |
148 149 150 151 152 153 154 155 156 157 158 159 160 |
} break; } case ATMMPC_CTRL: case ATMMPC_DATA: request_module("mpoa"); break; case ATMARPD_CTRL: request_module("clip"); break; case ATMLEC_CTRL: request_module("lec"); break; |
1da177e4c Linux-2.6.12-rc2 |
161 |
} |
1da177e4c Linux-2.6.12-rc2 |
162 |
error = -ENOIOCTLCMD; |
4a3e2f711 [NET] sem2mutex: ... |
163 |
mutex_lock(&ioctl_mutex); |
1da177e4c Linux-2.6.12-rc2 |
164 |
list_for_each(pos, &ioctl_list) { |
5ff7ef791 net/atm/ioctl.c: ... |
165 |
struct atm_ioctl *ic = list_entry(pos, struct atm_ioctl, list); |
1da177e4c Linux-2.6.12-rc2 |
166 167 168 169 170 171 172 |
if (try_module_get(ic->owner)) { error = ic->ioctl(sock, cmd, arg); module_put(ic->owner); if (error != -ENOIOCTLCMD) break; } } |
4a3e2f711 [NET] sem2mutex: ... |
173 |
mutex_unlock(&ioctl_mutex); |
1da177e4c Linux-2.6.12-rc2 |
174 175 176 |
if (error != -ENOIOCTLCMD) goto done; |
8865c418c atm: 32-bit ioctl... |
177 |
error = atm_dev_ioctl(cmd, argp, compat); |
1da177e4c Linux-2.6.12-rc2 |
178 179 180 181 |
done: return error; } |
8865c418c atm: 32-bit ioctl... |
182 |
|
8865c418c atm: 32-bit ioctl... |
183 184 185 186 187 188 |
int vcc_ioctl(struct socket *sock, unsigned int cmd, unsigned long arg) { return do_vcc_ioctl(sock, cmd, arg, 0); } #ifdef CONFIG_COMPAT |
805003a41 net/atm: move all... |
189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 |
/* * FIXME: * The compat_ioctl handling is duplicated, using both these conversion * routines and the compat argument to the actual handlers. Both * versions are somewhat incomplete and should be merged, e.g. by * moving the ioctl number translation into the actual handlers and * killing the conversion code. * * -arnd, November 2009 */ #define ATM_GETLINKRATE32 _IOW('a', ATMIOC_ITF+1, struct compat_atmif_sioc) #define ATM_GETNAMES32 _IOW('a', ATMIOC_ITF+3, struct compat_atm_iobuf) #define ATM_GETTYPE32 _IOW('a', ATMIOC_ITF+4, struct compat_atmif_sioc) #define ATM_GETESI32 _IOW('a', ATMIOC_ITF+5, struct compat_atmif_sioc) #define ATM_GETADDR32 _IOW('a', ATMIOC_ITF+6, struct compat_atmif_sioc) #define ATM_RSTADDR32 _IOW('a', ATMIOC_ITF+7, struct compat_atmif_sioc) #define ATM_ADDADDR32 _IOW('a', ATMIOC_ITF+8, struct compat_atmif_sioc) #define ATM_DELADDR32 _IOW('a', ATMIOC_ITF+9, struct compat_atmif_sioc) #define ATM_GETCIRANGE32 _IOW('a', ATMIOC_ITF+10, struct compat_atmif_sioc) #define ATM_SETCIRANGE32 _IOW('a', ATMIOC_ITF+11, struct compat_atmif_sioc) #define ATM_SETESI32 _IOW('a', ATMIOC_ITF+12, struct compat_atmif_sioc) #define ATM_SETESIF32 _IOW('a', ATMIOC_ITF+13, struct compat_atmif_sioc) #define ATM_GETSTAT32 _IOW('a', ATMIOC_SARCOM+0, struct compat_atmif_sioc) #define ATM_GETSTATZ32 _IOW('a', ATMIOC_SARCOM+1, struct compat_atmif_sioc) #define ATM_GETLOOP32 _IOW('a', ATMIOC_SARCOM+2, struct compat_atmif_sioc) #define ATM_SETLOOP32 _IOW('a', ATMIOC_SARCOM+3, struct compat_atmif_sioc) #define ATM_QUERYLOOP32 _IOW('a', ATMIOC_SARCOM+4, struct compat_atmif_sioc) static struct { unsigned int cmd32; unsigned int cmd; } atm_ioctl_map[] = { { ATM_GETLINKRATE32, ATM_GETLINKRATE }, { ATM_GETNAMES32, ATM_GETNAMES }, { ATM_GETTYPE32, ATM_GETTYPE }, { ATM_GETESI32, ATM_GETESI }, { ATM_GETADDR32, ATM_GETADDR }, { ATM_RSTADDR32, ATM_RSTADDR }, { ATM_ADDADDR32, ATM_ADDADDR }, { ATM_DELADDR32, ATM_DELADDR }, { ATM_GETCIRANGE32, ATM_GETCIRANGE }, { ATM_SETCIRANGE32, ATM_SETCIRANGE }, { ATM_SETESI32, ATM_SETESI }, { ATM_SETESIF32, ATM_SETESIF }, { ATM_GETSTAT32, ATM_GETSTAT }, { ATM_GETSTATZ32, ATM_GETSTATZ }, { ATM_GETLOOP32, ATM_GETLOOP }, { ATM_SETLOOP32, ATM_SETLOOP }, { ATM_QUERYLOOP32, ATM_QUERYLOOP }, }; #define NR_ATM_IOCTL ARRAY_SIZE(atm_ioctl_map) static int do_atm_iobuf(struct socket *sock, unsigned int cmd, unsigned long arg) { struct atm_iobuf __user *iobuf; struct compat_atm_iobuf __user *iobuf32; u32 data; void __user *datap; int len, err; iobuf = compat_alloc_user_space(sizeof(*iobuf)); iobuf32 = compat_ptr(arg); if (get_user(len, &iobuf32->length) || get_user(data, &iobuf32->buffer)) return -EFAULT; datap = compat_ptr(data); if (put_user(len, &iobuf->length) || put_user(datap, &iobuf->buffer)) return -EFAULT; err = do_vcc_ioctl(sock, cmd, (unsigned long) iobuf, 0); if (!err) { if (copy_in_user(&iobuf32->length, &iobuf->length, sizeof(int))) err = -EFAULT; } return err; } static int do_atmif_sioc(struct socket *sock, unsigned int cmd, unsigned long arg) { struct atmif_sioc __user *sioc; struct compat_atmif_sioc __user *sioc32; u32 data; void __user *datap; int err; sioc = compat_alloc_user_space(sizeof(*sioc)); sioc32 = compat_ptr(arg); |
5ff7ef791 net/atm/ioctl.c: ... |
284 285 |
if (copy_in_user(&sioc->number, &sioc32->number, 2 * sizeof(int)) || get_user(data, &sioc32->arg)) |
805003a41 net/atm: move all... |
286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 |
return -EFAULT; datap = compat_ptr(data); if (put_user(datap, &sioc->arg)) return -EFAULT; err = do_vcc_ioctl(sock, cmd, (unsigned long) sioc, 0); if (!err) { if (copy_in_user(&sioc32->length, &sioc->length, sizeof(int))) err = -EFAULT; } return err; } static int do_atm_ioctl(struct socket *sock, unsigned int cmd32, unsigned long arg) { int i; unsigned int cmd = 0; switch (cmd32) { case SONET_GETSTAT: case SONET_GETSTATZ: case SONET_GETDIAG: case SONET_SETDIAG: case SONET_CLRDIAG: case SONET_SETFRAMING: case SONET_GETFRAMING: case SONET_GETFRSENSE: return do_atmif_sioc(sock, cmd32, arg); } for (i = 0; i < NR_ATM_IOCTL; i++) { if (cmd32 == atm_ioctl_map[i].cmd32) { cmd = atm_ioctl_map[i].cmd; break; } } if (i == NR_ATM_IOCTL) return -EINVAL; switch (cmd) { case ATM_GETNAMES: return do_atm_iobuf(sock, cmd, arg); case ATM_GETLINKRATE: case ATM_GETTYPE: case ATM_GETESI: case ATM_GETADDR: case ATM_RSTADDR: case ATM_ADDADDR: case ATM_DELADDR: case ATM_GETCIRANGE: case ATM_SETCIRANGE: case ATM_SETESI: case ATM_SETESIF: case ATM_GETSTAT: case ATM_GETSTATZ: case ATM_GETLOOP: case ATM_SETLOOP: case ATM_QUERYLOOP: return do_atmif_sioc(sock, cmd, arg); } return -EINVAL; } int vcc_compat_ioctl(struct socket *sock, unsigned int cmd, unsigned long arg) |
8865c418c atm: 32-bit ioctl... |
356 |
{ |
805003a41 net/atm: move all... |
357 358 359 360 361 362 363 |
int ret; ret = do_vcc_ioctl(sock, cmd, arg, 1); if (ret != -ENOIOCTLCMD) return ret; return do_atm_ioctl(sock, cmd, arg); |
8865c418c atm: 32-bit ioctl... |
364 365 |
} #endif |