]> git.ipfire.org Git - thirdparty/asterisk.git/commitdiff
Fix timeouts for ast_waitfordigit[_full].
authorDavid M. Lee <dlee@digium.com>
Thu, 13 Sep 2012 18:44:30 +0000 (18:44 +0000)
committerDavid M. Lee <dlee@digium.com>
Thu, 13 Sep 2012 18:44:30 +0000 (18:44 +0000)
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/
........

Merged revisions 373024 from http://svn.asterisk.org/svn/asterisk/branches/1.8

git-svn-id: https://origsvn.digium.com/svn/asterisk/branches/10@373025 65c4cc65-6c06-0410-ace0-fbb531ad65f3

include/asterisk/channel.h
main/channel.c

index 1ff14220fb5d2d74c12c7d316e19e1bf16309dc4..8966f541b187acc49321e7c4e40941c4dd40c04a 100644 (file)
@@ -1910,7 +1910,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);
@@ -1919,7 +1919,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
index 93e9fc0f46088607418f7a470742181df24c0023..c7c4e27f6e2cd662498cf57f453531188df13145 100644 (file)
@@ -3566,7 +3566,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);
@@ -3615,8 +3614,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;
@@ -3624,15 +3625,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;