Skip to content

Commit 709566d

Browse files
arndbdavem330
authored andcommitted
net: socket: rework SIOC?IFMAP ioctls
SIOCGIFMAP and SIOCSIFMAP currently require compat_alloc_user_space() and copy_in_user() for compat mode. Move the compat handling into the location where the structures are actually used, to avoid using those interfaces and get a clearer implementation. Reviewed-by: Christoph Hellwig <hch@lst.de> Signed-off-by: Arnd Bergmann <arnd@arndb.de> Signed-off-by: David S. Miller <davem@davemloft.net>
1 parent dd98d28 commit 709566d

File tree

2 files changed

+54
-50
lines changed

2 files changed

+54
-50
lines changed

net/core/dev_ioctl.c

Lines changed: 52 additions & 13 deletions
Original file line numberDiff line numberDiff line change
@@ -98,6 +98,56 @@ int dev_ifconf(struct net *net, struct ifconf *ifc, int size)
9898
return 0;
9999
}
100100

101+
static int dev_getifmap(struct net_device *dev, struct ifreq *ifr)
102+
{
103+
struct ifmap *ifmap = &ifr->ifr_map;
104+
105+
if (in_compat_syscall()) {
106+
struct compat_ifmap *cifmap = (struct compat_ifmap *)ifmap;
107+
108+
cifmap->mem_start = dev->mem_start;
109+
cifmap->mem_end = dev->mem_end;
110+
cifmap->base_addr = dev->base_addr;
111+
cifmap->irq = dev->irq;
112+
cifmap->dma = dev->dma;
113+
cifmap->port = dev->if_port;
114+
115+
return 0;
116+
}
117+
118+
ifmap->mem_start = dev->mem_start;
119+
ifmap->mem_end = dev->mem_end;
120+
ifmap->base_addr = dev->base_addr;
121+
ifmap->irq = dev->irq;
122+
ifmap->dma = dev->dma;
123+
ifmap->port = dev->if_port;
124+
125+
return 0;
126+
}
127+
128+
static int dev_setifmap(struct net_device *dev, struct ifreq *ifr)
129+
{
130+
struct compat_ifmap *cifmap = (struct compat_ifmap *)&ifr->ifr_map;
131+
132+
if (!dev->netdev_ops->ndo_set_config)
133+
return -EOPNOTSUPP;
134+
135+
if (in_compat_syscall()) {
136+
struct ifmap ifmap = {
137+
.mem_start = cifmap->mem_start,
138+
.mem_end = cifmap->mem_end,
139+
.base_addr = cifmap->base_addr,
140+
.irq = cifmap->irq,
141+
.dma = cifmap->dma,
142+
.port = cifmap->port,
143+
};
144+
145+
return dev->netdev_ops->ndo_set_config(dev, &ifmap);
146+
}
147+
148+
return dev->netdev_ops->ndo_set_config(dev, &ifr->ifr_map);
149+
}
150+
101151
/*
102152
* Perform the SIOCxIFxxx calls, inside rcu_read_lock()
103153
*/
@@ -128,13 +178,7 @@ static int dev_ifsioc_locked(struct net *net, struct ifreq *ifr, unsigned int cm
128178
break;
129179

130180
case SIOCGIFMAP:
131-
ifr->ifr_map.mem_start = dev->mem_start;
132-
ifr->ifr_map.mem_end = dev->mem_end;
133-
ifr->ifr_map.base_addr = dev->base_addr;
134-
ifr->ifr_map.irq = dev->irq;
135-
ifr->ifr_map.dma = dev->dma;
136-
ifr->ifr_map.port = dev->if_port;
137-
return 0;
181+
return dev_getifmap(dev, ifr);
138182

139183
case SIOCGIFINDEX:
140184
ifr->ifr_ifindex = dev->ifindex;
@@ -275,12 +319,7 @@ static int dev_ifsioc(struct net *net, struct ifreq *ifr, unsigned int cmd)
275319
return 0;
276320

277321
case SIOCSIFMAP:
278-
if (ops->ndo_set_config) {
279-
if (!netif_device_present(dev))
280-
return -ENODEV;
281-
return ops->ndo_set_config(dev, &ifr->ifr_map);
282-
}
283-
return -EOPNOTSUPP;
322+
return dev_setifmap(dev, ifr);
284323

285324
case SIOCADDMULTI:
286325
if (!ops->ndo_set_rx_mode ||

net/socket.c

Lines changed: 2 additions & 37 deletions
Original file line numberDiff line numberDiff line change
@@ -3241,40 +3241,6 @@ static int compat_ifreq_ioctl(struct net *net, struct socket *sock,
32413241
return err;
32423242
}
32433243

3244-
static int compat_sioc_ifmap(struct net *net, unsigned int cmd,
3245-
struct compat_ifreq __user *uifr32)
3246-
{
3247-
struct ifreq ifr;
3248-
struct compat_ifmap __user *uifmap32;
3249-
int err;
3250-
3251-
uifmap32 = &uifr32->ifr_ifru.ifru_map;
3252-
err = copy_from_user(&ifr, uifr32, sizeof(ifr.ifr_name));
3253-
err |= get_user(ifr.ifr_map.mem_start, &uifmap32->mem_start);
3254-
err |= get_user(ifr.ifr_map.mem_end, &uifmap32->mem_end);
3255-
err |= get_user(ifr.ifr_map.base_addr, &uifmap32->base_addr);
3256-
err |= get_user(ifr.ifr_map.irq, &uifmap32->irq);
3257-
err |= get_user(ifr.ifr_map.dma, &uifmap32->dma);
3258-
err |= get_user(ifr.ifr_map.port, &uifmap32->port);
3259-
if (err)
3260-
return -EFAULT;
3261-
3262-
err = dev_ioctl(net, cmd, &ifr, NULL);
3263-
3264-
if (cmd == SIOCGIFMAP && !err) {
3265-
err = copy_to_user(uifr32, &ifr, sizeof(ifr.ifr_name));
3266-
err |= put_user(ifr.ifr_map.mem_start, &uifmap32->mem_start);
3267-
err |= put_user(ifr.ifr_map.mem_end, &uifmap32->mem_end);
3268-
err |= put_user(ifr.ifr_map.base_addr, &uifmap32->base_addr);
3269-
err |= put_user(ifr.ifr_map.irq, &uifmap32->irq);
3270-
err |= put_user(ifr.ifr_map.dma, &uifmap32->dma);
3271-
err |= put_user(ifr.ifr_map.port, &uifmap32->port);
3272-
if (err)
3273-
err = -EFAULT;
3274-
}
3275-
return err;
3276-
}
3277-
32783244
/* Since old style bridge ioctl's endup using SIOCDEVPRIVATE
32793245
* for some operations; this forces use of the newer bridge-utils that
32803246
* use compatible ioctls
@@ -3308,9 +3274,6 @@ static int compat_sock_ioctl_trans(struct file *file, struct socket *sock,
33083274
return compat_dev_ifconf(net, argp);
33093275
case SIOCWANDEV:
33103276
return compat_siocwandev(net, argp);
3311-
case SIOCGIFMAP:
3312-
case SIOCSIFMAP:
3313-
return compat_sioc_ifmap(net, cmd, argp);
33143277
case SIOCGSTAMP_OLD:
33153278
case SIOCGSTAMPNS_OLD:
33163279
if (!sock->ops->gettstamp)
@@ -3340,6 +3303,8 @@ static int compat_sock_ioctl_trans(struct file *file, struct socket *sock,
33403303

33413304
case SIOCGIFFLAGS:
33423305
case SIOCSIFFLAGS:
3306+
case SIOCGIFMAP:
3307+
case SIOCSIFMAP:
33433308
case SIOCGIFMETRIC:
33443309
case SIOCSIFMETRIC:
33453310
case SIOCGIFMTU:

0 commit comments

Comments
 (0)