[<prev] [next>] [<thread-prev] [thread-next>] [day] [month] [year] [list]
Message-ID: <4E1DDE62.3080503@hp.com>
Date: Wed, 13 Jul 2011 11:05:22 -0700
From: Rick Jones <rick.jones2@...com>
To: Eric Dumazet <eric.dumazet@...il.com>
CC: Chris Friesen <chris.friesen@...band.com>, netdev@...r.kernel.org
Subject: Re: any way to let host act as TCP server OR client on same IP/port?
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@...dy:~$ 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@...dy:~$ ./both
bind active: Address already in use
(nor HP-UX 11.31)
unless I botched the test code somehow:
raj@...dy:~$ 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
--
To unsubscribe from this list: send the line "unsubscribe netdev" in
the body of a message to majordomo@...r.kernel.org
More majordomo info at http://vger.kernel.org/majordomo-info.html
Powered by blists - more mailing lists