프로그래밍의 기초/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함수는 윈도우의 중첩 입출력에 의해 구현 가능

반응형