tcp: add tcp_sock_set_keepidle
Add a helper to directly set the TCP_KEEP_IDLE sockopt from kernel space without going through a fake uaccess. Signed-off-by: Christoph Hellwig <hch@lst.de> Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:

committed by
David S. Miller

parent
c488aeadcb
commit
71c48eb81c
@@ -498,6 +498,7 @@ int tcp_skb_shift(struct sk_buff *to, struct sk_buff *from, int pcount,
|
|||||||
int shiftlen);
|
int shiftlen);
|
||||||
|
|
||||||
void tcp_sock_set_cork(struct sock *sk, bool on);
|
void tcp_sock_set_cork(struct sock *sk, bool on);
|
||||||
|
int tcp_sock_set_keepidle(struct sock *sk, int val);
|
||||||
void tcp_sock_set_nodelay(struct sock *sk);
|
void tcp_sock_set_nodelay(struct sock *sk);
|
||||||
void tcp_sock_set_quickack(struct sock *sk, int val);
|
void tcp_sock_set_quickack(struct sock *sk, int val);
|
||||||
int tcp_sock_set_syncnt(struct sock *sk, int val);
|
int tcp_sock_set_syncnt(struct sock *sk, int val);
|
||||||
|
@@ -2901,6 +2901,39 @@ void tcp_sock_set_user_timeout(struct sock *sk, u32 val)
|
|||||||
}
|
}
|
||||||
EXPORT_SYMBOL(tcp_sock_set_user_timeout);
|
EXPORT_SYMBOL(tcp_sock_set_user_timeout);
|
||||||
|
|
||||||
|
static int __tcp_sock_set_keepidle(struct sock *sk, int val)
|
||||||
|
{
|
||||||
|
struct tcp_sock *tp = tcp_sk(sk);
|
||||||
|
|
||||||
|
if (val < 1 || val > MAX_TCP_KEEPIDLE)
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
tp->keepalive_time = val * HZ;
|
||||||
|
if (sock_flag(sk, SOCK_KEEPOPEN) &&
|
||||||
|
!((1 << sk->sk_state) & (TCPF_CLOSE | TCPF_LISTEN))) {
|
||||||
|
u32 elapsed = keepalive_time_elapsed(tp);
|
||||||
|
|
||||||
|
if (tp->keepalive_time > elapsed)
|
||||||
|
elapsed = tp->keepalive_time - elapsed;
|
||||||
|
else
|
||||||
|
elapsed = 0;
|
||||||
|
inet_csk_reset_keepalive_timer(sk, elapsed);
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int tcp_sock_set_keepidle(struct sock *sk, int val)
|
||||||
|
{
|
||||||
|
int err;
|
||||||
|
|
||||||
|
lock_sock(sk);
|
||||||
|
err = __tcp_sock_set_keepidle(sk, val);
|
||||||
|
release_sock(sk);
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL(tcp_sock_set_keepidle);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Socket option code for TCP.
|
* Socket option code for TCP.
|
||||||
*/
|
*/
|
||||||
@@ -3070,21 +3103,7 @@ static int do_tcp_setsockopt(struct sock *sk, int level,
|
|||||||
break;
|
break;
|
||||||
|
|
||||||
case TCP_KEEPIDLE:
|
case TCP_KEEPIDLE:
|
||||||
if (val < 1 || val > MAX_TCP_KEEPIDLE)
|
err = __tcp_sock_set_keepidle(sk, val);
|
||||||
err = -EINVAL;
|
|
||||||
else {
|
|
||||||
tp->keepalive_time = val * HZ;
|
|
||||||
if (sock_flag(sk, SOCK_KEEPOPEN) &&
|
|
||||||
!((1 << sk->sk_state) &
|
|
||||||
(TCPF_CLOSE | TCPF_LISTEN))) {
|
|
||||||
u32 elapsed = keepalive_time_elapsed(tp);
|
|
||||||
if (tp->keepalive_time > elapsed)
|
|
||||||
elapsed = tp->keepalive_time - elapsed;
|
|
||||||
else
|
|
||||||
elapsed = 0;
|
|
||||||
inet_csk_reset_keepalive_timer(sk, elapsed);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
break;
|
break;
|
||||||
case TCP_KEEPINTVL:
|
case TCP_KEEPINTVL:
|
||||||
if (val < 1 || val > MAX_TCP_KEEPINTVL)
|
if (val < 1 || val > MAX_TCP_KEEPINTVL)
|
||||||
|
@@ -52,10 +52,7 @@ int rds_tcp_keepalive(struct socket *sock)
|
|||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
goto bail;
|
goto bail;
|
||||||
|
|
||||||
ret = kernel_setsockopt(sock, IPPROTO_TCP, TCP_KEEPIDLE,
|
tcp_sock_set_keepidle(sock->sk, keepidle);
|
||||||
(char *)&keepidle, sizeof(keepidle));
|
|
||||||
if (ret < 0)
|
|
||||||
goto bail;
|
|
||||||
|
|
||||||
/* KEEPINTVL is the interval between successive probes. We follow
|
/* KEEPINTVL is the interval between successive probes. We follow
|
||||||
* the model in xs_tcp_finish_connecting() and re-use keepidle.
|
* the model in xs_tcp_finish_connecting() and re-use keepidle.
|
||||||
|
@@ -2107,8 +2107,7 @@ static void xs_tcp_set_socket_timeouts(struct rpc_xprt *xprt,
|
|||||||
|
|
||||||
/* TCP Keepalive options */
|
/* TCP Keepalive options */
|
||||||
sock_set_keepalive(sock->sk);
|
sock_set_keepalive(sock->sk);
|
||||||
kernel_setsockopt(sock, SOL_TCP, TCP_KEEPIDLE,
|
tcp_sock_set_keepidle(sock->sk, keepidle);
|
||||||
(char *)&keepidle, sizeof(keepidle));
|
|
||||||
kernel_setsockopt(sock, SOL_TCP, TCP_KEEPINTVL,
|
kernel_setsockopt(sock, SOL_TCP, TCP_KEEPINTVL,
|
||||||
(char *)&keepidle, sizeof(keepidle));
|
(char *)&keepidle, sizeof(keepidle));
|
||||||
kernel_setsockopt(sock, SOL_TCP, TCP_KEEPCNT,
|
kernel_setsockopt(sock, SOL_TCP, TCP_KEEPCNT,
|
||||||
|
Reference in New Issue
Block a user