freetdm: ftmod_zt: Refactor event processing to store OOB events that arrive during read/write operations

This commit is contained in:
Moises Silva 2013-11-01 02:07:10 -04:00
parent e945042c4d
commit 53d16f6193
1 changed files with 83 additions and 32 deletions

View File

@ -951,6 +951,21 @@ static FIO_GET_ALARMS_FUNCTION(zt_get_alarms)
return FTDM_SUCCESS; return FTDM_SUCCESS;
} }
#define ftdm_zt_set_event_pending(fchan) \
do { \
ftdm_set_io_flag(fchan, FTDM_CHANNEL_IO_EVENT); \
fchan->last_event_time = ftdm_current_time_in_ms(); \
} while (0);
#define ftdm_zt_store_chan_event(fchan, revent) \
do { \
if (fchan->io_data) { \
ftdm_log_chan(fchan, FTDM_LOG_WARNING, "Dropping event %d, not retrieved on time\n", revent); \
} \
fchan->io_data = (void *)zt_event_id; \
ftdm_zt_set_event_pending(fchan); \
} while (0);
/** /**
* \brief Waits for an event on a Zaptel/DAHDI channel * \brief Waits for an event on a Zaptel/DAHDI channel
* \param ftdmchan Channel to open * \param ftdmchan Channel to open
@ -1015,7 +1030,7 @@ pollagain:
*flags |= FTDM_WRITE; *flags |= FTDM_WRITE;
} }
if (inflags & POLLPRI) { if ((inflags & POLLPRI) || (ftdmchan->io_data && (*flags & FTDM_EVENTS))) {
*flags |= FTDM_EVENTS; *flags |= FTDM_EVENTS;
} }
@ -1046,15 +1061,24 @@ FIO_SPAN_POLL_EVENT_FUNCTION(zt_poll_event)
if (r == 0) { if (r == 0) {
return FTDM_TIMEOUT; return FTDM_TIMEOUT;
} else if (r < 0 || (pfds[i-1].revents & POLLERR)) { } else if (r < 0) {
snprintf(span->last_error, sizeof(span->last_error), "%s", strerror(errno)); snprintf(span->last_error, sizeof(span->last_error), "%s", strerror(errno));
return FTDM_FAIL; return FTDM_FAIL;
} }
for(i = 1; i <= span->chan_count; i++) { for(i = 1; i <= span->chan_count; i++) {
if (pfds[i-1].revents & POLLPRI) {
ftdm_set_io_flag(span->channels[i], FTDM_CHANNEL_IO_EVENT); ftdm_channel_lock(span->channels[i]);
span->channels[i]->last_event_time = ftdm_current_time_in_ms();
if (pfds[i-1].revents & POLLERR) {
ftdm_log_chan(span->channels[i], FTDM_LOG_ERROR, "POLLERR, flags=%d\n", pfds[i-1].events);
ftdm_channel_unlock(span->channels[i]);
continue;
}
if ((pfds[i-1].revents & POLLPRI) || (span->channels[i]->io_data)) {
ftdm_zt_set_event_pending(span->channels[i]);
k++; k++;
} }
if (pfds[i-1].revents & POLLIN) { if (pfds[i-1].revents & POLLIN) {
@ -1063,6 +1087,9 @@ FIO_SPAN_POLL_EVENT_FUNCTION(zt_poll_event)
if (pfds[i-1].revents & POLLOUT) { if (pfds[i-1].revents & POLLOUT) {
ftdm_set_io_flag(span->channels[i], FTDM_CHANNEL_IO_WRITE); ftdm_set_io_flag(span->channels[i], FTDM_CHANNEL_IO_WRITE);
} }
ftdm_channel_unlock(span->channels[i]);
} }
if (!k) { if (!k) {
@ -1241,7 +1268,10 @@ FIO_CHANNEL_NEXT_EVENT_FUNCTION(zt_channel_next_event)
ftdm_clear_io_flag(ftdmchan, FTDM_CHANNEL_IO_EVENT); ftdm_clear_io_flag(ftdmchan, FTDM_CHANNEL_IO_EVENT);
} }
if (ioctl(ftdmchan->sockfd, codes.GETEVENT, &zt_event_id) == -1) { if (ftdmchan->io_data) {
zt_event_id = (zt_event_t)ftdmchan->io_data;
ftdmchan->io_data = NULL;
} else if (ioctl(ftdmchan->sockfd, codes.GETEVENT, &zt_event_id) == -1) {
ftdm_log_chan(ftdmchan, FTDM_LOG_ERROR, "Failed retrieving event from channel: %s\n", ftdm_log_chan(ftdmchan, FTDM_LOG_ERROR, "Failed retrieving event from channel: %s\n",
strerror(errno)); strerror(errno));
return FTDM_FAIL; return FTDM_FAIL;
@ -1274,29 +1304,47 @@ FIO_SPAN_NEXT_EVENT_FUNCTION(zt_next_event)
for (i = 1; i <= span->chan_count; i++) { for (i = 1; i <= span->chan_count; i++) {
ftdm_channel_t *fchan = span->channels[i]; ftdm_channel_t *fchan = span->channels[i];
if (ftdm_test_io_flag(fchan, FTDM_CHANNEL_IO_EVENT)) {
ftdm_clear_io_flag(fchan, FTDM_CHANNEL_IO_EVENT);
if (ioctl(fchan->sockfd, codes.GETEVENT, &zt_event_id) == -1) {
snprintf(span->last_error, sizeof(span->last_error), "%s", strerror(errno));
return FTDM_FAIL;
}
ftdm_channel_lock(fchan); ftdm_channel_lock(fchan);
if (!ftdm_test_io_flag(fchan, FTDM_CHANNEL_IO_EVENT)) {
ftdm_channel_unlock(fchan);
continue;
}
ftdm_clear_io_flag(fchan, FTDM_CHANNEL_IO_EVENT);
if (fchan->io_data) {
zt_event_id = (zt_event_t)fchan->io_data;
fchan->io_data = NULL;
} else if (ioctl(fchan->sockfd, codes.GETEVENT, &zt_event_id) == -1) {
ftdm_log_chan(fchan, FTDM_LOG_ERROR, "Failed to retrieve DAHDI event from channel: %s\n", strerror(errno));
ftdm_channel_unlock(fchan);
continue;
}
if ((zt_channel_process_event(fchan, &event_id, zt_event_id)) != FTDM_SUCCESS) { if ((zt_channel_process_event(fchan, &event_id, zt_event_id)) != FTDM_SUCCESS) {
ftdm_log_chan(fchan, FTDM_LOG_ERROR, "Failed to process DAHDI event %d from channel\n", zt_event_id); ftdm_log_chan(fchan, FTDM_LOG_ERROR, "Failed to process DAHDI event %d from channel\n", zt_event_id);
ftdm_channel_unlock(fchan); ftdm_channel_unlock(fchan);
return FTDM_FAIL; return FTDM_FAIL;
} }
ftdm_channel_unlock(fchan);
fchan->last_event_time = 0; fchan->last_event_time = 0;
span->event_header.e_type = FTDM_EVENT_OOB; span->event_header.e_type = FTDM_EVENT_OOB;
span->event_header.enum_id = event_id; span->event_header.enum_id = event_id;
span->event_header.channel = fchan; span->event_header.channel = fchan;
*event = &span->event_header; *event = &span->event_header;
ftdm_channel_unlock(fchan);
return FTDM_SUCCESS; return FTDM_SUCCESS;
} }
}
return FTDM_FAIL; return FTDM_FAIL;
} }
@ -1346,11 +1394,13 @@ static FIO_READ_FUNCTION(zt_read)
} }
if (handle_dtmf_event(ftdmchan, zt_event_id)) { if (handle_dtmf_event(ftdmchan, zt_event_id)) {
/* we should enqueue this event somewhere so it can be retrieved by the user, for now, dropping it to see what it is! */ /* Enqueue this event for later */
ftdm_log_chan(ftdmchan, FTDM_LOG_WARNING, "Event %d is not dmtf related. Skipping one media read cycle\n", zt_event_id); ftdm_log_chan(ftdmchan, FTDM_LOG_DEBUG, "Deferring event %d to be able to read data\n", zt_event_id);
ftdm_zt_store_chan_event(ftdmchan, zt_event_id);
} else {
ftdm_log_chan_msg(ftdmchan, FTDM_LOG_DEBUG, "Skipping one IO read cycle due to DTMF event processing\n");
} }
ftdm_log_chan_msg(ftdmchan, FTDM_LOG_WARNING, "Skipping one IO read cycle due to events pending in the driver queue\n"); continue;
break;
} }
/* Read error, keep going unless to many errors force us to abort ...*/ /* Read error, keep going unless to many errors force us to abort ...*/
@ -1403,8 +1453,9 @@ tryagain:
} }
if (handle_dtmf_event(ftdmchan, zt_event_id)) { if (handle_dtmf_event(ftdmchan, zt_event_id)) {
/* we should enqueue this event somewhere so it can be retrieved by the user, for now, dropping it to see what it is! */ /* Enqueue this event for later */
ftdm_log_chan(ftdmchan, FTDM_LOG_ERROR, "Dropping event %d to be able to write data\n", zt_event_id); ftdm_log_chan(ftdmchan, FTDM_LOG_DEBUG, "Deferring event %d to be able to write data\n", zt_event_id);
ftdm_zt_store_chan_event(ftdmchan, zt_event_id);
} }
goto tryagain; goto tryagain;