[질문] SELECT 함수를 이용하여 Non-Block 소켓만들 경우 writefds를 설정 안하면 트리거링을 못합니다.
안녕하세요.
한창 소켓 공부 중인 학생 입니다.
서버로 연결을 할 경우에 아래와 같은 코드를 이용하고 있습니다.
이런 경우 select함수에 wset부분을 NULL로 셋팅하고, FD_ISSET에서도 rset만 검사하면
connect함수에 의하여 서버로 [SYN]패킷을 보내고 [SYN,ACK]패킷을 받고 다시 [ACK]를 보냈음에도 불구하고
내부적으로는 [SYN,ACK] 패킷을 받은거에 대한 인지를 못합니다.
서버에서 연결 후 일정 시간 동안 클라이언트가 아무런 메시지를 보내지 않을 경우 연결을 종료 하는데,
이에 따라 [FIN]메시지를 보내주는데, [FIN]메시지를 받은 다음에야 FD_ISSET에서 인지를 합니다 -_- ;;
궁금한것이 rset은 입력스트림에 변화를 감지 하는 것으로 알고 있는데 왜 서버로부터의 [SYN,ACK]를 감지 못하는지 궁금합니다.
wset을 설정 했을 경우에 정상적으로 [SYN,ACK]를 받은 후 이를 인지하고 정상 동작 하는데,
wset은 블로킹 되지 않고 바로 전송 가능한 경우 감지 되는 것으로 알고 있습니다.
그렇다면 [SYN,ACK]패킷이 들어왔을 경우 FD_ISSET에서 rset이 아닌 wset이 설정 된 것을 감지 하는 것입니까?
[SYN,ACK]의 경우 FD_ISSET에서 rset으로 왜 감지가안되는지 궁금합니다.
질문이 좀 두서가 없는데... 지식이 모자라서.. 위의 내용에 대하여 좀 전반적으로 이해하지 못하고 있어서
질문도 좀 딱 부러지지 못합니다 죄송합니다.
위의 현상에 대하여 총체적으로 설명 부탁 드리겠습니다.
ㅠ _ㅠ
int connect_nonb(int sockfd, const struct sockaddr *saptr, socklen_t salen, int nsec) { int flags, status, error; socklen_t len; fd_set rset, wset; struct timeval tval; int fd_max; flags = fcntl(sockfd, F_GETFL, 0); if (fcntl(sockfd, F_SETFL, flags | O_NONBLOCK) < 0) { printf("[Error] Non block mode set fail !\n"); return -1; } error = 0; printf("[INFO] connect function load\n"); status = connect(sockfd, (struct sockaddr *)saptr, salen); if ((status < 0) && (error == EINPROGRESS)) { printf("[INFO] connection fail !!\n"); return(-1); } /* Do whatever we want while the connect is taking place. */ if (status == 0) goto done; /* connect completed immediately */ FD_ZERO (&rset); FD_SET (sockfd, &rset); wset = rset; tval.tv_sec = nsec; tval.tv_usec = 0; printf("[INFO] select function load\n"); if ((status = select(sockfd + 1, &rset, &wset, NULL, nsec ? & tval : NULL)) == 0) { printf("[INFO] select return time out\n"); close(sockfd); error = ETIMEDOUT; return (-1); } if (FD_ISSET(sockfd, &rset) || FD_ISSET(sockfd, &wset)) { len = sizeof(error); if (getsockopt(sockfd, SOL_SOCKET, SO_ERROR, &error, &len) < 0) return (-1); } else { printf("[Error] select error: sockfd not set\n"); return (-1); } done: fcntl(sockfd, F_SETFL, flags); /* restore file status flags */ if (error) { printf("[Error] socket error\n"); close(sockfd); /* just in case */ return(-1); } printf("[Info] nonb connection complete !! \n"); return(0); }
댓글 달기