From: David M. Lee Date: Thu, 13 Sep 2012 18:39:40 +0000 (+0000) Subject: Fix timeouts for ast_waitfordigit[_full]. X-Git-Tag: 1.8.18.0-rc1~51 X-Git-Url: http://git.ipfire.org/cgi-bin/gitweb.cgi?a=commitdiff_plain;h=5cf7e22c0824bdc3bec8cc5ab800bb291f9b6788;p=thirdparty%2Fasterisk.git Fix timeouts for ast_waitfordigit[_full]. ast_waitfordigit_full would simply pass its timeout to ast_waitfor_nandfds, expecting it to decrement the timeout by however many milliseconds were waited. This is a problem if it consistently waits less than 1ms. The timeout will never be decremented, and we wait... FOREVER! This patch makes ast_waitfordigit_full manage the timeout itself. It maintains the previously undocumented behavior that negative timeouts wait forever. (closes issue ASTERISK-20375) Reported by: Mark Michelson Tested by: Mark Michelson Review: https://reviewboard.asterisk.org/r/2109/ git-svn-id: https://origsvn.digium.com/svn/asterisk/branches/1.8@373024 65c4cc65-6c06-0410-ace0-fbb531ad65f3 --- diff --git a/include/asterisk/channel.h b/include/asterisk/channel.h index 60ab9d18ff..cefb37b852 100644 --- a/include/asterisk/channel.h +++ b/include/asterisk/channel.h @@ -1874,7 +1874,7 @@ char *ast_recvtext(struct ast_channel *chan, int timeout); /*! * \brief Waits for a digit * \param c channel to wait for a digit on - * \param ms how many milliseconds to wait + * \param ms how many milliseconds to wait (<0 for indefinite). * \return Returns <0 on error, 0 on no entry, and the digit on success. */ int ast_waitfordigit(struct ast_channel *c, int ms); @@ -1883,7 +1883,7 @@ int ast_waitfordigit(struct ast_channel *c, int ms); * \brief Wait for a digit * Same as ast_waitfordigit() with audio fd for outputting read audio and ctrlfd to monitor for reading. * \param c channel to wait for a digit on - * \param ms how many milliseconds to wait + * \param ms how many milliseconds to wait (<0 for indefinite). * \param audiofd audio file descriptor to write to if audio frames are received * \param ctrlfd control file descriptor to monitor for reading * \return Returns 1 if ctrlfd becomes available diff --git a/main/channel.c b/main/channel.c index 7307065c3a..e69cde0c26 100644 --- a/main/channel.c +++ b/main/channel.c @@ -3515,7 +3515,6 @@ int ast_waitfor(struct ast_channel *c, int ms) return ms; } -/* XXX never to be called with ms = -1 */ int ast_waitfordigit(struct ast_channel *c, int ms) { return ast_waitfordigit_full(c, ms, -1, -1); @@ -3564,8 +3563,10 @@ int ast_settimeout(struct ast_channel *c, unsigned int rate, int (*func)(const v return res; } -int ast_waitfordigit_full(struct ast_channel *c, int ms, int audiofd, int cmdfd) +int ast_waitfordigit_full(struct ast_channel *c, int timeout_ms, int audiofd, int cmdfd) { + struct timeval start = ast_tvnow(); + /* Stop if we're a zombie or need a soft hangup */ if (ast_test_flag(c, AST_FLAG_ZOMBIE) || ast_check_hangup(c)) return -1; @@ -3573,15 +3574,29 @@ int ast_waitfordigit_full(struct ast_channel *c, int ms, int audiofd, int cmdfd) /* Only look for the end of DTMF, don't bother with the beginning and don't emulate things */ ast_set_flag(c, AST_FLAG_END_DTMF_ONLY); - /* Wait for a digit, no more than ms milliseconds total. */ - - while (ms) { + /* Wait for a digit, no more than timeout_ms milliseconds total. + * Or, wait indefinitely if timeout_ms is <0. + */ + while (ast_tvdiff_ms(ast_tvnow(), start) < timeout_ms || timeout_ms < 0) { struct ast_channel *rchan; int outfd=-1; + int ms; + + if (timeout_ms < 0) { + ms = timeout_ms; + } else { + ms = timeout_ms - ast_tvdiff_ms(ast_tvnow(), start); + if (ms < 0) { + ms = 0; + } + } errno = 0; + /* While ast_waitfor_nandfds tries to help by reducing the timeout by how much was waited, + * it is unhelpful if it waited less than a millisecond. + */ rchan = ast_waitfor_nandfds(&c, 1, &cmdfd, (cmdfd > -1) ? 1 : 0, NULL, &outfd, &ms); - + if (!rchan && outfd < 0 && ms) { if (errno == 0 || errno == EINTR) continue;