[svn-commits] moy: branch moy/mfcr2-1.2 r145016 - /team/moy/mfcr2-1.2/channels/chan_zap.c
SVN commits to the Digium repositories
svn-commits at lists.digium.com
Sat Sep 27 19:32:41 CDT 2008
Author: moy
Date: Sat Sep 27 19:32:40 2008
New Revision: 145016
URL: http://svn.digium.com/view/asterisk?view=rev&rev=145016
Log:
Added support to match extensions after each digit received. This will improve call setup time.
Modified:
team/moy/mfcr2-1.2/channels/chan_zap.c
Modified: team/moy/mfcr2-1.2/channels/chan_zap.c
URL: http://svn.digium.com/view/asterisk/team/moy/mfcr2-1.2/channels/chan_zap.c?view=diff&rev=145016&r1=145015&r2=145016
==============================================================================
--- team/moy/mfcr2-1.2/channels/chan_zap.c (original)
+++ team/moy/mfcr2-1.2/channels/chan_zap.c Sat Sep 27 19:32:40 2008
@@ -807,6 +807,9 @@
openr2_calling_party_category_t mfcr2_category;
int mfcr2_allow_collect_calls;
int mfcr2_forced_release;
+ int mfcr2_dnis_index;
+ int mfcr2_ani_index;
+ int mfcr2_dnis_matched;
#endif
struct zt_distRings drings;
@@ -1341,11 +1344,27 @@
static void zt_r2_on_call_init(openr2_chan_t *r2chan)
{
- /* TODO: how do we know that an external thread (just finished zt_request) is not requesting this very same
- interface but has not yet seized the line, and if we DO know, what to do here? */
struct zt_pvt *p = openr2_chan_get_client_data(r2chan);
ast_mutex_lock(&p->lock);
+ if (p->mfcr2call) {
+ ast_mutex_unlock(&p->lock);
+ /* TODO: This can happen when some other thread just finished zt_request requesting this very same
+ interface but has not yet seized the line (zt_call), and the far end wins and seize the line,
+ can we avoid this somehow?, at this point when zt_call send the seize, it is likely that since
+ the other end will see our seize as a forced release and drop the call, we will see an invalid
+ pattern that will be seen and treated as protocol error. */
+ ast_log(LOG_ERROR, "Collision of calls on chan %d detected!.\n", openr2_chan_get_number(r2chan));
+ return;
+ }
p->mfcr2call = 1;
+ /* better safe than sorry ... */
+ p->cid_name[0] = 0;
+ p->cid_num[0] = 0;
+ p->rdnis[0] = 0;
+ p->exten[0] = 0;
+ p->mfcr2_ani_index = 0;
+ p->mfcr2_dnis_index = 0;
+ p->mfcr2_dnis_matched = 0;
ast_mutex_unlock(&p->lock);
ast_log(LOG_NOTICE, "New MFC/R2 call detected on chan %d.\n", openr2_chan_get_number(r2chan));
}
@@ -1391,19 +1410,17 @@
}
ast_mutex_lock(&p->lock);
p->mfcr2_recvd_category = category;
- if (p->use_callerid && ani) {
- ast_copy_string(p->cid_num, ani, sizeof(p->cid_num));
- ast_copy_string(p->cid_name, ani, sizeof(p->cid_name));
- } else {
+ /* if we're not supposed to use CID, clear whatever we have */
+ if (!p->use_callerid) {
+ ast_log(LOG_DEBUG, "No CID allowed in configuration, CID is being cleared!\n");
p->cid_num[0] = 0;
- p->cid_num[0] = 0;
- }
- ast_copy_string(p->rdnis, dnis, sizeof(p->rdnis));
+ p->cid_name[0] = 0;
+ }
+ /* if we're supposed to answer immediately, clear DNIS and set 's' exten */
if (p->immediate || !openr2_context_get_max_dnis(openr2_chan_get_context(r2chan))) {
- ast_log(LOG_DEBUG, "setting exten => s because of immediate or 0 DNIS configured\n");
- ast_copy_string(p->exten, "s", sizeof(p->exten));
- } else {
- ast_copy_string(p->exten, dnis, sizeof(p->exten));
+ ast_log(LOG_DEBUG, "Setting exten => s because of immediate or 0 DNIS configured\n");
+ p->exten[0] = 's';
+ p->exten[1] = 0;
}
ast_mutex_unlock(&p->lock);
if (!ast_exists_extension(NULL, p->context, p->exten, 1, p->cid_num)) {
@@ -1504,7 +1521,7 @@
}
}
-static void zt_r2_on_call_disconnected(openr2_chan_t *r2chan, openr2_call_disconnect_cause_t cause)
+static void zt_r2_on_call_disconnect(openr2_chan_t *r2chan, openr2_call_disconnect_cause_t cause)
{
struct zt_pvt *p = openr2_chan_get_client_data(r2chan);
ast_log(LOG_NOTICE, "MFC/R2 call disconnected on chan %d\n", openr2_chan_get_number(r2chan));
@@ -1581,7 +1598,7 @@
ast_log(LOG_NOTICE, "Far end unblocked on chan %d\n", openr2_chan_get_number(r2chan));
}
-static void zt_r2_on_context_logging(openr2_context_t *r2context, openr2_log_level_t level, const char *fmt, va_list ap)
+static void zt_r2_on_context_log(openr2_context_t *r2context, openr2_log_level_t level, const char *fmt, va_list ap)
{
char logmsg[256];
char completemsg[sizeof(logmsg)+50];
@@ -1599,20 +1616,64 @@
zt_r2_write_log(level, completemsg);
}
+static int zt_r2_on_dnis_digit_received(openr2_chan_t *r2chan, char digit)
+{
+ struct zt_pvt *p = openr2_chan_get_client_data(r2chan);
+ /* if 'immediate' is set, let's stop requesting DNIS */
+ if (p->immediate) {
+ return 0;
+ }
+ p->exten[p->mfcr2_dnis_index] = digit;
+ p->rdnis[p->mfcr2_dnis_index] = digit;
+ p->mfcr2_dnis_index++;
+ p->exten[p->mfcr2_dnis_index] = 0;
+ p->rdnis[p->mfcr2_dnis_index] = 0;
+ /*
+ ast_log(LOG_DEBUG, "Got digit %c in zap, dnis so far: %s\n", digit, p->exten);
+ int ret;
+ ret = ast_exists_extension(NULL, p->context, p->exten, 1, p->cid_num);
+ ast_log(LOG_DEBUG, "ast_exists_extension(%s, %s, 1, %s) = %d\n", p->context, p->exten, p->cid_num, ret);
+ ret = ast_matchmore_extension(NULL, p->context, p->exten, 1, p->cid_num);
+ ast_log(LOG_DEBUG, "ast_matchmore_extension(%s, %s, 1, %s) = %d\n", p->context, p->exten, p->cid_num, ret);
+ */
+ /* if the DNIS is a match and cannot match more, stop requesting DNIS */
+ if ((p->mfcr2_dnis_matched ||
+ (ast_exists_extension(NULL, p->context, p->exten, 1, p->cid_num) && (p->mfcr2_dnis_matched = 1))) &&
+ !ast_matchmore_extension(NULL, p->context, p->exten, 1, p->cid_num)) {
+ return 0;
+ }
+ /* otherwise keep going */
+ return 1;
+}
+
+static void zt_r2_on_ani_digit_received(openr2_chan_t *r2chan, char digit)
+{
+ struct zt_pvt *p = openr2_chan_get_client_data(r2chan);
+ p->cid_num[p->mfcr2_ani_index] = digit;
+ p->cid_name[p->mfcr2_ani_index] = digit;
+ p->mfcr2_ani_index++;
+ p->cid_num[p->mfcr2_ani_index] = 0;
+ p->cid_name[p->mfcr2_ani_index] = 0;
+}
+
static openr2_event_interface_t zt_r2_event_iface = {
- zt_r2_on_call_init,
- zt_r2_on_call_offered,
- zt_r2_on_call_accepted,
- zt_r2_on_call_answered,
- zt_r2_on_call_disconnected,
- zt_r2_on_call_end,
- zt_r2_on_call_read,
- zt_r2_on_hardware_alarm,
- zt_r2_on_os_error,
- zt_r2_on_protocol_error,
- zt_r2_on_line_blocked,
- zt_r2_on_line_idle,
- zt_r2_on_context_logging
+ .on_call_init = zt_r2_on_call_init,
+ .on_call_offered = zt_r2_on_call_offered,
+ .on_call_accepted = zt_r2_on_call_accepted,
+ .on_call_answered = zt_r2_on_call_answered,
+ .on_call_disconnect = zt_r2_on_call_disconnect,
+ .on_call_end = zt_r2_on_call_end,
+ .on_call_read = zt_r2_on_call_read,
+ .on_hardware_alarm = zt_r2_on_hardware_alarm,
+ .on_os_error = zt_r2_on_os_error,
+ .on_protocol_error = zt_r2_on_protocol_error,
+ .on_line_blocked = zt_r2_on_line_blocked,
+ .on_line_idle = zt_r2_on_line_idle,
+ .on_context_log = zt_r2_on_context_log,
+ .on_dnis_digit_received = zt_r2_on_dnis_digit_received,
+ .on_ani_digit_received = zt_r2_on_ani_digit_received,
+ /* so far we do nothing with billing pulses */
+ .on_billing_pulse_received = NULL
};
static inline int16_t zt_r2_alaw_to_linear(uint8_t sample)
@@ -7532,6 +7593,8 @@
tmp->mfcr2_category = mfcr2_cur_category;
tmp->mfcr2 = zap_r2;
tmp->mfcr2call = 0;
+ tmp->mfcr2_ani_index = 0;
+ tmp->mfcr2_dnis_index = 0;
tmp->mfcr2_allow_collect_calls = mfcr2_cur_allow_collect_calls;
tmp->mfcr2_forced_release = mfcr2_cur_forced_release;
zap_r2->monitored_count++;
@@ -11497,8 +11560,14 @@
mfcr2_cur_call_files = ast_true(v->value);
} else if (!strcasecmp(v->name, "mfcr2_max_ani")) {
mfcr2_cur_max_ani = atoi(v->value);
+ if (mfcr2_cur_max_ani >= AST_MAX_EXTENSION) {
+ mfcr2_cur_max_ani = AST_MAX_EXTENSION - 1;
+ }
} else if (!strcasecmp(v->name, "mfcr2_max_dnis")) {
mfcr2_cur_max_dnis = atoi(v->value);
+ if (mfcr2_cur_max_dnis >= AST_MAX_EXTENSION) {
+ mfcr2_cur_max_dnis = AST_MAX_EXTENSION - 1;
+ }
} else if (!strcasecmp(v->name, "mfcr2_category")) {
mfcr2_cur_category = openr2_proto_get_category(v->value);
if (OR2_CALLING_PARTY_CATEGORY_UNKNOWN == mfcr2_cur_category) {
More information about the svn-commits
mailing list