<p>Friendly Automation <strong>submitted</strong> this change.</p><p><a href="https://gerrit.asterisk.org/c/asterisk/+/16368">View Change</a></p><div style="white-space:pre-wrap">Approvals:
  N A: Looks good to me, but someone else must approve
  Kevin Harwell: Looks good to me, approved
  Friendly Automation: Approved for Submit

</div><pre style="font-family: monospace,monospace; white-space: pre-wrap;">sig_analog: Changes to improve electromechanical signalling compatibility<br><br>This changeset is intended to address compatibility issues encountered<br>when interfacing Asterisk to electromechanical telephone switches that<br>implement ANI-B, ANI-C, or ANI-D.<br><br>In particular the behaviours that this impacts include:<br><br> - FGC-CAMA did not work at all when using MF signaling. Modified the<br>   switch case block to send calls to the correct part of the<br>   signaling-handling state machine.<br><br> - For FGC-CAMA operation, the delay between called number ST and<br>   second wink for ANI spill has been made configurable; previously<br>   all calls were made to wait for one full second.<br><br> - After the ANI spill, previous behavior was to require a 'ST' tone<br>   to advance the call.  This has been changed to allow 'STP' 'ST2P'<br>   or 'ST3P' as well, for compatibility with ANI-D.<br><br> - Store ANI2 (ANI INFO) digits in the CALLERID(ANI2) channel variable.<br><br> - For calls with an ANI failure, No. 1 Crossbar switches will send<br>   forward a single-digit failure code, with no calling number digits<br>   and no ST pulse to terminate the spill.  I've made the ANI timeout<br>   configurable so to reduce dead air time on calls with ANI fail.<br><br> - ANI info digits configurable.  Modern digital switches will send 2<br>   digits, but ANI-B sends only a single info digit.  This caused the<br>   ANI reported by Asterisk to be misaligned.<br><br> - Changed a confusing log message to be more informative.<br><br>ASTERISK-29518<br><br>Change-Id: Ib7e27d987aee4ed9bc3663c57ef413e21b404256<br>---<br>M channels/chan_dahdi.c<br>M channels/chan_dahdi.h<br>M channels/sig_analog.c<br>M channels/sig_analog.h<br>M configs/samples/chan_dahdi.conf.sample<br>5 files changed, 106 insertions(+), 10 deletions(-)<br><br></pre><pre style="font-family: monospace,monospace; white-space: pre-wrap;"><span>diff --git a/channels/chan_dahdi.c b/channels/chan_dahdi.c</span><br><span>index 2c7b673..74c76f7 100644</span><br><span>--- a/channels/chan_dahdi.c</span><br><span>+++ b/channels/chan_dahdi.c</span><br><span>@@ -953,6 +953,10 @@</span><br><span>                    .parkinglot = "",</span><br><span>                  .transfertobusy = 1,</span><br><span> </span><br><span style="color: hsl(120, 100%, 40%);">+                      .ani_info_digits = 2,</span><br><span style="color: hsl(120, 100%, 40%);">+                 .ani_wink_time = 1000,</span><br><span style="color: hsl(120, 100%, 40%);">+                        .ani_timeout = 10000,</span><br><span style="color: hsl(120, 100%, 40%);">+</span><br><span>                      .cid_signalling = CID_SIG_BELL,</span><br><span>                      .cid_start = CID_START_RING,</span><br><span>                         .dahditrcallerid = 0,</span><br><span>@@ -12839,6 +12843,9 @@</span><br><span> </span><br><span>          tmp->polarityonanswerdelay = conf->chan.polarityonanswerdelay;</span><br><span>                 tmp->answeronpolarityswitch = conf->chan.answeronpolarityswitch;</span><br><span style="color: hsl(120, 100%, 40%);">+                tmp->ani_info_digits = conf->chan.ani_info_digits;</span><br><span style="color: hsl(120, 100%, 40%);">+              tmp->ani_wink_time = conf->chan.ani_wink_time;</span><br><span style="color: hsl(120, 100%, 40%);">+          tmp->ani_timeout = conf->chan.ani_timeout;</span><br><span>             tmp->hanguponpolarityswitch = conf->chan.hanguponpolarityswitch;</span><br><span>               tmp->sendcalleridafter = conf->chan.sendcalleridafter;</span><br><span>                 ast_cc_copy_config_params(tmp->cc_params, conf->chan.cc_params);</span><br><span>@@ -12944,6 +12951,9 @@</span><br><span>                             analog_p->channel = tmp->channel;</span><br><span>                              analog_p->polarityonanswerdelay = conf->chan.polarityonanswerdelay;</span><br><span>                            analog_p->answeronpolarityswitch = conf->chan.answeronpolarityswitch;</span><br><span style="color: hsl(120, 100%, 40%);">+                           analog_p->ani_info_digits = conf->chan.ani_info_digits;</span><br><span style="color: hsl(120, 100%, 40%);">+                         analog_p->ani_timeout = conf->chan.ani_timeout;</span><br><span style="color: hsl(120, 100%, 40%);">+                         analog_p->ani_wink_time = conf->chan.ani_wink_time;</span><br><span>                            analog_p->hanguponpolarityswitch = conf->chan.hanguponpolarityswitch;</span><br><span>                          analog_p->permcallwaiting = conf->chan.callwaiting; /* permcallwaiting possibly modified in analog_config_complete */</span><br><span>                          analog_p->callreturn = conf->chan.callreturn;</span><br><span>@@ -18241,6 +18251,12 @@</span><br><span>                       confp->chan.polarityonanswerdelay = atoi(v->value);</span><br><span>            } else if (!strcasecmp(v->name, "answeronpolarityswitch")) {</span><br><span>                    confp->chan.answeronpolarityswitch = ast_true(v->value);</span><br><span style="color: hsl(120, 100%, 40%);">+                } else if (!strcasecmp(v->name, "ani_info_digits")) {</span><br><span style="color: hsl(120, 100%, 40%);">+                    confp->chan.ani_info_digits = atoi(v->value);</span><br><span style="color: hsl(120, 100%, 40%);">+           } else if (!strcasecmp(v->name, "ani_wink_time")) {</span><br><span style="color: hsl(120, 100%, 40%);">+                      confp->chan.ani_wink_time = atoi(v->value);</span><br><span style="color: hsl(120, 100%, 40%);">+             } else if (!strcasecmp(v->name, "ani_timeout")) {</span><br><span style="color: hsl(120, 100%, 40%);">+                        confp->chan.ani_timeout = atoi(v->value);</span><br><span>              } else if (!strcasecmp(v->name, "hanguponpolarityswitch")) {</span><br><span>                    confp->chan.hanguponpolarityswitch = ast_true(v->value);</span><br><span>               } else if (!strcasecmp(v->name, "sendcalleridafter")) {</span><br><span>diff --git a/channels/chan_dahdi.h b/channels/chan_dahdi.h</span><br><span>index f98ebdc..de813f2 100644</span><br><span>--- a/channels/chan_dahdi.h</span><br><span>+++ b/channels/chan_dahdi.h</span><br><span>@@ -182,6 +182,22 @@</span><br><span>          */</span><br><span>  unsigned int answeronpolarityswitch:1;</span><br><span>       /*!</span><br><span style="color: hsl(120, 100%, 40%);">+    * \brief INTEGER, number of ANI INFO digits on a CAMA trunk.</span><br><span style="color: hsl(120, 100%, 40%);">+  * older switches use 1 INFO digit, newer switches use 2 INFO digits</span><br><span style="color: hsl(120, 100%, 40%);">+   * \note Set from the "ani_info_digits" value read in from chan_dahdi.conf</span><br><span style="color: hsl(120, 100%, 40%);">+   */</span><br><span style="color: hsl(120, 100%, 40%);">+   unsigned int ani_info_digits:8;</span><br><span style="color: hsl(120, 100%, 40%);">+       /*!</span><br><span style="color: hsl(120, 100%, 40%);">+    * \brief INTEGER, length of ANI failure timeout in ms.</span><br><span style="color: hsl(120, 100%, 40%);">+        * \note Set from the "ani_timeout" value read in from chan_dahdi.conf</span><br><span style="color: hsl(120, 100%, 40%);">+       */</span><br><span style="color: hsl(120, 100%, 40%);">+   unsigned int ani_timeout:16;</span><br><span style="color: hsl(120, 100%, 40%);">+  /*!</span><br><span style="color: hsl(120, 100%, 40%);">+    * \brief INTEGER, length of time to wait before sending ANI wink in ms.</span><br><span style="color: hsl(120, 100%, 40%);">+       * \note Set from the "ani_wink_time" value read in from chan_dahdi.conf</span><br><span style="color: hsl(120, 100%, 40%);">+     */</span><br><span style="color: hsl(120, 100%, 40%);">+   unsigned int ani_wink_time:16;</span><br><span style="color: hsl(120, 100%, 40%);">+        /*!</span><br><span>   * \brief TRUE if busy detection is enabled.</span><br><span>          * (Listens for the beep-beep busy pattern.)</span><br><span>          * \note Set from the "busydetect" value read in from chan_dahdi.conf</span><br><span>diff --git a/channels/sig_analog.c b/channels/sig_analog.c</span><br><span>index 4356e02..5bb5649 100644</span><br><span>--- a/channels/sig_analog.c</span><br><span>+++ b/channels/sig_analog.c</span><br><span>@@ -1864,7 +1864,6 @@</span><br><span>                                     if (p->sig == ANALOG_SIG_E911) {</span><br><span>                                          analog_off_hook(p);</span><br><span>                                  }</span><br><span style="color: hsl(0, 100%, 40%);">-                                       res = analog_my_getsigstr(chan, dtmfbuf + strlen(dtmfbuf), "#", 3000);</span><br><span>                             }</span><br><span>                            if (res < 1) {</span><br><span>                                    analog_dsp_reset_and_flush_digits(p);</span><br><span>@@ -1927,21 +1926,64 @@</span><br><span>                      goto quit;</span><br><span>           }</span><br><span> </span><br><span style="color: hsl(0, 100%, 40%);">-           if (p->sig == ANALOG_SIG_FGC_CAMA) {</span><br><span style="color: hsl(0, 100%, 40%);">-                 char anibuf[100];</span><br><span style="color: hsl(120, 100%, 40%);">+             if (p->sig == ANALOG_SIG_FGC_CAMA || p->sig == ANALOG_SIG_FGC_CAMAMF) {</span><br><span style="color: hsl(120, 100%, 40%);">+                 /* This if block is where we process ANI for CAMA */</span><br><span> </span><br><span style="color: hsl(0, 100%, 40%);">-                        if (ast_safe_sleep(chan,1000) == -1) {</span><br><span style="color: hsl(120, 100%, 40%);">+                        char anibuf[100];</span><br><span style="color: hsl(120, 100%, 40%);">+                     struct ast_party_caller *caller;</span><br><span style="color: hsl(120, 100%, 40%);">+</span><br><span style="color: hsl(120, 100%, 40%);">+                    /* cnoffset is the point at which we pull the calling number out</span><br><span style="color: hsl(120, 100%, 40%);">+                       * of anibuf. Must be the number of ani_info_digits + 1 to account</span><br><span style="color: hsl(120, 100%, 40%);">+                     * for the KP, which is considered a digit.  */</span><br><span style="color: hsl(120, 100%, 40%);">+</span><br><span style="color: hsl(120, 100%, 40%);">+                     /* The 1XB with ANI-B will send a full 10 digits</span><br><span style="color: hsl(120, 100%, 40%);">+                      *  or 2 digits in case of ANI failure.</span><br><span style="color: hsl(120, 100%, 40%);">+                        * (CD-95811-01 Section II, page 10)</span><br><span style="color: hsl(120, 100%, 40%);">+                   * 10 digit string example:  *08320123#</span><br><span style="color: hsl(120, 100%, 40%);">+                        * 2 digit string example:   *2</span><br><span style="color: hsl(120, 100%, 40%);">+                        * KP (*) and ST (#) are considered to be digits */</span><br><span style="color: hsl(120, 100%, 40%);">+</span><br><span style="color: hsl(120, 100%, 40%);">+                  int cnoffset = p->ani_info_digits + 1;</span><br><span style="color: hsl(120, 100%, 40%);">+                     ast_debug(1, "cnoffset: %d\n", cnoffset);</span><br><span style="color: hsl(120, 100%, 40%);">+</span><br><span style="color: hsl(120, 100%, 40%);">+                 /* This is how long to wait before the wink to start ANI spill</span><br><span style="color: hsl(120, 100%, 40%);">+                         * Pulled from chan_dahdi.conf, default is 1000ms */</span><br><span style="color: hsl(120, 100%, 40%);">+                  if (ast_safe_sleep(chan,p->ani_wink_time) == -1) {</span><br><span>                                ast_hangup(chan);</span><br><span>                            goto quit;</span><br><span>                   }</span><br><span>                    analog_off_hook(p);</span><br><span style="color: hsl(120, 100%, 40%);">+                   ast_debug(1, "Sent wink to signal ANI start\n");</span><br><span>                   analog_dsp_set_digitmode(p, ANALOG_DIGITMODE_MF);</span><br><span style="color: hsl(0, 100%, 40%);">-                       res = analog_my_getsigstr(chan, anibuf, "#", 10000);</span><br><span style="color: hsl(0, 100%, 40%);">-                  if ((res > 0) && (strlen(anibuf) > 2)) {</span><br><span style="color: hsl(0, 100%, 40%);">-                          if (anibuf[strlen(anibuf) - 1] == '#') {</span><br><span style="color: hsl(120, 100%, 40%);">+</span><br><span style="color: hsl(120, 100%, 40%);">+                    /* ani_timeout is configured in chan_dahdi.conf. default is 10000ms.</span><br><span style="color: hsl(120, 100%, 40%);">+                   * ST, STP, ST2P, ST3P can all signal transmission complete, regardless of timeout */</span><br><span style="color: hsl(120, 100%, 40%);">+                 res = analog_my_getsigstr(chan, anibuf, "#ABC", p->ani_timeout);</span><br><span style="color: hsl(120, 100%, 40%);">+</span><br><span style="color: hsl(120, 100%, 40%);">+                       /* so we can work with the ani buffer */</span><br><span style="color: hsl(120, 100%, 40%);">+                      pbx_builtin_setvar_helper(chan, "ANIBUF", anibuf);</span><br><span style="color: hsl(120, 100%, 40%);">+</span><br><span style="color: hsl(120, 100%, 40%);">+                        /* as long as we get a terminating pulse OR the length of ANI buffer is at least >=2, we can treat</span><br><span style="color: hsl(120, 100%, 40%);">+                  * this as a complete spill for the purposes of setting anistart */</span><br><span style="color: hsl(120, 100%, 40%);">+                   if ((res > 0) || (strlen(anibuf) >= 2)) {</span><br><span style="color: hsl(120, 100%, 40%);">+                               char anistart[2] = "X";</span><br><span style="color: hsl(120, 100%, 40%);">+                             char f[10] = {0};</span><br><span style="color: hsl(120, 100%, 40%);">+                             if (strchr("#ABC", anibuf[strlen(anibuf) - 1])) {</span><br><span style="color: hsl(120, 100%, 40%);">+                                   anistart[0] = anibuf[strlen(anibuf) - 1];</span><br><span>                                    anibuf[strlen(anibuf) - 1] = 0;</span><br><span>                              }</span><br><span style="color: hsl(0, 100%, 40%);">-                               ast_set_callerid(chan, anibuf + 2, NULL, anibuf + 2);</span><br><span style="color: hsl(120, 100%, 40%);">+                         ast_set_callerid(chan, anibuf + cnoffset, NULL, anibuf + cnoffset);</span><br><span style="color: hsl(120, 100%, 40%);">+</span><br><span style="color: hsl(120, 100%, 40%);">+                         caller = ast_channel_caller(chan);</span><br><span style="color: hsl(120, 100%, 40%);">+                            strncpy(f, &(anibuf[1]), MIN((int)(p->ani_info_digits), sizeof(f)-1));</span><br><span style="color: hsl(120, 100%, 40%);">+                         caller->ani2 = atoi(f);</span><br><span style="color: hsl(120, 100%, 40%);">+</span><br><span style="color: hsl(120, 100%, 40%);">+                          anibuf[cnoffset] = 0;</span><br><span style="color: hsl(120, 100%, 40%);">+</span><br><span style="color: hsl(120, 100%, 40%);">+                               /* so we can work with the different start pulses as used in ANI-D */</span><br><span style="color: hsl(120, 100%, 40%);">+                         pbx_builtin_setvar_helper(chan, "ANISTART", anistart);</span><br><span style="color: hsl(120, 100%, 40%);">+                              /* so we can use our ANI INFO digits in our dialplan */</span><br><span style="color: hsl(120, 100%, 40%);">+                               pbx_builtin_setvar_helper(chan, "ANI2", anibuf + 1);</span><br><span>                       }</span><br><span>                    analog_dsp_set_digitmode(p, ANALOG_DIGITMODE_DTMF);</span><br><span>          }</span><br><span>@@ -2023,7 +2065,7 @@</span><br><span>                                    ast_copy_string(exten, "911", sizeof(exten));</span><br><span>                              }</span><br><span>                    } else {</span><br><span style="color: hsl(0, 100%, 40%);">-                                ast_log(LOG_WARNING, "Got a non-E911/FGC CAMA input on channel %d.  Assuming E&M Wink instead\n", p->channel);</span><br><span style="color: hsl(120, 100%, 40%);">+                               ast_log(LOG_WARNING, "A KP was expected to start signaling for Feature Group C CAMA-MF, but we got something else. Received: %s on channel %d\n", exten, p->channel);</span><br><span>                   }</span><br><span>            }</span><br><span>            if (p->sig == ANALOG_SIG_FEATB) {</span><br><span>@@ -2035,7 +2077,7 @@</span><br><span>                                 s1 = strsep(&stringp, "#");</span><br><span>                            ast_copy_string(exten, exten2 + 1, sizeof(exten));</span><br><span>                   } else {</span><br><span style="color: hsl(0, 100%, 40%);">-                                ast_log(LOG_WARNING, "Got a non-Feature Group B input on channel %d.  Assuming E&M Wink instead\n", p->channel);</span><br><span style="color: hsl(120, 100%, 40%);">+                             ast_log(LOG_WARNING, "A KP was expected to start signaling for Feature Group B, but we got something else. Received: %s on channel %d\n", exten, p->channel);</span><br><span>                   }</span><br><span>            }</span><br><span>            if ((p->sig == ANALOG_SIG_FEATDMF) || (p->sig == ANALOG_SIG_FEATDMF_TA)) {</span><br><span>diff --git a/channels/sig_analog.h b/channels/sig_analog.h</span><br><span>index 3b9dcf5..488be36 100644</span><br><span>--- a/channels/sig_analog.h</span><br><span>+++ b/channels/sig_analog.h</span><br><span>@@ -277,6 +277,10 @@</span><br><span>     int msgstate;</span><br><span> </span><br><span>    /* XXX: Option Variables - Set by allocator of private structure */</span><br><span style="color: hsl(120, 100%, 40%);">+   unsigned int ani_info_digits:8;                 /* Older switches use 1 INFO digit, newer switches use 2 */</span><br><span style="color: hsl(120, 100%, 40%);">+   unsigned int ani_timeout:16;                    /* Time in ms before we give up waiting for ANI spill */</span><br><span style="color: hsl(120, 100%, 40%);">+      unsigned int ani_wink_time:16;                  /* Safe wait time before we wink to start ANI spill */</span><br><span style="color: hsl(120, 100%, 40%);">+</span><br><span>     unsigned int answeronpolarityswitch:1;</span><br><span>       unsigned int callreturn:1;</span><br><span>   unsigned int cancallforward:1;</span><br><span>diff --git a/configs/samples/chan_dahdi.conf.sample b/configs/samples/chan_dahdi.conf.sample</span><br><span>index d24c22b..5c1d2ee 100644</span><br><span>--- a/configs/samples/chan_dahdi.conf.sample</span><br><span>+++ b/configs/samples/chan_dahdi.conf.sample</span><br><span>@@ -1098,6 +1098,24 @@</span><br><span> ; The default is -1: not to set anything.</span><br><span> ;tonezone = 0 ; 0 is US</span><br><span> ;</span><br><span style="color: hsl(120, 100%, 40%);">+; The number of ANI info digits to expect before the main ANI spill.</span><br><span style="color: hsl(120, 100%, 40%);">+; Switches using ANI-B, -C, and -D will usually send 1 digit. Modern digital</span><br><span style="color: hsl(120, 100%, 40%);">+; systems will send 2, following NANPA ANI II requirements.</span><br><span style="color: hsl(120, 100%, 40%);">+;</span><br><span style="color: hsl(120, 100%, 40%);">+;ani_info_digits=2</span><br><span style="color: hsl(120, 100%, 40%);">+;</span><br><span style="color: hsl(120, 100%, 40%);">+; Time in ms to wait before asterisk sends wink to start ANI spill. Can be</span><br><span style="color: hsl(120, 100%, 40%);">+; shortened if your switch supports it.</span><br><span style="color: hsl(120, 100%, 40%);">+;</span><br><span style="color: hsl(120, 100%, 40%);">+;ani_wink_time=1000</span><br><span style="color: hsl(120, 100%, 40%);">+;</span><br><span style="color: hsl(120, 100%, 40%);">+; Time in ms to wait for each digit in the spill including the ST pulse.</span><br><span style="color: hsl(120, 100%, 40%);">+; This value can affect how long it takes to recognize ANI failures that do</span><br><span style="color: hsl(120, 100%, 40%);">+; not send a ST pulse. If ANI failures take too long to recognize, you can</span><br><span style="color: hsl(120, 100%, 40%);">+; lower this value.</span><br><span style="color: hsl(120, 100%, 40%);">+;</span><br><span style="color: hsl(120, 100%, 40%);">+;ani_timeout=10000</span><br><span style="color: hsl(120, 100%, 40%);">+;</span><br><span> ; FXO (FXS signalled) devices must have a timeout to determine if there was a</span><br><span> ; hangup before the line was answered.  This value can be tweaked to shorten</span><br><span> ; how long it takes before DAHDI considers a non-ringing line to have hungup.</span><br><span></span><br></pre><div style="white-space:pre-wrap"></div><p>To view, visit <a href="https://gerrit.asterisk.org/c/asterisk/+/16368">change 16368</a>. To unsubscribe, or for help writing mail filters, visit <a href="https://gerrit.asterisk.org/settings">settings</a>.</p><div itemscope itemtype="http://schema.org/EmailMessage"><div itemscope itemprop="action" itemtype="http://schema.org/ViewAction"><link itemprop="url" href="https://gerrit.asterisk.org/c/asterisk/+/16368"/><meta itemprop="name" content="View Change"/></div></div>

<div style="display:none"> Gerrit-Project: asterisk </div>
<div style="display:none"> Gerrit-Branch: 19 </div>
<div style="display:none"> Gerrit-Change-Id: Ib7e27d987aee4ed9bc3663c57ef413e21b404256 </div>
<div style="display:none"> Gerrit-Change-Number: 16368 </div>
<div style="display:none"> Gerrit-PatchSet: 2 </div>
<div style="display:none"> Gerrit-Owner: Sarah Autumn <sarah@endlesstemple.org> </div>
<div style="display:none"> Gerrit-Reviewer: Friendly Automation </div>
<div style="display:none"> Gerrit-Reviewer: Kevin Harwell <kharwell@digium.com> </div>
<div style="display:none"> Gerrit-Reviewer: N A <mail@interlinked.x10host.com> </div>
<div style="display:none"> Gerrit-MessageType: merged </div>