https://hyeonski.tistory.com/9
와
https://tjdahr25.tistory.com/46
코드 비교 하면서 해보자
즉 tjdahr25님의 코드는 IO멀티플렉싱이 안 되어 있고
서버부분의 코드
#define BUFSIZE 1024
void error_handling(char *message);
int main(int argc, char **argv){
int serv_sock;
int clnt_sock;
char message[BUFSIZE];
int str_len;
struct sockaddr_in serv_addr;
struct sockaddr_in clnt_addr;
int clnt_addr_size;
if(argc!=2){
printf("Usage : %s <port>\n", argv[0]);
exit(1);
}
serv_sock=socket(PF_INET, SOCK_STREAM, 0);
if(serv_sock == -1) {
error_handling("socket() error");
}
memset(&serv_addr, 0, sizeof(serv_addr));
serv_addr.sin_family=AF_INET;
serv_addr.sin_addr.s_addr=htonl(INADDR_ANY);
serv_addr.sin_port=htons(atoi(argv[1]));
if(bind(serv_sock, (struct sockaddr*) &serv_addr, sizeof(serv_addr))==-1) {
error_handling("bind() error");
}
if(listen(serv_sock, 5)==-1){
error_handling("listen() error");
}
clnt_addr_size=sizeof(clnt_addr);
clnt_sock=accept(serv_sock, (struct sockaddr*)&clnt_addr,&clnt_addr_size);
if(clnt_sock==-1){
error_handling("accept() error");
}
/* 데이터 수신 및 전송 */
while( (str_len=read(clnt_sock,message, BUFSIZE)) != 0){
write(clnt_sock, message, str_len);
write(1, message, str_len);
}
close(clnt_sock);
return 0;
}
void error_handling(char *message)
{
fputs(message, stderr);
fputc('\n', stderr);
exit(1);
}
클라이언트 부분의 코드
#include <stdio.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <stdlib.h>
#include <string.h>
#define BUFSIZE 1024
void error_handling(char *message);
int main(int argc, char **argv) {
int sock;
char message[BUFSIZE];
int str_len;
struct sockaddr_in serv_addr;
if(argc!=3){
printf("Usage : %s <IP> <port>\n", argv[0]);
exit(1);
}
sock=socket(PF_INET, SOCK_STREAM, 0);
if(sock == -1) {
error_handling("socket() error");
}
memset(&serv_addr, 0, sizeof(serv_addr));
serv_addr.sin_family=AF_INET;
serv_addr.sin_addr.s_addr=inet_addr(argv[1]);
serv_addr.sin_port=htons(atoi(argv[2]));
if(connect(sock, (struct sockaddr*)&serv_addr, sizeof(serv_addr))==-1){
error_handling("connect() error!");
}
while(1) {
/* 메세지 입력, 전송 */
fputs("전송할 메시지를 입력 하세요 (q to quit) : ", stdout);
fgets(message, BUFSIZE, stdin);
if(!strcmp(message,"q\n")) break;
write(sock, message, strlen(message));
/* 메세지 수신, 출력 */
str_len=read(sock, message, BUFSIZE-1);
message[str_len]=0;
printf("서버로부터 전송된 메시지 : %s \n", message);
}
close(sock);
return 0;
}
void error_handling(char *message){
fputs(message, stderr);
fputc('\n', stderr);
exit(1);
}
hyeonski님은 멀티플렉싱을 했다
서버 부분의 코드
#include <sys/types.h>
#include <sys/event.h>
#include <sys/time.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <unistd.h>
#include <fcntl.h>
#include <iostream>
#include <map>
#include <vector>
using namespace std;
void exit_with_perror(const string& msg)
{
cerr << msg << endl;
exit(EXIT_FAILURE);
}
void change_events(vector<struct kevent>& change_list, uintptr_t ident, int16_t filter,
uint16_t flags, uint32_t fflags, intptr_t data, void *udata)
{
struct kevent temp_event;
EV_SET(&temp_event, ident, filter, flags, fflags, data, udata);
change_list.push_back(temp_event);
}
void disconnect_client(int client_fd, map<int, string>& clients)
{
cout << "client disconnected: " << client_fd << endl;
close(client_fd);
clients.erase(client_fd);
}
int main()
{
/* init server socket and listen */
int server_socket;
struct sockaddr_in server_addr;
if ((server_socket = socket(PF_INET, SOCK_STREAM, 0)) == -1)
exit_with_perror("socket() error\n" + string(strerror(errno)));
memset(&server_addr, 0, sizeof(server_addr));
server_addr.sin_family = AF_INET;
server_addr.sin_addr.s_addr = htonl(INADDR_ANY);
server_addr.sin_port = htons(8080);
if (bind(server_socket, (struct sockaddr*)&server_addr, sizeof(server_addr)) == -1)
exit_with_perror("bind() error\n" + string(strerror(errno)));
if (listen(server_socket, 5) == -1)
exit_with_perror("listen() error\n" + string(strerror(errno)));
fcntl(server_socket, F_SETFL, O_NONBLOCK);
/* init kqueue */
int kq;
if ((kq = kqueue()) == -1)
exit_with_perror("kqueue() error\n" + string(strerror(errno)));
map<int, string> clients; // map for client socket:data
vector<struct kevent> change_list; // kevent vector for changelist
struct kevent event_list[8]; // kevent array for eventlist
/* add event for server socket */
change_events(change_list, server_socket, EVFILT_READ, EV_ADD | EV_ENABLE, 0, 0, NULL);
cout << "echo server started" << endl;
/* main loop */
int new_events;
struct kevent* curr_event;
while (1)
{
/* apply changes and return new events(pending events) */
new_events = kevent(kq, &change_list[0], change_list.size(), event_list, 8, NULL);
if (new_events == -1)
exit_with_perror("kevent() error\n" + string(strerror(errno)));
change_list.clear(); // clear change_list for new changes
for (int i = 0; i < new_events; ++i)
{
curr_event = &event_list[i];
/* check error event return */
if (curr_event->flags & EV_ERROR)
{
if (curr_event->ident == server_socket)
exit_with_perror("server socket error");
else
{
cerr << "client socket error" << endl;
disconnect_client(curr_event->ident, clients);
}
}
else if (curr_event->filter == EVFILT_READ)
{
if (curr_event->ident == server_socket)
{
/* accept new client */
int client_socket;
if ((client_socket = accept(server_socket, NULL, NULL)) == -1)
exit_with_perror("accept() error\n" + string(strerror(errno)));
cout << "accept new client: " << client_socket << endl;
fcntl(client_socket, F_SETFL, O_NONBLOCK);
/* add event for client socket - add read && write event */
change_events(change_list, client_socket, EVFILT_READ, EV_ADD | EV_ENABLE, 0, 0, NULL);
change_events(change_list, client_socket, EVFILT_WRITE, EV_ADD | EV_ENABLE, 0, 0, NULL);
clients[client_socket] = "";
}
else if (clients.find(curr_event->ident)!= clients.end())
{
/* read data from client */
char buf[1024];
int n = read(curr_event->ident, buf, sizeof(buf));
if (n <= 0)
{
if (n < 0)
cerr << "client read error!" << endl;
disconnect_client(curr_event->ident, clients);
}
else
{
buf[n] = '\0';
clients[curr_event->ident] += buf;
cout << "received data from " << curr_event->ident << ": " << clients[curr_event->ident] << endl;
}
}
}
else if (curr_event->filter == EVFILT_WRITE)
{
/* send data to client */
map<int, string>::iterator it = clients.find(curr_event->ident);
if (it != clients.end())
{
if (clients[curr_event->ident] != "")
{
int n;
if ((n = write(curr_event->ident, clients[curr_event->ident].c_str(),
clients[curr_event->ident].size()) == -1))
{
cerr << "client write error!" << endl;
disconnect_client(curr_event->ident, clients);
}
else
clients[curr_event->ident].clear();
}
}
}
}
}
return (0);
}
그리고 거기서 더나아가 지현님이 포트를 여러개 받아서 가능하게 하고자 했다.