[PATCH 4/10]: Basic support for passive-close

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



[DCCP]: Basic support for passive-close

This implements necessary state transitions for the two forms of passive-close

 * PASSIVE_1, which is entered when a host receives Close;
 * PASSIVE_2, which is entered when a client receives CloseReq.

The handling is such that these passive connection-termination requests are
enqueued as `fin' packets so that dccp_recvmsg() can later detect them. The
only thing I am not entirely sure about is the wake_async in dccp_rcv_closereq();
this has been copied from dccp_rcv_close() (but looks ok from the principle).

The completed state transition diagram is on 
http://www.erg.abdn.ac.uk/users/gerrit/dccp/notes/closing_states/

NB: Regrettably, it is no longer easily possible to use a table-based transition as
    before. The reason is that the role of the host (server or client) also plays a
    `role' in the state transition. Therefore DCCP_ACTION_FIN has also been removed.
    Another difference is that the transition into DCCP_CLOSED is now after sending
    the Close/CloseReq.

Signed-off-by: Gerrit Renker <gerrit@xxxxxxxxxxxxxx>
---
 include/linux/dccp.h |    1 
 net/dccp/input.c     |   34 +++++++++++-----
 net/dccp/proto.c     |  108 ++++++++++++++++++++++++++++++++++++---------------
 3 files changed, 102 insertions(+), 41 deletions(-)

--- a/net/dccp/proto.c
+++ b/net/dccp/proto.c
@@ -55,6 +55,9 @@ EXPORT_SYMBOL_GPL(dccp_hashinfo);
 /* the maximum queue length for tx in packets. 0 is no limit */
 int sysctl_dccp_tx_qlen __read_mostly = 5;
 
+/* Forward Declarations. */
+static void dccp_handle_passive_close(struct sock *sk);
+
 void dccp_set_state(struct sock *sk, const int state)
 {
 	const int oldstate = sk->sk_state;
@@ -71,7 +74,8 @@ void dccp_set_state(struct sock *sk, con
 		break;
 
 	case DCCP_CLOSED:
-		if (oldstate == DCCP_CLOSING || oldstate == DCCP_OPEN)
+		if (oldstate == DCCP_CLOSING  ||
+		    oldstate == DCCP_CLOSEREQ || oldstate == DCCP_OPEN)
 			DCCP_INC_STATS(DCCP_MIB_ESTABRESETS);
 
 		sk->sk_prot->unhash(sk);
@@ -723,19 +727,26 @@ int dccp_recvmsg(struct kiocb *iocb, str
 
 		dh = dccp_hdr(skb);
 
-		if (dh->dccph_type == DCCP_PKT_DATA ||
-		    dh->dccph_type == DCCP_PKT_DATAACK)
+		switch (dh->dccph_type) {
+		case DCCP_PKT_DATA:
+		case DCCP_PKT_DATAACK:
 			goto found_ok_skb;
 
-		if (dh->dccph_type == DCCP_PKT_RESET ||
-		    dh->dccph_type == DCCP_PKT_CLOSE) {
-			dccp_pr_debug("found fin ok!\n");
+		case DCCP_PKT_CLOSE:
+		case DCCP_PKT_CLOSEREQ:
+			if (!(flags & MSG_PEEK))
+				dccp_handle_passive_close(sk);
+			/* fall through */
+		case DCCP_PKT_RESET:
+			dccp_pr_debug("found fin (%s) ok!\n",
+				      dccp_packet_name(dh->dccph_type));
 			len = 0;
 			goto found_fin_ok;
+		default:
+			dccp_pr_debug("packet_type=%s\n",
+				      dccp_packet_name(dh->dccph_type));
+			sk_eat_skb(sk, skb, 0);
 		}
-		dccp_pr_debug("packet_type=%s\n",
-			      dccp_packet_name(dh->dccph_type));
-		sk_eat_skb(sk, skb, 0);
 verify_sock_status:
 		if (sock_flag(sk, SOCK_DONE)) {
 			len = 0;
@@ -837,28 +848,64 @@ out:
 
 EXPORT_SYMBOL_GPL(inet_dccp_listen);
 
-static const unsigned char dccp_new_state[] = {
-	/* current state:   new state:      action:	*/
-	[0]		  = DCCP_CLOSED,
-	[DCCP_OPEN]	  = DCCP_CLOSING | DCCP_ACTION_FIN,
-	[DCCP_REQUESTING] = DCCP_CLOSED,
-	[DCCP_PARTOPEN]	  = DCCP_CLOSING | DCCP_ACTION_FIN,
-	[DCCP_LISTEN]	  = DCCP_CLOSED,
-	[DCCP_RESPOND]	  = DCCP_CLOSED,
-	[DCCP_CLOSING]	  = DCCP_CLOSED,
-	[DCCP_TIME_WAIT]  = DCCP_CLOSED,
-	[DCCP_CLOSED]	  = DCCP_CLOSED,
-};
-
-static int dccp_close_state(struct sock *sk)
+void dccp_handle_passive_close(struct sock *sk)
 {
-	const int next = dccp_new_state[sk->sk_state];
-	const int ns = next & DCCP_STATE_MASK;
+	switch (sk->sk_state) {
+	case DCCP_PASSIVE_1:
+		/* Node (client or server) has received Close packet. */
+		dccp_set_state(sk, DCCP_CLOSED);
+		dccp_send_reset(sk, DCCP_RESET_CODE_CLOSED);
+		break;
+	case DCCP_PASSIVE_2:
+		/*
+		 * Client passive-close by receiving a CloseReq packet.
+		 * We need to set `active' when sending the Close, since 8.3 in
+		 * RFC 4340 requires nodes in the CLOSING state to retransmit
+		 * the Close/CloseReq packets. A conforming peer implementation
+		 * will in turn retransmit its CloseReq packets as well.
+		 */
+		dccp_set_state(sk, DCCP_CLOSING);
+		dccp_send_close(sk, 1);
+		break;
+	default:
+		return;
+	}
+}
 
-	if (ns != sk->sk_state)
-		dccp_set_state(sk, ns);
+static void dccp_handle_close(struct sock *sk)
+{
+	u8 next_state = DCCP_CLOSED;
 
-	return next & DCCP_ACTION_FIN;
+	switch (sk->sk_state) {
+	case DCCP_CLOSED:
+		return;
+	case DCCP_PASSIVE_1:
+	case DCCP_PASSIVE_2:
+		dccp_handle_passive_close(sk);
+		break;
+	case DCCP_PARTOPEN:
+		/*
+		 * Clear PARTOPEN timer [RFC 4340, 8.1.5]; we will be sending a
+		 * Close presently, which will be (re-)transmitted anyhow (8.3).
+		 */
+		dccp_pr_debug("Stop PARTOPEN timer (%p)\n", sk);
+		inet_csk_clear_xmit_timer(sk, ICSK_TIME_DACK);
+		/* fall through */
+	case DCCP_OPEN:
+		/*
+		 * An active close need only be sent in these two states
+		 */
+		dccp_send_close(sk, 1);
+
+		if (dccp_sk(sk)->dccps_role == DCCP_ROLE_SERVER)
+			next_state = DCCP_CLOSEREQ;
+		else
+			next_state = DCCP_CLOSING;
+		/* fall through */
+	default:
+		dccp_set_state(sk, next_state);
+		break;
+	}
 }
 
 void dccp_close(struct sock *sk, long timeout)
@@ -895,9 +942,8 @@ void dccp_close(struct sock *sk, long ti
 	if (sock_flag(sk, SOCK_LINGER) && !sk->sk_lingertime) {
 		/* Check zero linger _after_ checking for unread data. */
 		sk->sk_prot->disconnect(sk, 0);
-	} else if (dccp_close_state(sk)) {
-		dccp_send_close(sk, 1);
-	}
+	} else	/* Passive/active close after all data has been read. */
+		dccp_handle_close(sk);
 
 	sk_stream_wait_close(sk, timeout);
 
--- a/net/dccp/input.c
+++ b/net/dccp/input.c
@@ -31,10 +31,23 @@ static void dccp_fin(struct sock *sk, st
 
 static void dccp_rcv_close(struct sock *sk, struct sk_buff *skb)
 {
-	dccp_send_reset(sk, DCCP_RESET_CODE_CLOSED);
-	dccp_fin(sk, skb);
-	dccp_set_state(sk, DCCP_CLOSED);
-	sk_wake_async(sk, 1, POLL_HUP);
+	if (sk->sk_state == DCCP_CLOSEREQ) {
+		/* Server performed active close */
+		dccp_send_reset(sk, DCCP_RESET_CODE_CLOSED);
+		dccp_done(sk);
+	} else {
+		/*
+		 * Passive-close: only the first Close is enqueued. There is no
+		 * point in putting several Closes on the queue - in case of an
+		 * application bug, the input queue may not be emptied at all.
+		 * In this case we wait until the peer sends the terminal Reset.
+		 */
+		if (sk->sk_state != DCCP_PASSIVE_1) {
+			dccp_fin(sk, skb);
+			dccp_set_state(sk, DCCP_PASSIVE_1);
+		}
+		sk_wake_async(sk, 1, POLL_HUP);
+	}
 }
 
 static void dccp_rcv_closereq(struct sock *sk, struct sk_buff *skb)
@@ -50,9 +63,12 @@ static void dccp_rcv_closereq(struct soc
 		return;
 	}
 
-	if (sk->sk_state != DCCP_CLOSING)
-		dccp_set_state(sk, DCCP_CLOSING);
-	dccp_send_close(sk, 0);
+	/* Do not enqueue CloseReq twice: see comments above for PASSIVE_1 */
+	if (sk->sk_state != DCCP_PASSIVE_2) {
+		dccp_fin(sk, skb);
+		dccp_set_state(sk, DCCP_PASSIVE_2);
+	}
+	sk_wake_async(sk, 1, POLL_HUP);
 }
 
 static void dccp_event_ack_recv(struct sock *sk, struct sk_buff *skb)
@@ -192,7 +208,7 @@ static int __dccp_rcv_established(struct
 		return 0;
 	case DCCP_PKT_CLOSEREQ:
 		dccp_rcv_closereq(sk, skb);
-		goto discard;
+		return 0;
 	case DCCP_PKT_CLOSE:
 		dccp_rcv_close(sk, skb);
 		return 0;
@@ -548,7 +564,7 @@ int dccp_rcv_state_process(struct sock *
 		goto discard;
 	} else if (dh->dccph_type == DCCP_PKT_CLOSEREQ) {
 		dccp_rcv_closereq(sk, skb);
-		goto discard;
+		return 0;
 	} else if (dh->dccph_type == DCCP_PKT_CLOSE) {
 		dccp_rcv_close(sk, skb);
 		return 0;
--- a/include/linux/dccp.h
+++ b/include/linux/dccp.h
@@ -259,7 +259,6 @@ enum dccp_state {
 };
 
 #define DCCP_STATE_MASK 0x1f
-#define DCCP_ACTION_FIN (1<<7)
 
 enum {
 	DCCPF_OPEN	 = TCPF_ESTABLISHED,
-
To unsubscribe from this list: send the line "unsubscribe dccp" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at  http://vger.kernel.org/majordomo-info.html

[Index of Archives]     [Linux Kernel]     [IETF DCCP]     [Linux Networking]     [Git]     [Security]     [Linux Assembly]     [Bugtraq]     [Yosemite]     [MIPS Linux]     [ARM Linux]     [Linux Security]     [Linux RAID]     [Linux SCSI]

  Powered by Linux