diff options
Diffstat (limited to 'net/ipx/af_ipx.c')
-rw-r--r-- | net/ipx/af_ipx.c | 62 |
1 files changed, 51 insertions, 11 deletions
diff --git a/net/ipx/af_ipx.c b/net/ipx/af_ipx.c index 66c7a20011f3..da3d21c41d90 100644 --- a/net/ipx/af_ipx.c +++ b/net/ipx/af_ipx.c | |||
@@ -40,6 +40,7 @@ | |||
40 | #include <linux/net.h> | 40 | #include <linux/net.h> |
41 | #include <linux/netdevice.h> | 41 | #include <linux/netdevice.h> |
42 | #include <linux/uio.h> | 42 | #include <linux/uio.h> |
43 | #include <linux/slab.h> | ||
43 | #include <linux/skbuff.h> | 44 | #include <linux/skbuff.h> |
44 | #include <linux/smp_lock.h> | 45 | #include <linux/smp_lock.h> |
45 | #include <linux/socket.h> | 46 | #include <linux/socket.h> |
@@ -1298,6 +1299,7 @@ static int ipx_setsockopt(struct socket *sock, int level, int optname, | |||
1298 | int opt; | 1299 | int opt; |
1299 | int rc = -EINVAL; | 1300 | int rc = -EINVAL; |
1300 | 1301 | ||
1302 | lock_kernel(); | ||
1301 | if (optlen != sizeof(int)) | 1303 | if (optlen != sizeof(int)) |
1302 | goto out; | 1304 | goto out; |
1303 | 1305 | ||
@@ -1312,6 +1314,7 @@ static int ipx_setsockopt(struct socket *sock, int level, int optname, | |||
1312 | ipx_sk(sk)->type = opt; | 1314 | ipx_sk(sk)->type = opt; |
1313 | rc = 0; | 1315 | rc = 0; |
1314 | out: | 1316 | out: |
1317 | unlock_kernel(); | ||
1315 | return rc; | 1318 | return rc; |
1316 | } | 1319 | } |
1317 | 1320 | ||
@@ -1323,6 +1326,7 @@ static int ipx_getsockopt(struct socket *sock, int level, int optname, | |||
1323 | int len; | 1326 | int len; |
1324 | int rc = -ENOPROTOOPT; | 1327 | int rc = -ENOPROTOOPT; |
1325 | 1328 | ||
1329 | lock_kernel(); | ||
1326 | if (!(level == SOL_IPX && optname == IPX_TYPE)) | 1330 | if (!(level == SOL_IPX && optname == IPX_TYPE)) |
1327 | goto out; | 1331 | goto out; |
1328 | 1332 | ||
@@ -1343,6 +1347,7 @@ static int ipx_getsockopt(struct socket *sock, int level, int optname, | |||
1343 | 1347 | ||
1344 | rc = 0; | 1348 | rc = 0; |
1345 | out: | 1349 | out: |
1350 | unlock_kernel(); | ||
1346 | return rc; | 1351 | return rc; |
1347 | } | 1352 | } |
1348 | 1353 | ||
@@ -1352,12 +1357,13 @@ static struct proto ipx_proto = { | |||
1352 | .obj_size = sizeof(struct ipx_sock), | 1357 | .obj_size = sizeof(struct ipx_sock), |
1353 | }; | 1358 | }; |
1354 | 1359 | ||
1355 | static int ipx_create(struct net *net, struct socket *sock, int protocol) | 1360 | static int ipx_create(struct net *net, struct socket *sock, int protocol, |
1361 | int kern) | ||
1356 | { | 1362 | { |
1357 | int rc = -ESOCKTNOSUPPORT; | 1363 | int rc = -ESOCKTNOSUPPORT; |
1358 | struct sock *sk; | 1364 | struct sock *sk; |
1359 | 1365 | ||
1360 | if (net != &init_net) | 1366 | if (!net_eq(net, &init_net)) |
1361 | return -EAFNOSUPPORT; | 1367 | return -EAFNOSUPPORT; |
1362 | 1368 | ||
1363 | /* | 1369 | /* |
@@ -1390,6 +1396,7 @@ static int ipx_release(struct socket *sock) | |||
1390 | if (!sk) | 1396 | if (!sk) |
1391 | goto out; | 1397 | goto out; |
1392 | 1398 | ||
1399 | lock_kernel(); | ||
1393 | if (!sock_flag(sk, SOCK_DEAD)) | 1400 | if (!sock_flag(sk, SOCK_DEAD)) |
1394 | sk->sk_state_change(sk); | 1401 | sk->sk_state_change(sk); |
1395 | 1402 | ||
@@ -1397,6 +1404,7 @@ static int ipx_release(struct socket *sock) | |||
1397 | sock->sk = NULL; | 1404 | sock->sk = NULL; |
1398 | sk_refcnt_debug_release(sk); | 1405 | sk_refcnt_debug_release(sk); |
1399 | ipx_destroy_socket(sk); | 1406 | ipx_destroy_socket(sk); |
1407 | unlock_kernel(); | ||
1400 | out: | 1408 | out: |
1401 | return 0; | 1409 | return 0; |
1402 | } | 1410 | } |
@@ -1424,7 +1432,8 @@ static __be16 ipx_first_free_socketnum(struct ipx_interface *intrfc) | |||
1424 | return htons(socketNum); | 1432 | return htons(socketNum); |
1425 | } | 1433 | } |
1426 | 1434 | ||
1427 | static int ipx_bind(struct socket *sock, struct sockaddr *uaddr, int addr_len) | 1435 | static int __ipx_bind(struct socket *sock, |
1436 | struct sockaddr *uaddr, int addr_len) | ||
1428 | { | 1437 | { |
1429 | struct sock *sk = sock->sk; | 1438 | struct sock *sk = sock->sk; |
1430 | struct ipx_sock *ipxs = ipx_sk(sk); | 1439 | struct ipx_sock *ipxs = ipx_sk(sk); |
@@ -1519,6 +1528,17 @@ out: | |||
1519 | return rc; | 1528 | return rc; |
1520 | } | 1529 | } |
1521 | 1530 | ||
1531 | static int ipx_bind(struct socket *sock, struct sockaddr *uaddr, int addr_len) | ||
1532 | { | ||
1533 | int rc; | ||
1534 | |||
1535 | lock_kernel(); | ||
1536 | rc = __ipx_bind(sock, uaddr, addr_len); | ||
1537 | unlock_kernel(); | ||
1538 | |||
1539 | return rc; | ||
1540 | } | ||
1541 | |||
1522 | static int ipx_connect(struct socket *sock, struct sockaddr *uaddr, | 1542 | static int ipx_connect(struct socket *sock, struct sockaddr *uaddr, |
1523 | int addr_len, int flags) | 1543 | int addr_len, int flags) |
1524 | { | 1544 | { |
@@ -1531,6 +1551,7 @@ static int ipx_connect(struct socket *sock, struct sockaddr *uaddr, | |||
1531 | sk->sk_state = TCP_CLOSE; | 1551 | sk->sk_state = TCP_CLOSE; |
1532 | sock->state = SS_UNCONNECTED; | 1552 | sock->state = SS_UNCONNECTED; |
1533 | 1553 | ||
1554 | lock_kernel(); | ||
1534 | if (addr_len != sizeof(*addr)) | 1555 | if (addr_len != sizeof(*addr)) |
1535 | goto out; | 1556 | goto out; |
1536 | addr = (struct sockaddr_ipx *)uaddr; | 1557 | addr = (struct sockaddr_ipx *)uaddr; |
@@ -1550,7 +1571,7 @@ static int ipx_connect(struct socket *sock, struct sockaddr *uaddr, | |||
1550 | IPX_NODE_LEN); | 1571 | IPX_NODE_LEN); |
1551 | #endif /* CONFIG_IPX_INTERN */ | 1572 | #endif /* CONFIG_IPX_INTERN */ |
1552 | 1573 | ||
1553 | rc = ipx_bind(sock, (struct sockaddr *)&uaddr, | 1574 | rc = __ipx_bind(sock, (struct sockaddr *)&uaddr, |
1554 | sizeof(struct sockaddr_ipx)); | 1575 | sizeof(struct sockaddr_ipx)); |
1555 | if (rc) | 1576 | if (rc) |
1556 | goto out; | 1577 | goto out; |
@@ -1577,6 +1598,7 @@ static int ipx_connect(struct socket *sock, struct sockaddr *uaddr, | |||
1577 | ipxrtr_put(rt); | 1598 | ipxrtr_put(rt); |
1578 | rc = 0; | 1599 | rc = 0; |
1579 | out: | 1600 | out: |
1601 | unlock_kernel(); | ||
1580 | return rc; | 1602 | return rc; |
1581 | } | 1603 | } |
1582 | 1604 | ||
@@ -1592,6 +1614,7 @@ static int ipx_getname(struct socket *sock, struct sockaddr *uaddr, | |||
1592 | 1614 | ||
1593 | *uaddr_len = sizeof(struct sockaddr_ipx); | 1615 | *uaddr_len = sizeof(struct sockaddr_ipx); |
1594 | 1616 | ||
1617 | lock_kernel(); | ||
1595 | if (peer) { | 1618 | if (peer) { |
1596 | rc = -ENOTCONN; | 1619 | rc = -ENOTCONN; |
1597 | if (sk->sk_state != TCP_ESTABLISHED) | 1620 | if (sk->sk_state != TCP_ESTABLISHED) |
@@ -1626,6 +1649,19 @@ static int ipx_getname(struct socket *sock, struct sockaddr *uaddr, | |||
1626 | 1649 | ||
1627 | rc = 0; | 1650 | rc = 0; |
1628 | out: | 1651 | out: |
1652 | unlock_kernel(); | ||
1653 | return rc; | ||
1654 | } | ||
1655 | |||
1656 | static unsigned int ipx_datagram_poll(struct file *file, struct socket *sock, | ||
1657 | poll_table *wait) | ||
1658 | { | ||
1659 | int rc; | ||
1660 | |||
1661 | lock_kernel(); | ||
1662 | rc = datagram_poll(file, sock, wait); | ||
1663 | unlock_kernel(); | ||
1664 | |||
1629 | return rc; | 1665 | return rc; |
1630 | } | 1666 | } |
1631 | 1667 | ||
@@ -1700,6 +1736,7 @@ static int ipx_sendmsg(struct kiocb *iocb, struct socket *sock, | |||
1700 | int rc = -EINVAL; | 1736 | int rc = -EINVAL; |
1701 | int flags = msg->msg_flags; | 1737 | int flags = msg->msg_flags; |
1702 | 1738 | ||
1739 | lock_kernel(); | ||
1703 | /* Socket gets bound below anyway */ | 1740 | /* Socket gets bound below anyway */ |
1704 | /* if (sk->sk_zapped) | 1741 | /* if (sk->sk_zapped) |
1705 | return -EIO; */ /* Socket not bound */ | 1742 | return -EIO; */ /* Socket not bound */ |
@@ -1723,7 +1760,7 @@ static int ipx_sendmsg(struct kiocb *iocb, struct socket *sock, | |||
1723 | memcpy(uaddr.sipx_node, ipxs->intrfc->if_node, | 1760 | memcpy(uaddr.sipx_node, ipxs->intrfc->if_node, |
1724 | IPX_NODE_LEN); | 1761 | IPX_NODE_LEN); |
1725 | #endif | 1762 | #endif |
1726 | rc = ipx_bind(sock, (struct sockaddr *)&uaddr, | 1763 | rc = __ipx_bind(sock, (struct sockaddr *)&uaddr, |
1727 | sizeof(struct sockaddr_ipx)); | 1764 | sizeof(struct sockaddr_ipx)); |
1728 | if (rc) | 1765 | if (rc) |
1729 | goto out; | 1766 | goto out; |
@@ -1751,6 +1788,7 @@ static int ipx_sendmsg(struct kiocb *iocb, struct socket *sock, | |||
1751 | if (rc >= 0) | 1788 | if (rc >= 0) |
1752 | rc = len; | 1789 | rc = len; |
1753 | out: | 1790 | out: |
1791 | unlock_kernel(); | ||
1754 | return rc; | 1792 | return rc; |
1755 | } | 1793 | } |
1756 | 1794 | ||
@@ -1765,6 +1803,7 @@ static int ipx_recvmsg(struct kiocb *iocb, struct socket *sock, | |||
1765 | struct sk_buff *skb; | 1803 | struct sk_buff *skb; |
1766 | int copied, rc; | 1804 | int copied, rc; |
1767 | 1805 | ||
1806 | lock_kernel(); | ||
1768 | /* put the autobinding in */ | 1807 | /* put the autobinding in */ |
1769 | if (!ipxs->port) { | 1808 | if (!ipxs->port) { |
1770 | struct sockaddr_ipx uaddr; | 1809 | struct sockaddr_ipx uaddr; |
@@ -1779,7 +1818,7 @@ static int ipx_recvmsg(struct kiocb *iocb, struct socket *sock, | |||
1779 | memcpy(uaddr.sipx_node, ipxs->intrfc->if_node, IPX_NODE_LEN); | 1818 | memcpy(uaddr.sipx_node, ipxs->intrfc->if_node, IPX_NODE_LEN); |
1780 | #endif /* CONFIG_IPX_INTERN */ | 1819 | #endif /* CONFIG_IPX_INTERN */ |
1781 | 1820 | ||
1782 | rc = ipx_bind(sock, (struct sockaddr *)&uaddr, | 1821 | rc = __ipx_bind(sock, (struct sockaddr *)&uaddr, |
1783 | sizeof(struct sockaddr_ipx)); | 1822 | sizeof(struct sockaddr_ipx)); |
1784 | if (rc) | 1823 | if (rc) |
1785 | goto out; | 1824 | goto out; |
@@ -1823,6 +1862,7 @@ static int ipx_recvmsg(struct kiocb *iocb, struct socket *sock, | |||
1823 | out_free: | 1862 | out_free: |
1824 | skb_free_datagram(sk, skb); | 1863 | skb_free_datagram(sk, skb); |
1825 | out: | 1864 | out: |
1865 | unlock_kernel(); | ||
1826 | return rc; | 1866 | return rc; |
1827 | } | 1867 | } |
1828 | 1868 | ||
@@ -1834,6 +1874,7 @@ static int ipx_ioctl(struct socket *sock, unsigned int cmd, unsigned long arg) | |||
1834 | struct sock *sk = sock->sk; | 1874 | struct sock *sk = sock->sk; |
1835 | void __user *argp = (void __user *)arg; | 1875 | void __user *argp = (void __user *)arg; |
1836 | 1876 | ||
1877 | lock_kernel(); | ||
1837 | switch (cmd) { | 1878 | switch (cmd) { |
1838 | case TIOCOUTQ: | 1879 | case TIOCOUTQ: |
1839 | amount = sk->sk_sndbuf - sk_wmem_alloc_get(sk); | 1880 | amount = sk->sk_sndbuf - sk_wmem_alloc_get(sk); |
@@ -1896,6 +1937,7 @@ static int ipx_ioctl(struct socket *sock, unsigned int cmd, unsigned long arg) | |||
1896 | rc = -ENOIOCTLCMD; | 1937 | rc = -ENOIOCTLCMD; |
1897 | break; | 1938 | break; |
1898 | } | 1939 | } |
1940 | unlock_kernel(); | ||
1899 | 1941 | ||
1900 | return rc; | 1942 | return rc; |
1901 | } | 1943 | } |
@@ -1927,13 +1969,13 @@ static int ipx_compat_ioctl(struct socket *sock, unsigned int cmd, unsigned long | |||
1927 | * Socket family declarations | 1969 | * Socket family declarations |
1928 | */ | 1970 | */ |
1929 | 1971 | ||
1930 | static struct net_proto_family ipx_family_ops = { | 1972 | static const struct net_proto_family ipx_family_ops = { |
1931 | .family = PF_IPX, | 1973 | .family = PF_IPX, |
1932 | .create = ipx_create, | 1974 | .create = ipx_create, |
1933 | .owner = THIS_MODULE, | 1975 | .owner = THIS_MODULE, |
1934 | }; | 1976 | }; |
1935 | 1977 | ||
1936 | static const struct proto_ops SOCKOPS_WRAPPED(ipx_dgram_ops) = { | 1978 | static const struct proto_ops ipx_dgram_ops = { |
1937 | .family = PF_IPX, | 1979 | .family = PF_IPX, |
1938 | .owner = THIS_MODULE, | 1980 | .owner = THIS_MODULE, |
1939 | .release = ipx_release, | 1981 | .release = ipx_release, |
@@ -1942,7 +1984,7 @@ static const struct proto_ops SOCKOPS_WRAPPED(ipx_dgram_ops) = { | |||
1942 | .socketpair = sock_no_socketpair, | 1984 | .socketpair = sock_no_socketpair, |
1943 | .accept = sock_no_accept, | 1985 | .accept = sock_no_accept, |
1944 | .getname = ipx_getname, | 1986 | .getname = ipx_getname, |
1945 | .poll = datagram_poll, | 1987 | .poll = ipx_datagram_poll, |
1946 | .ioctl = ipx_ioctl, | 1988 | .ioctl = ipx_ioctl, |
1947 | #ifdef CONFIG_COMPAT | 1989 | #ifdef CONFIG_COMPAT |
1948 | .compat_ioctl = ipx_compat_ioctl, | 1990 | .compat_ioctl = ipx_compat_ioctl, |
@@ -1957,8 +1999,6 @@ static const struct proto_ops SOCKOPS_WRAPPED(ipx_dgram_ops) = { | |||
1957 | .sendpage = sock_no_sendpage, | 1999 | .sendpage = sock_no_sendpage, |
1958 | }; | 2000 | }; |
1959 | 2001 | ||
1960 | SOCKOPS_WRAP(ipx_dgram, PF_IPX); | ||
1961 | |||
1962 | static struct packet_type ipx_8023_packet_type __read_mostly = { | 2002 | static struct packet_type ipx_8023_packet_type __read_mostly = { |
1963 | .type = cpu_to_be16(ETH_P_802_3), | 2003 | .type = cpu_to_be16(ETH_P_802_3), |
1964 | .func = ipx_rcv, | 2004 | .func = ipx_rcv, |