프로그래밍의 기초/TCP | IP
윈도우 기반으로 writev & readv 함수 구현하기
Kim나현
2022. 1. 28. 14:53
반응형
- 윈도우에서는 Linux에서 보인 형태의 시그널 핸들링이 존재하지 않는다.
- MSG_OOB 옵션에 대한 이벤트 핸들링이 윈도우 기반에서는 불가능
→ select 함수를 통해 가능
select 함수의 관찰항목: [수신한 데이터, 데이터 전송이 가능한 socket, 예외상황의 발생]
- out-of-band 데이터의 수신도 예외상황에 해당
// msg_oob 를 이용해 readv 구현
#define BUF_SZ 40
int main(int argc, char* argv[])
{
WSADATA wsaData;
int serv_sock, clnt_sock;
SOCKADDR_IN serv_addr, clnt_addr;
int adrsz, len;
char buf[BUF_SZ];
int result;
fd_set reads, except, retemp, extemp;
struct timeval timeout;
if (argc != 2)
{
printf("usage: %s <addr>\n", argv[0]);
exit(1);
}
if (WSAStartup(MAKEWORD(2, 2), &wsaData) != 0)
err_handling("startup() error");
serv_sock = socket(PF_INET, SOCK_STREAM, 0);
if (serv_sock == INVALID_SOCKET)
err_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, &serv_addr, sizeof(serv_addr)) == SOCKET_ERROR)
err_handling("bind() error");
if (listen(serv_sock, 5) == SOCKET_ERROR)
err_handling("listen() error");
adrsz = sizeof(clnt_addr);
clnt_sock = accept(serv_sock, &clnt_addr, &adrsz);
FD_ZERO(&reads);
FD_ZERO(&except);
FD_SET(clnt_sock, &reads);
FD_SET(clnt_sock, &except);
while (1)
{
retemp = reads;
extemp = except;
timeout.tv_sec = 5;
timeout.tv_usec = 0;
result = select(0, &retemp, 0, &extemp, &timeout);
if (result > 0)
{
if (FD_ISSET(clnt_sock, &extemp))
{
len = recv(clnt_sock, buf, BUF_SZ - 1, MSG_OOB);
buf[len] = 0;
printf("urgent msg: %s\n", buf);
}
if (FD_ISSET(clnt_sock, &retemp))
{
len = recv(clnt_sock, buf, BUF_SZ - 1, 0);
if (len == 0)
{
break;
closesocket(clnt_sock);
}
else
{
buf[len] = 0;
printf("msg: %s\n", buf);
}
}
}
}
closesocket(serv_sock);
WSACleanup();
}
// msg_oob를 이용해 writev를 구현
#define BUF_SZ 40
int main(int argc, char* argv[])
{
SOCKET sock;
SOCKADDR_IN serv_adr;
WSADATA wsaData;
char msg[BUF_SZ] = { 0 };
if (argc != 3)
{
printf("usage: %s <ip> <port>\n", argv[0]);
exit(1);
}
if (WSAStartup(MAKEWORD(2, 2), &wsaData) != 0)
err_handling("startup() error");
sock = socket(PF_INET, SOCK_STREAM, 0);
if (sock == INVALID_SOCKET)
err_handling("socket() eror");
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, &serv_adr, sizeof(serv_adr)) == SOCKET_ERROR)
err_handling("connect() eror");
send(sock, "123", 3, 0);
send(sock, "4", 1, MSG_OOB);
send(sock, "567", 3, 0);
send(sock, "890", 3, MSG_OOB);
closesocket(sock);
WSACleanup();
return 0;
}
// 실행창
* 다만 Linux의 wirtev & readv함수는 윈도우의 중첩 입출력에 의해 구현 가능
반응형