TCP/IP 서버에서 send() 시 EPIPE 에러가 발생하지 않게 하려면?
TCP/IP 서버를 만들고 있습니다.
메인에서 서버 소켓을 생성하고, 여러 개의 쓰레드에서 데이터 송수신을 담당합니다.
서버 소켓 생성 시 bind() 하기 전에 다음과 같이 합니다.
int optval = 1;
int nRet = 0;
int opt_value;
opt_value = 1024 * 1024;
if ( setsockopt(m_nSvrSd,SOL_SOCKET,SO_RCVBUF,(char *)&opt_value, sizeof optval) < 0 )
{
throw errno;
}
if ( setsockopt(m_nSvrSd, SOL_SOCKET,SO_SNDBUF,(char *)&opt_value, sizeof optval) < 0 )
{
throw errno;
}
if ( setsockopt( m_nSvrSd , SOL_SOCKET , SO_REUSEADDR , &optval , sizeof optval ) < 0 )
{
throw errno;
}
select(), accept() 하기 전에는 다음과 같이 하구요.
fcntl(m_nSvrSd, F_SETFL, O_NDELAY);
FD_ZERO(&readFDS);
FD_SET(m_nSvrSd, &readFDS);
평소에는 괜찮은데, 송수신되는 데이터가 너무 많으면 send() 시 EPIPE 에러가 자주 발생합니다.
(자세하게 못 듣긴 했는데, 초당 17만건인가, 더 된다고 했나, 그랬던 듯...)
이 경우에도 recv() 하는데는 문제가 없어 보입니다.
프로세스를 내렸다 올리면 또 괜찮구요.
에러가 발생하면 연결을 끊고 재접속을 기다리도록 해서 프로세스가 죽거나 하지는 않습니다만,
클라이언트측에서는 보낼 데이터는 많은데 자꾸 재접속을 하니 데이터가 유실된다고 ㅈㄹ입니다.
EPIPE 에러가 발생하지 않도록 하려면 어떻게 해야 하나요?
내일이 오픈인데 참... 미치겠슴다... ㅠ_ㅠ
Server/Client 간에
Server/Client 간에 send/recv 가 딱 들어맞아야 EPIPE 가 안 뜨더군요.
그런데 이렇게 하기가 참 어렵죠. 특히나 하나의 Server, 여러개의 Client 의 경우는 ...
그런고로 그냥 EPIPE 의 경우는 EAGAIN 이나 EPROGRESS(?) 처럼 다시 시도하게 처리했던 기억이 있습니다.
리눅스상에서 SIGPIPE 가 뜨는 경우를 대비해서 SIGPIPE 시그널 무시하게 셋팅도 하고 해서 말이죠...
딱히 퍼포먼스상에 큰 문제가 있다거나 하지는 않았던거 같네요.
처음에는 EPIPE를
처음에는 EPIPE를 무시했었는데요, 한 번 FAIL 나면 계속 안되더라구요.
그래서 어쩔 수 없이 연결을 끊게 했죠. ㅠ_ㅠ
댓글 달기