리눅스 소켓프로그래밍 엣지 트리거
질문 제목 : 리눅스 소켓프로그래밍 엣지 트리거 질문
질문 요약 :
질문 내용 :
윤성우 저자책 TCP/IP 소켓프로그래밍에서 p383 ~ p386
리눅스 epoll함수로 엣지트리거를 구현하는 건데요(버퍼사이즈를 일부러 낮게 조절)
두대의 리눅스에서 테스트를 하면 클라이언트가 전송한 문자열을 서버는
어느때는 문자열을 다 읽고 , 어느때는 문자열 일부만 읽어서 클라이언트에게 전송한 결과
클라이언트의 화면에서는 이상하게 문자열이 출력 됩니다
로컬에서 서버와 클라이언트 프로그램을 테스트를 하면 클라이언트가 전송한 문자열을
서버는 입력버퍼에 데이터가 없을 때까지 읽어서 클라이언트에게 전송합니다. 그 결과
클라이언트의 화면에서는 클라이언트가 전송한 문자열이 그대로 출력이 됩니다
서버소스 , 클라이언트 소스 같고 테스트환경이 로컬에서 하느냐 두대의 시스템에서 하느냐의
차이인데 왜 이런차이가 나죠???
왜 이런 차이가 나는지 제 머리로는 잘 이해가 안갑니다.
소스의 문제라면 고칠 텐데 소스의 문제인가요???
소스의 문제가 아니라면 epoll함수의 문제인가요???
■ 두대의 리눅스에서 테스트
리눅스 이미지1(IP : 10.10.0.123) : 서버
[root@localhost backup]# ./server 9000
return epoll_wait
connected client: 5
return epoll_wait
return epoll_wait
return epoll_wait
return epoll_wait
return epoll_wait
closed client: 5
리눅스 이미지2(IP : 10.10.0.122) : 클라이언트
[root@localhost backup]# ./client 10.10.0.123 9000
Connected...........
Input message(Q to quit): i like computer programming??? // 사용자가 문자열을 입력
Message from server: i like computer programming??? // 서버로부터 받은 메시지
Input message(Q to quit): Do you like computer programmin???
Message from server: Do yInput message(Q to quit): I like hacking!!!
Message from server: ou like computer programmin???
Input message(Q to quit): Good bye
Message from server: I like hacking!!!
Input message(Q to quit): q
[root@localhost backup]#
비정상적인 출력결과...
================================================
■ 한대의 리눅스에서 테스트(즉, 로컬에서 테스트)
리눅스 이미지1(IP : 10.10.0.123) 에서 테스트(서버,클라이언트 모두)
[root@localhost backup]# ./server 9000
return epoll_wait
connected client: 5
return epoll_wait
return epoll_wait
return epoll_wait
return epoll_wait
return epoll_wait
closed client: 5
[root@localhost backup]# ./client 127.0.0.1 9000
Connected...........
Input message(Q to quit): I like computer programming???
Message from server: I like computer programming???
Input message(Q to quit): Do you like computer programming???
Message from server: Do you like computer programming???
Input message(Q to quit): I like hacking!!!
Message from server: I like hacking!!!
Input message(Q to quit): Good bye
Message from server: Good bye
Input message(Q to quit): q
[root@localhost backup]#
정상적인 출력 결과.
================================================
server.c
#include
#include
#include
#include
#include
#include
#include
#include
#include
#define BUF_SIZE 4
#define EPOLL_SIZE 50
void setnonblockingmode(int fd);
void error_handling(char *buf);
int main(int argc, char *argv[])
{
int serv_sock, clnt_sock;
struct sockaddr_in serv_adr, clnt_adr;
socklen_t adr_sz;
int str_len, i;
char buf[BUF_SIZE];
struct epoll_event *ep_events;
struct epoll_event event;
int epfd, event_cnt;
if(argc!=2) {
printf("Usage : %s \n", argv[0]);
exit(1);
}
serv_sock=socket(PF_INET, SOCK_STREAM, 0);
memset(&serv_adr, 0, sizeof(serv_adr));
serv_adr.sin_family=AF_INET;
serv_adr.sin_addr.s_addr=htonl(INADDR_ANY);
serv_adr.sin_port=htons(atoi(argv[1]));
if(bind(serv_sock, (struct sockaddr*) &serv_adr, sizeof(serv_adr))==-1)
error_handling("bind() error");
if(listen(serv_sock, 5)==-1)
error_handling("listen() error");
epfd=epoll_create(EPOLL_SIZE);
ep_events=malloc(sizeof(struct epoll_event)*EPOLL_SIZE);
setnonblockingmode(serv_sock);
event.events=EPOLLIN;
event.data.fd=serv_sock;
epoll_ctl(epfd, EPOLL_CTL_ADD, serv_sock, &event);
while(1)
{
event_cnt=epoll_wait(epfd, ep_events, EPOLL_SIZE, -1);
if(event_cnt==-1)
{
puts("epoll_wait() error");
break;
}
puts("return epoll_wait");
for(i=0; i
{
if(ep_events[i].data.fd==serv_sock)
{
adr_sz=sizeof(clnt_adr);
clnt_sock=accept(serv_sock, (struct sockaddr*)&clnt_adr, &adr_sz);
setnonblockingmode(clnt_sock);
event.events=EPOLLIN|EPOLLET;
event.data.fd=clnt_sock;
epoll_ctl(epfd, EPOLL_CTL_ADD, clnt_sock, &event);
printf("connected client: %d \n", clnt_sock);
}
else
{
while(1)
{
str_len=read(ep_events[i].data.fd, buf, BUF_SIZE);
if(str_len==0) // close request!
{
epoll_ctl(epfd, EPOLL_CTL_DEL, ep_events[i].data.fd, NULL);
close(ep_events[i].data.fd);
printf("closed client: %d \n", ep_events[i].data.fd);
break;
}
else if(str_len<0)
{
if(errno==EAGAIN)
break;
}
else
{
write(ep_events[i].data.fd, buf, str_len); // echo!
}
}
}
}
}
close(serv_sock);
close(epfd);
return 0;
}
void setnonblockingmode(int fd)
{
int flag=fcntl(fd, F_GETFL, 0);
fcntl(fd, F_SETFL, flag|O_NONBLOCK);
}
void error_handling(char *buf)
{
fputs(buf, stderr);
fputc('\n', stderr);
exit(1);
}
================================================
client.c
#include
#include
#include
#include
#include
#include
#define BUF_SIZE 1024
void error_handling(char *message);
int main(int argc, char *argv[])
{
int sock;
char message[BUF_SIZE];
int str_len;
struct sockaddr_in serv_adr;
if(argc!=3) {
printf("Usage : %s \n", argv[0]);
exit(1);
}
sock=socket(PF_INET, SOCK_STREAM, 0);
if(sock==-1)
error_handling("socket() error");
memset(&serv_adr, 0, sizeof(serv_adr));
serv_adr.sin_family=AF_INET;
serv_adr.sin_addr.s_addr=inet_addr(argv[1]);
serv_adr.sin_port=htons(atoi(argv[2]));
if(connect(sock, (struct sockaddr*)&serv_adr, sizeof(serv_adr))==-1)
error_handling("connect() error!");
else
puts("Connected...........");
while(1)
{
fputs("Input message(Q to quit): ", stdout);
fgets(message, BUF_SIZE, stdin);
if(!strcmp(message,"q\n") || !strcmp(message,"Q\n"))
break;
write(sock, message, strlen(message));
str_len=read(sock, message, BUF_SIZE-1);
message[str_len]=0;
printf("Message from server: %s", message);
}
close(sock);
return 0;
}
void error_handling(char *message)
{
fputs(message, stderr);
fputc('\n', stderr);
exit(1);
}
message 버퍼를 초기화 하세요.
.
Knowleage Creator.
댓글 달기