connect-timeout.patch 3.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121
  1. Index: netcat-openbsd-1.89/netcat.c
  2. ===================================================================
  3. --- netcat-openbsd-1.89.orig/netcat.c 2008-01-22 16:17:27.000000000 -0500
  4. +++ netcat-openbsd-1.89/netcat.c 2008-01-22 16:17:30.000000000 -0500
  5. @@ -65,6 +65,10 @@
  6. #define PORT_MAX 65535
  7. #define PORT_MAX_LEN 6
  8. +#define CONNECTION_SUCCESS 0
  9. +#define CONNECTION_FAILED 1
  10. +#define CONNECTION_TIMEOUT 2
  11. +
  12. /* Command Line Options */
  13. int dflag; /* detached, no stdin */
  14. int iflag; /* Interval Flag */
  15. @@ -104,6 +108,9 @@
  16. int parse_iptos(char *);
  17. void usage(int);
  18. +static int connect_with_timeout(int fd, const struct sockaddr *sa,
  19. + socklen_t salen, int ctimeout);
  20. +
  21. int
  22. main(int argc, char *argv[])
  23. {
  24. @@ -508,13 +515,15 @@
  25. }
  26. set_common_sockopts(s);
  27. -
  28. - if (connect(s, res0->ai_addr, res0->ai_addrlen) == 0)
  29. + if ((error = connect_with_timeout(s, res0->ai_addr, res0->ai_addrlen, timeout)) == CONNECTION_SUCCESS)
  30. break;
  31. - else if (vflag)
  32. + else if (vflag && error == CONNECTION_FAILED)
  33. warn("connect to %s port %s (%s) failed", host, port,
  34. uflag ? "udp" : "tcp");
  35. -
  36. + else if (vflag && error == CONNECTION_TIMEOUT)
  37. + warn("connect to %s port %s (%s) timed out", host, port,
  38. + uflag ? "udp" : "tcp");
  39. +
  40. close(s);
  41. s = -1;
  42. } while ((res0 = res0->ai_next) != NULL);
  43. @@ -524,6 +533,74 @@
  44. return (s);
  45. }
  46. +static int connect_with_timeout(int fd, const struct sockaddr *sa,
  47. + socklen_t salen, int ctimeout)
  48. +{
  49. + int err;
  50. + struct timeval tv, *tvp = NULL;
  51. + fd_set connect_fdset;
  52. + socklen_t len;
  53. + int orig_flags;
  54. +
  55. + orig_flags = fcntl(fd, F_GETFL, 0);
  56. + if (fcntl(fd, F_SETFL, orig_flags | O_NONBLOCK) < 0 ) {
  57. + warn("can't set O_NONBLOCK - timeout not avaliable");
  58. + if (connect(fd, sa, salen) == 0)
  59. + return CONNECTION_SUCCESS;
  60. + else
  61. + return CONNECTION_FAILED;
  62. + }
  63. +
  64. + /* set connect timeout */
  65. + if (ctimeout > 0) {
  66. + tv.tv_sec = (time_t)ctimeout/1000;
  67. + tv.tv_usec = 0;
  68. + tvp = &tv;
  69. + }
  70. +
  71. + /* attempt the connection */
  72. + err = connect(fd, sa, salen);
  73. +
  74. + if (err != 0 && errno == EINPROGRESS) {
  75. + /* connection is proceeding
  76. + * it is complete (or failed) when select returns */
  77. +
  78. + /* initialize connect_fdset */
  79. + FD_ZERO(&connect_fdset);
  80. + FD_SET(fd, &connect_fdset);
  81. +
  82. + /* call select */
  83. + do {
  84. + err = select(fd + 1, NULL, &connect_fdset,
  85. + NULL, tvp);
  86. + } while (err < 0 && errno == EINTR);
  87. +
  88. + /* select error */
  89. + if (err < 0)
  90. + errx(1,"select error: %s", strerror(errno));
  91. +
  92. + /* we have reached a timeout */
  93. + if (err == 0)
  94. + return CONNECTION_TIMEOUT;
  95. +
  96. + /* select returned successfully, but we must test socket
  97. + * error for result */
  98. + len = sizeof(err);
  99. + if (getsockopt(fd, SOL_SOCKET, SO_ERROR, &err, &len) < 0)
  100. + errx(1, "getsockopt error: %s", strerror(errno));
  101. +
  102. + /* setup errno according to the result returned by
  103. + * getsockopt */
  104. + if (err != 0)
  105. + errno = err;
  106. + }
  107. +
  108. + /* return aborted if an error occured, and valid otherwise */
  109. + fcntl(fd, F_SETFL, orig_flags);
  110. + return (err != 0)? CONNECTION_FAILED : CONNECTION_SUCCESS;
  111. +}
  112. +
  113. +
  114. /*
  115. * local_listen()
  116. * Returns a socket listening on a local port, binds to specified source