diff options
Diffstat (limited to 'net/kcm/kcmsock.c')
-rw-r--r-- | net/kcm/kcmsock.c | 33 |
1 files changed, 23 insertions, 10 deletions
diff --git a/net/kcm/kcmsock.c b/net/kcm/kcmsock.c index a6cd0712e063..516cfad71b85 100644 --- a/net/kcm/kcmsock.c +++ b/net/kcm/kcmsock.c | |||
@@ -1381,24 +1381,32 @@ static int kcm_attach(struct socket *sock, struct socket *csock, | |||
1381 | .parse_msg = kcm_parse_func_strparser, | 1381 | .parse_msg = kcm_parse_func_strparser, |
1382 | .read_sock_done = kcm_read_sock_done, | 1382 | .read_sock_done = kcm_read_sock_done, |
1383 | }; | 1383 | }; |
1384 | int err; | 1384 | int err = 0; |
1385 | 1385 | ||
1386 | csk = csock->sk; | 1386 | csk = csock->sk; |
1387 | if (!csk) | 1387 | if (!csk) |
1388 | return -EINVAL; | 1388 | return -EINVAL; |
1389 | 1389 | ||
1390 | lock_sock(csk); | ||
1391 | |||
1390 | /* Only allow TCP sockets to be attached for now */ | 1392 | /* Only allow TCP sockets to be attached for now */ |
1391 | if ((csk->sk_family != AF_INET && csk->sk_family != AF_INET6) || | 1393 | if ((csk->sk_family != AF_INET && csk->sk_family != AF_INET6) || |
1392 | csk->sk_protocol != IPPROTO_TCP) | 1394 | csk->sk_protocol != IPPROTO_TCP) { |
1393 | return -EOPNOTSUPP; | 1395 | err = -EOPNOTSUPP; |
1396 | goto out; | ||
1397 | } | ||
1394 | 1398 | ||
1395 | /* Don't allow listeners or closed sockets */ | 1399 | /* Don't allow listeners or closed sockets */ |
1396 | if (csk->sk_state == TCP_LISTEN || csk->sk_state == TCP_CLOSE) | 1400 | if (csk->sk_state == TCP_LISTEN || csk->sk_state == TCP_CLOSE) { |
1397 | return -EOPNOTSUPP; | 1401 | err = -EOPNOTSUPP; |
1402 | goto out; | ||
1403 | } | ||
1398 | 1404 | ||
1399 | psock = kmem_cache_zalloc(kcm_psockp, GFP_KERNEL); | 1405 | psock = kmem_cache_zalloc(kcm_psockp, GFP_KERNEL); |
1400 | if (!psock) | 1406 | if (!psock) { |
1401 | return -ENOMEM; | 1407 | err = -ENOMEM; |
1408 | goto out; | ||
1409 | } | ||
1402 | 1410 | ||
1403 | psock->mux = mux; | 1411 | psock->mux = mux; |
1404 | psock->sk = csk; | 1412 | psock->sk = csk; |
@@ -1407,7 +1415,7 @@ static int kcm_attach(struct socket *sock, struct socket *csock, | |||
1407 | err = strp_init(&psock->strp, csk, &cb); | 1415 | err = strp_init(&psock->strp, csk, &cb); |
1408 | if (err) { | 1416 | if (err) { |
1409 | kmem_cache_free(kcm_psockp, psock); | 1417 | kmem_cache_free(kcm_psockp, psock); |
1410 | return err; | 1418 | goto out; |
1411 | } | 1419 | } |
1412 | 1420 | ||
1413 | write_lock_bh(&csk->sk_callback_lock); | 1421 | write_lock_bh(&csk->sk_callback_lock); |
@@ -1420,7 +1428,8 @@ static int kcm_attach(struct socket *sock, struct socket *csock, | |||
1420 | strp_stop(&psock->strp); | 1428 | strp_stop(&psock->strp); |
1421 | strp_done(&psock->strp); | 1429 | strp_done(&psock->strp); |
1422 | kmem_cache_free(kcm_psockp, psock); | 1430 | kmem_cache_free(kcm_psockp, psock); |
1423 | return -EALREADY; | 1431 | err = -EALREADY; |
1432 | goto out; | ||
1424 | } | 1433 | } |
1425 | 1434 | ||
1426 | psock->save_data_ready = csk->sk_data_ready; | 1435 | psock->save_data_ready = csk->sk_data_ready; |
@@ -1456,7 +1465,10 @@ static int kcm_attach(struct socket *sock, struct socket *csock, | |||
1456 | /* Schedule RX work in case there are already bytes queued */ | 1465 | /* Schedule RX work in case there are already bytes queued */ |
1457 | strp_check_rcv(&psock->strp); | 1466 | strp_check_rcv(&psock->strp); |
1458 | 1467 | ||
1459 | return 0; | 1468 | out: |
1469 | release_sock(csk); | ||
1470 | |||
1471 | return err; | ||
1460 | } | 1472 | } |
1461 | 1473 | ||
1462 | static int kcm_attach_ioctl(struct socket *sock, struct kcm_attach *info) | 1474 | static int kcm_attach_ioctl(struct socket *sock, struct kcm_attach *info) |
@@ -1508,6 +1520,7 @@ static void kcm_unattach(struct kcm_psock *psock) | |||
1508 | 1520 | ||
1509 | if (WARN_ON(psock->rx_kcm)) { | 1521 | if (WARN_ON(psock->rx_kcm)) { |
1510 | write_unlock_bh(&csk->sk_callback_lock); | 1522 | write_unlock_bh(&csk->sk_callback_lock); |
1523 | release_sock(csk); | ||
1511 | return; | 1524 | return; |
1512 | } | 1525 | } |
1513 | 1526 | ||