From: Rick Jones <rick.jones2@hp.com>
To: Eric Dumazet <eric.dumazet@gmail.com>
Cc: Chris Friesen <chris.friesen@genband.com>, netdev@vger.kernel.org
Subject: Re: any way to let host act as TCP server OR client on same IP/port?
Date: Wed, 13 Jul 2011 11:05:22 -0700 [thread overview]
Message-ID: <4E1DDE62.3080503@hp.com> (raw)
In-Reply-To: <1310579520.2509.17.camel@edumazet-laptop>
On 07/13/2011 10:52 AM, Eric Dumazet wrote:
> Le mercredi 13 juillet 2011 à 10:30 -0600, Chris Friesen a écrit :
>> I've been asked an interesting question about TCP. We have some people
>> that want to set up a TCP socket that can listen for connections on a
>> given IP/port, but also initiate connections from that same IP/port.
>> (Only one at a time, of course.)
>>
>> The TCP state machine seems to allow this (moving from LISTEN to
>> SYN_SENT) but it's not a normal transition.
>>
>> Is there any way to do this using the socket API?
>>
>> I thought up a hack whereby we could use NFQUEUE to detect an incoming
>> SYN and delay it while we call listen() on the socket. Is there any
>> better way to do this?
>
> Could you try SO_REUSEADDR, on both listener and connect attempt ?
I was thinking the same thing, but it appears to not work under:
raj@tardy:~$ uname -a
Linux tardy 2.6.35-30-generic #54-Ubuntu SMP Tue Jun 7 18:41:54 UTC 2011
x86_64 GNU/Linux
raj@tardy:~$ ./both
bind active: Address already in use
(nor HP-UX 11.31)
unless I botched the test code somehow:
raj@tardy:~$ cat both.c
#include <stdio.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <string.h>
#include <arpa/inet.h>
#include <stdlib.h>
int
main(int argc, char *argv[]) {
struct sockaddr_in me,him;
int on = 1;
socklen_t len;
int listener, active, client;
if ((listener = socket(AF_INET,SOCK_STREAM,0)) < 0) {
perror("socket listener");
exit(-1);
}
if ((active = socket(AF_INET,SOCK_STREAM,0)) < 0) {
perror("socket active");
exit(-1);
}
if ((client = socket(AF_INET,SOCK_STREAM,0)) < 0) {
perror("socket client");
exit(-1);
}
/* setup the listen endpoint */
memset(&me,0,sizeof(me));
if (!inet_pton(AF_INET,"127.0.0.1",&(me.sin_addr.s_addr))) {
perror("inet_pton me");
exit(-1);
}
me.sin_port = htons(23456);
me.sin_family = AF_INET;
if (setsockopt(listener,
SOL_SOCKET,
SO_REUSEADDR,
(char *)&on ,
sizeof(on)) < 0) {
perror("setsockopt SO_REUSEADDR listener");
exit(-1);
}
if (bind(listener,
(struct sockaddr *)&me,
sizeof(me)) < 0) {
perror("bind listener");
exit(-1);
}
if (listen(listener,128) < 0) {
perror("listen listener");
exit(-1);
}
/* connect something to it */
if (connect(client,(struct sockaddr *)&me,sizeof(me)) < 0) {
perror("connect client");
exit(-1);
}
/* now try to make an active connection from "me" */
memset(&him,0,sizeof(him));
if (!inet_pton(AF_INET,"127.0.0.1",&(him.sin_addr.s_addr))) {
perror("inet_pton him");
exit(-1);
}
him.sin_port = htons(9); /* assume someone is listening on
"discard"*/
him.sin_family = AF_INET;
if (setsockopt(active,
SOL_SOCKET,
SO_REUSEADDR,
(char *)&on ,
sizeof(on)) < 0) {
perror("setsockopt SO_REUSEADDR active");
exit(-1);
}
if (bind(active,
(struct sockaddr *)&me,
sizeof(me)) < 0) {
perror("bind active");
exit(-1);
}
if (connect(active,(struct sockaddr *)&him,sizeof(him)) < 0) {
perror("connect active");
exit(-1);
}
printf("Successfully created a listen endpoint and an active
connection from the same IP/port pair\n");
}
rick jones
next prev parent reply other threads:[~2011-07-13 18:05 UTC|newest]
Thread overview: 9+ messages / expand[flat|nested] mbox.gz Atom feed top
2011-07-13 16:30 any way to let host act as TCP server OR client on same IP/port? Chris Friesen
2011-07-13 17:52 ` Eric Dumazet
2011-07-13 18:05 ` Rick Jones [this message]
2011-07-13 19:16 ` Chris Friesen
2011-07-13 20:36 ` Rick Jones
2011-07-13 22:28 ` Chris Friesen
2011-07-14 14:55 ` Chris Friesen
2011-07-14 16:45 ` Rick Jones
2011-07-14 17:33 ` Chris Friesen
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=4E1DDE62.3080503@hp.com \
--to=rick.jones2@hp.com \
--cc=chris.friesen@genband.com \
--cc=eric.dumazet@gmail.com \
--cc=netdev@vger.kernel.org \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is an external index of several public inboxes,
see mirroring instructions on how to clone and mirror
all data and code used by this external index.