Отвечая на ICMP в избранном - PullRequest
       78

Отвечая на ICMP в избранном

3 голосов
/ 05 декабря 2011

Базовая кодовая последовательность, которая мне интересна: (псевдокод)

sendto(some host); // host may be unreachable for now which is normal
...
if(select(readfs, timeout)) // there are some data to read
  recvfrom();

Начиная с Win2000, пакет ICMP, который отправляется обратно после отправки дейтаграммы UDP на недоступный порт, запускается триггерами, после которых происходит сбой recvfromс WSAECONNRESET.Такое поведение нежелательно для меня, потому что в этом случае я хочу выбрать для окончания тайм-аут (нет данных для чтения).В Windows это можно решить с помощью WSAIoctl SIO_UDP_CONNRESET (http://support.microsoft.com/kb/263823).

Мои вопросы:

  1. Является ли SIO_UDP_CONNRESET лучшим способом в этой ситуации?
  2. Существуют ли другие способы игнорировать ICMP для «выбора» или фильтровать его для recvfrom (может быть, игнорируя ошибку WSAECONNRESET в Windows, трактуя ее как тайм-аут, может ли эта ошибка быть запущена в другом случае)?
  3. Есть ли похожие проблемы в Linux и Unix (Solaris, OpenBSD)?

1 Ответ

2 голосов
/ 05 декабря 2011
Набор

select() readfds на самом деле просто сообщает, что read() в сокете не будет блокировать - он не обещает ничего о том, есть или нет фактические доступные данныечитать.

Я не знаю, что конкретно вы пытаетесь достичь с помощью двухсекундного тайм-аута, а не просто спать вечно - и почему вы не можете просто добавить блок if для проверкидля WSAECONNRESET из recvfrom() - но такое чувство, что у вас слишком сложный дизайн, если он плохо справляется с этим делом.

Страница select_tut(2)во многих системах Linux есть некоторые рекомендации по правильному использованию select().Вот несколько правил, которые кажутся наиболее подходящими для вашей ситуации:

   1.  You should always try to use select() without a timeout.
       Your program should have nothing to do if there is no
       data available.  Code that depends on timeouts is not
       usually portable and is difficult to debug.

   ...

   3.  No file descriptor must be added to any set if you do not
       intend to check its result after the select() call, and
       respond appropriately.  See next rule.

   4.  After select() returns, all file descriptors in all sets
       should be checked to see if they are ready.
...