[질문] "Address family not supported by protocl"
글쓴이: nTachyon / 작성시간: 월, 2005/10/10 - 3:15오후
제가 소켓 통신을 함에 있어
"Address family not supported by protocl" (errno:84)가 가끔 발생하여 질문 드립니다.
일단 상황이 좀 독특합니다.
저는 connection 하기전에 소켓을 non-block 상태로 만들어서 connection 하고,
연결이 되면 다시 block 상태로 소켓을 만듭니다.
non-block과 block 사이에서
주소 구조체를 만들고
connect() 함수를 호출하면
위와 같은 에러가 계속해서 발생합니다.(연결될 때 까지 주기적으로 connection 시도하거든요)
그런데 웃기는건... 한참을 내버려두면 자기혼자 또 connection이 이루어집니다.
재가동하면 또 바로 잘 됩니다.
자주 발생하는 상황이 아니라 가끔씩 발생하는 상황인데다가
84번 에러에 대한 일반적인 대처방법으로 설명되는 것도 아니고...
조금은 특수하다 할 수 있는 상황(제 입장에서는)이라서
이런 경우를 겪으신 분이 계시거나,
짐작 가시는 분이 계시거나,
대처 코드에 대한 아이디어가 계신 분이 있으시면
조언을 구하고자 질문 올립니다.
아래는 코드입니다.
static void set_nonblock(int sock)
{
int val;
val = fcntl(sock, F_GETFL, 0);
assert(val != -1);
val |= O_NONBLOCK;
val = fcntl(sock, F_SETFL, val);
assert(val != -1);
}
static void set_block(int sock)
{
int val;
val = fcntl(sock, F_GETFL, 0);
assert(val != -1);
val &= ~O_NONBLOCK;
val = fcntl(sock, F_SETFL, val);
assert(val != -1);
}
int init_tcp(char *ip, int port, struct sockaddr_in *psa)
{
int sock = INVALID_SOCKET;
int one = 1;
struct linger option;
sock = socket(AF_INET, SOCK_STREAM, 0);
if(sock <= 0)
goto RET;
if(setsockopt(sock, SOL_SOCKET, SO_REUSEADDR, (char*)&one, sizeof(one)) < 0)
{
sock = INVALID_SOCKET;
goto RET;
}
option.l_onoff = 1;
option.l_linger = 3;
if(setsockopt(sock, SOL_SOCKET, SO_LINGER, (char*)&option, sizeof(option)) < 0)
{
close(sock);
sock = INVALID_SOCKET;
goto RET;
}
memset((void*) psa, 0x00, sizeof(struct sockaddr_in));
psa->sin_family = AF_INET;
psa->sin_addr.s_addr = inet_addr(ip);
psa->sin_port = htons(port);
RET:
return(sock);
}
static int timed_nonblock_connect(int sock, struct sockaddr *sa, int nsa, int tv_sec, char* szErrMsg)
{
int rc = FAIL;
int r, soerr, nsoerr;
struct timeval timeval;
fd_set fds;
set_nonblock(sock);
if(connect(sock, sa, nsa) == 0)
{
/* success */
goto END;
}
if(errno != EINPROGRESS)
{
sprintf(szErrMsg, "Non-block 연결 실패(@connect).(원인:%d-%s)\n", errno, strerror(errno));
goto END;
}
errno = 0;
FD_ZERO(&fds);
FD_SET(sock, &fds);
timeval.tv_sec = tv_sec;
timeval.tv_usec = 0;
r = select(sock + 1, NULL, &fds, NULL, &timeval);
if(r == 0)
{
errno = ETIMEDOUT;
sprintf(szErrMsg, "Non-block 연결 실패(@select).(원인:%d-%s)", errno, strerror(errno));
goto END;
}
else if(r < 0)
{
sprintf(szErrMsg, "Non-block 연결 실패(@select).(원인:%d-%s)", errno, strerror(errno));
goto END;
}
nsoerr = sizeof(soerr);
r = getsockopt(sock, SOL_SOCKET, SO_ERROR, &soerr, &nsoerr);
assert(r == 0);
if(soerr != 0)
{
errno = soerr;
sprintf(szErrMsg, "Non-block 연결 실패(@getsockopt).(원인:%d-%s)", errno, strerror(errno));
goto END;
}
rc = SUCCESS;
END:
set_block(sock);
return(rc);
}
int main(...)
{
sock = init_tcp(pinfo->remote_ip, pinfo->remote_port, &sa);
if(sock == INVALID_SOCKET)
goto END;
if(timed_nonblock_connect(sock, (struct sockaddr *) &sa, sizeof sa, pinfo->timeout_conn, buf_err_msg) == FAIL)
goto END;
...
...
...
}
Forums:


댓글 달기