Установить тайм-аут для winsock recvfrom - PullRequest
16 голосов
/ 01 декабря 2009

Я пытаюсь настроить сокет блокировки на тайм-аут после 16 мс попытки recvfrom () на порту Платформа Windows. Я просмотрел множество примеров в Интернете, и мне кажется, что все просто, я просто не могу заставить его работать. Любая помощь будет оценена!

#include <winsock2.h>
#include <string>

#pragma comment(lib, "ws2_32.lib")

#define PORT_NUM 8001

int main(void)
{
  std::string localIP;
  sockaddr_in localAddr;
  sockaddr_in remoteAddr;
  hostent* localhost;
  char buffer[1024];
  WSADATA wsData;

  int result = WSAStartup(MAKEWORD(2,2), &wsData);  // winsock version 2

  localhost = gethostbyname("");
  localIP   = inet_ntoa(*(in_addr*)*localhost->h_addr_list);

  localAddr.sin_family       = AF_INET;
  localAddr.sin_port         = htons(PORT_NUM);             // Set Port Number
  localAddr.sin_addr.s_addr  = inet_addr(localIP.c_str());  // Set IP Address

  int mHandle = WSASocket(AF_INET, SOCK_DGRAM, IPPROTO_UDP, NULL, 0, 0);

  if(mHandle == INVALID_SOCKET)
    return 1;


  if(bind(mHandle, (SOCKADDR*)&localAddr, sizeof(localAddr)) == SOCKET_ERROR)
    return 1;

  timeval tv;
  tv.tv_sec  = 0;
  tv.tv_usec = 1600;

    // Set Timeout for recv call
  if(setsockopt(mHandle, SOL_SOCKET, SO_RCVTIMEO, 
                reinterpret_cast<char*>(&tv), sizeof(timeval)))
    return 1;

  int length = sizeof(remoteAddr);

  // <-- Blocks here forever
  recvfrom(mHandle, buffer, 1024, 0, (SOCKADDR*)&remoteAddr, &length);  

  return 0;
}

/*  I've also tried passing the time like so:
int ms = 16;

if(setsockopt(mHandle, SOL_SOCKET, SO_RCVTIMEO, reinterpret_cast<char*>(&ms), sizeof(int)))
  return 1; */

Ответы [ 4 ]

23 голосов
/ 02 декабря 2009

Я посмотрел на функцию выбора, и, как сказала Лора, я должен это сделать и заставил ее работать очень легко! Спасибо!

fd_set fds ;
int n ;
struct timeval tv ;

// Set up the file descriptor set.
FD_ZERO(&fds) ;
FD_SET(mHandle, &fds) ;

// Set up the struct timeval for the timeout.
tv.tv_sec = 10 ;
tv.tv_usec = 0 ;

// Wait until timeout or data received.
n = select ( mHandle, &fds, NULL, NULL, &tv ) ;
if ( n == 0)
{ 
  printf("Timeout..\n");
  return 0 ;
}
else if( n == -1 )
{
  printf("Error..\n");
  return 1;   
}

int length = sizeof(remoteAddr);

recvfrom(mHandle, buffer, 1024, 0, (SOCKADDR*)&remoteAddr, &length); 
15 голосов
/ 07 ноября 2012

Я попробовал, передав его как фоллинг

     int iTimeout = 1600;
     iRet = setsockopt( pSapManager->m_cSocket,
                        SOL_SOCKET,
                        SO_RCVTIMEO,
                        /*
                        reinterpret_cast<char*>(&tv),
                        sizeof(timeval) );
                        */
                        (const char *)&iTimeout,
                        sizeof(iTimeout) );

и запусти !!

11 голосов
/ 25 марта 2012

WINDOWS: Значение тайм-аута - это DWORD в миллисекундах, адрес, переданный setsockopt (), является постоянным символом *

LINUX: Значение тайм-аута является структурным временным интервалом, адрес, переданный setsockopt (), является константным void *

Источник: http://forums.codeguru.com/showthread.php?t=353217

1 голос
/ 01 декабря 2009

Я предполагаю, что Windows из вызова WSASocket ().Если это так, вы неправильно передаете время ожидания.

MSDN говорит, что SO_RCVTIMEO принимает параметр типа int, который указывает время ожидания в мс.

...