리눅스 소켓프로그래밍 엣지 트리거
질문 제목 : 리눅스 소켓프로그래밍 엣지 트리거 질문
질문 요약 :   
질문 내용 :  
윤성우 저자책 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.
댓글 달기