[질문] 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);
}

댓글 달기