analog for zt works, still need some more for wanpipe
git-svn-id: http://svn.openzap.org/svn/openzap/trunk@190 a93c3328-9c30-0410-af19-c9cd2b2d52af
This commit is contained in:
parent
69acc51920
commit
d9ad22285a
|
@ -470,7 +470,7 @@ static switch_status_t channel_write_frame(switch_core_session_t *session, switc
|
||||||
}
|
}
|
||||||
|
|
||||||
len = frame->datalen;
|
len = frame->datalen;
|
||||||
if (zap_channel_write(tech_pvt->zchan, frame->data, &len) != ZAP_SUCCESS) {
|
if (zap_channel_write(tech_pvt->zchan, frame->data, frame->buflen, &len) != ZAP_SUCCESS) {
|
||||||
if (++tech_pvt->wr_error > 10) {
|
if (++tech_pvt->wr_error > 10) {
|
||||||
return SWITCH_STATUS_GENERR;
|
return SWITCH_STATUS_GENERR;
|
||||||
}
|
}
|
||||||
|
@ -764,11 +764,21 @@ static switch_core_session_t *zap_channel_get_session(zap_channel_t *channel, in
|
||||||
static ZIO_SIGNAL_CB_FUNCTION(on_fxo_signal)
|
static ZIO_SIGNAL_CB_FUNCTION(on_fxo_signal)
|
||||||
{
|
{
|
||||||
switch_core_session_t *session = NULL;
|
switch_core_session_t *session = NULL;
|
||||||
|
switch_channel_t *channel = NULL;
|
||||||
zap_status_t status;
|
zap_status_t status;
|
||||||
|
|
||||||
zap_log(ZAP_LOG_DEBUG, "got fxo sig [%s]\n", zap_signal_event2str(sigmsg->event_id));
|
zap_log(ZAP_LOG_DEBUG, "got fxo sig [%s]\n", zap_signal_event2str(sigmsg->event_id));
|
||||||
|
|
||||||
switch(sigmsg->event_id) {
|
switch(sigmsg->event_id) {
|
||||||
|
case ZAP_SIGEVENT_UP:
|
||||||
|
{
|
||||||
|
if ((session = zap_channel_get_session(sigmsg->channel, 0))) {
|
||||||
|
channel = switch_core_session_get_channel(session);
|
||||||
|
switch_channel_mark_answered(channel);
|
||||||
|
switch_core_session_rwunlock(session);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
break;
|
||||||
case ZAP_SIGEVENT_START:
|
case ZAP_SIGEVENT_START:
|
||||||
{
|
{
|
||||||
status = zap_channel_from_event(sigmsg, &session);
|
status = zap_channel_from_event(sigmsg, &session);
|
||||||
|
|
|
@ -248,6 +248,7 @@ struct zap_channel {
|
||||||
zap_channel_state_t last_state;
|
zap_channel_state_t last_state;
|
||||||
zap_mutex_t *mutex;
|
zap_mutex_t *mutex;
|
||||||
teletone_dtmf_detect_state_t dtmf_detect;
|
teletone_dtmf_detect_state_t dtmf_detect;
|
||||||
|
uint32_t dtmf_delay;
|
||||||
zap_event_t event_header;
|
zap_event_t event_header;
|
||||||
char last_error[256];
|
char last_error[256];
|
||||||
zio_event_cb_t event_callback;
|
zio_event_cb_t event_callback;
|
||||||
|
@ -260,9 +261,12 @@ struct zap_channel {
|
||||||
teletone_generation_session_t tone_session;
|
teletone_generation_session_t tone_session;
|
||||||
zap_time_t last_event_time;
|
zap_time_t last_event_time;
|
||||||
char tokens[ZAP_MAX_TOKENS+1][ZAP_TOKEN_STRLEN];
|
char tokens[ZAP_MAX_TOKENS+1][ZAP_TOKEN_STRLEN];
|
||||||
|
uint8_t need_tone[ZAP_TONEMAP_INVALID+1];
|
||||||
uint32_t token_count;
|
uint32_t token_count;
|
||||||
char chan_name[128];
|
char chan_name[128];
|
||||||
char chan_number[32];
|
char chan_number[32];
|
||||||
|
zap_tonemap_t detected_tone;
|
||||||
|
zap_tonemap_t last_detected_tone;
|
||||||
struct zap_caller_data caller_data;
|
struct zap_caller_data caller_data;
|
||||||
struct zap_span *span;
|
struct zap_span *span;
|
||||||
struct zap_io_interface *zio;
|
struct zap_io_interface *zio;
|
||||||
|
@ -310,6 +314,8 @@ struct zap_span {
|
||||||
zap_event_t event_header;
|
zap_event_t event_header;
|
||||||
char last_error[256];
|
char last_error[256];
|
||||||
char tone_map[ZAP_TONEMAP_INVALID+1][ZAP_TONEMAP_LEN];
|
char tone_map[ZAP_TONEMAP_INVALID+1][ZAP_TONEMAP_LEN];
|
||||||
|
teletone_tone_map_t tone_detect_map[ZAP_TONEMAP_INVALID+1];
|
||||||
|
teletone_multi_tone_t tone_finder[ZAP_TONEMAP_INVALID+1];
|
||||||
zap_channel_t channels[ZAP_MAX_CHANNELS_SPAN];
|
zap_channel_t channels[ZAP_MAX_CHANNELS_SPAN];
|
||||||
zio_channel_outgoing_call_t outgoing_call;
|
zio_channel_outgoing_call_t outgoing_call;
|
||||||
void *app_data;
|
void *app_data;
|
||||||
|
@ -359,7 +365,7 @@ zap_status_t zap_channel_use(zap_channel_t *zchan);
|
||||||
zap_status_t zap_channel_command(zap_channel_t *zchan, zap_command_t command, void *obj);
|
zap_status_t zap_channel_command(zap_channel_t *zchan, zap_command_t command, void *obj);
|
||||||
zap_status_t zap_channel_wait(zap_channel_t *zchan, zap_wait_flag_t *flags, int32_t to);
|
zap_status_t zap_channel_wait(zap_channel_t *zchan, zap_wait_flag_t *flags, int32_t to);
|
||||||
zap_status_t zap_channel_read(zap_channel_t *zchan, void *data, zap_size_t *datalen);
|
zap_status_t zap_channel_read(zap_channel_t *zchan, void *data, zap_size_t *datalen);
|
||||||
zap_status_t zap_channel_write(zap_channel_t *zchan, void *data, zap_size_t *datalen);
|
zap_status_t zap_channel_write(zap_channel_t *zchan, void *data, zap_size_t datasize, zap_size_t *datalen);
|
||||||
zap_status_t zap_global_init(void);
|
zap_status_t zap_global_init(void);
|
||||||
zap_status_t zap_global_destroy(void);
|
zap_status_t zap_global_destroy(void);
|
||||||
void zap_global_set_logger(zap_logger_t logger);
|
void zap_global_set_logger(zap_logger_t logger);
|
||||||
|
|
|
@ -57,13 +57,14 @@ typedef uint64_t zap_time_t;
|
||||||
|
|
||||||
#define ZAP_TONEMAP_LEN 128
|
#define ZAP_TONEMAP_LEN 128
|
||||||
typedef enum {
|
typedef enum {
|
||||||
|
ZAP_TONEMAP_NONE,
|
||||||
ZAP_TONEMAP_DIAL,
|
ZAP_TONEMAP_DIAL,
|
||||||
ZAP_TONEMAP_RING,
|
ZAP_TONEMAP_RING,
|
||||||
ZAP_TONEMAP_BUSY,
|
ZAP_TONEMAP_BUSY,
|
||||||
ZAP_TONEMAP_ATTN,
|
ZAP_TONEMAP_ATTN,
|
||||||
ZAP_TONEMAP_INVALID
|
ZAP_TONEMAP_INVALID
|
||||||
} zap_tonemap_t;
|
} zap_tonemap_t;
|
||||||
#define TONEMAP_STRINGS "DIAL", "RING", "BUSY", "ATTN", "INVALID"
|
#define TONEMAP_STRINGS "NONE", "DIAL", "RING", "BUSY", "ATTN", "INVALID"
|
||||||
ZAP_STR2ENUM_P(zap_str2zap_tonemap, zap_tonemap2str, zap_tonemap_t)
|
ZAP_STR2ENUM_P(zap_str2zap_tonemap, zap_tonemap2str, zap_tonemap_t)
|
||||||
|
|
||||||
typedef enum {
|
typedef enum {
|
||||||
|
@ -108,10 +109,11 @@ typedef enum {
|
||||||
ZAP_SIGEVENT_PROGRESS,
|
ZAP_SIGEVENT_PROGRESS,
|
||||||
ZAP_SIGEVENT_PROGRESS_MEDIA,
|
ZAP_SIGEVENT_PROGRESS_MEDIA,
|
||||||
ZAP_SIGEVENT_NOTIFY,
|
ZAP_SIGEVENT_NOTIFY,
|
||||||
|
ZAP_SIGEVENT_TONE_DETECTED,
|
||||||
ZAP_SIGEVENT_MISC,
|
ZAP_SIGEVENT_MISC,
|
||||||
ZAP_SIGEVENT_INVALID
|
ZAP_SIGEVENT_INVALID
|
||||||
} zap_signal_event_t;
|
} zap_signal_event_t;
|
||||||
#define SIGNAL_STRINGS "START", "STOP", "TRANSFER", "ANSWER", "UP", "FLASH", "PROGRESS", "PROGRESS_MEDIA", "NOTIFY", "MISC", "INVALID"
|
#define SIGNAL_STRINGS "START", "STOP", "TRANSFER", "ANSWER", "UP", "FLASH", "PROGRESS", "PROGRESS_MEDIA", "NOTIFY", "TONE_DETECTED", "MISC", "INVALID"
|
||||||
ZAP_STR2ENUM_P(zap_str2zap_signal_event, zap_signal_event2str, zap_signal_event_t)
|
ZAP_STR2ENUM_P(zap_str2zap_signal_event, zap_signal_event2str, zap_signal_event_t)
|
||||||
|
|
||||||
typedef enum {
|
typedef enum {
|
||||||
|
@ -173,6 +175,8 @@ typedef enum {
|
||||||
ZAP_COMMAND_GENERATE_RING_OFF,
|
ZAP_COMMAND_GENERATE_RING_OFF,
|
||||||
ZAP_COMMAND_OFFHOOK,
|
ZAP_COMMAND_OFFHOOK,
|
||||||
ZAP_COMMAND_ONHOOK,
|
ZAP_COMMAND_ONHOOK,
|
||||||
|
ZAP_COMMAND_ENABLE_PROGRESS_DETECT,
|
||||||
|
ZAP_COMMAND_DISABLE_PROGRESS_DETECT,
|
||||||
ZAP_COMMAND_COUNT
|
ZAP_COMMAND_COUNT
|
||||||
} zap_command_t;
|
} zap_command_t;
|
||||||
|
|
||||||
|
@ -208,9 +212,10 @@ typedef enum {
|
||||||
ZAP_CHANNEL_STATE_ATTN,
|
ZAP_CHANNEL_STATE_ATTN,
|
||||||
ZAP_CHANNEL_STATE_IDLE,
|
ZAP_CHANNEL_STATE_IDLE,
|
||||||
ZAP_CHANNEL_STATE_GENRING,
|
ZAP_CHANNEL_STATE_GENRING,
|
||||||
|
ZAP_CHANNEL_STATE_DIALING,
|
||||||
ZAP_CHANNEL_STATE_INVALID
|
ZAP_CHANNEL_STATE_INVALID
|
||||||
} zap_channel_state_t;
|
} zap_channel_state_t;
|
||||||
#define CHANNEL_STATE_STRINGS "DOWN", "UP", "HANGUP", "HOLD", "DIALTONE", "COLLECT", "RING", "BUSY", "ATTN", "IDLE", "GENRING", "INVALID"
|
#define CHANNEL_STATE_STRINGS "DOWN", "UP", "HANGUP", "HOLD", "DIALTONE", "COLLECT", "RING", "BUSY", "ATTN", "IDLE", "GENRING", "DIALING", "INVALID"
|
||||||
ZAP_STR2ENUM_P(zap_str2zap_channel_state, zap_channel_state2str, zap_channel_state_t)
|
ZAP_STR2ENUM_P(zap_str2zap_channel_state, zap_channel_state2str, zap_channel_state_t)
|
||||||
|
|
||||||
typedef enum {
|
typedef enum {
|
||||||
|
@ -229,7 +234,8 @@ typedef enum {
|
||||||
ZAP_CHANNEL_HOLD = (1 << 12),
|
ZAP_CHANNEL_HOLD = (1 << 12),
|
||||||
ZAP_CHANNEL_INUSE = (1 << 13),
|
ZAP_CHANNEL_INUSE = (1 << 13),
|
||||||
ZAP_CHANNEL_OFFHOOK = (1 << 14),
|
ZAP_CHANNEL_OFFHOOK = (1 << 14),
|
||||||
ZAP_CHANNEL_RINGING = (1 << 15)
|
ZAP_CHANNEL_RINGING = (1 << 15),
|
||||||
|
ZAP_CHANNEL_PROGRESS_DETECT = (1 << 16)
|
||||||
} zap_channel_flag_t;
|
} zap_channel_flag_t;
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -207,7 +207,7 @@ typedef enum {
|
||||||
#define ZT_SETGAINS _IOWR (ZT_CODE, 17, struct zt_gains) /* Set Channel audio gains */
|
#define ZT_SETGAINS _IOWR (ZT_CODE, 17, struct zt_gains) /* Set Channel audio gains */
|
||||||
|
|
||||||
#define ZT_AUDIOMODE _IOW (ZT_CODE, 32, int) /* Set a clear channel into audio mode */
|
#define ZT_AUDIOMODE _IOW (ZT_CODE, 32, int) /* Set a clear channel into audio mode */
|
||||||
|
#define ZT_ECHOCANCEL _IOW (ZT_CODE, 33, int) /* Control Echo Canceller */
|
||||||
#define ZT_HDLCRAWMODE _IOW (ZT_CODE, 36, int) /* Set a clear channel into HDLC w/out FCS checking/calculation mode */
|
#define ZT_HDLCRAWMODE _IOW (ZT_CODE, 36, int) /* Set a clear channel into HDLC w/out FCS checking/calculation mode */
|
||||||
#define ZT_HDLCFCSMODE _IOW (ZT_CODE, 37, int) /* Set a clear channel into HDLC w/ FCS mode */
|
#define ZT_HDLCFCSMODE _IOW (ZT_CODE, 37, int) /* Set a clear channel into HDLC w/ FCS mode */
|
||||||
|
|
||||||
|
@ -221,7 +221,7 @@ typedef enum {
|
||||||
#define ZT_SETLINEAR _IOW (ZT_CODE, 40, int)
|
#define ZT_SETLINEAR _IOW (ZT_CODE, 40, int)
|
||||||
|
|
||||||
#define ZT_GETCONFMUTE _IOR (ZT_CODE, 49, int) /* Get Conference to mute mode */
|
#define ZT_GETCONFMUTE _IOR (ZT_CODE, 49, int) /* Get Conference to mute mode */
|
||||||
|
#define ZT_ECHOTRAIN _IOW (ZT_CODE, 50, int) /* Control Echo Trainer */
|
||||||
|
|
||||||
|
|
||||||
/* Openzap ZT hardware interface functions */
|
/* Openzap ZT hardware interface functions */
|
||||||
|
|
|
@ -25,7 +25,7 @@ static void *test_call(zap_thread_t *me, void *obj)
|
||||||
if (flags & ZAP_READ) {
|
if (flags & ZAP_READ) {
|
||||||
if (zap_channel_read(chan, frame, &len) == ZAP_SUCCESS) {
|
if (zap_channel_read(chan, frame, &len) == ZAP_SUCCESS) {
|
||||||
//zap_log(ZAP_LOG_DEBUG, "WRITE %d\n", len);
|
//zap_log(ZAP_LOG_DEBUG, "WRITE %d\n", len);
|
||||||
zap_channel_write(chan, frame, &len);
|
zap_channel_write(chan, frame, sizeof(frame), &len);
|
||||||
} else {
|
} else {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
|
@ -8,11 +8,6 @@ int main(int argc, char *argv[])
|
||||||
zap_codec_t codec = ZAP_CODEC_SLIN;
|
zap_codec_t codec = ZAP_CODEC_SLIN;
|
||||||
unsigned runs = 1;
|
unsigned runs = 1;
|
||||||
|
|
||||||
if (!argv[1]) {
|
|
||||||
printf("usage %s <modname>\n", argv[0]);
|
|
||||||
exit(-1);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
if (zap_global_init() != ZAP_SUCCESS) {
|
if (zap_global_init() != ZAP_SUCCESS) {
|
||||||
fprintf(stderr, "Error loading OpenZAP\n");
|
fprintf(stderr, "Error loading OpenZAP\n");
|
||||||
|
@ -23,7 +18,7 @@ int main(int argc, char *argv[])
|
||||||
|
|
||||||
top:
|
top:
|
||||||
//if (zap_channel_open_any("wanpipe", 0, ZAP_TOP_DOWN, &chan) == ZAP_SUCCESS) {
|
//if (zap_channel_open_any("wanpipe", 0, ZAP_TOP_DOWN, &chan) == ZAP_SUCCESS) {
|
||||||
if (zap_channel_open(argv[1], 1, 1, &chan) == ZAP_SUCCESS) {
|
if (zap_channel_open(1, 1, &chan) == ZAP_SUCCESS) {
|
||||||
int x = 0;
|
int x = 0;
|
||||||
printf("opened channel %d:%d\n", chan->span_id, chan->chan_id);
|
printf("opened channel %d:%d\n", chan->span_id, chan->chan_id);
|
||||||
|
|
||||||
|
|
|
@ -38,6 +38,14 @@ static void *zap_analog_channel_run(zap_thread_t *me, void *obj);
|
||||||
|
|
||||||
static ZIO_CHANNEL_OUTGOING_CALL_FUNCTION(analog_fxo_outgoing_call)
|
static ZIO_CHANNEL_OUTGOING_CALL_FUNCTION(analog_fxo_outgoing_call)
|
||||||
{
|
{
|
||||||
|
if (!zap_test_flag(zchan, ZAP_CHANNEL_OFFHOOK) && !zap_test_flag(zchan, ZAP_CHANNEL_INTHREAD)) {
|
||||||
|
zap_channel_command(zchan, ZAP_COMMAND_OFFHOOK, NULL);
|
||||||
|
zap_channel_command(zchan, ZAP_COMMAND_ENABLE_PROGRESS_DETECT, NULL);
|
||||||
|
zchan->need_tone[ZAP_TONEMAP_DIAL] = 1;
|
||||||
|
zap_set_state_locked(zchan, ZAP_CHANNEL_STATE_DIALING);
|
||||||
|
zap_thread_create_detached(zap_analog_channel_run, zchan);
|
||||||
|
return ZAP_SUCCESS;
|
||||||
|
}
|
||||||
return ZAP_FAIL;
|
return ZAP_FAIL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -114,6 +122,7 @@ static void *zap_analog_channel_run(zap_thread_t *me, void *obj)
|
||||||
zap_sigmsg_t sig;
|
zap_sigmsg_t sig;
|
||||||
zap_status_t status;
|
zap_status_t status;
|
||||||
|
|
||||||
|
|
||||||
zap_log(ZAP_LOG_DEBUG, "ANALOG CHANNEL thread starting.\n");
|
zap_log(ZAP_LOG_DEBUG, "ANALOG CHANNEL thread starting.\n");
|
||||||
|
|
||||||
ts.buffer = NULL;
|
ts.buffer = NULL;
|
||||||
|
@ -156,6 +165,17 @@ static void *zap_analog_channel_run(zap_thread_t *me, void *obj)
|
||||||
|
|
||||||
if (!zap_test_flag(chan, ZAP_CHANNEL_STATE_CHANGE)) {
|
if (!zap_test_flag(chan, ZAP_CHANNEL_STATE_CHANGE)) {
|
||||||
switch(chan->state) {
|
switch(chan->state) {
|
||||||
|
case ZAP_CHANNEL_STATE_DIALING:
|
||||||
|
{
|
||||||
|
if (state_counter > 5000) {
|
||||||
|
if (chan->need_tone[ZAP_TONEMAP_DIAL]) {
|
||||||
|
zap_set_state_locked(chan, ZAP_CHANNEL_STATE_BUSY);
|
||||||
|
} else {
|
||||||
|
zap_set_state_locked(chan, ZAP_CHANNEL_STATE_UP);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
break;
|
||||||
case ZAP_CHANNEL_STATE_GENRING:
|
case ZAP_CHANNEL_STATE_GENRING:
|
||||||
{
|
{
|
||||||
if (state_counter > 60000) {
|
if (state_counter > 60000) {
|
||||||
|
@ -183,7 +203,11 @@ static void *zap_analog_channel_run(zap_thread_t *me, void *obj)
|
||||||
case ZAP_CHANNEL_STATE_HANGUP:
|
case ZAP_CHANNEL_STATE_HANGUP:
|
||||||
{
|
{
|
||||||
if (state_counter > 2000) {
|
if (state_counter > 2000) {
|
||||||
zap_set_state_locked(chan, ZAP_CHANNEL_STATE_BUSY);
|
if (chan->type == ZAP_CHAN_TYPE_FXO) {
|
||||||
|
zap_set_state_locked(chan, ZAP_CHANNEL_STATE_DOWN);
|
||||||
|
} else {
|
||||||
|
zap_set_state_locked(chan, ZAP_CHANNEL_STATE_BUSY);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
@ -231,6 +255,11 @@ static void *zap_analog_channel_run(zap_thread_t *me, void *obj)
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
case ZAP_CHANNEL_STATE_DIALING:
|
||||||
|
{
|
||||||
|
zap_channel_use(chan);
|
||||||
|
}
|
||||||
|
break;
|
||||||
case ZAP_CHANNEL_STATE_IDLE:
|
case ZAP_CHANNEL_STATE_IDLE:
|
||||||
{
|
{
|
||||||
zap_channel_use(chan);
|
zap_channel_use(chan);
|
||||||
|
@ -323,6 +352,55 @@ static void *zap_analog_channel_run(zap_thread_t *me, void *obj)
|
||||||
goto done;
|
goto done;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (chan->detected_tone) {
|
||||||
|
zap_sigmsg_t sig;
|
||||||
|
memset(&sig, 0, sizeof(sig));
|
||||||
|
sig.chan_id = chan->chan_id;
|
||||||
|
sig.span_id = chan->span_id;
|
||||||
|
sig.channel = chan;
|
||||||
|
sig.event_id = ZAP_SIGEVENT_TONE_DETECTED;
|
||||||
|
zap_log(ZAP_LOG_DEBUG, "Detected tone %s\n", zap_tonemap2str(chan->detected_tone));
|
||||||
|
data->sig_cb(&sig);
|
||||||
|
|
||||||
|
if (chan->type == ZAP_CHAN_TYPE_FXO) {
|
||||||
|
|
||||||
|
if (chan->need_tone[chan->detected_tone]) {
|
||||||
|
switch (chan->detected_tone) {
|
||||||
|
case ZAP_TONEMAP_DIAL:
|
||||||
|
{
|
||||||
|
zap_channel_command(chan, ZAP_COMMAND_SEND_DTMF, chan->caller_data.ani);
|
||||||
|
state_counter = 0;
|
||||||
|
chan->need_tone[ZAP_TONEMAP_RING] = 1;
|
||||||
|
chan->need_tone[ZAP_TONEMAP_BUSY] = 1;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case ZAP_TONEMAP_RING:
|
||||||
|
{
|
||||||
|
zap_set_state_locked(chan, ZAP_CHANNEL_STATE_UP);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case ZAP_TONEMAP_BUSY:
|
||||||
|
{
|
||||||
|
zap_set_state_locked(chan, ZAP_CHANNEL_STATE_BUSY);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
chan->need_tone[chan->detected_tone] = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
chan->detected_tone = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (chan->dtmf_buffer && zap_buffer_inuse(chan->dtmf_buffer)) {
|
||||||
|
rlen = len;
|
||||||
|
memset(frame, 0, len);
|
||||||
|
zap_channel_write(chan, frame, sizeof(frame), &rlen);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
if (!indicate) {
|
if (!indicate) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
@ -337,6 +415,7 @@ static void *zap_analog_channel_run(zap_thread_t *me, void *obj)
|
||||||
|
|
||||||
rlen = zap_buffer_read_loop(dt_buffer, frame, len);
|
rlen = zap_buffer_read_loop(dt_buffer, frame, len);
|
||||||
|
|
||||||
|
|
||||||
if (chan->effective_codec != ZAP_CODEC_SLIN) {
|
if (chan->effective_codec != ZAP_CODEC_SLIN) {
|
||||||
zio_codec_t codec_func = NULL;
|
zio_codec_t codec_func = NULL;
|
||||||
|
|
||||||
|
@ -354,7 +433,7 @@ static void *zap_analog_channel_run(zap_thread_t *me, void *obj)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
zap_channel_write(chan, frame, &rlen);
|
zap_channel_write(chan, frame, sizeof(frame), &rlen);
|
||||||
}
|
}
|
||||||
|
|
||||||
done:
|
done:
|
||||||
|
|
|
@ -271,9 +271,30 @@ zap_status_t zap_span_load_tones(zap_span_t *span, char *mapname)
|
||||||
while (zap_config_next_pair(&cfg, &var, &val)) {
|
while (zap_config_next_pair(&cfg, &var, &val)) {
|
||||||
if (!strcasecmp(cfg.category, mapname) && var && val) {
|
if (!strcasecmp(cfg.category, mapname) && var && val) {
|
||||||
int index = zap_str2zap_tonemap(var);
|
int index = zap_str2zap_tonemap(var);
|
||||||
if (index > ZAP_TONEMAP_INVALID) {
|
char valbuf[512];
|
||||||
|
|
||||||
|
if (index > ZAP_TONEMAP_INVALID || index == ZAP_TONEMAP_NONE) {
|
||||||
zap_log(ZAP_LOG_WARNING, "Unknown tone name %s\n", var);
|
zap_log(ZAP_LOG_WARNING, "Unknown tone name %s\n", var);
|
||||||
} else {
|
} else {
|
||||||
|
char *p, *next;
|
||||||
|
int i = 0;
|
||||||
|
zap_set_string(valbuf, val);
|
||||||
|
val = valbuf;
|
||||||
|
if ((p = strchr(val, ':'))) {
|
||||||
|
*p++ = '\0';
|
||||||
|
do {
|
||||||
|
teletone_process_t this;
|
||||||
|
next = strchr(p, ',');
|
||||||
|
this = atof(p);
|
||||||
|
span->tone_detect_map[index].freqs[i++] = this;
|
||||||
|
if (next) {
|
||||||
|
p = next + 1;
|
||||||
|
}
|
||||||
|
} while (next);
|
||||||
|
|
||||||
|
zap_copy_string(span->tone_map[index], val, sizeof(span->tone_map[index]));
|
||||||
|
|
||||||
|
}
|
||||||
zap_log(ZAP_LOG_DEBUG, "added tone [%s] = [%s]\n", var, val);
|
zap_log(ZAP_LOG_DEBUG, "added tone [%s] = [%s]\n", var, val);
|
||||||
zap_copy_string(span->tone_map[index], val, sizeof(span->tone_map[index]));
|
zap_copy_string(span->tone_map[index], val, sizeof(span->tone_map[index]));
|
||||||
x++;
|
x++;
|
||||||
|
@ -718,7 +739,7 @@ static zap_status_t zchan_activate_dtmf_buffer(zap_channel_t *zchan)
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!zchan->dtmf_on) {
|
if (!zchan->dtmf_on) {
|
||||||
zchan->dtmf_on = 150;
|
zchan->dtmf_on = 250;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!zchan->dtmf_off) {
|
if (!zchan->dtmf_off) {
|
||||||
|
@ -726,11 +747,16 @@ static zap_status_t zchan_activate_dtmf_buffer(zap_channel_t *zchan)
|
||||||
}
|
}
|
||||||
|
|
||||||
memset(&zchan->tone_session, 0, sizeof(zchan->tone_session));
|
memset(&zchan->tone_session, 0, sizeof(zchan->tone_session));
|
||||||
teletone_init_session(&zchan->tone_session, 1024, NULL, NULL);
|
teletone_init_session(&zchan->tone_session, 0, NULL, NULL);
|
||||||
|
|
||||||
zchan->tone_session.rate = 8000;
|
zchan->tone_session.rate = 8000;
|
||||||
zchan->tone_session.duration = zchan->dtmf_on * (zchan->tone_session.rate / 1000);
|
zchan->tone_session.duration = zchan->dtmf_on * (zchan->tone_session.rate / 1000);
|
||||||
zchan->tone_session.wait = zchan->dtmf_off * (zchan->tone_session.rate / 1000);
|
zchan->tone_session.wait = zchan->dtmf_off * (zchan->tone_session.rate / 1000);
|
||||||
|
/*
|
||||||
|
zchan->tone_session.debug = 1;
|
||||||
|
zchan->tone_session.debug_stream = stdout;
|
||||||
|
*/
|
||||||
|
|
||||||
return ZAP_SUCCESS;
|
return ZAP_SUCCESS;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -809,6 +835,22 @@ zap_status_t zap_channel_command(zap_channel_t *zchan, zap_command_t command, vo
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
case ZAP_COMMAND_ENABLE_PROGRESS_DETECT:
|
||||||
|
{
|
||||||
|
/* if they don't have thier own, use ours */
|
||||||
|
teletone_multi_tone_init(&zchan->span->tone_finder[ZAP_TONEMAP_DIAL], &zchan->span->tone_detect_map[ZAP_TONEMAP_DIAL]);
|
||||||
|
teletone_multi_tone_init(&zchan->span->tone_finder[ZAP_TONEMAP_RING], &zchan->span->tone_detect_map[ZAP_TONEMAP_RING]);
|
||||||
|
teletone_multi_tone_init(&zchan->span->tone_finder[ZAP_TONEMAP_BUSY], &zchan->span->tone_detect_map[ZAP_TONEMAP_BUSY]);
|
||||||
|
zap_set_flag(zchan, ZAP_CHANNEL_PROGRESS_DETECT);
|
||||||
|
GOTO_STATUS(done, ZAP_SUCCESS);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case ZAP_COMMAND_DISABLE_PROGRESS_DETECT:
|
||||||
|
{
|
||||||
|
zap_clear_flag_locked(zchan, ZAP_CHANNEL_PROGRESS_DETECT);
|
||||||
|
GOTO_STATUS(done, ZAP_SUCCESS);
|
||||||
|
}
|
||||||
|
break;
|
||||||
case ZAP_COMMAND_ENABLE_TONE_DETECT:
|
case ZAP_COMMAND_ENABLE_TONE_DETECT:
|
||||||
{
|
{
|
||||||
/* if they don't have thier own, use ours */
|
/* if they don't have thier own, use ours */
|
||||||
|
@ -816,8 +858,8 @@ zap_status_t zap_channel_command(zap_channel_t *zchan, zap_command_t command, vo
|
||||||
zap_tone_type_t tt = ZAP_COMMAND_OBJ_INT;
|
zap_tone_type_t tt = ZAP_COMMAND_OBJ_INT;
|
||||||
if (tt == ZAP_TONE_DTMF) {
|
if (tt == ZAP_TONE_DTMF) {
|
||||||
teletone_dtmf_detect_init (&zchan->dtmf_detect, 8000);
|
teletone_dtmf_detect_init (&zchan->dtmf_detect, 8000);
|
||||||
zap_set_flag(zchan, ZAP_CHANNEL_DTMF_DETECT);
|
zap_set_flag_locked(zchan, ZAP_CHANNEL_DTMF_DETECT);
|
||||||
zap_set_flag(zchan, ZAP_CHANNEL_SUPRESS_DTMF);
|
zap_set_flag_locked(zchan, ZAP_CHANNEL_SUPRESS_DTMF);
|
||||||
GOTO_STATUS(done, ZAP_SUCCESS);
|
GOTO_STATUS(done, ZAP_SUCCESS);
|
||||||
} else {
|
} else {
|
||||||
snprintf(zchan->last_error, sizeof(zchan->last_error), "invalid command");
|
snprintf(zchan->last_error, sizeof(zchan->last_error), "invalid command");
|
||||||
|
@ -890,6 +932,8 @@ zap_status_t zap_channel_command(zap_channel_t *zchan, zap_command_t command, vo
|
||||||
if (!zap_channel_test_feature(zchan, ZAP_CHANNEL_FEATURE_DTMF)) {
|
if (!zap_channel_test_feature(zchan, ZAP_CHANNEL_FEATURE_DTMF)) {
|
||||||
char *cur;
|
char *cur;
|
||||||
char *digits = ZAP_COMMAND_OBJ_CHAR_P;
|
char *digits = ZAP_COMMAND_OBJ_CHAR_P;
|
||||||
|
int x = 0;
|
||||||
|
|
||||||
if (!zchan->dtmf_buffer) {
|
if (!zchan->dtmf_buffer) {
|
||||||
if ((status = zchan_activate_dtmf_buffer(zchan)) != ZAP_SUCCESS) {
|
if ((status = zchan_activate_dtmf_buffer(zchan)) != ZAP_SUCCESS) {
|
||||||
GOTO_STATUS(done, status);
|
GOTO_STATUS(done, status);
|
||||||
|
@ -901,10 +945,11 @@ zap_status_t zap_channel_command(zap_channel_t *zchan, zap_command_t command, vo
|
||||||
int wrote = 0;
|
int wrote = 0;
|
||||||
if ((wrote = teletone_mux_tones(&zchan->tone_session, &zchan->tone_session.TONES[(int)*cur]))) {
|
if ((wrote = teletone_mux_tones(&zchan->tone_session, &zchan->tone_session.TONES[(int)*cur]))) {
|
||||||
zap_buffer_write(zchan->dtmf_buffer, zchan->tone_session.buffer, wrote * 2);
|
zap_buffer_write(zchan->dtmf_buffer, zchan->tone_session.buffer, wrote * 2);
|
||||||
|
x++;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
zchan->skip_read_frames = 200;
|
zchan->skip_read_frames = 200 * x;
|
||||||
GOTO_STATUS(done, ZAP_SUCCESS);
|
GOTO_STATUS(done, ZAP_SUCCESS);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1169,7 +1214,7 @@ zap_status_t zap_channel_read(zap_channel_t *zchan, void *data, zap_size_t *data
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (zap_test_flag(zchan, ZAP_CHANNEL_DTMF_DETECT)) {
|
if (zap_test_flag(zchan, ZAP_CHANNEL_DTMF_DETECT) || zap_test_flag(zchan, ZAP_CHANNEL_PROGRESS_DETECT)) {
|
||||||
uint8_t sln_buf[1024] = {0};
|
uint8_t sln_buf[1024] = {0};
|
||||||
int16_t *sln;
|
int16_t *sln;
|
||||||
zap_size_t slen = 0;
|
zap_size_t slen = 0;
|
||||||
|
@ -1203,47 +1248,62 @@ zap_status_t zap_channel_read(zap_channel_t *zchan, void *data, zap_size_t *data
|
||||||
slen = len;
|
slen = len;
|
||||||
}
|
}
|
||||||
|
|
||||||
teletone_dtmf_detect(&zchan->dtmf_detect, sln, (int)slen);
|
if (zap_test_flag(zchan, ZAP_CHANNEL_PROGRESS_DETECT)) {
|
||||||
teletone_dtmf_get(&zchan->dtmf_detect, digit_str, sizeof(digit_str));
|
uint32_t i;
|
||||||
|
|
||||||
if(*digit_str) {
|
for (i = 0; i < ZAP_TONEMAP_INVALID+1; i++) {
|
||||||
zio_event_cb_t event_callback = NULL;
|
if (zchan->span->tone_finder[i].tone_count) {
|
||||||
|
if (teletone_multi_tone_detect(&zchan->span->tone_finder[i], sln, (int)slen) && i != zchan->last_detected_tone) {
|
||||||
zap_channel_queue_dtmf(zchan, digit_str);
|
zchan->detected_tone = i;
|
||||||
|
zchan->last_detected_tone = i;
|
||||||
if (zchan->span->event_callback) {
|
}
|
||||||
event_callback = zchan->span->event_callback;
|
}
|
||||||
} else if (zchan->event_callback) {
|
|
||||||
event_callback = zchan->event_callback;
|
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (event_callback) {
|
if (zap_test_flag(zchan, ZAP_CHANNEL_DTMF_DETECT)) {
|
||||||
zchan->event_header.channel = zchan;
|
|
||||||
zchan->event_header.e_type = ZAP_EVENT_DTMF;
|
teletone_dtmf_detect(&zchan->dtmf_detect, sln, (int)slen);
|
||||||
zchan->event_header.data = digit_str;
|
teletone_dtmf_get(&zchan->dtmf_detect, digit_str, sizeof(digit_str));
|
||||||
event_callback(zchan, &zchan->event_header);
|
|
||||||
zchan->event_header.e_type = ZAP_EVENT_NONE;
|
if(*digit_str) {
|
||||||
zchan->event_header.data = NULL;
|
zio_event_cb_t event_callback = NULL;
|
||||||
}
|
|
||||||
if (zap_test_flag(zchan, ZAP_CHANNEL_SUPRESS_DTMF)) {
|
zap_channel_queue_dtmf(zchan, digit_str);
|
||||||
zchan->skip_read_frames = 20;
|
|
||||||
}
|
if (zchan->span->event_callback) {
|
||||||
if (zchan->skip_read_frames > 0) {
|
event_callback = zchan->span->event_callback;
|
||||||
memset(data, 0, *datalen);
|
} else if (zchan->event_callback) {
|
||||||
zchan->skip_read_frames--;
|
event_callback = zchan->event_callback;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (event_callback) {
|
||||||
|
zchan->event_header.channel = zchan;
|
||||||
|
zchan->event_header.e_type = ZAP_EVENT_DTMF;
|
||||||
|
zchan->event_header.data = digit_str;
|
||||||
|
event_callback(zchan, &zchan->event_header);
|
||||||
|
zchan->event_header.e_type = ZAP_EVENT_NONE;
|
||||||
|
zchan->event_header.data = NULL;
|
||||||
|
}
|
||||||
|
if (zap_test_flag(zchan, ZAP_CHANNEL_SUPRESS_DTMF)) {
|
||||||
|
zchan->skip_read_frames = 20;
|
||||||
|
}
|
||||||
|
if (zchan->skip_read_frames > 0) {
|
||||||
|
memset(data, 0, *datalen);
|
||||||
|
zchan->skip_read_frames--;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return status;
|
return status;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
zap_status_t zap_channel_write(zap_channel_t *zchan, void *data, zap_size_t *datalen)
|
zap_status_t zap_channel_write(zap_channel_t *zchan, void *data, zap_size_t datasize, zap_size_t *datalen)
|
||||||
{
|
{
|
||||||
zap_status_t status = ZAP_FAIL;
|
zap_status_t status = ZAP_FAIL;
|
||||||
zio_codec_t codec_func = NULL;
|
zio_codec_t codec_func = NULL;
|
||||||
zap_size_t dtmf_blen, max = *datalen;
|
zap_size_t dtmf_blen, max = datasize;
|
||||||
|
|
||||||
assert(zchan != NULL);
|
assert(zchan != NULL);
|
||||||
assert(zchan->zio != NULL);
|
assert(zchan->zio != NULL);
|
||||||
|
@ -1277,7 +1337,7 @@ zap_status_t zap_channel_write(zap_channel_t *zchan, void *data, zap_size_t *dat
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (zchan->dtmf_buffer && (dtmf_blen = zap_buffer_inuse(zchan->dtmf_buffer))) {
|
if (zchan->dtmf_buffer && (dtmf_blen = zap_buffer_inuse(zchan->dtmf_buffer)) && (!zchan->dtmf_delay || --zchan->dtmf_delay == 0)) {
|
||||||
zap_size_t dlen = *datalen;
|
zap_size_t dlen = *datalen;
|
||||||
uint8_t auxbuf[1024];
|
uint8_t auxbuf[1024];
|
||||||
zap_size_t len, br;
|
zap_size_t len, br;
|
||||||
|
@ -1305,6 +1365,7 @@ zap_status_t zap_channel_write(zap_channel_t *zchan, void *data, zap_size_t *dat
|
||||||
zio_slin2alaw(data, max, datalen);
|
zio_slin2alaw(data, max, datalen);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
status = zchan->zio->write(zchan, data, datalen);
|
status = zchan->zio->write(zchan, data, datalen);
|
||||||
|
|
|
@ -95,7 +95,7 @@ static int zap_isdn_921_21(void *pvt, L2UCHAR *msg, L2INT mlen)
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
assert(span != NULL);
|
assert(span != NULL);
|
||||||
return zap_channel_write(span->isdn_data->dchan, msg, &len) == ZAP_SUCCESS ? 0 : -1;
|
return zap_channel_write(span->isdn_data->dchan, msg, len, &len) == ZAP_SUCCESS ? 0 : -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void *zap_isdn_run(zap_thread_t *me, void *obj)
|
static void *zap_isdn_run(zap_thread_t *me, void *obj)
|
||||||
|
|
|
@ -59,6 +59,14 @@ static unsigned zt_open_range(zap_span_t *span, unsigned start, unsigned end, za
|
||||||
sockfd = open(path, O_RDWR);
|
sockfd = open(path, O_RDWR);
|
||||||
|
|
||||||
if (sockfd != ZT_INVALID_SOCKET && zap_span_add_channel(span, sockfd, type, &chan) == ZAP_SUCCESS) {
|
if (sockfd != ZT_INVALID_SOCKET && zap_span_add_channel(span, sockfd, type, &chan) == ZAP_SUCCESS) {
|
||||||
|
len = 64;
|
||||||
|
if (ioctl(chan->sockfd, ZT_ECHOCANCEL, &len)) {
|
||||||
|
zap_log(ZAP_LOG_INFO, "failure configuring device %s as OpenZAP device %d:%d fd:%d err:%s\n",
|
||||||
|
path, chan->span_id, chan->chan_id, sockfd, strerror(errno));
|
||||||
|
close(sockfd);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
len = zt_globals.codec_ms * 8;
|
len = zt_globals.codec_ms * 8;
|
||||||
if (ioctl(chan->sockfd, ZT_SET_BLOCKSIZE, &len)) {
|
if (ioctl(chan->sockfd, ZT_SET_BLOCKSIZE, &len)) {
|
||||||
zap_log(ZAP_LOG_INFO, "failure configuring device %s as OpenZAP device %d:%d fd:%d err:%s\n",
|
zap_log(ZAP_LOG_INFO, "failure configuring device %s as OpenZAP device %d:%d fd:%d err:%s\n",
|
||||||
|
|
Loading…
Reference in New Issue