In a TCP segment with the URG flag up there might be normal data as well. How does the receiving host handles the urgent packet? How does it acknowledge the urgent data if it is not part of the data stream? Does it acknowledge the rest of it?
I understand that it is not usually used, but if both hosts support the same RFC about the URG flag, how do they handle out-of-band data?
If the urgent data is an abort message, the receiver will drop all other data, but the sender will still want an acknowledgment that the message was received.
A bit of background:
The TCP urgent mechanism permits a point in the data stream to be designated as the end of urgent information. Thus we have the Urgent Pointer that contains a positive offset from the sequence number in this tcp segment. This field is significant only with the URG control bit set.
Discrepancies about the Urgent Pointer:
RFC 793 (1981, page 17):
RFC 1011 (1987, page 8):
The same thing in RFC 1122 (1989, page 84):
The intelligible RFC 6093 (2011, pages 6-7) says:
Thus the updating RFC 793, RFC 1011, and RFC 1122 is
It meets virtually all existing TCP implementations.
Note: Linux provides the
net.ipv4.tcp_stdurg
sysctl
to override the default behaviour but thissysctl
only affects the processing of incoming segments. The Urgent Pointer in outgoing segments will still be set as specified in RFC 793.About the data handling
You can gain urgent data in two ways (keep in mind that the TCP concept of "urgent data" is mapped to the socket API as "out-of-band data"):
using
recv
withMSG_OOB
flag set. (normally you should establish ownership of the socket with something likefcntl(sock, F_SETOWN, getpid());
and establish a signal handler forSIGURG
). Thus you will be notified withSIGURG
signal. The data will be read separately from the normal data stream.using
recv
withoutMSG_OOB
flag set. Previously, you should setSO_OOBINLINE
socket option such way:int so_oobinline = 1; /* true */
setsockopt(sock, SOL_SOCKET, SO_OOBINLINE, &so_oobinline, sizeof so_oobinline);
The data remain "in-line". And you can determine the Urgent Pointer with a help of
ioctl
:int flag; /* True when at mark */
ioctl(sock, SIOCATMARK, &flag);
Besides it is recommended for new applications
not to use the mechanism of urgent data at allto use (if so) receiving in-line, as mentioned above.From RFC 1122:
Also from RFC 793:
So you can handle as you want. It is an application level issue.
Accordingly, the answer to your question about acknowledgements when all other data was dropped is "You can implement it in your application".
As for tcp-ack, I found nothing special about it in the case of urgent data.
About the length of "Urgent Data"
Almost all implementations really can provide only one byte of "out-of-band data".
RFC 6093 says:
So TCP urgent mode and its urgent pointer cannot provide marking the boundaries of the urgent data in practice.
Rumor has it that there are some implementations that queue each of the received urgent bytes. Some of them have been known to fail to enforce any limits on the amount of "urgent data", that they queue. Thus, they become vulnerable to trivial resource exhaustion attacks.
P. S. All of the above probably covers a little more than was asked, but that's only to make it clear for people unfamiliar with this issue.
Some more useful links:
TCP Urgent Pointer, buffer management, and the "Send" call
Difference between push and urgent flags in TCP
Understanding the urgent pointer