[질문] "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:
댓글 달기