리눅스 fork를 이용한 채팅 프로그램 질문드립니다.
미즈
질문 제목 :리눅스 fork를 이용한 채팅 프로그램 질문드립니다.리눅스에서 채팅프로그램을 짜는데 아래는 server의 코드입니다.
client도 비슷한데요 문제는 bye를 보냈을때 client와 server 둘다 종료가 되야 하는데
받는 사람은 종료가 되는데 보내는 사람은 종료가 안됩니다.
밑에 소스에서 뭐가 문제가 되는지 좀 알려주세요질문 내용 : #include stdio.h
#include signal.h
#include sys/types.h
#include sys/socket.h
#include netinet/in.h
#include arpa/inet.h#define serv_tcp_port 10623
#define serv_addr 165.246.38.151void dis_connect(int s1, int s2, char buf[], int child){
if(strncmp(buf,bye,3)==0){
close(s2);
close(s1);
kill(getpid(),sigterm);
if(child==0)
kill(getppid(),sigterm);
}
}void fork_error(int child){
if(child0)
exit(1);
}void receive_m(char buf[], int s2, int s1, int child){
int y;
y=read(s2, buf, 50);
buf[y]=0;
printf(client %s, buf);
dis_connect(s1, s2, buf, child);
}void send_m(char buf[], int s2, int s1, int child){
fgets(buf,50,stdin);
write(s2,buf,strlen(buf));
dis_connect(s1, s2, buf, child);
}void main(){
int s1, s2, x, child, i;
struct sockaddr_in serv_addr, cli_addr;
char buf[50];
size_t xx; printf(hi, i am the server\n); bzero((char *)&serv_addr, sizeof(serv_addr));
serv_addr.sin_family=pf_inet;
serv_addr.sin_addr.s_addr=inet_addr(serv_addr);
serv_addr.sin_port=htons(serv_tcp_port); if((s1=socket(pf_inet, sock_stream, 0))0){
printf(socket creation error\n);
exit(1);
} printf(socket created successfully. socket num is %d\n, s1); x=bind(s1, (struct sockaddr *)&serv_addr, sizeof(serv_addr));
if(x0){
printf(binding failed\n);
exit(1);
} printf(binding passed\n);
listen(s1, 5);
xx=sizeof(cli_addr);
s2=accept(s1,(struct sockaddr *)&cli_addr, &xx);
printf(we passed accept. new socket num is %d\n, s2); child=fork();
fork_error(child); for(i=0;i5;i++){
if(child==0){
send_m(buf, s2, s1, child);
else
receive_m(buf, s2, s1, child);
}
}
-
도담
소켓서버를 정확하게 구현하는것은 초보자에게 어려울거에요.
서버에서 fork 후 부모의 역활은 일반적으로 연결요청 수신 후 자식생성과 종료입니다.
자식의 역확을 연결된 클라이언트와 통신(수신 후 송신이 일반적이겠죠)입니다.
좀비가 발생하는 문제 해결을 위해서 우선은
fork 이전에 시그널 처리 또는
fork 후 자식 프로세스 로직을 exec로 구현이라고 생각하세요. -
일진오빠
좀비가 발생하는 경우는 exit() 등으로 프로세스 종료시에 부모 프로세스가 시그널을 처리해주지 않을때 발생합니다.
자신이 죽는것을 자신을 만든 부모에게 알리는데 부모가 확인을 안해줘서 대기하는거라고 생각하시면되요.
fork 에서 시그널 처리를 추가해주면 좀비는 생기지 않습니다.
실제 운영중인 대형사이트에서도 비슷한 경우를 많이 봤습니다.
아래의 코드와 같이 짜는것이 정상적으로 동작을하게됩니다.
(로직을 설명하기 위한 코드니까 참고만하세요)
sigs -
맑은
네 부모가 좀비로 남아있어서 저런식으로 코드를 짰습니다. 그럼 자식일 경우에 exit로 종료하고 ppid를 죽이면 되는건가요?
-
그린나래
예제 감사합니다. 근데 부모에서는 kill로 pid를 죽이고 break를 걸었는데 kill로 지금 부모의 pid를 죽였는데 또 break를 걸어야 하나요??
그리고 자식에서는 그냥 break만 걸었는데 자식의 pid를 죽이고 부모의 pid(ppid)를 죽여야 하는게 아닌가요???
제 코드가 그런식으로 되어있는데 뭐가 잘못되었는지 좀 부탁드립니다. -
찬바리
서버, 클라이언트라고 하기엔 단촐하지만 서버를 실행하고 클라이언트에서 서버에 연결하는 방식으로 프로그램을 짰습니다. 위에 코드는 서버만 올려놓은 거구요
-
Creator
fork 를 이용한 채팅프로그램인데 왜 서버프로그램/클라이언트프로그램이 따로있죠??