WSASend WSASend 함수는 접속된 상대방 소켓에게 지정한 데이터를 보내는 함수입니다. int WSASend (
Parameters s lpBuffers dwBufferCount lpNumberOfBytesSent dwFlags lpOverlapped lpCompletionRoutine
Remarks WSASend 함수는 기존의 send 함수의 기능을 모두 다 수행 하면서, 아래의 두가지 항목을 추가적으로 지원합니다. 1. 중복된 전송 연산을 수행하도록 중복(overlapped)소켓을 가지고 작업할 수 있는 기능 WSASend 함수는 s 매개변수로 로 지정된 접속지향형 소켓에서 한 개 또는 그 이상의 데이터를 전송 하기위해서 사용하는 함수입니다. 이 함수는 비접속 지향형 소켓(SOCK_DGRAM 과 같은 소켓이겠죠?) 에서도 사용 가능합니다. 하지만, 이렇게 사용하려 하는 경우에, 소켓은 connect 함수나 WSAConnect 함수에 의해서 상대방 어드레스가 지정되어 있어야 합니다. 오버랩 소켓(WSASocket 함수를 WSA_FLAG_OVERLAPPED 플래그를 두어 생성한 소켓)은 오버랩 입/출력을 사용하여 정보를 전송하게 됩니다. 하지만, lpOverlapped 와 lpCompleteRoutine 매개변수가 NULL인 경우 소켓은 넌-오버랩 소켓으로 간주되어 처리되게 됩니다. 전송하는데 지정된 버퍼가 다 사용되었을 때 완료루틴이 호출되거나, 이벤트 객체가 셋팅되는 방식으로 작업이 완료 되었다는 것을 알 수 있습니다. 만약 작업이 바로 완료되지 않는 경우엔 완료루틴이나, WSAGetOverlappedResult 함수에 의해서 마지막 완료 상태를 얻어낼 수 있습니다. 넌-오버랩 소켓에서는 마지막 두 매개변수(lpOverlapped, lpCompletetionRoutine)은 무시됩니다. 그리고, WSASend 함수는 send 함수와 같은 동작을 하도록 사용되어 집니다. 데이터는 지정된 버퍼로부터 전송버퍼로 복사됩니다. 만약 소켓이 비동기 모드인 스트림 소켓이고, 전송버퍼에 충분한 공간이 남아있지 않다면, WSASend 함수는 전송하는데 사용되었던 버퍼 만큼만을 반환 할 것입니다. 블록킹 소켓의 경우는 전송버퍼 공간이 남아서 지정된 버퍼를 전송버퍼로 다 복사 할 때 까지 블록킹 됩니다. lpBuffers 매개변수에 의해서 포인트되는 WSABUF 구조체 배열은 일시적인 공간입니다. 전송연산이 오버랩 처리를 완료했다면, 이 공간은 서비스 프로바이더의 책임입니다. 즉, WSASend 함수가 반환하기전에 이 WSABUF 구조체를 서비스 프로바이더가 캡쳐해야 하는 것이죠. 이러한 처리방식은 WSABUF 배열을 스택기반으로 이루어질 수 있도록 하게 됩니다. 메시지 지향형 소켓(비접속 지향형 소켓)의 경우 서비스 프로바이더에서 지원할 수 있는 메시지의 최대 크기를 넘지 않도록 조심해야 합니다. 이 최대값은 SO_MAX_MSG_SIZE 옵션으로 얻어낼 수 있죠. 만약 전송할 데이터가 전송하기에 너무 큰데이터라면, WSAEMSGSIZE 라는 에러를 반환 하게 됩니다. 글구... 아무런 데이터도 전송되지 않게 됩니다. Note : WSASend 함수를 성공적으로 수행 했다고 해서 데이터가 성공적으로 상대방에게 전송되었다고 확신 할 수는 없습니다. dwFlags 매개변수는 함수를 호출했을 때 어떻게 데이터를 전송할 것인지 세부적인 함수의 행동을 설정하는데 사용하는 값입니다. 즉, 이 함수의 동작을 결정하는 것은 소켓 옵션과 dwFlags 매개변수라고 할 수 있습니다. dwFlag 매개변수에 들어가는 값은 아래에 제시한 값들을 OR 로 구성하여 넘겨주면 됩니다.
Overlapped socket I/O 오버랩 연산이 바로 완료 되었다면, WSASend 함수는 0 값을 반환 합니다. 그리고, lpNumberOfBytesSent 매개변수는 전송된 바이트의 수를 포인트 합니다. 오버랩 연산이 성공적으로 시작되었으나, 나중에 완료될 것이라면, WSASend 함수는 SOCKET_ERROR을 반환하고, WSA_IO_PENDING 이라는 에러코드를 발생할 것입니다. 이러한 경우에 lpNumberOfBytesSent 매개변수는 전송된 값을 포인트 하지 않습니다. 오버랩 연산이 완료되었을 때, 전송된 데이터의 수치는 완료루틴의 cbTransferred 매개변수나 WSAGetOverlappedResult 함수의 lpcbTransfer 매개변수를 가지고 알 수 있습니다. WSASend 함수는 이전에 호출한 WSARecv, WSARecvFrom, WSASend, WSASendTo 함수들의 완료함수 내부에서 호출될 수 있습니다. 만약 여러개의 I/O 연산이 동시에 일어난 다면, 각각의 연산은 서로다른 WSAOVERLAPPED 구조체를 참조해야 합니다. lpCompletionRoutine 매개변수가 NULL이라면, lpOverlapped 매개변수의 hEvent 필드는 오버랩 연산이 완료 되었을 때 신호를 받게 됩니다. 어플레케이션은 WSAWaitForMultipleEvents 나 WSAGetOverlappedResult 함수를 사용해서 이벤트 객체에 대해서 대기하거나 폴링할 수 있습니다. lpCompletionRoutine 매개변수가 NULL이 아니 이라면, hEvent 필드는 무시되고, 완료루틴을 위한 흐름정보(context information)를 사용할 수 있게 됩니다. 이렇게 NULL 이 아닌 lpCompletionRoutine 을 넘겨주는 콜러와 나중에 같은 오버랩 I/O 요청을 위해 호출되는 WSAGetOverlappedResult 함수는 WSAGetOverlappedResult 함수를 TRUE로 호출하기위한 fWait 맵개변수를 셋팅하지 않습니다. 이러한 경우에 hEvent 필드를 사용하는 특별한 방법은 정의되지 않습니다. 그리고, hEvent 필드를 사용해 대기하려 하면, 예측할 수 없는 결과를 가져 올 수도 있습니다. 완료루틴은 Win32 파일 I/O 완료루틴과 같은 규칙을 따르게 됩니다. 완료루틴은 WSAWaitForMultipleEvents 함수가 fAlertable 매개변수를 TRUE로 설정하여 호출될 때와 같이 작업쓰레드가 반응할수있는(alertable) 대기 상태에 있을 때 까지 호출되지 않을 것입니다. 완료루틴의 프로토타입은 아래와 같습니다. void CALLBACK CompletionROUTINE( CompletionRoutine 함수는 어플리케이션-정의 또는 라이브러리 정의 함수 이름으로 사용됩니다. dwError 매개변수는 lpOverlapped 에 의해서 표현되는 오버랩 연산에 대한 완료의 상태를 가리킵니다. cbTransferred 매개변수는 전송한 데이터의 바이트 수를 나타냅니다. 일반적으로 dwFlags 매개변수는 잘 사용되지 않습니다. 이값은 0 값을 가지게 될 것입니다. 이 함수는 반환값이 없습니다.
Return Values 에러가 발생하지 않고, 전송연산이 바로 완료되었다면, WSASend 함수는 0을 반환 합니다. 이러한 경우에 완료루틴은 호출쓰레드가 반응할 수 있는(alertable) 상태로 호출되기 위한 준비를 이미 같추었다고 볼 수 있습니다. 에러가 발생한 경우 SOCKET_ERROR 값이 반환 됩니다. 그리고, WSAGetLastError 함수를 호출해 특정한 에러코드를 얻어낼 수 있습니다.
Error Codes
QuickInfo Windows NT : 사용가능 See Alsooverview, WSACloseEvent, WSACreateEvent, WSAGetOverlappedResult, WSASocket, WSAWaitForMultipleEvents |
'C/C++' 카테고리의 다른 글
WSAEventSelect 모델 (0) | 2014.05.02 |
---|---|
WSAEventSelect (0) | 2014.05.02 |
소켓 입출력 모델 - Overlapped 모델 (0) | 2014.05.02 |
iocp주의점 (0) | 2014.05.02 |
중첩된 io (0) | 2014.05.02 |