Функция accept
из стандартной библиотеки сокетов позволяет серверу извлечь из очереди соединений информацию о подключившемся клиенте и создать сокет для его обслуживания. Эти действия выполняются безусловно, для любых подключившихся клиентов. Если сервер допускает подключение не любых клиентов, а только тех, которые отвечают некоторым условиям (для протокола TCP эти условия могут заключаться в том, какие IP-адреса и какие порты допустимо использовать клиентам), сразу после установления соединения его приходится разрывать, если клиент не удовлетворяет этим условиям. Для упрощения этой операции в WinSock 2 предусмотрена функция WSAAccept
, прототип которой приведен в листинге 2.40.
Листинг 2.40. Функция WSAAccept
// ***** Описание на C++ *****
SOCKET WSAAccept(SOCKET S, struct sockaddr FAR* addr, LPINT addrlen, LPCONDITIONPROC lpfnCondition, dwCallbackData);
// ***** описание на Delphi *****
function WSAAccept( S: TSocket; Addr: PSockAddr; AddrLen: PInteger; lpfnCondition: TConditionProc; dwCallbackData: DWORD): TSocket;
По сравнению с уже известной нам функцией accept
функция WSAAccept
имеет два новых параметра: lpfnCondition
и dwCallbackData
. lpfnCondition
является указателем на функцию обратного вызова. Эта функция объявляется и реализуется программой. WSAAccept
вызывает ее внутри себя и в зависимости от ее результата принимает или отклоняет соединение. Параметр dwCallbackData
не имеет смысла для самой функции WSAAccept
и передается без изменений в функцию обратного вызова. Тип TConditionProc
должен быть объявлен следующим образом (листинг 2.41).
Листинг 2.41. Тип TConditionProc
// ***** Описание на C++ *****
typedef (int*)(LPWSABUF lpCallerId, LPWSABUF lpCallerData, LPQOS lpSQOS, LPQOS lpGQOS, LPWSABUF lpCalleeId, LPWSABUF lpCalleeData, GROUP FAR* g, DWORD dwCallbackData) LPCONDITIONPROC;
// ***** Описание на Delphi *****
TConditionProc = function(lpCallerId, lpCallerData: PWSABuf; lpSQOS, lpGQOS: PQOS; lpCalleeID, lpCalleeData: PWSABuf; g: PGroup; dwCallbackData: DWORD): Integer; stdcall;
Параметр lpCallerId
указывает на буфер, в котором хранится адрес подключившегося клиента. При работе со стеком TCP/IP lpCallerId^.Len
будет равен SizeOf(TSockAddr)
, a lpCallerId^.Buf
будет указывать на структуру TSockAddr
, содержащую адрес клиента. Параметр lpCallerData
определяет буфер, в котором хранятся данные, переданные клиентом при соединении. Как уже отмечалось, протоколы стека TCP/IP не поддерживают передачу данных при соединении, поэтому для них этот параметр будет равен nil. Параметры lpSQOS
и lpGQOS
задают требуемое клиентом качество обслуживания для сокета и для группы соответственно. Так как группы сокетов в текущей реализации WinSock не поддерживаются, параметр lpGQOS
будет равен nil
. Параметр lpSQOS
тоже будет равен nil
, если клиент не задал качество обслуживания при соединении.
Параметр lpCalleeId
содержит адрес интерфейса, принявшего соединение (поля структуры при этом используются так же, как у параметра lpCallerId
). Ранее уже обсуждалось, что сокет, привязанный к адресу INADDR_ANY
, прослушивает все сетевые интерфейсы, имеющиеся на компьютере, но каждое подключение, созданное с его помощью, использует конкретный интерфейс. Параметр lpCalleeId
содержит адрес, привязанный к конкретному соединению. Параметр lpCalleeData
указывает на буфер, в который сервер может поместить данные для отправки клиенту. Этот параметр также не имеет смысла для протокола TCP, не поддерживающего отправку данных при соединении.
Параметр g
выходной, он позволяет управлять присоединением создаваемого функцией WSAAccept
сокета к группе. Параметр, как и все, связанное с группами, зарезервирован для использования в будущем.
Примечание
Если вы пользуетесь старой версией MSDN, то можете не обнаружить там описания параметра g
— оно там отсутствует. Видимо, просто по ошибке.
И наконец, через параметр dwCallbackData
в функцию обратного вызова передается значение параметра dwCallbackData
, переданное в функцию WSAAccept
. Программист должен сам решить, как ему интерпретировать это значение.
Функция должна вернуть CF_ACCEPT
(0), если соединение принимается, CF_REJECT
(1), если оно отклоняется, и CF_DEFER
(2), если решение о разрешении или запрете соединения откладывается. Если функция обратного вызова вернула CF_REJECT
, to WSAAccept
завершается с ошибкой WSAECONNREFUSED
, если CF_DEFER
— то с ошибкой WSATRY_AGAIN
(в последнем случае соединение остаётся в очереди, и информация о нем вновь будет передана в функцию обратного вызова при следующем вызове WSAAccept
). Обе эти ошибки не фатальные, сокет остается в режиме ожидания соединения и может принимать подключения от новых клиентов.
Читать дальше
Конец ознакомительного отрывка
Купить книгу