diff --git a/channels/chan_dahdi.c b/channels/chan_dahdi.c index 85855e6a40d6b5b49bcb852c503bd1164fc11857..d2a5671354999f451c62f9bd5369c6839f404180 100644 --- a/channels/chan_dahdi.c +++ b/channels/chan_dahdi.c @@ -2737,6 +2737,52 @@ static void my_set_needringing(void *pvt, int value) p->subs[SUB_REAL].needringing = value; } +static void my_set_polarity(void *pvt, int value) +{ + struct dahdi_pvt *p = pvt; + + if (p->channel == CHAN_PSEUDO) { + return; + } + p->polarity = value; + ioctl(p->subs[SUB_REAL].dfd, DAHDI_SETPOLARITY, &value); +} + +static void my_start_polarityswitch(void *pvt) +{ + struct dahdi_pvt *p = pvt; + + if (p->answeronpolarityswitch || p->hanguponpolarityswitch) { + my_set_polarity(pvt, 0); + } +} + +static void my_answer_polarityswitch(void *pvt) +{ + struct dahdi_pvt *p = pvt; + + if (!p->answeronpolarityswitch) { + return; + } + + my_set_polarity(pvt, 1); +} + +static void my_hangup_polarityswitch(void *pvt) +{ + struct dahdi_pvt *p = pvt; + + if (!p->hanguponpolarityswitch) { + return; + } + + if (p->answeronpolarityswitch) { + my_set_polarity(pvt, 0); + } else { + my_set_polarity(pvt, 1); + } +} + static int my_start(void *pvt) { struct dahdi_pvt *p = pvt; @@ -3454,6 +3500,10 @@ static struct analog_callback dahdi_analog_callbacks = .set_pulsedial = my_set_pulsedial, .get_orig_dialstring = my_get_orig_dialstring, .set_needringing = my_set_needringing, + .set_polarity = my_set_polarity, + .start_polarityswitch = my_start_polarityswitch, + .answer_polarityswitch = my_answer_polarityswitch, + .hangup_polarityswitch = my_hangup_polarityswitch, }; /*! Round robin search locations. */ diff --git a/channels/sig_analog.c b/channels/sig_analog.c index 68677f561bc610938abc9442aeb24b26dc4f50f6..3214681bda928c276b7c4771a8112fc40e6d7734 100644 --- a/channels/sig_analog.c +++ b/channels/sig_analog.c @@ -537,6 +537,35 @@ static void analog_set_needringing(struct analog_pvt *p, int value) } } +#if 0 +static void analog_set_polarity(struct analog_pvt *p, int value) +{ + if (p->calls->set_polarity) { + return p->calls->set_polarity(p->chan_pvt, value); + } +} +#endif + +static void analog_start_polarityswitch(struct analog_pvt *p) +{ + if (p->calls->start_polarityswitch) { + return p->calls->start_polarityswitch(p->chan_pvt); + } +} +static void analog_answer_polarityswitch(struct analog_pvt *p) +{ + if (p->calls->answer_polarityswitch) { + return p->calls->answer_polarityswitch(p->chan_pvt); + } +} + +static void analog_hangup_polarityswitch(struct analog_pvt *p) +{ + if (p->calls->hangup_polarityswitch) { + return p->calls->hangup_polarityswitch(p->chan_pvt); + } +} + static int analog_dsp_set_digitmode(struct analog_pvt *p, enum analog_dsp_digitmode mode) { if (p->calls->dsp_set_digitmode) { @@ -1269,6 +1298,7 @@ int analog_hangup(struct analog_pvt *p, struct ast_channel *ast) case ANALOG_SIG_FXOKS: /* If they're off hook, try playing congestion */ if (analog_is_off_hook(p)) { + analog_hangup_polarityswitch(p); analog_play_tone(p, ANALOG_SUB_REAL, ANALOG_TONE_CONGESTION); } else { analog_play_tone(p, ANALOG_SUB_REAL, -1); @@ -1360,9 +1390,21 @@ int analog_answer(struct analog_pvt *p, struct ast_channel *ast) p->owner = p->subs[ANALOG_SUB_REAL].owner; } } - if ((p->sig == ANALOG_SIG_FXSLS) || (p->sig == ANALOG_SIG_FXSKS) || (p->sig == ANALOG_SIG_FXSGS)) { + + switch (p->sig) { + case ANALOG_SIG_FXSLS: + case ANALOG_SIG_FXSKS: + case ANALOG_SIG_FXSGS: analog_set_echocanceller(p, 1); analog_train_echocanceller(p); + break; + case ANALOG_SIG_FXOLS: + case ANALOG_SIG_FXOKS: + case ANALOG_SIG_FXOGS: + analog_answer_polarityswitch(p); + break; + default: + break; } break; default: @@ -2524,6 +2566,7 @@ static struct ast_frame *__analog_handle_event(struct analog_pvt *p, struct ast_ case ANALOG_SIG_FXOLS: case ANALOG_SIG_FXOGS: case ANALOG_SIG_FXOKS: + analog_start_polarityswitch(p); p->fxsoffhookstate = 0; p->onhooktime = time(NULL); p->msgstate = -1; @@ -2705,6 +2748,7 @@ static struct ast_frame *__analog_handle_event(struct analog_pvt *p, struct ast_ ast_setstate(ast, AST_STATE_DIALING); } else { ast_setstate(ast, AST_STATE_UP); + analog_answer_polarityswitch(p); } return &p->subs[index].f; case AST_STATE_DOWN: @@ -3443,6 +3487,7 @@ void *analog_handle_init_event(struct analog_pvt *i, int event) case ANALOG_SIG_FXOLS: case ANALOG_SIG_FXOGS: i->fxsoffhookstate = 0; + analog_start_polarityswitch(i); case ANALOG_SIG_FEATD: case ANALOG_SIG_FEATDMF: case ANALOG_SIG_FEATDMF_TA: diff --git a/channels/sig_analog.h b/channels/sig_analog.h index e7f1ae2be9b6da38e332fb3af87340d96ea1748e..ff62ed52ddf04114d91e46f96986d96385def90f 100644 --- a/channels/sig_analog.h +++ b/channels/sig_analog.h @@ -147,6 +147,14 @@ struct analog_callback { /*! \brief Set channel off hook */ int (* const off_hook)(void *pvt); void (* const set_needringing)(void *pvt, int value); + /*! \brief Set FXS line polarity to 0=IDLE NZ=REVERSED */ + void (* const set_polarity)(void *pvt, int value); + /*! \brief Reset FXS line polarity to IDLE, based on answeronpolarityswitch and hanguponpolarityswitch */ + void (* const start_polarityswitch)(void *pvt); + /*! \brief Switch FXS line polarity, based on answeronpolarityswitch=yes */ + void (* const answer_polarityswitch)(void *pvt); + /*! \brief Switch FXS line polarity, based on answeronpolarityswitch and hanguponpolarityswitch */ + void (* const hangup_polarityswitch)(void *pvt); /* We're assuming that we're going to only wink on ANALOG_SUB_REAL - even though in the code there's an argument to the index * function */ int (* const wink)(void *pvt, enum analog_sub sub); diff --git a/configs/chan_dahdi.conf.sample b/configs/chan_dahdi.conf.sample index 0bc4ac82eb04aa8edb65002fa4277723f5672a38..56794320fa613452b48d45f2b352bb6e11d7ab0c 100644 --- a/configs/chan_dahdi.conf.sample +++ b/configs/chan_dahdi.conf.sample @@ -820,14 +820,22 @@ pickupgroup=1 ; useful to use the ztmonitor utility to record the audio that main/dsp.c ; is receiving after the caller hangs up. ; -; Use a polarity reversal to mark when a outgoing call is answered by the -; remote party. +; For FXS (FXO signalled) ports +; switch the line polarity to signal the connected PBX that an outgoing +; call was answered by the remote party. +; For FXO (FXS signalled) ports +; watch for a polarity reversal to mark when a outgoing call is +; answered by the remote party. ; ;answeronpolarityswitch=yes ; -; In some countries, a polarity reversal is used to signal the disconnect of a -; phone line. If the hanguponpolarityswitch option is selected, the call will -; be considered "hung up" on a polarity reversal. +; For FXS (FXO signalled) ports +; switch the line polarity to signal the connected PBX that the current +; call was "hung up" by the remote party +; For FXO (FXS signalled) ports +; In some countries, a polarity reversal is used to signal the disconnect of a +; phone line. If the hanguponpolarityswitch option is selected, the call will +; be considered "hung up" on a polarity reversal. ; ;hanguponpolarityswitch=yes ;