RTS API Documentation  1.10.11
switch_channel.c
Go to the documentation of this file.
1 /*
2  * FreeSWITCH Modular Media Switching Software Library / Soft-Switch Application
3  * Copyright (C) 2005-2020, Anthony Minessale II <anthm@freeswitch.org>
4  *
5  * Version: MPL 1.1
6  *
7  * The contents of this file are subject to the Mozilla Public License Version
8  * 1.1 (the "License"); you may not use this file except in compliance with
9  * the License. You may obtain a copy of the License at
10  * http://www.mozilla.org/MPL/
11  *
12  * Software distributed under the License is distributed on an "AS IS" basis,
13  * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
14  * for the specific language governing rights and limitations under the
15  * License.
16  *
17  * The Original Code is FreeSWITCH Modular Media Switching Software Library / Soft-Switch Application
18  *
19  * The Initial Developer of the Original Code is
20  * Anthony Minessale II <anthm@freeswitch.org>
21  * Portions created by the Initial Developer are Copyright (C)
22  * the Initial Developer. All Rights Reserved.
23  *
24  * Contributor(s):
25  *
26  * Anthony Minessale II <anthm@freeswitch.org>
27  * Michael Jerris <mike@jerris.com>
28  *
29  *
30  * switch_channel.c -- Media Channel Interface
31  *
32  */
33 
34 #include <switch.h>
35 #include <switch_channel.h>
36 #include <pcre.h>
37 
39  const char *name;
41 };
42 
45  void *user_data;
48 
49 static struct {
54 } globals;
55 
56 static struct switch_cause_table CAUSE_CHART[] = {
57  {"NONE", SWITCH_CAUSE_NONE},
58  {"UNALLOCATED_NUMBER", SWITCH_CAUSE_UNALLOCATED_NUMBER},
59  {"NO_ROUTE_TRANSIT_NET", SWITCH_CAUSE_NO_ROUTE_TRANSIT_NET},
60  {"NO_ROUTE_DESTINATION", SWITCH_CAUSE_NO_ROUTE_DESTINATION},
61  {"CHANNEL_UNACCEPTABLE", SWITCH_CAUSE_CHANNEL_UNACCEPTABLE},
62  {"CALL_AWARDED_DELIVERED", SWITCH_CAUSE_CALL_AWARDED_DELIVERED},
63  {"NORMAL_CLEARING", SWITCH_CAUSE_NORMAL_CLEARING},
64  {"USER_BUSY", SWITCH_CAUSE_USER_BUSY},
65  {"NO_USER_RESPONSE", SWITCH_CAUSE_NO_USER_RESPONSE},
66  {"NO_ANSWER", SWITCH_CAUSE_NO_ANSWER},
67  {"SUBSCRIBER_ABSENT", SWITCH_CAUSE_SUBSCRIBER_ABSENT},
68  {"CALL_REJECTED", SWITCH_CAUSE_CALL_REJECTED},
69  {"NUMBER_CHANGED", SWITCH_CAUSE_NUMBER_CHANGED},
70  {"REDIRECTION_TO_NEW_DESTINATION", SWITCH_CAUSE_REDIRECTION_TO_NEW_DESTINATION},
71  {"EXCHANGE_ROUTING_ERROR", SWITCH_CAUSE_EXCHANGE_ROUTING_ERROR},
72  {"DESTINATION_OUT_OF_ORDER", SWITCH_CAUSE_DESTINATION_OUT_OF_ORDER},
73  {"INVALID_NUMBER_FORMAT", SWITCH_CAUSE_INVALID_NUMBER_FORMAT},
74  {"FACILITY_REJECTED", SWITCH_CAUSE_FACILITY_REJECTED},
75  {"RESPONSE_TO_STATUS_ENQUIRY", SWITCH_CAUSE_RESPONSE_TO_STATUS_ENQUIRY},
76  {"NORMAL_UNSPECIFIED", SWITCH_CAUSE_NORMAL_UNSPECIFIED},
77  {"NORMAL_CIRCUIT_CONGESTION", SWITCH_CAUSE_NORMAL_CIRCUIT_CONGESTION},
78  {"NETWORK_OUT_OF_ORDER", SWITCH_CAUSE_NETWORK_OUT_OF_ORDER},
79  {"NORMAL_TEMPORARY_FAILURE", SWITCH_CAUSE_NORMAL_TEMPORARY_FAILURE},
80  {"SWITCH_CONGESTION", SWITCH_CAUSE_SWITCH_CONGESTION},
81  {"ACCESS_INFO_DISCARDED", SWITCH_CAUSE_ACCESS_INFO_DISCARDED},
82  {"REQUESTED_CHAN_UNAVAIL", SWITCH_CAUSE_REQUESTED_CHAN_UNAVAIL},
83  {"PRE_EMPTED", SWITCH_CAUSE_PRE_EMPTED},
84  {"FACILITY_NOT_SUBSCRIBED", SWITCH_CAUSE_FACILITY_NOT_SUBSCRIBED},
85  {"OUTGOING_CALL_BARRED", SWITCH_CAUSE_OUTGOING_CALL_BARRED},
86  {"INCOMING_CALL_BARRED", SWITCH_CAUSE_INCOMING_CALL_BARRED},
87  {"BEARERCAPABILITY_NOTAUTH", SWITCH_CAUSE_BEARERCAPABILITY_NOTAUTH},
88  {"BEARERCAPABILITY_NOTAVAIL", SWITCH_CAUSE_BEARERCAPABILITY_NOTAVAIL},
89  {"SERVICE_UNAVAILABLE", SWITCH_CAUSE_SERVICE_UNAVAILABLE},
90  {"BEARERCAPABILITY_NOTIMPL", SWITCH_CAUSE_BEARERCAPABILITY_NOTIMPL},
91  {"CHAN_NOT_IMPLEMENTED", SWITCH_CAUSE_CHAN_NOT_IMPLEMENTED},
92  {"FACILITY_NOT_IMPLEMENTED", SWITCH_CAUSE_FACILITY_NOT_IMPLEMENTED},
93  {"SERVICE_NOT_IMPLEMENTED", SWITCH_CAUSE_SERVICE_NOT_IMPLEMENTED},
94  {"INVALID_CALL_REFERENCE", SWITCH_CAUSE_INVALID_CALL_REFERENCE},
95  {"INCOMPATIBLE_DESTINATION", SWITCH_CAUSE_INCOMPATIBLE_DESTINATION},
96  {"INVALID_MSG_UNSPECIFIED", SWITCH_CAUSE_INVALID_MSG_UNSPECIFIED},
97  {"MANDATORY_IE_MISSING", SWITCH_CAUSE_MANDATORY_IE_MISSING},
98  {"MESSAGE_TYPE_NONEXIST", SWITCH_CAUSE_MESSAGE_TYPE_NONEXIST},
99  {"WRONG_MESSAGE", SWITCH_CAUSE_WRONG_MESSAGE},
100  {"IE_NONEXIST", SWITCH_CAUSE_IE_NONEXIST},
101  {"INVALID_IE_CONTENTS", SWITCH_CAUSE_INVALID_IE_CONTENTS},
102  {"WRONG_CALL_STATE", SWITCH_CAUSE_WRONG_CALL_STATE},
103  {"RECOVERY_ON_TIMER_EXPIRE", SWITCH_CAUSE_RECOVERY_ON_TIMER_EXPIRE},
104  {"MANDATORY_IE_LENGTH_ERROR", SWITCH_CAUSE_MANDATORY_IE_LENGTH_ERROR},
105  {"PROTOCOL_ERROR", SWITCH_CAUSE_PROTOCOL_ERROR},
106  {"INTERWORKING", SWITCH_CAUSE_INTERWORKING},
107  {"SUCCESS", SWITCH_CAUSE_SUCCESS},
108  {"ORIGINATOR_CANCEL", SWITCH_CAUSE_ORIGINATOR_CANCEL},
109  {"CRASH", SWITCH_CAUSE_CRASH},
110  {"SYSTEM_SHUTDOWN", SWITCH_CAUSE_SYSTEM_SHUTDOWN},
111  {"LOSE_RACE", SWITCH_CAUSE_LOSE_RACE},
112  {"MANAGER_REQUEST", SWITCH_CAUSE_MANAGER_REQUEST},
113  {"BLIND_TRANSFER", SWITCH_CAUSE_BLIND_TRANSFER},
114  {"ATTENDED_TRANSFER", SWITCH_CAUSE_ATTENDED_TRANSFER},
115  {"ALLOTTED_TIMEOUT", SWITCH_CAUSE_ALLOTTED_TIMEOUT},
116  {"USER_CHALLENGE", SWITCH_CAUSE_USER_CHALLENGE},
117  {"MEDIA_TIMEOUT", SWITCH_CAUSE_MEDIA_TIMEOUT},
118  {"PICKED_OFF", SWITCH_CAUSE_PICKED_OFF},
119  {"USER_NOT_REGISTERED", SWITCH_CAUSE_USER_NOT_REGISTERED},
120  {"PROGRESS_TIMEOUT", SWITCH_CAUSE_PROGRESS_TIMEOUT},
121  {"INVALID_GATEWAY", SWITCH_CAUSE_INVALID_GATEWAY},
122  {"GATEWAY_DOWN", SWITCH_CAUSE_GATEWAY_DOWN},
123  {"INVALID_URL", SWITCH_CAUSE_INVALID_URL},
124  {"INVALID_PROFILE", SWITCH_CAUSE_INVALID_PROFILE},
125  {"NO_PICKUP", SWITCH_CAUSE_NO_PICKUP},
126  {"SRTP_READ_ERROR", SWITCH_CAUSE_SRTP_READ_ERROR},
127  {"BOWOUT", SWITCH_CAUSE_BOWOUT},
128  {"BUSY_EVERYWHERE", SWITCH_CAUSE_BUSY_EVERYWHERE},
129  {"DECLINE", SWITCH_CAUSE_DECLINE},
130  {"DOES_NOT_EXIST_ANYWHERE", SWITCH_CAUSE_DOES_NOT_EXIST_ANYWHERE},
131  {"NOT_ACCEPTABLE", SWITCH_CAUSE_NOT_ACCEPTABLE},
132  {"UNWANTED", SWITCH_CAUSE_UNWANTED},
133  {"NO_IDENTITY", SWITCH_CAUSE_NO_IDENTITY},
134  {"BAD_IDENTITY_INFO", SWITCH_CAUSE_BAD_IDENTITY_INFO},
135  {"UNSUPPORTED_CERTIFICATE", SWITCH_CAUSE_UNSUPPORTED_CERTIFICATE},
136  {"INVALID_IDENTITY", SWITCH_CAUSE_INVALID_IDENTITY},
137  {"STALE_DATE", SWITCH_CAUSE_STALE_DATE},
138  {"REJECT_ALL", SWITCH_CAUSE_REJECT_ALL},
139  {NULL, 0}
140 };
141 
142 typedef enum {
143  OCF_HANGUP = (1 << 0)
145 
146 typedef enum {
151 
153  char *name;
167  uint32_t flags[CF_FLAG_MAX];
168  uint32_t caps[CC_FLAG_MAX];
169  uint8_t state_flags[CF_FLAG_MAX];
170  uint32_t private_flags;
179  int vi;
190  char *device_id;
192 };
193 
194 static void process_device_hup(switch_channel_t *channel);
196 
198 {
199  return channel->hold_record;
200 }
201 
203 {
204  uint8_t x;
205  const char *str = "UNKNOWN";
206 
207  for (x = 0; x < (sizeof(CAUSE_CHART) / sizeof(struct switch_cause_table)) - 1; x++) {
208  if (CAUSE_CHART[x].cause == cause) {
209  str = CAUSE_CHART[x].name;
210  break;
211  }
212  }
213 
214  return str;
215 }
216 
218 {
219  uint8_t x;
221 
222  if (!zstr(str)) {
223  if (*str > 47 && *str < 58) {
224  cause = atoi(str);
225  } else {
226  for (x = 0; x < (sizeof(CAUSE_CHART) / sizeof(struct switch_cause_table)) - 1 && CAUSE_CHART[x].name; x++) {
227  if (!strcasecmp(CAUSE_CHART[x].name, str)) {
228  cause = CAUSE_CHART[x].cause;
229  break;
230  }
231  }
232  }
233  }
234 
235  return cause;
236 }
237 
239 {
240  return channel->hangup_cause;
241 }
242 
243 
245 {
246  return &channel->hangup_cause;
247 }
248 
249 
251  const char *name;
253 };
255  {"DOWN", CCS_DOWN},
256  {"DIALING", CCS_DIALING},
257  {"RINGING", CCS_RINGING},
258  {"EARLY", CCS_EARLY},
259  {"ACTIVE", CCS_ACTIVE},
260  {"HELD", CCS_HELD},
261  {"RING_WAIT", CCS_RING_WAIT},
262  {"HANGUP", CCS_HANGUP},
263  {"UNHELD", CCS_UNHELD},
264  {NULL, 0}
265 };
266 
268  const char *name;
270 };
272  {"DOWN", SDS_DOWN},
273  {"RINGING", SDS_RINGING},
274  {"ACTIVE", SDS_ACTIVE},
275  {"ACTIVE_MULTI", SDS_ACTIVE_MULTI},
276  {"HELD", SDS_HELD},
277  {"UNHELD", SDS_UNHELD},
278  {"HANGUP", SDS_HANGUP},
279  {NULL, 0}
280 };
281 
282 
284  const char *file, const char *func, int line)
285 {
286  switch_event_t *event;
287  switch_channel_callstate_t o_callstate = channel->callstate;
288 
289  if (o_callstate == callstate || o_callstate == CCS_HANGUP) return;
290 
291  channel->callstate = callstate;
292  if (channel->device_node) {
293  channel->device_node->callstate = callstate;
294  }
296  "(%s) Callstate Change %s -> %s\n", channel->name,
298 
299  switch_channel_check_device_state(channel, channel->callstate);
300 
302  switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "Original-Channel-Call-State", switch_channel_callstate2str(o_callstate));
303  switch_event_add_header(event, SWITCH_STACK_BOTTOM, "Channel-Call-State-Number", "%d", callstate);
304  switch_channel_event_set_data(channel, event);
305  switch_event_fire(&event);
306  }
307 }
308 
310 {
311  return channel->callstate;
312 }
313 
314 
316 {
317  uint8_t x;
318  const char *str = "UNKNOWN";
319 
320  for (x = 0; x < (sizeof(CALLSTATE_CHART) / sizeof(struct switch_cause_table)) - 1; x++) {
321  if (CALLSTATE_CHART[x].callstate == callstate) {
322  str = CALLSTATE_CHART[x].name;
323  break;
324  }
325  }
326 
327  return str;
328 }
329 
331 {
332  uint8_t x;
333  const char *str = "UNKNOWN";
334 
335  for (x = 0; x < (sizeof(DEVICE_STATE_CHART) / sizeof(struct switch_cause_table)) - 1; x++) {
336  if (DEVICE_STATE_CHART[x].device_state == device_state) {
337  str = DEVICE_STATE_CHART[x].name;
338  break;
339  }
340  }
341 
342  return str;
343 }
344 
345 
347 {
348  uint8_t x;
350 
351  if (*str > 47 && *str < 58) {
352  callstate = atoi(str);
353  } else {
354  for (x = 0; x < (sizeof(CALLSTATE_CHART) / sizeof(struct switch_callstate_table)) - 1 && CALLSTATE_CHART[x].name; x++) {
355  if (!strcasecmp(CALLSTATE_CHART[x].name, str)) {
356  callstate = CALLSTATE_CHART[x].callstate;
357  break;
358  }
359  }
360  }
361  return callstate;
362 }
363 
364 
365 
366 SWITCH_DECLARE(void) switch_channel_perform_audio_sync(switch_channel_t *channel, const char *file, const char *func, int line)
367 {
368  if (switch_channel_media_up(channel)) {
369  switch_core_session_message_t *msg = NULL;
370 
371  msg = switch_core_session_alloc(channel->session, sizeof(*msg));
372  MESSAGE_STAMP_FFL(msg);
374  msg->from = channel->name;
375  msg->_file = file;
376  msg->_func = func;
377  msg->_line = line;
378 
380  }
381 }
382 
383 
384 SWITCH_DECLARE(void) switch_channel_perform_video_sync(switch_channel_t *channel, const char *file, const char *func, int line)
385 {
386 
387  if (switch_channel_media_up(channel)) {
388  switch_core_session_message_t *msg = NULL;
389 
390  msg = switch_core_session_alloc(channel->session, sizeof(*msg));
391  MESSAGE_STAMP_FFL(msg);
393  msg->from = channel->name;
394  msg->_file = file;
395  msg->_func = func;
396  msg->_line = line;
397 
400  }
401 }
402 
403 
404 
406 {
407  if (cause <= SWITCH_CAUSE_INTERWORKING) {
408  return cause;
409  } else {
411  }
412 }
413 
415 {
416  return switch_channel_cause_q850(channel->hangup_cause);
417 }
418 
420 {
421  switch_channel_timetable_t *times = NULL;
422 
423  if (channel->caller_profile) {
425  times = channel->caller_profile->times;
427  }
428 
429  return times;
430 }
431 
433 {
434  if (!switch_core_session_in_thread(channel->session)) {
435  channel->direction = channel->logical_direction = direction;
436  switch_channel_set_variable(channel, "direction", switch_channel_direction(channel) == SWITCH_CALL_DIRECTION_OUTBOUND ? "outbound" : "inbound");
437  }
438 }
439 
441 {
442  return channel->direction;
443 }
444 
446 {
447  return channel->logical_direction;
448 }
449 
451 {
452  switch_assert(pool != NULL);
453 
454  if (((*channel) = switch_core_alloc(pool, sizeof(switch_channel_t))) == 0) {
455  return SWITCH_STATUS_MEMERR;
456  }
457 
458  switch_event_create_plain(&(*channel)->variables, SWITCH_EVENT_CHANNEL_DATA);
459 
460  switch_core_hash_init(&(*channel)->private_hash);
461  switch_queue_create(&(*channel)->dtmf_queue, SWITCH_DTMF_LOG_LEN, pool);
462  switch_queue_create(&(*channel)->dtmf_log_queue, SWITCH_DTMF_LOG_LEN, pool);
463 
464  switch_mutex_init(&(*channel)->dtmf_mutex, SWITCH_MUTEX_NESTED, pool);
465  switch_mutex_init(&(*channel)->flag_mutex, SWITCH_MUTEX_NESTED, pool);
466  switch_mutex_init(&(*channel)->state_mutex, SWITCH_MUTEX_NESTED, pool);
467  switch_mutex_init(&(*channel)->thread_mutex, SWITCH_MUTEX_NESTED, pool);
468  switch_mutex_init(&(*channel)->profile_mutex, SWITCH_MUTEX_NESTED, pool);
469  (*channel)->hangup_cause = SWITCH_CAUSE_NONE;
470  (*channel)->name = "";
471  (*channel)->direction = (*channel)->logical_direction = direction;
472  switch_channel_set_variable(*channel, "direction", switch_channel_direction(*channel) == SWITCH_CALL_DIRECTION_OUTBOUND ? "outbound" : "inbound");
473 
474  return SWITCH_STATUS_SUCCESS;
475 }
476 
478 {
479  return switch_mutex_lock(channel->dtmf_mutex);
480 }
481 
483 {
484  return switch_mutex_trylock(channel->dtmf_mutex);
485 }
486 
488 {
489  return switch_mutex_unlock(channel->dtmf_mutex);
490 }
491 
493 {
494  switch_size_t has;
495 
496  switch_mutex_lock(channel->dtmf_mutex);
497  has = switch_queue_size(channel->dtmf_queue);
499 
500  return has;
501 }
502 
504 {
505  switch_status_t status;
506  void *pop;
507  switch_dtmf_t new_dtmf = { 0 };
509 
510  switch_assert(dtmf);
511 
512  switch_mutex_lock(channel->dtmf_mutex);
513  new_dtmf = *dtmf;
514 
515  if (sensitive) {
516  switch_set_flag((&new_dtmf), DTMF_FLAG_SENSITIVE);
517  }
518 
519  if ((status = switch_core_session_recv_dtmf(channel->session, dtmf) != SWITCH_STATUS_SUCCESS)) {
520  goto done;
521  }
522 
523  if (is_dtmf(new_dtmf.digit)) {
524  switch_dtmf_t *dt;
525  int x = 0;
526 
527  if (!sensitive) {
528  switch_log_printf(SWITCH_CHANNEL_CHANNEL_LOG(channel), SWITCH_LOG_INFO, "RECV DTMF %c:%d\n", new_dtmf.digit, new_dtmf.duration);
529  }
530 
531  if (new_dtmf.digit != 'w' && new_dtmf.digit != 'W') {
532  if (new_dtmf.duration > switch_core_max_dtmf_duration(0)) {
533  switch_log_printf(SWITCH_CHANNEL_CHANNEL_LOG(channel), SWITCH_LOG_DEBUG, "%s EXCESSIVE DTMF DIGIT LEN [%d]\n",
534  switch_channel_get_name(channel), new_dtmf.duration);
536  } else if (new_dtmf.duration < switch_core_min_dtmf_duration(0)) {
537  switch_log_printf(SWITCH_CHANNEL_CHANNEL_LOG(channel), SWITCH_LOG_DEBUG, "%s SHORT DTMF DIGIT LEN [%d]\n",
538  switch_channel_get_name(channel), new_dtmf.duration);
540  }
541  }
542 
543  if (!new_dtmf.duration) {
545  }
546 
547 
548  switch_zmalloc(dt, sizeof(*dt));
549  *dt = new_dtmf;
550 
551 
552  while (switch_queue_trypush(channel->dtmf_queue, dt) != SWITCH_STATUS_SUCCESS) {
553  if (switch_queue_trypop(channel->dtmf_queue, &pop) == SWITCH_STATUS_SUCCESS) {
554  free(pop);
555  }
556  if (++x > 100) {
557  status = SWITCH_STATUS_FALSE;
558  free(dt);
559  goto done;
560  }
561  }
562  }
563 
564  status = SWITCH_STATUS_SUCCESS;
565 
566  done:
567 
569 
571 
572  return status;
573 }
574 
576 {
577  char *p;
579  int sent = 0, dur, bad_input = 0;
580  char *string;
581  int i, argc;
582  char *argv[256];
583 
584  if (zstr(dtmf_string)) {
585  return SWITCH_STATUS_GENERR;
586  }
587 
588 
590 
591  if (*dtmf_string == '~') {
592  dtmf_string++;
593  dtmf.flags = 0;
594  }
595 
596  string = switch_core_session_strdup(channel->session, dtmf_string);
597  argc = switch_separate_string(string, '+', argv, (sizeof(argv) / sizeof(argv[0])));
598 
599  for (i = 0; i < argc; i++) {
602  if ((p = strchr(argv[i], '@'))) {
603  *p++ = '\0';
604  if ((dur = atoi(p)) > (int)switch_core_min_dtmf_duration(0) / 8) {
605  dtmf.duration = dur * 8;
606  }
607  }
608 
609  for (p = argv[i]; p && *p; p++) {
610  if (is_dtmf(*p)) {
611  dtmf.digit = *p;
612 
613  if (dtmf.duration > switch_core_max_dtmf_duration(0)) {
614  switch_log_printf(SWITCH_CHANNEL_CHANNEL_LOG(channel), SWITCH_LOG_WARNING, "EXCESSIVE DTMF DIGIT LEN %c %d\n", dtmf.digit, dtmf.duration);
616  } else if (dtmf.duration < switch_core_min_dtmf_duration(0)) {
617  switch_log_printf(SWITCH_CHANNEL_CHANNEL_LOG(channel), SWITCH_LOG_WARNING, "SHORT DTMF DIGIT LEN %c %d\n", dtmf.digit, dtmf.duration);
619  } else if (!dtmf.duration) {
621  }
622 
623  if (switch_channel_queue_dtmf(channel, &dtmf) == SWITCH_STATUS_SUCCESS) {
624  switch_log_printf(SWITCH_CHANNEL_CHANNEL_LOG(channel), SWITCH_LOG_DEBUG, "%s Queue dtmf\ndigit=%c ms=%u samples=%u\n",
625  switch_channel_get_name(channel), dtmf.digit, dur, dtmf.duration);
626  sent++;
627  }
628  } else {
629  bad_input++;
630  }
631  }
632 
633  }
634  if (sent) {
635  return SWITCH_STATUS_SUCCESS;
636  }
637  return bad_input ? SWITCH_STATUS_GENERR : SWITCH_STATUS_FALSE;
638 }
639 
641 {
642  switch_event_t *event;
643  void *pop;
644  switch_dtmf_t *dt;
646  int sensitive = 0;
647 
648  switch_mutex_lock(channel->dtmf_mutex);
649 
650  if (switch_queue_trypop(channel->dtmf_queue, &pop) == SWITCH_STATUS_SUCCESS) {
651  dt = (switch_dtmf_t *) pop;
652  *dtmf = *dt;
653  sensitive = switch_test_flag(dtmf, DTMF_FLAG_SENSITIVE);
654 
655  if (!sensitive && switch_queue_trypush(channel->dtmf_log_queue, dt) != SWITCH_STATUS_SUCCESS) {
656  free(dt);
657  }
658 
659  dt = NULL;
660 
661  if (dtmf->duration > switch_core_max_dtmf_duration(0)) {
662  switch_log_printf(SWITCH_CHANNEL_CHANNEL_LOG(channel), SWITCH_LOG_WARNING, "%s EXCESSIVE DTMF DIGIT [%c] LEN [%d]\n",
663  switch_channel_get_name(channel), sensitive ? 'S' : dtmf->digit, dtmf->duration);
665  } else if (dtmf->duration < switch_core_min_dtmf_duration(0)) {
666  switch_log_printf(SWITCH_CHANNEL_CHANNEL_LOG(channel), SWITCH_LOG_WARNING, "%s SHORT DTMF DIGIT [%c] LEN [%d]\n",
667  switch_channel_get_name(channel), sensitive ? 'S' : dtmf->digit, dtmf->duration);
669  } else if (!dtmf->duration) {
671  }
672 
673  status = SWITCH_STATUS_SUCCESS;
674  }
676 
677  if (!sensitive && status == SWITCH_STATUS_SUCCESS && switch_event_create(&event, SWITCH_EVENT_DTMF) == SWITCH_STATUS_SUCCESS) {
678  const char *dtmf_source_str = NULL;
679  switch_channel_event_set_data(channel, event);
680  switch_event_add_header(event, SWITCH_STACK_BOTTOM, "DTMF-Digit", "%c", dtmf->digit);
681  switch_event_add_header(event, SWITCH_STACK_BOTTOM, "DTMF-Duration", "%u", dtmf->duration);
682  switch(dtmf->source) {
683  case SWITCH_DTMF_INBAND_AUDIO: /* From audio */
684  dtmf_source_str = "INBAND_AUDIO";
685  break;
686  case SWITCH_DTMF_RTP: /* From RTP as a telephone event */
687  dtmf_source_str = "RTP";
688  break;
689  case SWITCH_DTMF_ENDPOINT: /* From endpoint signaling */
690  dtmf_source_str = "ENDPOINT";
691  break;
692  case SWITCH_DTMF_APP: /* Injected by application */
693  dtmf_source_str = "APP";
694  break;
695  case SWITCH_DTMF_UNKNOWN: /* Unknown source */
696  default:
697  dtmf_source_str = "UNKNOWN";
698  break;
699  }
700  switch_event_add_header(event, SWITCH_STACK_BOTTOM, "DTMF-Source", "%s", dtmf_source_str);
702  switch_core_session_queue_event(channel->session, &event);
703  } else {
704  switch_event_fire(&event);
705  }
706  }
707 
708  return status;
709 }
710 
712 {
713  switch_size_t x = 0;
714  switch_dtmf_t dtmf = { 0 };
715 
716  memset(dtmf_str, 0, len);
717 
718  while (x < len - 1 && switch_channel_dequeue_dtmf(channel, &dtmf) == SWITCH_STATUS_SUCCESS) {
719  dtmf_str[x++] = dtmf.digit;
720  }
721 
722  return x;
723 
724 }
725 
727 {
728  void *pop;
729 
730  switch_mutex_lock(channel->dtmf_mutex);
731  while (switch_queue_trypop(channel->dtmf_queue, &pop) == SWITCH_STATUS_SUCCESS) {
732  switch_dtmf_t *dt = (switch_dtmf_t *) pop;
733  if (channel->state >= CS_HANGUP || switch_queue_trypush(channel->dtmf_log_queue, dt) != SWITCH_STATUS_SUCCESS) {
734  free(dt);
735  }
736  }
738 }
739 
741 {
742  void *pop;
743  switch_channel_flush_dtmf(channel);
744  while (switch_queue_trypop(channel->dtmf_log_queue, &pop) == SWITCH_STATUS_SUCCESS) {
745  switch_safe_free(pop);
746  }
747 
748  if (channel->private_hash) {
750  }
751 
752  if (channel->app_flag_hash) {
754  }
755 
757  switch_event_destroy(&channel->variables);
758  switch_event_destroy(&channel->api_list);
759  switch_event_destroy(&channel->var_list);
760  switch_event_destroy(&channel->app_list);
761  if (channel->log_tags) {
762  switch_event_destroy(&channel->log_tags);
763  }
765 }
766 
769 {
770  switch_assert(channel != NULL);
771  channel->state = state;
772  switch_channel_set_flag(channel, flag);
773  channel->session = session;
774  channel->running_state = CS_NONE;
775  return SWITCH_STATUS_SUCCESS;
776 }
777 
778 SWITCH_DECLARE(void) switch_channel_perform_presence(switch_channel_t *channel, const char *rpid, const char *status, const char *id,
779  const char *file, const char *func, int line)
780 {
781  switch_event_t *event;
783  const char *call_info = NULL;
784  char *call_info_state = "active";
785 
787  return;
788  }
789 
790  if (!status) {
792  status = "idle";
793  }
794 
795  if (!id) {
796  id = switch_channel_get_variable(channel, "presence_id");
797  }
798 
799  if (!id) {
800  return;
801  }
802 
803  call_info = switch_channel_get_variable(channel, "presence_call_info");
804 
805  if (switch_event_create(&event, type) == SWITCH_STATUS_SUCCESS) {
806  switch_channel_event_set_data(channel, event);
807  switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "proto", "any");
808  switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "login", __FILE__);
810  if (type == SWITCH_EVENT_PRESENCE_IN) {
811  if (!rpid) {
812  rpid = "unknown";
813  }
815  switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "status", status);
816  }
817  switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "event_type", "presence");
818  switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "alt_event_type", "dialog");
819 
820 
821  if (!strcasecmp(status, "idle") || !switch_channel_up_nosig(channel)) {
822  call_info_state = "idle";
823  } else if (!strcasecmp(status, "hold-private")) {
824  call_info_state = "held-private";
825  } else if (!strcasecmp(status, "hold")) {
826  call_info_state = "held";
827  } else if (!switch_channel_test_flag(channel, CF_ANSWERED)) {
828  if (channel->direction == SWITCH_CALL_DIRECTION_OUTBOUND) {
829  call_info_state = "progressing";
830  } else {
832  call_info_state = "idle";
833  } else {
834  call_info_state = "alerting";
835  }
836  }
837  }
838 
839  switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "presence-call-info-state", call_info_state);
840 
841  if (call_info) {
842  switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "presence-call-info", call_info);
843  }
844 
845  switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "presence-call-direction",
846  channel->direction == SWITCH_CALL_DIRECTION_OUTBOUND ? "outbound" : "inbound");
847 
848  switch_event_add_header(event, SWITCH_STACK_BOTTOM, "event_count", "%d", channel->event_count++);
849  switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "Presence-Calling-File", file);
850  switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "Presence-Calling-Function", func);
851  switch_event_add_header(event, SWITCH_STACK_BOTTOM, "Presence-Calling-Line", "%d", line);
852 
853  if (switch_true(switch_channel_get_variable(channel, "presence_privacy"))) {
854  switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "Presence-Privacy", "true");
855  }
856 
857  switch_event_fire(&event);
858  }
859 }
860 
862 {
863  switch_event_t *event;
864 
865  if (!!on == !!switch_channel_test_flag(channel, CF_LEG_HOLDING)) {
866  goto end;
867  }
868 
869  if (on) {
871  } else {
873  }
874 
876  switch_channel_event_set_data(channel, event);
877  switch_event_fire(&event);
878  }
879 
880  end:
881 
882  if (on) {
883  if (switch_true(switch_channel_get_variable(channel, "flip_record_on_hold"))) {
884  switch_core_session_t *other_session;
885  if (switch_core_session_get_partner(channel->session, &other_session) == SWITCH_STATUS_SUCCESS) {
886  switch_ivr_transfer_recordings(channel->session, other_session);
887  switch_core_session_rwunlock(other_session);
888  }
889  }
890  }
891 
892 }
893 
895 {
896  const char *var;
897 
900  }
901 
902  if (!zstr(var)) {
903  char *expanded = switch_channel_expand_variables(channel, var);
904 
905  if (expanded != var) {
906  var = switch_core_session_strdup(channel->session, expanded);
907  free(expanded);
908  }
909  }
910 
911 
912  return var;
913 }
914 
916 {
917  switch_core_session_t *session;
918  const char *r = NULL;
919 
923  }
924 
925  return r;
926 }
927 
929 {
931 
932  if (event && *event) { /* push */
933  (*event)->next = channel->scope_variables;
934  channel->scope_variables = *event;
935  *event = NULL;
936  } else if (channel->scope_variables) { /* pop */
937  switch_event_t *top_event = channel->scope_variables;
938  channel->scope_variables = channel->scope_variables->next;
939  switch_event_destroy(&top_event);
940  }
941 
943 
944 }
945 
947 {
949  switch_event_t *new_event;
950 
952  if (channel->scope_variables) {
953  switch_event_t *ep;
955 
957  status = SWITCH_STATUS_SUCCESS;
958  *event = new_event;
959 
960  for (ep = channel->scope_variables; ep; ep = ep->next) {
961  for (hp = ep->headers; hp; hp = hp->next) {
962  if (!switch_event_get_header(new_event, hp->value)) {
964  }
965  }
966  }
967  }
969 
970  return status;
971 }
972 
973 SWITCH_DECLARE(const char *) switch_channel_get_variable_dup(switch_channel_t *channel, const char *varname, switch_bool_t dup, int idx)
974 {
975  const char *v = NULL, *r = NULL, *vdup = NULL;
976  switch_assert(channel != NULL);
977 
979 
980  if (!zstr(varname)) {
981  if (channel->scope_variables) {
982  switch_event_t *ep;
983 
984  for (ep = channel->scope_variables; ep; ep = ep->next) {
985  if ((v = switch_event_get_header_idx(ep, varname, idx))) {
986  break;
987  }
988  }
989  }
990 
991  if (!v && (!channel->variables || !(v = switch_event_get_header_idx(channel->variables, varname, idx)))) {
993 
994  if (cp) {
995  if (!strncmp(varname, "aleg_", 5)) {
996  cp = cp->originator_caller_profile;
997  varname += 5;
998  } else if (!strncmp(varname, "bleg_", 5)) {
999  cp = cp->originatee_caller_profile;
1000  varname += 5;
1001  }
1002  }
1003 
1004  if (!cp || !(v = switch_caller_get_field_by_name(cp, varname))) {
1005  if ((vdup = switch_core_get_variable_pdup(varname, switch_core_session_get_pool(channel->session)))) {
1006  v = vdup;
1007  }
1008  }
1009  }
1010  }
1011 
1012  if (dup && v != vdup) {
1013  if (v) {
1014  r = switch_core_session_strdup(channel->session, v);
1015  }
1016  } else {
1017  r = v;
1018  }
1019 
1021 
1022  return r;
1023 }
1024 
1025 SWITCH_DECLARE(const char *) switch_channel_get_variable_strdup(switch_channel_t *channel, const char *varname)
1026 {
1027  const char *value = switch_channel_get_variable_dup(channel, varname, SWITCH_FALSE, -1);
1028 
1029  return value ? (const char *)strdup(value) : NULL;
1030 }
1031 
1033 {
1034  const char *value = switch_channel_get_variable_dup(channel, varname, SWITCH_FALSE, -1);
1035 
1036  if (value && buf && buflen && switch_copy_string(buf, value, buflen)) {
1037  return SWITCH_STATUS_SUCCESS;
1038  }
1039 
1040  return SWITCH_STATUS_FALSE;
1041 }
1042 
1043 SWITCH_DECLARE(const char *) switch_channel_get_variable_partner(switch_channel_t *channel, const char *varname)
1044 {
1045  const char *uuid;
1046  const char *val = NULL, *r = NULL;
1047  switch_assert(channel != NULL);
1048 
1049  if (!zstr(varname)) {
1050  if ((uuid = switch_channel_get_partner_uuid(channel))) {
1051  switch_core_session_t *session;
1052  if ((session = switch_core_session_locate(uuid))) {
1054  val = switch_channel_get_variable(tchannel, varname);
1056  }
1057  }
1058  }
1059 
1060  if (val)
1061  r = switch_core_session_strdup(channel->session, val);
1062 
1063  return r;
1064 }
1065 
1066 
1068 {
1069  switch_assert(channel != NULL);
1070  if (!channel->vi) {
1071  return;
1072  }
1073  channel->vi = 0;
1075 
1076 }
1077 
1079 {
1080  switch_event_header_t *hi = NULL;
1081 
1082  switch_assert(channel != NULL);
1083  switch_mutex_lock(channel->profile_mutex);
1084  if (channel->variables && (hi = channel->variables->headers)) {
1085  channel->vi = 1;
1086  } else {
1088  }
1089 
1090  return hi;
1091 }
1092 
1093 SWITCH_DECLARE(switch_status_t) switch_channel_set_private(switch_channel_t *channel, const char *key, const void *private_info)
1094 {
1095  switch_assert(channel != NULL);
1096  switch_core_hash_insert_locked(channel->private_hash, key, private_info, channel->profile_mutex);
1097  return SWITCH_STATUS_SUCCESS;
1098 }
1099 
1101 {
1102  void *val;
1103  switch_assert(channel != NULL);
1104  val = switch_core_hash_find_locked(channel->private_hash, key, channel->profile_mutex);
1105  return val;
1106 }
1107 
1109 {
1110  const char *uuid;
1111  void *val = NULL;
1112 
1113  switch_assert(channel != NULL);
1114 
1115  if ((uuid = switch_channel_get_partner_uuid(channel))) {
1116  switch_core_session_t *session;
1117  if ((session = switch_core_session_locate(uuid))) {
1118  val = switch_core_hash_find_locked(channel->private_hash, key, channel->profile_mutex);
1120  }
1121  }
1122 
1123  return val;
1124 }
1125 
1127 {
1128  const char *old = NULL;
1129 
1130  switch_assert(channel != NULL);
1131  if (!zstr(channel->name)) {
1132  old = channel->name;
1133  }
1134  channel->name = NULL;
1135  if (name) {
1136  char *uuid = switch_core_session_get_uuid(channel->session);
1137  channel->name = switch_core_session_strdup(channel->session, name);
1139  if (old) {
1140  switch_log_printf(SWITCH_CHANNEL_CHANNEL_LOG(channel), SWITCH_LOG_NOTICE, "Rename Channel %s->%s [%s]\n", old, name, uuid);
1141  } else {
1142  switch_log_printf(SWITCH_CHANNEL_CHANNEL_LOG(channel), SWITCH_LOG_NOTICE, "New Channel %s [%s]\n", name, uuid);
1143  }
1144  }
1145  return SWITCH_STATUS_SUCCESS;
1146 }
1147 
1149 {
1150  switch_assert(channel != NULL);
1151  return (!zstr(channel->name)) ? channel->name : "N/A";
1152 }
1153 
1155 {
1156  char *v;
1158 
1159  switch_mutex_lock(channel->profile_mutex);
1160 
1161 
1162  if (!strcasecmp(name, "device_id") && !zstr(val)) {
1163  const char *device_id;
1164  if (!(device_id = switch_channel_set_device_id(channel, val))) {
1165  /* one time setting */
1167  return status;
1168  }
1169 
1170  val = device_id;
1171  }
1172 
1173  if (!zstr(val)) {
1174  v = switch_core_strdup(channel->caller_profile->pool, val);
1175  } else {
1176  v = SWITCH_BLANK_STRING;
1177  }
1178 
1179  if (!strcasecmp(name, "dialplan")) {
1180  channel->caller_profile->dialplan = v;
1181  } else if (!strcasecmp(name, "username")) {
1182  channel->caller_profile->username = v;
1183  } else if (!strcasecmp(name, "caller_id_name")) {
1184  channel->caller_profile->caller_id_name = v;
1185  } else if (!strcasecmp(name, "caller_id_number")) {
1186  channel->caller_profile->caller_id_number = v;
1187  } else if (!strcasecmp(name, "callee_id_name")) {
1188  channel->caller_profile->callee_id_name = v;
1189  } else if (!strcasecmp(name, "callee_id_number")) {
1190  channel->caller_profile->callee_id_number = v;
1191  } else if (val && !strcasecmp(name, "caller_ton")) {
1192  channel->caller_profile->caller_ton = (uint8_t) atoi(v);
1193  } else if (val && !strcasecmp(name, "caller_numplan")) {
1194  channel->caller_profile->caller_numplan = (uint8_t) atoi(v);
1195  } else if (val && !strcasecmp(name, "destination_number_ton")) {
1196  channel->caller_profile->destination_number_ton = (uint8_t) atoi(v);
1197  } else if (val && !strcasecmp(name, "destination_number_numplan")) {
1198  channel->caller_profile->destination_number_numplan = (uint8_t) atoi(v);
1199  } else if (!strcasecmp(name, "ani")) {
1200  channel->caller_profile->ani = v;
1201  } else if (!strcasecmp(name, "aniii")) {
1202  channel->caller_profile->aniii = v;
1203  } else if (!strcasecmp(name, "network_addr")) {
1204  channel->caller_profile->network_addr = v;
1205  } else if (!strcasecmp(name, "rdnis")) {
1206  channel->caller_profile->rdnis = v;
1207  } else if (!strcasecmp(name, "destination_number")) {
1208  channel->caller_profile->destination_number = v;
1209  } else if (!strcasecmp(name, "uuid")) {
1210  channel->caller_profile->uuid = v;
1211  } else if (!strcasecmp(name, "source")) {
1212  channel->caller_profile->source = v;
1213  } else if (!strcasecmp(name, "context")) {
1214  channel->caller_profile->context = v;
1215  } else if (!strcasecmp(name, "chan_name")) {
1216  channel->caller_profile->chan_name = v;
1217  } else {
1218  profile_node_t *pn, *n = switch_core_alloc(channel->caller_profile->pool, sizeof(*n));
1219  int var_found;
1220 
1221  n->var = switch_core_strdup(channel->caller_profile->pool, name);
1222  n->val = v;
1223 
1224  if (!channel->caller_profile->soft) {
1225  channel->caller_profile->soft = n;
1226  } else {
1227  var_found = 0;
1228 
1229  for(pn = channel->caller_profile->soft; pn ; pn = pn->next) {
1230  if (!strcasecmp(pn->var,n->var)) {
1231  pn->val = n->val;
1232  var_found = 1;
1233  break;
1234  }
1235 
1236  if(!pn->next) {
1237  break;
1238  }
1239  }
1240 
1241  if (pn && !pn->next && !var_found) {
1242  pn->next = n;
1243  }
1244  }
1245  }
1247 
1248  return status;
1249 }
1250 
1251 
1253  switch_event_t *var_event, const char *export_varname)
1254 {
1255 
1256  const char *export_vars = switch_channel_get_variable(channel, export_varname);
1257  char *cptmp = switch_core_session_strdup(channel->session, export_vars);
1258  int argc;
1259  char *argv[256];
1260 
1261  if (zstr(export_vars)) return;
1262 
1263 
1264  if (var_event) {
1265  switch_event_del_header(var_event, export_varname);
1266  switch_event_add_header_string(var_event, SWITCH_STACK_BOTTOM, export_varname, export_vars);
1267  }
1268 
1269  if (peer_channel) {
1270  switch_channel_set_variable(peer_channel, export_varname, export_vars);
1271  }
1272 
1273  if ((argc = switch_separate_string(cptmp, ',', argv, (sizeof(argv) / sizeof(argv[0]))))) {
1274  int x;
1275 
1276  for (x = 0; x < argc; x++) {
1277  const char *vval;
1278  if ((vval = switch_channel_get_variable(channel, argv[x]))) {
1279  char *vvar = argv[x];
1280  if (!strncasecmp(vvar, "nolocal:", 8)) { /* remove this later ? */
1281  vvar += 8;
1282  } else if (!strncasecmp(vvar, "_nolocal_", 9)) {
1283  vvar += 9;
1284  }
1285  if (var_event) {
1286  switch_event_del_header(var_event, vvar);
1287  switch_event_add_header_string(var_event, SWITCH_STACK_BOTTOM, vvar, vval);
1289  "%s EXPORTING[%s] [%s]=[%s] to event\n",
1290  switch_channel_get_name(channel),
1291  export_varname,
1292  vvar, vval);
1293  }
1294  if (peer_channel) {
1296  "%s EXPORTING[%s] [%s]=[%s] to %s\n",
1297  switch_channel_get_name(channel),
1298  export_varname,
1299  vvar, vval, switch_channel_get_name(peer_channel));
1300  switch_channel_set_variable(peer_channel, vvar, vval);
1301  }
1302  }
1303  }
1304  }
1305 
1306 
1307 }
1308 
1310  const char *varname, const char *val,
1311  const char *export_varname, switch_bool_t var_check)
1312 {
1313  char *var_name = NULL;
1314  const char *exports;
1315  char *var, *new_exports, *new_exports_d = NULL;
1316  int local = 1;
1317 
1318  exports = switch_channel_get_variable(channel, export_varname);
1319 
1320  var = switch_core_session_strdup(channel->session, varname);
1321 
1322  if (var) {
1323  if (!strncasecmp(var, "nolocal:", 8)) { /* remove this later ? */
1324  var_name = var + 8;
1325  local = 0;
1326  } else if (!strncasecmp(var, "_nolocal_", 9)) {
1327  var_name = var + 9;
1328  local = 0;
1329  } else {
1330  var_name = var;
1331  }
1332  }
1333 
1334  switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(channel->session), SWITCH_LOG_DEBUG, "EXPORT (%s) %s[%s]=[%s]\n",
1335  export_varname, local ? "" : "(REMOTE ONLY) ",
1336  var_name ? var_name : "", val ? val : "UNDEF");
1337 
1338 
1339  switch_channel_set_variable_var_check(channel, var, val, var_check);
1340 
1341  if (var && val) {
1342  if (exports) {
1343  new_exports_d = switch_mprintf("%s,%s", exports, var);
1344  new_exports = new_exports_d;
1345  } else {
1346  new_exports = var;
1347  }
1348 
1349  switch_channel_set_variable(channel, export_varname, new_exports);
1350 
1351  switch_safe_free(new_exports_d);
1352  }
1353 
1354  return SWITCH_STATUS_SUCCESS;
1355 }
1356 
1358  const char *export_varname, const char *fmt, ...)
1359 {
1361  char *data = NULL;
1362  va_list ap;
1363  int ret;
1364 
1365  switch_assert(channel != NULL);
1366 
1367  va_start(ap, fmt);
1368  ret = switch_vasprintf(&data, fmt, ap);
1369  va_end(ap);
1370 
1371  if (ret == -1) {
1372  return SWITCH_STATUS_FALSE;
1373  }
1374 
1375  status = switch_channel_export_variable(channel, varname, data, export_varname);
1376 
1377  free(data);
1378 
1379  return status;
1380 }
1381 
1382 
1384 {
1385  switch_event_t *event;
1387  uint32_t r = 0;
1388 
1389  switch_channel_get_variables(channel, &event);
1390 
1391  if (event) {
1392  for (hp = event->headers; hp; hp = hp->next) {
1393  if (zstr(prefix) || !strncasecmp(hp->name, prefix, strlen(prefix))) {
1394  switch_channel_set_variable(channel, hp->name, NULL);
1395  }
1396  }
1397  }
1398 
1399  switch_event_destroy(&event);
1400 
1401  return r;
1402 }
1403 
1405 {
1406  switch_event_header_t *hi = NULL;
1407  int x = 0;
1408 
1409  if ((hi = switch_channel_variable_first(orig_channel))) {
1410  for (; hi; hi = hi->next) {
1411  char *var = hi->name;
1412  char *val = hi->value;
1413 
1414  if (zstr(prefix) || !strncasecmp(var, prefix, strlen(prefix))) {
1415  x++;
1416  switch_channel_set_variable(new_channel, var, val);
1417  }
1418  }
1419  switch_channel_variable_last(orig_channel);
1420  }
1421 
1423 }
1424 
1425 SWITCH_DECLARE(void) switch_channel_set_presence_data_vals(switch_channel_t *channel, const char *presence_data_cols)
1426 {
1427  char *cols[128] = { 0 };
1428  char header_name[128] = "";
1429  int col_count = 0, i = 0;
1430  char *data_copy = NULL;
1431 
1432  if (zstr(presence_data_cols)) {
1433  presence_data_cols = switch_channel_get_variable_dup(channel, "presence_data_cols", SWITCH_FALSE, -1);
1434  if (zstr(presence_data_cols)) {
1435  return;
1436  }
1437  }
1438 
1439  data_copy = strdup(presence_data_cols);
1440 
1441  col_count = switch_split(data_copy, ':', cols);
1442 
1443  for (i = 0; i < col_count; i++) {
1444  const char *val = NULL;
1445  switch_snprintf(header_name, sizeof(header_name), "PD-%s", cols[i]);
1446  val = switch_channel_get_variable(channel, cols[i]);
1447  switch_channel_set_profile_var(channel, header_name, val);
1448  }
1449 
1450  switch_safe_free(data_copy);
1451 }
1452 
1453 SWITCH_DECLARE(switch_status_t) switch_channel_set_log_tag(switch_channel_t *channel, const char *tagname, const char *tagvalue)
1454 {
1456  switch_assert(channel != NULL);
1457  switch_mutex_lock(channel->profile_mutex);
1458  if (!zstr(tagname)) {
1459  if (!channel->log_tags) {
1461  }
1462  if (zstr(tagvalue)) {
1463  switch_event_del_header(channel->log_tags, tagname);
1464  } else {
1465  switch_event_add_header_string(channel->log_tags, SWITCH_STACK_BOTTOM, tagname, tagvalue);
1466  }
1467  status = SWITCH_STATUS_SUCCESS;
1468  }
1470  return status;
1471 }
1472 
1474 {
1476  switch_assert(channel != NULL);
1477  if (!channel->log_tags) {
1478  return status;
1479  }
1480  switch_mutex_lock(channel->profile_mutex);
1481  if (channel->log_tags && log_tags) {
1482  status = switch_event_dup(log_tags, channel->log_tags);
1483  }
1485  return status;
1486 }
1487 
1489  const char *varname, const char *value, switch_bool_t var_check)
1490 {
1492 
1493  switch_assert(channel != NULL);
1494 
1495  switch_mutex_lock(channel->profile_mutex);
1496  if (channel->variables && !zstr(varname)) {
1497  if (zstr(value)) {
1498  switch_event_del_header(channel->variables, varname);
1499  } else {
1500  int ok = 1;
1501 
1502  if (var_check) {
1503  ok = !switch_string_var_check_const(value);
1504  }
1505  if (ok) {
1507  } else {
1508  switch_log_printf(SWITCH_CHANNEL_CHANNEL_LOG(channel), SWITCH_LOG_CRIT, "Invalid data (${%s} contains a variable)\n", varname);
1509  }
1510  }
1511  status = SWITCH_STATUS_SUCCESS;
1512  }
1514 
1515  return status;
1516 }
1517 
1519  const char *varname, const char *value, switch_bool_t var_check)
1520 {
1522 
1523  switch_assert(channel != NULL);
1524 
1525  switch_mutex_lock(channel->profile_mutex);
1526  if (channel->variables && !zstr(varname)) {
1527  if (zstr(value)) {
1528  switch_event_del_header(channel->variables, varname);
1529  } else {
1530  int ok = 1;
1531  char *t = (char *)value;
1532  char *r = (char *)value;
1533  char *tmp = NULL;
1534 
1535  if (t && *t == '"') {
1536  t++;
1537  if (end_of(t) == '"') {
1538  r = tmp = strdup(t);
1539  switch_assert(r);
1540  end_of(r) = '\0';
1541  }
1542  }
1543 
1544  if (var_check) {
1546  }
1547  if (ok) {
1549  } else {
1550  switch_log_printf(SWITCH_CHANNEL_CHANNEL_LOG(channel), SWITCH_LOG_CRIT, "Invalid data (${%s} contains a variable)\n", varname);
1551  }
1552 
1553  switch_safe_free(tmp);
1554  }
1555  status = SWITCH_STATUS_SUCCESS;
1556  }
1558 
1559  return status;
1560 }
1561 
1563  const char *varname, const char *value, switch_bool_t var_check, switch_stack_t stack)
1564 {
1566 
1567  switch_assert(channel != NULL);
1568 
1569  switch_mutex_lock(channel->profile_mutex);
1570  if (channel->variables && !zstr(varname)) {
1571  if (zstr(value)) {
1572  switch_event_del_header(channel->variables, varname);
1573  } else {
1574  int ok = 1;
1575 
1576  if (var_check) {
1577  ok = !switch_string_var_check_const(value);
1578  }
1579  if (ok) {
1580  switch_event_add_header_string(channel->variables, stack, varname, value);
1581  } else {
1582  switch_log_printf(SWITCH_CHANNEL_CHANNEL_LOG(channel), SWITCH_LOG_CRIT, "Invalid data (${%s} contains a variable)\n", varname);
1583  }
1584  }
1585  status = SWITCH_STATUS_SUCCESS;
1586  }
1588 
1589  return status;
1590 }
1591 
1592 
1593 switch_status_t switch_event_base_add_header(switch_event_t *event, switch_stack_t stack, const char *header_name, char *data);
1594 
1596 {
1597  int ret = 0;
1598  char *data;
1599  va_list ap;
1601 
1602  switch_assert(channel != NULL);
1603 
1604  switch_mutex_lock(channel->profile_mutex);
1605  if (channel->variables && !zstr(varname)) {
1606  switch_event_del_header(channel->variables, varname);
1607 
1608  va_start(ap, fmt);
1609  ret = switch_vasprintf(&data, fmt, ap);
1610  va_end(ap);
1611 
1612  if (ret == -1) {
1614  return SWITCH_STATUS_MEMERR;
1615  }
1616 
1617  status = switch_channel_set_variable(channel, varname, data);
1618  free(data);
1619  }
1621 
1622  return status;
1623 }
1624 
1625 
1627 {
1628  int ret = 0;
1629  char *varname;
1630  va_list ap;
1632 
1633  switch_assert(channel != NULL);
1634 
1635  switch_mutex_lock(channel->profile_mutex);
1636 
1637  va_start(ap, fmt);
1638  ret = switch_vasprintf(&varname, fmt, ap);
1639  va_end(ap);
1640 
1641  if (ret == -1) {
1643  return SWITCH_STATUS_MEMERR;
1644  }
1645 
1646  status = switch_channel_set_variable(channel, varname, val);
1647 
1648  free(varname);
1649 
1651 
1652  return status;
1653 }
1654 
1655 
1657  const char *varname, const char *value, switch_bool_t var_check)
1658 {
1659  const char *uuid;
1660  switch_assert(channel != NULL);
1661 
1662  if (!zstr(varname)) {
1663  if ((uuid = switch_channel_get_partner_uuid(channel))) {
1664  switch_core_session_t *session;
1665  if ((session = switch_core_session_locate(uuid))) {
1667  switch_channel_set_variable_var_check(tchannel, varname, value, var_check);
1669  }
1670  return SWITCH_STATUS_SUCCESS;
1671  }
1672  }
1673 
1674  return SWITCH_STATUS_FALSE;
1675 }
1676 
1678 {
1679  uint32_t r = 0;
1680 
1681  switch_assert(channel != NULL);
1682 
1683  switch_mutex_lock(channel->flag_mutex);
1684  r = channel->flags[flag];
1685  switch_mutex_unlock(channel->flag_mutex);
1686 
1687  return r;
1688 }
1689 
1691 {
1692  const char *uuid;
1693 
1694  switch_assert(channel != NULL);
1695 
1696  if ((uuid = switch_channel_get_partner_uuid(channel))) {
1697  switch_core_session_t *session;
1698  if ((session = switch_core_session_locate(uuid))) {
1701  return SWITCH_TRUE;
1702  }
1703  }
1704 
1705  return SWITCH_FALSE;
1706 }
1707 
1709 {
1710  const char *uuid;
1711  int r = 0;
1712 
1713  switch_assert(channel != NULL);
1714 
1715  if ((uuid = switch_channel_get_partner_uuid(channel))) {
1716  switch_core_session_t *session;
1717  if ((session = switch_core_session_locate(uuid))) {
1720  }
1721  }
1722 
1723  return r;
1724 }
1725 
1727 {
1728  const char *uuid;
1729 
1730  switch_assert(channel != NULL);
1731 
1732  if ((uuid = switch_channel_get_partner_uuid(channel))) {
1733  switch_core_session_t *session;
1734  if ((session = switch_core_session_locate(uuid))) {
1737  return SWITCH_TRUE;
1738  }
1739  }
1740 
1741  return SWITCH_FALSE;
1742 }
1743 
1745 {
1746 
1747  switch_assert(channel);
1748 
1749  for (;;) {
1750  if ((channel->state < CS_HANGUP && channel->state == channel->running_state && channel->running_state == want_state) ||
1751  (other_channel && switch_channel_down_nosig(other_channel)) || switch_channel_down(channel)) {
1752  break;
1753  }
1754  switch_cond_next();
1755  }
1756 }
1757 
1758 
1760 {
1761 
1762  uint32_t count = 0;
1763 
1764  for (;;) {
1765 
1766  if ((channel->state == channel->running_state && channel->running_state == want_state) || channel->state >= CS_HANGUP) {
1767  break;
1768  }
1769 
1771 
1772  switch_cond_next();
1773 
1774  if (++count >= timeout) {
1775  break;
1776  }
1777  }
1778 }
1779 
1781  switch_channel_flag_t want_flag,
1782  switch_bool_t pres, uint32_t to, switch_channel_t *super_channel)
1783 {
1784 
1785  if (to) {
1786  to++;
1787  }
1788 
1789  for (;;) {
1790  if (pres) {
1791  if (switch_channel_test_flag(channel, want_flag)) {
1792  break;
1793  }
1794  } else {
1795  if (!switch_channel_test_flag(channel, want_flag)) {
1796  break;
1797  }
1798  }
1799 
1800  switch_cond_next();
1801 
1802  if (super_channel && !switch_channel_ready(super_channel)) {
1803  return SWITCH_STATUS_FALSE;
1804  }
1805 
1806  if (switch_channel_down(channel)) {
1807  return SWITCH_STATUS_FALSE;
1808  }
1809 
1810  if (to && !--to) {
1811  return SWITCH_STATUS_TIMEOUT;
1812  }
1813  }
1814 
1815  return SWITCH_STATUS_SUCCESS;
1816 }
1817 
1819  uint32_t app_flag,
1820  const char *key, switch_bool_t pres, uint32_t to)
1821 {
1822  int r = 0;
1823 
1824  if (to) {
1825  to++;
1826  }
1827 
1828  for (;;) {
1829  if (pres) {
1830  if ((r = switch_channel_test_app_flag_key(key, channel, app_flag))) {
1831  break;
1832  }
1833  } else {
1834  if (!(r = switch_channel_test_app_flag_key(key, channel, app_flag))) {
1835  break;
1836  }
1837  }
1838 
1839  switch_cond_next();
1840 
1841  if (switch_channel_down(channel)) {
1842  return r;
1843  }
1844 
1845  if (to && !--to) {
1846  return r;
1847  }
1848  }
1849 
1850  return r;
1851 }
1852 
1853 
1855 {
1856  switch_assert(channel);
1857  switch_assert(channel->flag_mutex);
1858 
1859  switch_mutex_lock(channel->flag_mutex);
1860  channel->caps[cap] = value;
1861  switch_mutex_unlock(channel->flag_mutex);
1862 }
1863 
1865 {
1866  switch_assert(channel != NULL);
1867  switch_assert(channel->flag_mutex);
1868 
1869  switch_mutex_lock(channel->flag_mutex);
1870  channel->caps[cap] = 0;
1871  switch_mutex_unlock(channel->flag_mutex);
1872 }
1873 
1875 {
1876  switch_assert(channel != NULL);
1877  return channel->caps[cap] ? 1 : 0;
1878 }
1879 
1881 {
1882  const char *uuid;
1883  int r = 0;
1884 
1885  switch_assert(channel != NULL);
1886 
1887  if ((uuid = switch_channel_get_partner_uuid(channel))) {
1888  switch_core_session_t *session;
1889  if ((session = switch_core_session_locate(uuid))) {
1892  }
1893  }
1894 
1895  return r;
1896 }
1897 
1899 {
1900  switch_stream_handle_t stream = { 0 };
1901  char *r;
1902  int i = 0;
1903 
1904  SWITCH_STANDARD_STREAM(stream);
1905 
1906  switch_mutex_lock(channel->flag_mutex);
1907  for (i = 0; i < CF_FLAG_MAX; i++) {
1908  if (channel->flags[i]) {
1909  stream.write_function(&stream, "%d=%d;", i, channel->flags[i]);
1910  }
1911  }
1912  switch_mutex_unlock(channel->flag_mutex);
1913 
1914  r = (char *) stream.data;
1915 
1916  if (end_of(r) == ';') {
1917  end_of(r) = '\0';
1918  }
1919 
1920  return r;
1921 
1922 }
1923 
1925 {
1926  switch_stream_handle_t stream = { 0 };
1927  char *r;
1928  int i = 0;
1929 
1930  SWITCH_STANDARD_STREAM(stream);
1931 
1932  switch_mutex_lock(channel->flag_mutex);
1933  for (i = 0; i < CC_FLAG_MAX; i++) {
1934  if (channel->caps[i]) {
1935  stream.write_function(&stream, "%d=%d;", i, channel->caps[i]);
1936  }
1937  }
1938  switch_mutex_unlock(channel->flag_mutex);
1939 
1940  r = (char *) stream.data;
1941 
1942  if (end_of(r) == ';') {
1943  end_of(r) = '\0';
1944  }
1945 
1946  return r;
1947 
1948 }
1949 
1951 {
1952  int HELD = 0;
1953  int just_set = 0;
1954 
1955  switch_assert(channel);
1956  switch_assert(channel->flag_mutex);
1957 
1958  switch_mutex_lock(channel->flag_mutex);
1959  if (flag == CF_LEG_HOLDING && !channel->flags[flag] && channel->flags[CF_ANSWERED]) {
1960  HELD = 1;
1961  }
1962  if (channel->flags[flag] != value) {
1963  just_set = 1;
1964  channel->flags[flag] = value;
1965  }
1966  switch_mutex_unlock(channel->flag_mutex);
1967 
1968  if (flag == CF_VIDEO_READY && just_set) {
1970  }
1971 
1972  if (flag == CF_ORIGINATOR && switch_channel_test_flag(channel, CF_ANSWERED) && switch_channel_up_nosig(channel)) {
1974  }
1975 
1976  if (flag == CF_DIALPLAN) {
1977  if (channel->direction == SWITCH_CALL_DIRECTION_INBOUND) {
1979  if (channel->device_node) {
1981  }
1982  } else {
1984  if (channel->device_node) {
1986  }
1987  }
1988  }
1989 
1990  if (HELD) {
1992  const char *brto = switch_channel_get_partner_uuid(channel);
1993 
1995  switch_mutex_lock(channel->profile_mutex);
1997 
1998  hr = switch_core_session_alloc(channel->session, sizeof(*hr));
1999  hr->on = switch_time_now();
2000  if (brto) {
2001  hr->uuid = switch_core_session_strdup(channel->session, brto);
2002  }
2003 
2004  if (channel->hold_record) {
2005  hr->next = channel->hold_record;
2006  }
2007  channel->hold_record = hr;
2008 
2010  }
2011 
2012  if (flag == CF_OUTBOUND) {
2013  switch_channel_set_variable(channel, "is_outbound", "true");
2014  }
2015 
2016  if (flag == CF_RECOVERED) {
2017  switch_channel_set_variable(channel, "recovered", "true");
2018  }
2019 
2020  if (flag == CF_VIDEO_ECHO || flag == CF_VIDEO_BLANK || flag == CF_VIDEO_DECODED_READ || flag == CF_VIDEO_PASSIVE) {
2022  }
2023 
2024  if (flag == CF_VIDEO_DECODED_READ && channel->flags[CF_VIDEO]) {
2026  }
2027 }
2028 
2030 {
2031  switch_assert(channel);
2032  switch_assert(channel->flag_mutex);
2033 
2034  switch_mutex_lock(channel->flag_mutex);
2035  channel->flags[flag]++;
2036  switch_mutex_unlock(channel->flag_mutex);
2037 
2038  if (flag == CF_OUTBOUND) {
2039  switch_channel_set_variable(channel, "is_outbound", "true");
2040  }
2041 
2042  if (flag == CF_RECOVERED) {
2043  switch_channel_set_variable(channel, "recovered", "true");
2044  }
2045 }
2046 
2047 
2049 {
2050  switch_assert(channel != NULL);
2051  switch_mutex_lock(channel->flag_mutex);
2052  channel->private_flags |= flags;
2053  switch_mutex_unlock(channel->flag_mutex);
2054 }
2055 
2057 {
2058  switch_assert(channel != NULL);
2059  switch_mutex_lock(channel->flag_mutex);
2060  channel->private_flags &= ~flags;
2061  switch_mutex_unlock(channel->flag_mutex);
2062 }
2063 
2065 {
2066  switch_assert(channel != NULL);
2067  return (channel->private_flags & flags);
2068 }
2069 
2070 SWITCH_DECLARE(void) switch_channel_set_app_flag_key(const char *key, switch_channel_t *channel, uint32_t flags)
2071 {
2072  uint32_t *flagp = NULL;
2073  switch_byte_t new = 0;
2074 
2075  switch_assert(channel != NULL);
2076  switch_mutex_lock(channel->flag_mutex);
2077 
2078  if (!channel->app_flag_hash) {
2080  new++;
2081  }
2082 
2083  if (new || !(flagp = switch_core_hash_find(channel->app_flag_hash, key))) {
2084  flagp = switch_core_session_alloc(channel->session, sizeof(uint32_t));
2085  switch_core_hash_insert(channel->app_flag_hash, key, flagp);
2086  }
2087 
2088  switch_assert(flagp);
2089  *flagp |= flags;
2090 
2091  switch_mutex_unlock(channel->flag_mutex);
2092 }
2093 
2094 SWITCH_DECLARE(void) switch_channel_clear_app_flag_key(const char *key, switch_channel_t *channel, uint32_t flags)
2095 {
2096  uint32_t *flagp = NULL;
2097 
2098  switch_assert(channel != NULL);
2099  switch_mutex_lock(channel->flag_mutex);
2100  if (channel->app_flag_hash && (flagp = switch_core_hash_find(channel->app_flag_hash, key))) {
2101  if (!flags) {
2102  *flagp = 0;
2103  } else {
2104  *flagp &= ~flags;
2105  }
2106  }
2107  switch_mutex_unlock(channel->flag_mutex);
2108 }
2109 
2110 SWITCH_DECLARE(int) switch_channel_test_app_flag_key(const char *key, switch_channel_t *channel, uint32_t flags)
2111 {
2112  int r = 0;
2113  uint32_t *flagp = NULL;
2114  switch_assert(channel != NULL);
2115 
2116  switch_mutex_lock(channel->flag_mutex);
2117  if (channel->app_flag_hash && (flagp = switch_core_hash_find(channel->app_flag_hash, key))) {
2118  r = (*flagp & flags);
2119  }
2120  switch_mutex_unlock(channel->flag_mutex);
2121 
2122 
2123  return r;
2124 }
2125 
2127 {
2128  switch_assert(channel != NULL);
2129 
2130  switch_mutex_lock(channel->flag_mutex);
2131  channel->state_flags[0] = 1;
2132  channel->state_flags[flag] = 1;
2133  switch_mutex_unlock(channel->flag_mutex);
2134 }
2135 
2137 {
2138  switch_assert(channel != NULL);
2139 
2140  switch_mutex_lock(channel->flag_mutex);
2141  channel->state_flags[flag] = 0;
2142  switch_mutex_unlock(channel->flag_mutex);
2143 }
2144 
2146 {
2147  int ACTIVE = 0;
2148  int CLEAR = 0;
2149 
2150  switch_assert(channel != NULL);
2151  switch_assert(channel->flag_mutex);
2152 
2153  switch_mutex_lock(channel->flag_mutex);
2154  if (flag == CF_LEG_HOLDING && channel->flags[flag] && channel->flags[CF_ANSWERED]) {
2155  ACTIVE = 1;
2156  }
2157 
2158  if (flag == CF_VIDEO_PASSIVE && channel->flags[CF_VIDEO]) {
2159  channel->flags[CF_VIDEO_READY] = 1;
2160 
2161  if (channel->flags[flag]) {
2162  CLEAR = 1;
2163  }
2164  }
2165 
2166  channel->flags[flag] = 0;
2167  switch_mutex_unlock(channel->flag_mutex);
2168 
2169  if (flag == CF_DIALPLAN) {
2170  if (channel->direction == SWITCH_CALL_DIRECTION_OUTBOUND) {
2172  if (channel->device_node) {
2174  }
2175  }
2176  }
2177 
2178  if (ACTIVE) {
2180  switch_mutex_lock(channel->profile_mutex);
2181  if (channel->caller_profile->times->last_hold) {
2183  }
2184 
2185  if (channel->hold_record) {
2186  channel->hold_record->off = switch_time_now();
2187  }
2188 
2191  }
2192 
2194  }
2195 
2196  if (flag == CF_ORIGINATOR && switch_channel_test_flag(channel, CF_ANSWERED) && switch_channel_up_nosig(channel)) {
2198  }
2199 
2200  if (flag == CF_OUTBOUND) {
2201  switch_channel_set_variable(channel, "is_outbound", NULL);
2202  }
2203 
2204  if (flag == CF_RECOVERED) {
2205  switch_channel_set_variable(channel, "recovered", NULL);
2206  }
2207 
2208  if (flag == CF_VIDEO_PASSIVE && CLEAR) {
2210  }
2211 
2212  if (flag == CF_RECOVERING && !channel->hangup_cause && !switch_channel_test_flag(channel, CF_NO_RECOVER)) {
2214  }
2215 
2216 }
2217 
2218 
2220 {
2221  switch_assert(channel != NULL);
2222  switch_assert(channel->flag_mutex);
2223 
2224  switch_mutex_lock(channel->flag_mutex);
2225  if (channel->flags[flag]) {
2226  channel->flags[flag]--;
2227  }
2228  switch_mutex_unlock(channel->flag_mutex);
2229 
2230  if (flag == CF_OUTBOUND) {
2231  switch_channel_set_variable(channel, "is_outbound", NULL);
2232  }
2233 }
2234 
2236 {
2238  switch_assert(channel != NULL);
2239 
2240  state = channel->state;
2241 
2242  return state;
2243 }
2244 
2246 {
2248  switch_assert(channel != NULL);
2249 
2250  state = channel->running_state;
2251 
2252  return state;
2253 }
2254 
2256 {
2258  return 0;
2259  }
2260 
2261  return channel->running_state != channel->state;
2262 }
2263 
2265 {
2266  switch_ivr_parse_signal_data(channel->session, SWITCH_FALSE, in_thread_only);
2267  return 0;
2268 }
2269 
2271 {
2272  int ret = 0;
2273 
2274  switch_assert(channel != NULL);
2275 
2277 
2278  if (check_media) {
2279  ret = ((switch_channel_test_flag(channel, CF_ANSWERED) ||
2282 
2283 
2284  if (!ret)
2285  return ret;
2286  }
2287 
2288  if (!check_ready)
2289  return ret;
2290 
2291  ret = 0;
2292 
2293  if (!channel->hangup_cause && channel->state > CS_ROUTING && channel->state < CS_HANGUP && channel->state != CS_RESET &&
2296  ret++;
2297  }
2298 
2299 
2300 
2301  return ret;
2302 }
2303 
2304 static const char *state_names[] = {
2305  "CS_NEW",
2306  "CS_INIT",
2307  "CS_ROUTING",
2308  "CS_SOFT_EXECUTE",
2309  "CS_EXECUTE",
2310  "CS_EXCHANGE_MEDIA",
2311  "CS_PARK",
2312  "CS_CONSUME_MEDIA",
2313  "CS_HIBERNATE",
2314  "CS_RESET",
2315  "CS_HANGUP",
2316  "CS_REPORTING",
2317  "CS_DESTROY",
2318  "CS_NONE",
2319  NULL
2320 };
2321 
2323 {
2324  return state_names[state];
2325 }
2326 
2327 
2329 {
2330  uint32_t x = 0;
2331  for (x = 0; state_names[x]; x++) {
2332  if (!strcasecmp(state_names[x], name)) {
2333  return (switch_channel_state_t) x;
2334  }
2335  }
2336 
2337  return CS_DESTROY;
2338 }
2339 
2341 
2343  *state = val;
2345  } else {
2347  int x = 0;
2348 
2349  for (x = 0; x < 100; x++) {
2351  *state = val;
2352  switch_mutex_unlock(mutex);
2353  break;
2354  } else {
2355  switch_cond_next();
2356  }
2357  }
2358 
2359  if (x == 100) {
2360  *state = val;
2361  }
2362 
2363  }
2364 }
2365 
2367  const char *file, const char *func, int line)
2368 {
2369  int x;
2370 
2371  switch_mutex_lock(channel->flag_mutex);
2372  if (channel->state_flags[0]) {
2373  for (x = 1; x < CF_FLAG_MAX; x++) {
2374  if (channel->state_flags[x]) {
2375  channel->flags[x] = 1;
2376  channel->state_flags[x] = 0;
2377  }
2378  }
2379  channel->state_flags[0] = 0;
2380  }
2381  switch_mutex_unlock(channel->flag_mutex);
2382 
2384 
2385 
2386  switch_log_printf(SWITCH_CHANNEL_ID_LOG, file, func, line, switch_channel_get_uuid(channel), SWITCH_LOG_DEBUG, "(%s) Running State Change %s (Cur %d Tot %" SWITCH_SIZE_T_FMT ")\n",
2388 
2389  switch_mutex_lock(channel->state_mutex);
2390 
2391  careful_set(channel, &channel->running_state, state);
2392 
2393  if (state <= CS_DESTROY) {
2394  switch_event_t *event;
2395 
2397  if (state < CS_HANGUP) {
2398  if (state == CS_ROUTING) {
2400  } else if (switch_channel_test_flag(channel, CF_ANSWERED)) {
2402  } else if (switch_channel_test_flag(channel, CF_EARLY_MEDIA)) {
2404  }
2405  }
2406  }
2407 
2409  switch_channel_event_set_data(channel, event);
2410  switch_event_fire(&event);
2411  }
2412  }
2413 
2414  switch_mutex_unlock(channel->state_mutex);
2415 
2417 }
2418 
2420  const char *file, const char *func, int line, switch_channel_state_t state)
2421 {
2422  switch_channel_state_t last_state;
2423  int ok = 0;
2424 
2425  switch_assert(channel != NULL);
2426  switch_assert(state <= CS_DESTROY);
2427  switch_mutex_lock(channel->state_mutex);
2428 
2429  last_state = channel->state;
2430  switch_assert(last_state <= CS_DESTROY);
2431 
2432  if (last_state == state) {
2433  goto done;
2434  }
2435 
2436  if (last_state >= CS_HANGUP && state < last_state) {
2437  goto done;
2438  }
2439 
2440  /* STUB for more dev
2441  case CS_INIT:
2442  switch(state) {
2443 
2444  case CS_NEW:
2445  case CS_INIT:
2446  case CS_EXCHANGE_MEDIA:
2447  case CS_SOFT_EXECUTE:
2448  case CS_ROUTING:
2449  case CS_EXECUTE:
2450  case CS_HANGUP:
2451  case CS_DESTROY:
2452 
2453  default:
2454  break;
2455  }
2456  break;
2457  */
2458 
2459  switch (last_state) {
2460  case CS_NEW:
2461  case CS_RESET:
2462  switch (state) {
2463  default:
2464  ok++;
2465  break;
2466  }
2467  break;
2468 
2469  case CS_INIT:
2470  switch (state) {
2471  case CS_EXCHANGE_MEDIA:
2472  case CS_SOFT_EXECUTE:
2473  case CS_ROUTING:
2474  case CS_EXECUTE:
2475  case CS_PARK:
2476  case CS_CONSUME_MEDIA:
2477  case CS_HIBERNATE:
2478  case CS_RESET:
2479  ok++;
2480  default:
2481  break;
2482  }
2483  break;
2484 
2485  case CS_EXCHANGE_MEDIA:
2486  switch (state) {
2487  case CS_SOFT_EXECUTE:
2488  case CS_ROUTING:
2489  case CS_EXECUTE:
2490  case CS_PARK:
2491  case CS_CONSUME_MEDIA:
2492  case CS_HIBERNATE:
2493  case CS_RESET:
2494  ok++;
2495  default:
2496  break;
2497  }
2498  break;
2499 
2500  case CS_SOFT_EXECUTE:
2501  switch (state) {
2502  case CS_EXCHANGE_MEDIA:
2503  case CS_ROUTING:
2504  case CS_EXECUTE:
2505  case CS_PARK:
2506  case CS_CONSUME_MEDIA:
2507  case CS_HIBERNATE:
2508  case CS_RESET:
2509  ok++;
2510  default:
2511  break;
2512  }
2513  break;
2514 
2515  case CS_PARK:
2516  switch (state) {
2517  case CS_EXCHANGE_MEDIA:
2518  case CS_ROUTING:
2519  case CS_EXECUTE:
2520  case CS_SOFT_EXECUTE:
2521  case CS_HIBERNATE:
2522  case CS_RESET:
2523  case CS_CONSUME_MEDIA:
2524  ok++;
2525  default:
2526  break;
2527  }
2528  break;
2529 
2530  case CS_CONSUME_MEDIA:
2531  switch (state) {
2532  case CS_EXCHANGE_MEDIA:
2533  case CS_ROUTING:
2534  case CS_EXECUTE:
2535  case CS_SOFT_EXECUTE:
2536  case CS_HIBERNATE:
2537  case CS_RESET:
2538  case CS_PARK:
2539  ok++;
2540  default:
2541  break;
2542  }
2543  break;
2544  case CS_HIBERNATE:
2545  switch (state) {
2546  case CS_EXCHANGE_MEDIA:
2547  case CS_INIT:
2548  case CS_ROUTING:
2549  case CS_EXECUTE:
2550  case CS_SOFT_EXECUTE:
2551  case CS_PARK:
2552  case CS_CONSUME_MEDIA:
2553  case CS_RESET:
2554  ok++;
2555  default:
2556  break;
2557  }
2558  break;
2559 
2560  case CS_ROUTING:
2561  switch (state) {
2562  case CS_EXCHANGE_MEDIA:
2563  case CS_EXECUTE:
2564  case CS_SOFT_EXECUTE:
2565  case CS_PARK:
2566  case CS_CONSUME_MEDIA:
2567  case CS_HIBERNATE:
2568  case CS_RESET:
2569  ok++;
2570  default:
2571  break;
2572  }
2573  break;
2574 
2575  case CS_EXECUTE:
2576  switch (state) {
2577  case CS_EXCHANGE_MEDIA:
2578  case CS_SOFT_EXECUTE:
2579  case CS_ROUTING:
2580  case CS_PARK:
2581  case CS_CONSUME_MEDIA:
2582  case CS_HIBERNATE:
2583  case CS_RESET:
2584  ok++;
2585  default:
2586  break;
2587  }
2588  break;
2589 
2590  case CS_HANGUP:
2591  switch (state) {
2592  case CS_REPORTING:
2593  case CS_DESTROY:
2594  ok++;
2595  default:
2596  break;
2597  }
2598  break;
2599 
2600  case CS_REPORTING:
2601  switch (state) {
2602  case CS_DESTROY:
2603  ok++;
2604  default:
2605  break;
2606  }
2607  break;
2608 
2609  default:
2610  break;
2611 
2612  }
2613 
2614  if (ok) {
2615  switch_log_printf(SWITCH_CHANNEL_ID_LOG, file, func, line, switch_channel_get_uuid(channel), SWITCH_LOG_DEBUG, "(%s) State Change %s -> %s\n",
2616  channel->name, state_names[last_state], state_names[state]);
2617 
2618  careful_set(channel, &channel->state, state);
2619 
2620  if (state == CS_HANGUP && !channel->hangup_cause) {
2622  }
2623 
2624  if (state <= CS_DESTROY) {
2626  }
2627  } else {
2629  "(%s) Invalid State Change %s -> %s\n", channel->name, state_names[last_state], state_names[state]);
2630  /* we won't tolerate an invalid state change so we can make sure we are as robust as a nice cup of dark coffee! */
2631  /* not cool lets crash this bad boy and figure out wtf is going on */
2632  switch_assert(channel->state >= CS_HANGUP);
2633  }
2634  done:
2635 
2636  switch_mutex_unlock(channel->state_mutex);
2637  return channel->state;
2638 }
2639 
2641 {
2642  switch_mutex_lock(channel->thread_mutex);
2643 }
2644 
2645 
2647 {
2648  return switch_mutex_trylock(channel->thread_mutex);
2649 }
2650 
2651 
2653 {
2655 }
2656 
2658 {
2659  switch_caller_profile_t *caller_profile, *originator_caller_profile = NULL, *originatee_caller_profile = NULL;
2660  switch_codec_implementation_t impl = { 0 };
2661  char state_num[25];
2662  const char *v;
2663 
2664  switch_mutex_lock(channel->profile_mutex);
2665 
2666  if ((caller_profile = channel->caller_profile)) {
2667  originator_caller_profile = caller_profile->originator_caller_profile;
2668  originatee_caller_profile = caller_profile->originatee_caller_profile;
2669  }
2670 
2673  switch_snprintf(state_num, sizeof(state_num), "%d", channel->state);
2674  switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "Channel-State-Number", state_num);
2678 
2679  switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "Call-Direction",
2680  channel->direction == SWITCH_CALL_DIRECTION_OUTBOUND ? "outbound" : "inbound");
2681  switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "Presence-Call-Direction",
2682  channel->direction == SWITCH_CALL_DIRECTION_OUTBOUND ? "outbound" : "inbound");
2683 
2684  switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "Channel-HIT-Dialplan",
2686  switch_channel_test_flag(channel, CF_DIALPLAN) ? "true" : "false");
2687 
2688 
2689  if ((v = switch_channel_get_variable_dup(channel, "presence_id", SWITCH_FALSE, -1))) {
2690  switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "Channel-Presence-ID", v);
2691  }
2692 
2693  if ((v = switch_channel_get_variable_dup(channel, "presence_data", SWITCH_FALSE, -1))) {
2694  switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "Channel-Presence-Data", v);
2695  }
2696 
2697 
2698  if ((v = switch_channel_get_variable_dup(channel, "presence_data_cols", SWITCH_FALSE, -1))) {
2699  switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "Presence-Data-Cols", v);
2700  switch_event_add_presence_data_cols(channel, event, "PD-");
2701  }
2702 
2703  if ((v = switch_channel_get_variable_dup(channel, "call_uuid", SWITCH_FALSE, -1))) {
2704  switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "Channel-Call-UUID", v);
2705  } else {
2707  }
2708 
2709  if (switch_channel_down_nosig(channel)) {
2710  switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "Answer-State", "hangup");
2711  } else if (switch_channel_test_flag(channel, CF_ANSWERED)) {
2712  switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "Answer-State", "answered");
2713  } else if (switch_channel_test_flag(channel, CF_EARLY_MEDIA)) {
2714  switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "Answer-State", "early");
2715  } else {
2716  switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "Answer-State", "ringing");
2717  }
2718 
2719  if (channel->hangup_cause) {
2721  }
2722 
2723 
2724  switch_core_session_get_read_impl(channel->session, &impl);
2725 
2726  if (impl.iananame) {
2727  switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "Channel-Read-Codec-Name", impl.iananame);
2728  switch_event_add_header(event, SWITCH_STACK_BOTTOM, "Channel-Read-Codec-Rate", "%u", impl.actual_samples_per_second);
2729  switch_event_add_header(event, SWITCH_STACK_BOTTOM, "Channel-Read-Codec-Bit-Rate", "%d", impl.bits_per_second);
2730  }
2731 
2733 
2734  if (impl.iananame) {
2735  switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "Channel-Write-Codec-Name", impl.iananame);
2736  switch_event_add_header(event, SWITCH_STACK_BOTTOM, "Channel-Write-Codec-Rate", "%u", impl.actual_samples_per_second);
2737  switch_event_add_header(event, SWITCH_STACK_BOTTOM, "Channel-Write-Codec-Bit-Rate", "%d", impl.bits_per_second);
2738  }
2739 
2740  /* Index Caller's Profile */
2741  if (caller_profile) {
2742  switch_caller_profile_event_set_data(caller_profile, "Caller", event);
2743  }
2744 
2745  /* Index Originator/ee's Profile */
2746  if (originator_caller_profile && channel->last_profile_type == LP_ORIGINATOR) {
2747  switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "Other-Type", "originator");
2748  switch_caller_profile_event_set_data(originator_caller_profile, "Other-Leg", event);
2749  } else if (originatee_caller_profile && channel->last_profile_type == LP_ORIGINATEE) { /* Index Originatee's Profile */
2750  switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "Other-Type", "originatee");
2751  switch_caller_profile_event_set_data(originatee_caller_profile, "Other-Leg", event);
2752  }
2753 
2754 
2756 }
2757 
2759 {
2761  int global_verbose_events = -1;
2762 
2763  switch_mutex_lock(channel->profile_mutex);
2764 
2765  switch_core_session_ctl(SCSC_VERBOSE_EVENTS, &global_verbose_events);
2766 
2767  if (global_verbose_events ||
2769  switch_event_get_header(event, "presence-data-cols") ||
2772  event->event_id == SWITCH_EVENT_CHANNEL_UUID ||
2774  event->event_id == SWITCH_EVENT_CHANNEL_PARK ||
2783  event->event_id == SWITCH_EVENT_CHANNEL_DATA ||
2788  event->event_id == SWITCH_EVENT_API ||
2789  event->event_id == SWITCH_EVENT_RECORD_START ||
2790  event->event_id == SWITCH_EVENT_RECORD_STOP ||
2793  event->event_id == SWITCH_EVENT_CALL_UPDATE ||
2796  event->event_id == SWITCH_EVENT_CHANNEL_HOLD ||
2798  event->event_id == SWITCH_EVENT_TEXT ||
2799  event->event_id == SWITCH_EVENT_CUSTOM) {
2800 
2801  /* Index Variables */
2802 
2803  if (channel->scope_variables) {
2804  switch_event_t *ep;
2805 
2806  for (ep = channel->scope_variables; ep; ep = ep->next) {
2807  for (hi = ep->headers; hi; hi = hi->next) {
2808  char buf[1024];
2809  char *vvar = NULL, *vval = NULL;
2810 
2811  vvar = (char *) hi->name;
2812  vval = (char *) hi->value;
2813 
2814  switch_assert(vvar && vval);
2815  switch_snprintf(buf, sizeof(buf), "scope_variable_%s", vvar);
2816 
2817  if (!switch_event_get_header(event, buf)) {
2819  }
2820  }
2821  }
2822  }
2823 
2824  if (channel->variables) {
2825  for (hi = channel->variables->headers; hi; hi = hi->next) {
2826  char buf[1024];
2827  char *vvar = NULL, *vval = NULL;
2828 
2829  vvar = (char *) hi->name;
2830  vval = (char *) hi->value;
2831 
2832  switch_assert(vvar && vval);
2833  switch_snprintf(buf, sizeof(buf), "variable_%s", vvar);
2835  }
2836  }
2837  }
2838 
2840 }
2841 
2842 
2844 {
2845  switch_mutex_lock(channel->profile_mutex);
2846  switch_channel_event_set_basic_data(channel, event);
2849 }
2850 
2852 {
2854 
2855 
2856  switch_mutex_lock(channel->profile_mutex);
2857  cp = switch_caller_profile_clone(channel->session, channel->caller_profile);
2859 
2860  switch_channel_set_caller_profile(channel, cp);
2861 }
2862 
2864 {
2865  char *uuid = NULL;
2866  switch_assert(channel != NULL);
2867  switch_assert(channel->session != NULL);
2868  switch_mutex_lock(channel->profile_mutex);
2869  switch_assert(caller_profile != NULL);
2870 
2871  caller_profile->direction = channel->direction;
2872  caller_profile->logical_direction = channel->logical_direction;
2873  uuid = switch_core_session_get_uuid(channel->session);
2874 
2875  if (!caller_profile->uuid || strcasecmp(caller_profile->uuid, uuid)) {
2876  caller_profile->uuid = switch_core_session_strdup(channel->session, uuid);
2877  }
2878 
2879  if (!caller_profile->chan_name || strcasecmp(caller_profile->chan_name, channel->name)) {
2880  caller_profile->chan_name = switch_core_session_strdup(channel->session, channel->name);
2881  }
2882 
2883  if (!caller_profile->context) {
2884  caller_profile->context = switch_core_session_strdup(channel->session, "default");
2885  }
2886 
2887  if (!caller_profile->times) {
2888  caller_profile->times = (switch_channel_timetable_t *) switch_core_session_alloc(channel->session, sizeof(*caller_profile->times));
2889  caller_profile->times->profile_created = switch_micro_time_now();
2890  }
2891 
2892  if (channel->caller_profile && channel->caller_profile->times) {
2893  channel->caller_profile->times->transferred = caller_profile->times->profile_created;
2894  caller_profile->times->answered = channel->caller_profile->times->answered;
2895  caller_profile->times->progress = channel->caller_profile->times->progress;
2896  caller_profile->times->progress_media = channel->caller_profile->times->progress_media;
2897  caller_profile->times->created = channel->caller_profile->times->created;
2898  caller_profile->times->hungup = channel->caller_profile->times->hungup;
2899  if (channel->caller_profile->caller_extension) {
2900  switch_caller_extension_clone(&caller_profile->caller_extension, channel->caller_profile->caller_extension, caller_profile->pool);
2901  }
2902  } else {
2903  caller_profile->times->created = switch_micro_time_now();
2904  }
2905 
2906 
2907  caller_profile->next = channel->caller_profile;
2908  channel->caller_profile = caller_profile;
2909  caller_profile->profile_index = switch_core_sprintf(caller_profile->pool, "%d", ++channel->profile_index);
2910 
2912 }
2913 
2915 {
2916  switch_caller_profile_t *profile;
2917  switch_assert(channel != NULL);
2918  switch_mutex_lock(channel->profile_mutex);
2919  if ((profile = channel->caller_profile) && profile->hunt_caller_profile) {
2920  profile = profile->hunt_caller_profile;
2921  }
2923  return profile;
2924 }
2925 
2927 {
2928  switch_assert(channel != NULL);
2929  switch_assert(channel->caller_profile != NULL);
2930  switch_mutex_lock(channel->profile_mutex);
2931 
2932  if (!caller_profile->times) {
2933  caller_profile->times = (switch_channel_timetable_t *) switch_core_alloc(caller_profile->pool, sizeof(*caller_profile->times));
2934  }
2935 
2936  if (channel->caller_profile) {
2937  caller_profile->next = channel->caller_profile->originator_caller_profile;
2938  channel->caller_profile->originator_caller_profile = caller_profile;
2939  channel->last_profile_type = LP_ORIGINATOR;
2940  }
2943 }
2944 
2946 {
2947  switch_assert(channel != NULL);
2948  switch_assert(channel->caller_profile != NULL);
2949 
2950  switch_mutex_lock(channel->profile_mutex);
2951 
2952  channel->caller_profile->hunt_caller_profile = NULL;
2953  if (channel->caller_profile && caller_profile) {
2954  caller_profile->direction = channel->direction;
2955  caller_profile->logical_direction = channel->logical_direction;
2956  channel->caller_profile->hunt_caller_profile = caller_profile;
2957  }
2959 }
2960 
2962 {
2963  switch_assert(channel != NULL);
2964  switch_assert(channel->caller_profile != NULL);
2965 
2966  switch_mutex_lock(channel->profile_mutex);
2967 
2968  if (channel->caller_profile) {
2969  caller_profile->next = channel->caller_profile->origination_caller_profile;
2970  channel->caller_profile->origination_caller_profile = caller_profile;
2971  }
2974 }
2975 
2977 {
2978  switch_caller_profile_t *profile = NULL;
2979  switch_assert(channel != NULL);
2980 
2981  switch_mutex_lock(channel->profile_mutex);
2982  if (channel->caller_profile) {
2983  profile = channel->caller_profile->origination_caller_profile;
2984  }
2986 
2987  return profile;
2988 }
2989 
2990 
2992 {
2993  switch_assert(channel != NULL);
2994  switch_assert(channel->caller_profile != NULL);
2995 
2996  switch_mutex_lock(channel->profile_mutex);
2997 
2998  if (channel->caller_profile) {
2999  caller_profile->next = channel->caller_profile->originatee_caller_profile;
3000  channel->caller_profile->originatee_caller_profile = caller_profile;
3001  channel->last_profile_type = LP_ORIGINATEE;
3002  }
3005 }
3006 
3008 {
3009  switch_caller_profile_t *profile = NULL;
3010  switch_assert(channel != NULL);
3011 
3012  switch_mutex_lock(channel->profile_mutex);
3013 
3014  if (channel->caller_profile) {
3015  profile = channel->caller_profile->originator_caller_profile;
3016  }
3018 
3019  return profile;
3020 }
3021 
3023 {
3024  switch_caller_profile_t *profile = NULL;
3025  switch_assert(channel != NULL);
3026 
3027  switch_mutex_lock(channel->profile_mutex);
3028  if (channel->caller_profile) {
3029  profile = channel->caller_profile->originatee_caller_profile;
3030  }
3032 
3033  return profile;
3034 }
3035 
3037 {
3038  switch_assert(channel != NULL);
3039  switch_assert(channel->session != NULL);
3040  return switch_core_session_get_uuid(channel->session);
3041 }
3042 
3044 {
3045  int x, index;
3046 
3047  switch_assert(channel != NULL);
3048  switch_mutex_lock(channel->state_mutex);
3049  for (x = 0; x < SWITCH_MAX_STATE_HANDLERS; x++) {
3050  if (channel->state_handlers[x] == state_handler) {
3051  index = x;
3052  goto end;
3053  }
3054  }
3055  index = channel->state_handler_index++;
3056 
3057  if (channel->state_handler_index >= SWITCH_MAX_STATE_HANDLERS) {
3058  index = -1;
3059  goto end;
3060  }
3061 
3062  channel->state_handlers[index] = state_handler;
3063 
3064  end:
3065  switch_mutex_unlock(channel->state_mutex);
3066  return index;
3067 }
3068 
3070 {
3071  const switch_state_handler_table_t *h = NULL;
3072 
3073  switch_assert(channel != NULL);
3074 
3075  switch_mutex_lock(channel->state_mutex);
3076 
3077  if (index < SWITCH_MAX_STATE_HANDLERS && index <= channel->state_handler_index) {
3078  h = channel->state_handlers[index];
3079  }
3080 
3081  switch_mutex_unlock(channel->state_mutex);
3082 
3083  return h;
3084 }
3085 
3087 {
3088  int index, i;
3089  const switch_state_handler_table_t *new_handlers[SWITCH_MAX_STATE_HANDLERS] = { 0 };
3090 
3091  switch_assert(channel != NULL);
3092 
3093  switch_mutex_lock(channel->state_mutex);
3094  i = channel->state_handler_index;
3095  channel->state_handler_index = 0;
3096 
3097  if (state_handler) {
3098  for (index = 0; index < i; index++) {
3099  if (channel->state_handlers[index] != state_handler) {
3100  new_handlers[channel->state_handler_index++] = channel->state_handlers[index];
3101  }
3102  }
3103  } else {
3104  for (index = 0; index < i; index++) {
3105  if (channel->state_handlers[index] && switch_test_flag(channel->state_handlers[index], SSH_FLAG_STICKY)) {
3106  new_handlers[channel->state_handler_index++] = channel->state_handlers[index];
3107  }
3108  }
3109  }
3110 
3111  for (index = 0; index < SWITCH_MAX_STATE_HANDLERS; index++) {
3112  channel->state_handlers[index] = NULL;
3113  }
3114 
3115  if (channel->state_handler_index > 0) {
3116  for (index = 0; index < channel->state_handler_index; index++) {
3117  channel->state_handlers[index] = new_handlers[index];
3118  }
3119  }
3120 
3121  switch_mutex_unlock(channel->state_mutex);
3122 }
3123 
3125 {
3129 }
3130 
3131 /* XXX This is a somewhat simple operation. Were essentially taking the extension that one channel
3132  was executing and generating a new extension for another channel that starts out where the
3133  original one left off with an optional forward offset. Since all we are really doing is
3134  copying a few basic pool-allocated structures from one channel to another there really is
3135  not much to worry about here in terms of threading since we use read-write locks.
3136  While the features are nice, they only really are needed in one specific crazy attended
3137  transfer scenario where one channel was in the middle of calling a particular extension
3138  when it was rudely cut off by a transfer key press. XXX */
3139 
3141 {
3142  switch_caller_profile_t *caller_profile;
3143  switch_caller_extension_t *extension = NULL, *orig_extension = NULL;
3146  switch_event_header_t *hi = NULL;
3147  const char *no_copy = switch_channel_get_variable(orig_channel, "attended_transfer_no_copy");
3148  char *dup;
3149  int i, argc = 0;
3150  char *argv[128];
3151 
3152  if (no_copy) {
3153  dup = switch_core_session_strdup(new_channel->session, no_copy);
3154  argc = switch_separate_string(dup, ',', argv, (sizeof(argv) / sizeof(argv[0])));
3155  }
3156 
3157 
3158  switch_mutex_lock(orig_channel->profile_mutex);
3159  switch_mutex_lock(new_channel->profile_mutex);
3160 
3161 
3162  caller_profile = switch_caller_profile_clone(new_channel->session, new_channel->caller_profile);
3163  switch_assert(caller_profile);
3164  extension = switch_caller_extension_new(new_channel->session, caller_profile->destination_number, caller_profile->destination_number);
3165  orig_extension = switch_channel_get_caller_extension(orig_channel);
3166 
3167 
3168  if (extension && orig_extension) {
3169  for (ap = orig_extension->current_application; ap && offset > 0; offset--) {
3170  ap = ap->next;
3171  }
3172 
3173  for (; ap; ap = ap->next) {
3175  }
3176 
3177  caller_profile->destination_number = switch_core_strdup(caller_profile->pool, orig_channel->caller_profile->destination_number);
3178  switch_channel_set_caller_profile(new_channel, caller_profile);
3179  switch_channel_set_caller_extension(new_channel, extension);
3180 
3181  for (hi = orig_channel->variables->headers; hi; hi = hi->next) {
3182  int ok = 1;
3183  for (i = 0; i < argc; i++) {
3184  if (!strcasecmp(argv[i], hi->name)) {
3185  ok = 0;
3186  break;
3187  }
3188  }
3189 
3190  if (!ok)
3191  continue;
3192 
3193  switch_channel_set_variable(new_channel, hi->name, hi->value);
3194  }
3195 
3196  status = SWITCH_STATUS_SUCCESS;
3197  }
3198 
3199 
3200  switch_mutex_unlock(new_channel->profile_mutex);
3201  switch_mutex_unlock(orig_channel->profile_mutex);
3202 
3203 
3204  return status;
3205 }
3206 
3208 {
3209  const char *tname, *tnum;
3211 
3212  cp = switch_channel_get_caller_profile(channel);
3213 
3214  tname = cp->caller_id_name;
3215  tnum = cp->caller_id_number;
3216 
3217 #ifdef DEEP_DEBUG_CID
3219 #endif
3220 
3221  cp->caller_id_name = cp->callee_id_name;
3223 
3224  cp->callee_id_name = tname;
3225  cp->callee_id_number = tnum;
3226 
3227  if (zstr(cp->caller_id_name)) {
3228  cp->caller_id_name = "Unknown";
3229  }
3230 
3231  if (zstr(cp->caller_id_number)) {
3232  cp->caller_id_number = "Unknown";
3233  }
3234 }
3235 
3236 
3238 {
3239  switch_event_t *event;
3240  const char *tmp = NULL;
3241 
3242  switch_mutex_lock(channel->profile_mutex);
3243 
3244  if (switch_channel_test_flag(channel, CF_RECOVERING) && switch_true(switch_channel_get_variable(channel, "channel_cid_flipped"))) {
3246  return;
3247  }
3248 
3249  if (channel->caller_profile->callee_id_name) {
3250  tmp = channel->caller_profile->caller_id_name;
3251  switch_channel_set_variable(channel, "pre_transfer_caller_id_name", channel->caller_profile->caller_id_name);
3253  }
3254 
3255  if (switch_channel_test_flag(channel, CF_BRIDGED)) {
3257  } else if (tmp) {
3258  channel->caller_profile->callee_id_name = tmp;
3259  }
3260 
3261  if (channel->caller_profile->callee_id_number) {
3262  tmp = channel->caller_profile->caller_id_number;
3263  switch_channel_set_variable(channel, "pre_transfer_caller_id_number", channel->caller_profile->caller_id_number);
3265  }
3266 
3267  if (switch_channel_test_flag(channel, CF_BRIDGED)) {
3269  } else if (tmp) {
3270  channel->caller_profile->callee_id_number = tmp;
3271  }
3272 
3273  switch_channel_set_variable(channel, "channel_cid_flipped", "yes");
3274 
3276 
3277 
3279  const char *uuid = switch_channel_get_partner_uuid(channel);
3280  switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "Direction", "RECV");
3281 
3282  if (uuid) {
3283  switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "Bridged-To", uuid);
3284  }
3285  switch_channel_event_set_data(channel, event);
3286  switch_event_fire(&event);
3287  }
3288 
3289 
3290  switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(channel->session), SWITCH_LOG_INFO, "%s Flipping CID from \"%s\" <%s> to \"%s\" <%s>\n",
3291  switch_channel_get_name(channel),
3292  switch_str_nil(switch_channel_get_variable(channel, "pre_transfer_caller_id_name")),
3293  switch_str_nil(switch_channel_get_variable(channel, "pre_transfer_caller_id_number")),
3294  channel->caller_profile->caller_id_name,
3296  );
3297 
3298 }
3299 
3301 {
3302 
3304  switch_channel_flip_cid(channel);
3308  switch_channel_flip_cid(channel);
3309  }
3310 }
3311 
3313 {
3314  switch_caller_extension_t *caller_extension;
3315 
3316  switch_mutex_lock(channel->profile_mutex);
3317  caller_extension = channel->queued_extension;
3318  channel->queued_extension = NULL;
3320 
3321  return caller_extension;
3322 }
3323 
3325 {
3326  switch_mutex_lock(channel->profile_mutex);
3327  channel->queued_extension = caller_extension;
3329 
3332 }
3333 
3335 {
3336  switch_assert(channel != NULL);
3337 
3338  switch_channel_sort_cid(channel);
3339 
3340  switch_mutex_lock(channel->profile_mutex);
3341  caller_extension->next = channel->caller_profile->caller_extension;
3342  channel->caller_profile->caller_extension = caller_extension;
3344 }
3345 
3346 
3348 {
3349  switch_caller_extension_t *extension = NULL;
3350 
3351  switch_assert(channel != NULL);
3352  switch_mutex_lock(channel->profile_mutex);
3353  if (channel->caller_profile) {
3354  extension = channel->caller_profile->caller_extension;
3355  }
3357  return extension;
3358 }
3359 
3360 
3362 {
3363  switch_mutex_lock(channel->profile_mutex);
3364  if (channel->caller_profile && channel->caller_profile->times) {
3366  }
3368 }
3369 
3370 
3372 {
3373  if (channel->caller_profile && channel->caller_profile->times && !channel->caller_profile->times->hungup) {
3374  switch_mutex_lock(channel->profile_mutex);
3377  }
3378 }
3379 
3380 
3382  const char *file, const char *func, int line, switch_call_cause_t hangup_cause)
3383 {
3384  int ok = 0;
3385 
3386  switch_assert(channel != NULL);
3387 
3388  /* one per customer */
3389  switch_mutex_lock(channel->state_mutex);
3390  if (!(channel->opaque_flags & OCF_HANGUP)) {
3391  channel->opaque_flags |= OCF_HANGUP;
3392  ok = 1;
3393  }
3394  switch_mutex_unlock(channel->state_mutex);
3395 
3396  if (switch_channel_test_flag(channel, CF_LEG_HOLDING)) {
3399  }
3400 
3401  if (!ok) {
3402  return channel->state;
3403  }
3404 
3406 
3407  if (channel->state < CS_HANGUP) {
3408  switch_channel_state_t last_state;
3409  switch_event_t *event;
3410  const char *var;
3411 
3412 
3413  switch_mutex_lock(channel->profile_mutex);
3414  if (channel->hold_record && !channel->hold_record->off) {
3415  channel->hold_record->off = switch_time_now();
3416  }
3418 
3419  switch_mutex_lock(channel->state_mutex);
3420  last_state = channel->state;
3421  channel->state = CS_HANGUP;
3422  switch_mutex_unlock(channel->state_mutex);
3423 
3424  channel->hangup_cause = hangup_cause;
3425  switch_log_printf(SWITCH_CHANNEL_ID_LOG, file, func, line, switch_channel_get_uuid(channel), SWITCH_LOG_NOTICE, "Hangup %s [%s] [%s]\n",
3426  channel->name, state_names[last_state], switch_channel_cause2str(channel->hangup_cause));
3427 
3428 
3429  switch_channel_set_variable_partner(channel, "last_bridge_hangup_cause", switch_channel_cause2str(hangup_cause));
3430 
3433  }
3434 
3436  switch_channel_set_variable(channel, "last_bridge_role", "originator");
3437  } else if (switch_channel_test_flag(channel, CF_BRIDGED)) {
3438  switch_channel_set_variable(channel, "last_bridge_role", "originatee");
3439  }
3440 
3441 
3444  }
3445 
3447  switch_channel_event_set_data(channel, event);
3448  switch_event_fire(&event);
3449  }
3450 
3454  }
3455 
3456  return channel->state;
3457 }
3458 
3459 static switch_status_t send_ind(switch_channel_t *channel, switch_core_session_message_types_t msg_id, const char *file, const char *func, int line)
3460 {
3461  switch_core_session_message_t msg = { 0 };
3462 
3463  msg.message_id = msg_id;
3464  msg.from = channel->name;
3465  return switch_core_session_perform_receive_message(channel->session, &msg, file, func, line);
3466 }
3467 
3469  const char *file, const char *func, int line)
3470 {
3471  send_ind(channel, SWITCH_MESSAGE_INDICATE_ACKNOWLEDGE_CALL, file, func, line);
3472 
3473  return SWITCH_STATUS_SUCCESS;
3474 }
3475 
3478  const char *file, const char *func, int line)
3479 {
3480  switch_event_t *event;
3481 
3483  switch_log_printf(SWITCH_CHANNEL_ID_LOG, file, func, line, switch_channel_get_uuid(channel), SWITCH_LOG_NOTICE, "Ring-Ready %s!\n", channel->name);
3485 
3486  switch_mutex_lock(channel->profile_mutex);
3487  if (channel->caller_profile && channel->caller_profile->times && !channel->caller_profile->times->progress) {
3489  if (channel->caller_profile->originator_caller_profile) {
3490  switch_core_session_t *other_session;
3491  if ((other_session = switch_core_session_locate(channel->caller_profile->originator_caller_profile->uuid))) {
3492  switch_channel_t *other_channel;
3493  other_channel = switch_core_session_get_channel(other_session);
3494  switch_mutex_lock(other_channel->profile_mutex);
3495  if (other_channel->caller_profile && !other_channel->caller_profile->times->progress) {
3496  other_channel->caller_profile->times->progress = channel->caller_profile->times->progress;
3497  }
3498  switch_mutex_unlock(other_channel->profile_mutex);
3499  switch_core_session_rwunlock(other_session);
3500  }
3502  }
3503  }
3505 
3507  switch_channel_event_set_data(channel, event);
3508  switch_event_fire(&event);
3509  }
3510 
3513 
3515 
3516  send_ind(channel, SWITCH_MESSAGE_RING_EVENT, file, func, line);
3517 
3518  return SWITCH_STATUS_SUCCESS;
3519  }
3520 
3521  return SWITCH_STATUS_FALSE;
3522 }
3523 
3524 SWITCH_DECLARE(switch_status_t) switch_channel_perform_mark_pre_answered(switch_channel_t *channel, const char *file, const char *func, int line)
3525 {
3526  switch_event_t *event;
3527 
3529  const char *uuid;
3530  switch_core_session_t *other_session;
3531 
3533 
3534  switch_log_printf(SWITCH_CHANNEL_ID_LOG, file, func, line, switch_channel_get_uuid(channel), SWITCH_LOG_NOTICE, "Pre-Answer %s!\n", channel->name);
3536 
3538 
3539  if (switch_true(switch_channel_get_variable(channel, "video_mirror_input"))) {
3541  }
3542 
3543  if (channel->caller_profile && channel->caller_profile->times) {
3544  switch_mutex_lock(channel->profile_mutex);
3546  if (channel->caller_profile->originator_caller_profile) {
3547  switch_core_session_t *osession;
3549  switch_channel_t *other_channel;
3550  other_channel = switch_core_session_get_channel(osession);
3551  if (other_channel->caller_profile) {
3552  other_channel->caller_profile->times->progress_media = channel->caller_profile->times->progress_media;
3553  }
3554  switch_core_session_rwunlock(osession);
3555  }
3557  }
3559  }
3560 
3562  switch_channel_event_set_data(channel, event);
3563  switch_event_fire(&event);
3564  }
3565 
3568 
3571 
3574  }
3575 
3576 
3577  /* if we're the child of another channel and the other channel is in a blocking read they will never realize we have answered so send
3578  a SWITCH_SIG_BREAK to interrupt any blocking reads on that channel
3579  */
3581  && (other_session = switch_core_session_locate(uuid))) {
3583  switch_core_session_rwunlock(other_session);
3584  }
3585 
3587 
3588  send_ind(channel, SWITCH_MESSAGE_PROGRESS_EVENT, file, func, line);
3589 
3591 
3592  return SWITCH_STATUS_SUCCESS;
3593  }
3594 
3595  return SWITCH_STATUS_FALSE;
3596 }
3597 
3598 SWITCH_DECLARE(switch_status_t) switch_channel_perform_pre_answer(switch_channel_t *channel, const char *file, const char *func, int line)
3599 {
3600  switch_core_session_message_t msg = { 0 };
3602 
3603  switch_assert(channel != NULL);
3604 
3605  if (channel->hangup_cause || channel->state >= CS_HANGUP) {
3606  return SWITCH_STATUS_FALSE;
3607  }
3608 
3609  if (switch_channel_test_flag(channel, CF_ANSWERED)) {
3610  return SWITCH_STATUS_SUCCESS;
3611  }
3612 
3613  if (switch_channel_test_flag(channel, CF_EARLY_MEDIA)) {
3614  return SWITCH_STATUS_SUCCESS;
3615  }
3616 
3619  msg.from = channel->name;
3620  status = switch_core_session_perform_receive_message(channel->session, &msg, file, func, line);
3621  }
3622 
3623  if (status == SWITCH_STATUS_SUCCESS) {
3624  switch_channel_perform_mark_pre_answered(channel, file, func, line);
3625  switch_channel_audio_sync(channel);
3626  } else {
3628  }
3629 
3630  return status;
3631 }
3632 
3634  const char *file, const char *func, int line)
3635 {
3636  switch_core_session_message_t msg = { 0 };
3638 
3639  switch_assert(channel != NULL);
3640 
3641  if (channel->hangup_cause || channel->state >= CS_HANGUP) {
3642  return SWITCH_STATUS_FALSE;
3643  }
3644 
3645  if (switch_channel_test_flag(channel, CF_ANSWERED)) {
3646  return SWITCH_STATUS_SUCCESS;
3647  }
3648 
3649  if (switch_channel_test_flag(channel, CF_EARLY_MEDIA)) {
3650  return SWITCH_STATUS_SUCCESS;
3651  }
3652 
3655  msg.from = channel->name;
3656  msg.numeric_arg = rv;
3657  status = switch_core_session_perform_receive_message(channel->session, &msg, file, func, line);
3658  }
3659 
3660  if (status == SWITCH_STATUS_SUCCESS) {
3661  switch_log_printf(SWITCH_CHANNEL_ID_LOG, file, func, line, switch_channel_get_uuid(channel), SWITCH_LOG_NOTICE, "Ring Ready %s!\n", channel->name);
3662  switch_channel_perform_mark_ring_ready_value(channel, rv, file, func, line);
3663  } else {
3665  }
3666 
3667  return status;
3668 }
3669 
3670 static void do_api_on(switch_channel_t *channel, const char *variable)
3671 {
3672  char *app;
3673  char *arg = NULL;
3674  char *expanded = NULL;
3675  switch_stream_handle_t stream = { 0 };
3676 
3677  app = switch_core_session_strdup(channel->session, variable);
3678 
3679  if ((arg = strchr(app, ' '))) {
3680  *arg++ = '\0';
3681  }
3682 
3683  if (zstr(arg)) {
3684  expanded = arg;
3685  } else {
3686  expanded = switch_channel_expand_variables(channel, arg);
3687  }
3688 
3689  SWITCH_STANDARD_STREAM(stream);
3690  switch_api_execute(app, expanded, NULL, &stream);
3691  switch_log_printf(SWITCH_CHANNEL_CHANNEL_LOG(channel), SWITCH_LOG_DEBUG, "%s process %s: %s(%s)\n%s\n",
3692  channel->name, variable, app, switch_str_nil(expanded), (char *) stream.data);
3693 
3694  if (expanded && expanded != arg) {
3695  free(expanded);
3696  }
3697 
3698  free(stream.data);
3699 }
3700 
3701 
3703 {
3705  switch_event_t *event;
3706  int x = 0;
3707 
3708 
3709  switch_channel_get_variables(channel, &event);
3710 
3711  for (hp = event->headers; hp; hp = hp->next) {
3712  char *var = hp->name;
3713  char *val = hp->value;
3714 
3715  if (!strncasecmp(var, variable_prefix, strlen(variable_prefix))) {
3716  if (hp->idx) {
3717  int i;
3718  for (i = 0; i < hp->idx; i++) {
3719  x++;
3720  do_api_on(channel, hp->array[i]);
3721  }
3722  } else {
3723  x++;
3724  do_api_on(channel, val);
3725  }
3726  }
3727  }
3728 
3729  switch_event_destroy(&event);
3730 
3732 }
3733 
3735 {
3736  switch_status_t status;
3737  char *arg = NULL;
3738  char *p;
3739  int bg = 0;
3740  char *app;
3741  char *expanded = NULL;
3742 
3743  app = switch_core_session_strdup(channel->session, variable_value);
3744 
3745  for(p = app; p && *p; p++) {
3746  if (*p == ' ' || (*p == ':' && (*(p+1) != ':'))) {
3747  *p++ = '\0';
3748  arg = p;
3749  break;
3750  } else if (*p == ':' && (*(p+1) == ':')) {
3751  bg++;
3752  break;
3753  }
3754  }
3755 
3756  switch_assert(app != NULL);
3757  if (!strncasecmp(app, "perl", 4)) {
3758  bg++;
3759  }
3760 
3761  if (zstr(arg)) {
3762  expanded = arg;
3763  } else {
3764  expanded = switch_channel_expand_variables(channel, arg);
3765  }
3766 
3767  if (bg) {
3768  status = switch_core_session_execute_application_async(channel->session, app, arg);
3769  } else {
3770  status = switch_core_session_execute_application(channel->session, app, arg);
3771  }
3772 
3773  if (expanded && expanded != arg) {
3774  free(expanded);
3775  }
3776 
3777  return status;
3778 }
3779 
3781 {
3783  switch_event_t *event, *cevent;
3784  int x = 0;
3785 
3786  switch_core_get_variables(&event);
3787  switch_channel_get_variables(channel, &cevent);
3788  switch_event_merge(event, cevent);
3789 
3790  for (hp = event->headers; hp; hp = hp->next) {
3791  char *var = hp->name;
3792  char *val = hp->value;
3793 
3794  if (!strncasecmp(var, variable_prefix, strlen(variable_prefix))) {
3795  if (hp->idx) {
3796  int i;
3797  for (i = 0; i < hp->idx; i++) {
3798  x++;
3799  switch_channel_execute_on_value(channel, hp->array[i]);
3800  }
3801  } else {
3802  x++;
3803  switch_channel_execute_on_value(channel, val);
3804  }
3805  }
3806  }
3807 
3808  switch_event_destroy(&event);
3809  switch_event_destroy(&cevent);
3810 
3812 }
3813 
3814 SWITCH_DECLARE(switch_status_t) switch_channel_perform_mark_answered(switch_channel_t *channel, const char *file, const char *func, int line)
3815 {
3816  switch_event_t *event;
3817  const char *uuid;
3818  switch_core_session_t *other_session;
3819  const char *var;
3820 
3821  switch_assert(channel != NULL);
3822 
3823  if (channel->hangup_cause || channel->state >= CS_HANGUP) {
3824  return SWITCH_STATUS_FALSE;
3825  }
3826 
3827  if (switch_channel_test_flag(channel, CF_ANSWERED)) {
3828  return SWITCH_STATUS_SUCCESS;
3829  }
3830 
3832 
3833  if (channel->caller_profile && channel->caller_profile->times) {
3834  switch_mutex_lock(channel->profile_mutex);
3837  }
3838 
3840 
3841  if (switch_true(switch_channel_get_variable(channel, "video_mirror_input"))) {
3843  //switch_channel_set_flag(channel, CF_VIDEO_DECODED_READ);
3844  }
3845 
3846 
3848  switch_channel_event_set_data(channel, event);
3849  switch_event_fire(&event);
3850  }
3851 
3852  /* if we're the child of another channel and the other channel is in a blocking read they will never realize we have answered so send
3853  a SWITCH_SIG_BREAK to interrupt any blocking reads on that channel
3854  */
3856  && (other_session = switch_core_session_locate(uuid))) {
3858  switch_core_session_rwunlock(other_session);
3859  }
3860 
3863  }
3864 
3866  uint32_t seconds = 60;
3867  int tmp;
3868 
3869  if (switch_is_number(var)) {
3870  tmp = atoi(var);
3871  if (tmp > 0) {
3872  seconds = tmp;
3873  }
3874  } else if (!switch_true(var)) {
3875  seconds = 0;
3876  }
3877 
3878  if (seconds) {
3879  switch_core_session_enable_heartbeat(channel->session, seconds);
3880  }
3881  }
3882 
3884  switch_log_printf(SWITCH_CHANNEL_ID_LOG, file, func, line, switch_channel_get_uuid(channel), SWITCH_LOG_NOTICE, "Channel [%s] has been answered\n",
3885  channel->name);
3886 
3887 
3888  if (switch_channel_get_variable(channel, "absolute_codec_string")) {
3889  /* inherit_codec == true will implicitly clear the absolute_codec_string
3890  variable if used since it was the reason it was set in the first place and is no longer needed */
3891  if (switch_true(switch_channel_get_variable(channel, "inherit_codec"))) {
3892  switch_channel_set_variable(channel, "absolute_codec_string", NULL);
3893  }
3894  }
3895 
3897 
3898  if (!switch_channel_test_flag(channel, CF_EARLY_MEDIA)) {
3901  }
3902 
3904 
3905  switch_channel_presence(channel, "unknown", "answered", NULL);
3906 
3907  //switch_channel_audio_sync(channel);
3908 
3909  if (!switch_channel_test_flag(channel, CF_NO_RECOVER)) {
3911  }
3912 
3914 
3915  send_ind(channel, SWITCH_MESSAGE_ANSWER_EVENT, file, func, line);
3916 
3918 
3919  if (switch_channel_test_flag(channel, CF_RTT)) {
3921  }
3922 
3923  return SWITCH_STATUS_SUCCESS;
3924 }
3925 
3926 SWITCH_DECLARE(switch_status_t) switch_channel_perform_answer(switch_channel_t *channel, const char *file, const char *func, int line)
3927 {
3928  switch_core_session_message_t msg = { 0 };
3930 
3931  switch_assert(channel != NULL);
3932 
3934  return SWITCH_STATUS_SUCCESS;
3935  }
3936 
3937  if (channel->hangup_cause || channel->state >= CS_HANGUP) {
3938  return SWITCH_STATUS_FALSE;
3939  }
3940 
3941  if (switch_channel_test_flag(channel, CF_ANSWERED)) {
3942  return SWITCH_STATUS_SUCCESS;
3943  }
3944 
3946  msg.from = channel->name;
3947  status = switch_core_session_perform_receive_message(channel->session, &msg, file, func, line);
3948 
3949 
3950  if (status == SWITCH_STATUS_SUCCESS) {
3951  switch_channel_perform_mark_answered(channel, file, func, line);
3952  if (!switch_channel_test_flag(channel, CF_EARLY_MEDIA)) {
3953  switch_channel_audio_sync(channel);
3954  }
3955  } else {
3957  }
3958 
3959 
3962  const char *delay;
3963 
3964  if ((delay = switch_channel_get_variable(channel, "answer_delay"))) {
3965  uint32_t msec = atoi(delay);
3966 
3967  if (msec) {
3968  switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(channel->session), SWITCH_LOG_DEBUG, "Answer delay for %u msec\n", msec);
3969  switch_ivr_sleep(channel->session, msec, SWITCH_TRUE, NULL);
3970  }
3971  }
3972  }
3973 
3974  return status;
3975 }
3976 
3977 #define resize(l) {\
3978  char *dp;\
3979  olen += (len + l + block);\
3980  cpos = c - data;\
3981  if ((dp = realloc(data, olen))) {\
3982  data = dp;\
3983  c = data + cpos;\
3984  memset(c, 0, olen - cpos);\
3985  }} \
3986 
3987 SWITCH_DECLARE(char *) switch_channel_expand_variables_check(switch_channel_t *channel, const char *in, switch_event_t *var_list, switch_event_t *api_list, uint32_t recur)
3988 {
3989  char *p, *c = NULL;
3990  char *data, *indup, *endof_indup;
3991  size_t sp = 0, len = 0, olen = 0, vtype = 0, br = 0, cpos, block = 128;
3992  char *cloned_sub_val = NULL, *sub_val = NULL, *expanded_sub_val = NULL;
3993  char *func_val = NULL, *sb = NULL;
3994  int nv = 0;
3995 
3996  if (recur > 100) {
3997  return (char *) in;
3998  }
3999 
4000  if (zstr(in)) {
4001  return (char *) in;
4002  }
4003 
4005 
4006  if (!nv) {
4007  return (char *) in;
4008  }
4009 
4010 
4011  nv = 0;
4012  olen = strlen(in) + 1;
4013  indup = strdup(in);
4014  endof_indup = end_of_p(indup) + 1;
4015 
4016  if ((data = malloc(olen))) {
4017  memset(data, 0, olen);
4018  c = data;
4019  for (p = indup; p && p < endof_indup && *p; p++) {
4020  int global = 0;
4021  vtype = 0;
4022 
4023  if (*p == '\\') {
4024  if (*(p + 1) == '$') {
4025  nv = 1;
4026  p++;
4027  if (*(p + 1) == '$') {
4028  p++;
4029  }
4030  } else if (*(p + 1) == '\'') {
4031  p++;
4032  continue;
4033  } else if (*(p + 1) == '\\') {
4034  if (len + 1 >= olen) {
4035  resize(1);
4036  }
4037 
4038  *c++ = *p++;
4039  len++;
4040  continue;
4041  }
4042  }
4043 
4044  if (*p == '$' && !nv) {
4045 
4046  if (*(p + 1) == '$') {
4047  p++;
4048  global++;
4049  }
4050 
4051  if (*(p + 1)) {
4052  if (*(p + 1) == '{') {
4053  vtype = global ? 3 : 1;
4054  } else {
4055  nv = 1;
4056  }
4057  } else {
4058  nv = 1;
4059  }
4060  }
4061 
4062  if (nv) {
4063  if (len + 1 >= olen) {
4064  resize(1);
4065  }
4066 
4067  *c++ = *p;
4068  len++;
4069  nv = 0;
4070  continue;
4071  }
4072 
4073  if (vtype) {
4074  char *s = p, *e, *vname, *vval = NULL;
4075  size_t nlen;
4076 
4077  s++;
4078 
4079  if ((vtype == 1 || vtype == 3) && *s == '{') {
4080  br = 1;
4081  s++;
4082  }
4083 
4084  e = s;
4085  vname = s;
4086  while (*e) {
4087  if (br == 1 && *e == '}') {
4088  br = 0;
4089  *e++ = '\0';
4090  break;
4091  }
4092 
4093  if (br > 0) {
4094  if (e != s && *e == '{') {
4095  br++;
4096  } else if (br > 1 && *e == '}') {
4097  br--;
4098  }
4099  }
4100 
4101  e++;
4102  }
4103  p = e > endof_indup ? endof_indup : e;
4104 
4105  vval = NULL;
4106  for(sb = vname; sb && *sb; sb++) {
4107  if (*sb == ' ') {
4108  vval = sb;
4109  break;
4110  } else if (*sb == '(') {
4111  vval = sb;
4112  br = 1;
4113  break;
4114  }
4115  }
4116 
4117  if (vval) {
4118  e = vval - 1;
4119  *vval++ = '\0';
4120  while (*e == ' ') {
4121  *e-- = '\0';
4122  }
4123  e = vval;
4124 
4125  while (e && *e) {
4126  if (*e == '(') {
4127  br++;
4128  } else if (br > 1 && *e == ')') {
4129  br--;
4130  } else if (br == 1 && *e == ')') {
4131  *e = '\0';
4132  break;
4133  }
4134  e++;
4135  }
4136 
4137  vtype = 2;
4138  }
4139 
4140  if (vtype == 1 || vtype == 3) {
4141  char *expanded = NULL;
4142  int offset = 0;
4143  int ooffset = 0;
4144  char *ptr;
4145  int idx = -1;
4146 
4147  if ((expanded = switch_channel_expand_variables_check(channel, (char *) vname, var_list, api_list, recur+1)) == vname) {
4148  expanded = NULL;
4149  } else {
4150  vname = expanded;
4151  }
4152 
4153  if ((ptr = strchr(vname, ':'))) {
4154  *ptr++ = '\0';
4155  offset = atoi(ptr);
4156  if ((ptr = strchr(ptr, ':'))) {
4157  ptr++;
4158  ooffset = atoi(ptr);
4159  }
4160  }
4161 
4162  if ((ptr = strchr(vname, '[')) && strchr(ptr, ']')) {
4163  *ptr++ = '\0';
4164  idx = atoi(ptr);
4165  }
4166 
4167  if ((sub_val = (char *) switch_channel_get_variable_dup(channel, vname, SWITCH_TRUE, idx))) {
4168  if (var_list && !switch_event_check_permission_list(var_list, vname)) {
4169  sub_val = "<Variable Expansion Permission Denied>";
4170  }
4171 
4172  if ((expanded_sub_val = switch_channel_expand_variables_check(channel, sub_val, var_list, api_list, recur+1)) == sub_val) {
4173  expanded_sub_val = NULL;
4174  } else {
4175  sub_val = expanded_sub_val;
4176  }
4177 
4178  if (offset || ooffset) {
4179  cloned_sub_val = strdup(sub_val);
4180  switch_assert(cloned_sub_val);
4181  sub_val = cloned_sub_val;
4182  }
4183 
4184  if (offset >= 0) {
4185  if ((size_t) offset > strlen(sub_val)) {
4186  *sub_val = '\0';
4187  } else {
4188  sub_val += offset;
4189  }
4190  } else if ((size_t) abs(offset) <= strlen(sub_val)) {
4191  sub_val = cloned_sub_val + (strlen(cloned_sub_val) + offset);
4192  }
4193 
4194  if (ooffset > 0 && (size_t) ooffset < strlen(sub_val)) {
4195  if ((ptr = (char *) sub_val + ooffset)) {
4196  *ptr = '\0';
4197  }
4198  }
4199  }
4200 
4201  switch_safe_free(expanded);
4202  } else {
4203  switch_stream_handle_t stream = { 0 };
4204  char *expanded = NULL;
4205 
4206  SWITCH_STANDARD_STREAM(stream);
4207 
4208  if (stream.data) {
4209  char *expanded_vname = NULL;
4210 
4211  if ((expanded_vname = switch_channel_expand_variables_check(channel, (char *) vname, var_list, api_list, recur+1)) == vname) {
4212  expanded_vname = NULL;
4213  } else {
4214  vname = expanded_vname;
4215  }
4216 
4217  if ((expanded = switch_channel_expand_variables_check(channel, vval, var_list, api_list, recur+1)) == vval) {
4218  expanded = NULL;
4219  } else {
4220  vval = expanded;
4221  }
4222 
4223  if (!switch_core_test_flag(SCF_API_EXPANSION) || (api_list && !switch_event_check_permission_list(api_list, vname))) {
4224  func_val = NULL;
4225  sub_val = "<API Execute Permission Denied>";
4226  free(stream.data);
4227  } else {
4228  if (switch_api_execute(vname, vval, channel->session, &stream) == SWITCH_STATUS_SUCCESS) {
4229  func_val = stream.data;
4230  sub_val = func_val;
4231  } else {
4232  free(stream.data);
4233  }
4234  }
4235 
4236  switch_safe_free(expanded);
4237  switch_safe_free(expanded_vname);
4238 
4239  } else {
4240  switch_log_printf(SWITCH_CHANNEL_CHANNEL_LOG(channel), SWITCH_LOG_CRIT, "Memory Error!\n");
4241  free(data);
4242  free(indup);
4243  return (char *) in;
4244  }
4245  }
4246  if ((nlen = sub_val ? strlen(sub_val) : 0)) {
4247  if (len + nlen >= olen) {
4248  resize(nlen);
4249  }
4250 
4251  len += nlen;
4252  strcat(c, sub_val);
4253  c += nlen;
4254  }
4255 
4256  switch_safe_free(func_val);
4257  switch_safe_free(cloned_sub_val);
4258  switch_safe_free(expanded_sub_val);
4259  sub_val = NULL;
4260  vname = NULL;
4261  br = 0;
4262  }
4263 
4264  if (sp) {
4265  if (len + 1 >= olen) {
4266  resize(1);
4267  }
4268 
4269  *c++ = ' ';
4270  sp = 0;
4271  len++;
4272  }
4273 
4274  if (*p == '$') {
4275  p--;
4276  } else {
4277  if (len + 1 >= olen) {
4278  resize(1);
4279  }
4280 
4281  *c++ = *p;
4282  len++;
4283  }
4284  }
4285  }
4286  free(indup);
4287 
4288  return data;
4289 }
4290 
4292 {
4293  switch_stream_handle_t stream = { 0 };
4294  switch_size_t encode_len = 1024, new_len = 0;
4295  char *encode_buf = NULL;
4296  const char *prof[13] = { 0 }, *prof_names[13] = {
4297  0};
4298  char *e = NULL;
4300  uint32_t x = 0;
4301 
4302  SWITCH_STANDARD_STREAM(stream);
4303 
4304  if (prefix) {
4305  stream.write_function(&stream, "%s&", prefix);
4306  }
4307 
4308  encode_buf = malloc(encode_len);
4309  switch_assert(encode_buf);
4310 
4311  if (!caller_profile) {
4312  caller_profile = switch_channel_get_caller_profile(channel);
4313  }
4314 
4315  switch_assert(caller_profile != NULL);
4316 
4317  prof[0] = caller_profile->context;
4318  prof[1] = caller_profile->destination_number;
4319  prof[2] = caller_profile->caller_id_name;
4320  prof[3] = caller_profile->caller_id_number;
4321  prof[4] = caller_profile->network_addr;
4322  prof[5] = caller_profile->ani;
4323  prof[6] = caller_profile->aniii;
4324  prof[7] = caller_profile->rdnis;
4325  prof[8] = caller_profile->source;
4326  prof[9] = caller_profile->chan_name;
4327  prof[10] = caller_profile->uuid;
4328  prof[11] = caller_profile->transfer_source;
4329 
4330  prof_names[0] = "context";
4331  prof_names[1] = "destination_number";
4332  prof_names[2] = "caller_id_name";
4333  prof_names[3] = "caller_id_number";
4334  prof_names[4] = "network_addr";
4335  prof_names[5] = "ani";
4336  prof_names[6] = "aniii";
4337  prof_names[7] = "rdnis";
4338  prof_names[8] = "source";
4339  prof_names[9] = "chan_name";
4340  prof_names[10] = "uuid";
4341  prof_names[11] = "transfer_source";
4342 
4343  for (x = 0; prof[x]; x++) {
4344  if (zstr(prof[x])) {
4345  continue;
4346  }
4347  new_len = (strlen(prof[x]) * 3) + 1;
4348  if (encode_len < new_len) {
4349  char *tmp;
4350 
4351  encode_len = new_len;
4352 
4353  if (!(tmp = realloc(encode_buf, encode_len))) {
4354  abort();
4355  }
4356 
4357  encode_buf = tmp;
4358  }
4359  switch_url_encode(prof[x], encode_buf, encode_len);
4360  stream.write_function(&stream, "%s=%s&", prof_names[x], encode_buf);
4361  }
4362 
4363  if (channel->caller_profile->soft) {
4364  profile_node_t *pn;
4365 
4366  for(pn = channel->caller_profile->soft; pn; pn = pn->next) {
4367  char *var = pn->var;
4368  char *val = pn->val;
4369 
4370  new_len = (strlen((char *) var) * 3) + 1;
4371  if (encode_len < new_len) {
4372  char *tmp;
4373 
4374  encode_len = new_len;
4375 
4376  tmp = realloc(encode_buf, encode_len);
4377  switch_assert(tmp);
4378  encode_buf = tmp;
4379  }
4380 
4381  switch_url_encode((char *) val, encode_buf, encode_len);
4382  stream.write_function(&stream, "%s=%s&", (char *) var, encode_buf);
4383 
4384  }
4385  }
4386 
4387  if ((hi = switch_channel_variable_first(channel))) {
4388  for (; hi; hi = hi->next) {
4389  char *var = hi->name;
4390  char *val = hi->value;
4391 
4392  new_len = (strlen((char *) var) * 3) + 1;
4393  if (encode_len < new_len) {
4394  char *tmp;
4395 
4396  encode_len = new_len;
4397 
4398  tmp = realloc(encode_buf, encode_len);
4399  switch_assert(tmp);
4400  encode_buf = tmp;
4401  }
4402 
4403  switch_url_encode((char *) val, encode_buf, encode_len);
4404  stream.write_function(&stream, "%s=%s&", (char *) var, encode_buf);
4405 
4406  }
4408  }
4409 
4410  e = (char *) stream.data + (strlen((char *) stream.data) - 1);
4411 
4412  if (e && *e == '&') {
4413  *e = '\0';
4414  }
4415 
4416  switch_safe_free(encode_buf);
4417 
4418  return stream.data;
4419 }
4420 
4422 {
4423  int x = 0;
4424 
4425  switch_assert(channel);
4426  switch_assert(other_channel);
4427 
4428  switch_mutex_lock(channel->profile_mutex);
4429  switch_mutex_lock(other_channel->profile_mutex);
4430 
4431  if (!zstr(channel->caller_profile->callee_id_name)) {
4433  x++;
4434  }
4435 
4436  if (!zstr(channel->caller_profile->callee_id_number)) {
4438  x++;
4439  }
4440 
4441  switch_mutex_unlock(other_channel->profile_mutex);
4443 
4445 }
4446 
4448 {
4449  switch_status_t status;
4450  switch_mutex_lock(channel->profile_mutex);
4451  if (channel->variables) {
4452  status = switch_event_dup(event, channel->variables);
4453  } else {
4455  }
4457  return status;
4458 }
4459 
4461 {
4463  switch_event_t *vars;
4464 
4466 
4467  switch_mutex_lock(channel->profile_mutex);
4468  if (channel->variables) {
4470 
4471  for (hi = channel->variables->headers; hi; hi = hi->next) {
4472  if (!strncmp(hi->name, prefix, strlen(prefix))) {
4474  }
4475  }
4476  }
4478 
4479  *event = vars;
4480  return status;
4481 }
4482 
4484 {
4485  switch_assert(channel);
4486  return channel->session;
4487 }
4488 
4490 {
4492  const char *cid_buf = NULL;
4493  switch_caller_profile_t *caller_profile;
4494  switch_app_log_t *app_log, *ap;
4495  char *last_app = NULL, *last_arg = NULL;
4496  char start[80] = "", resurrect[80] = "", answer[80] = "", hold[80],
4497  bridge[80] = "", progress[80] = "", progress_media[80] = "", end[80] = "", tmp[80] = "",
4498  profile_start[80] = "";
4499  int32_t duration = 0, legbillsec = 0, billsec = 0, mduration = 0, billmsec = 0, legbillmsec = 0, progressmsec = 0, progress_mediamsec = 0;
4500  int32_t answersec = 0, answermsec = 0, waitsec = 0, waitmsec = 0;
4501  switch_time_t answerusec = 0;
4502  switch_time_t uduration = 0, legbillusec = 0, billusec = 0, progresssec = 0, progressusec = 0, progress_mediasec = 0, progress_mediausec = 0, waitusec = 0;
4503  time_t tt_created = 0, tt_answered = 0, tt_resurrected = 0, tt_bridged, tt_last_hold, tt_hold_accum,
4504  tt_progress = 0, tt_progress_media = 0, tt_hungup = 0, mtt_created = 0, mtt_answered = 0, mtt_bridged = 0,
4505  mtt_hungup = 0, tt_prof_created, mtt_progress = 0, mtt_progress_media = 0;
4506  void *pop;
4507  char dtstr[SWITCH_DTMF_LOG_LEN + 1] = "";
4508  int x = 0;
4509 
4510  switch_mutex_lock(channel->profile_mutex);
4511 
4514  return SWITCH_STATUS_FALSE;
4515  }
4516 
4517  if (!(caller_profile = channel->caller_profile) || !channel->variables) {
4519  return SWITCH_STATUS_FALSE;
4520  }
4521 
4523 
4524  if ((app_log = switch_core_session_get_app_log(channel->session))) {
4525  for (ap = app_log; ap && ap->next; ap = ap->next);
4526  last_app = ap->app;
4527  last_arg = ap->arg;
4528  }
4529 
4530  if (!zstr(caller_profile->caller_id_name)) {
4531  cid_buf = switch_core_session_sprintf(channel->session, "\"%s\" <%s>", caller_profile->caller_id_name,
4532  switch_str_nil(caller_profile->caller_id_number));
4533  } else {
4534  cid_buf = caller_profile->caller_id_number;
4535  }
4536 
4538  switch_dtmf_t *dt = (switch_dtmf_t *) pop;
4539 
4540  if (dt) {
4541  dtstr[x++] = dt->digit;
4542  free(dt);
4543  dt = NULL;
4544  }
4545  }
4546 
4547  if (x) {
4548  const char *var = switch_channel_get_variable(channel, "digits_dialed_filter");
4549  char *digit_string = dtstr;
4550  char *X = NULL;
4551  switch_regex_t *re = NULL;
4552  char *substituted = NULL;
4553 
4554  if (!zstr(var)) {
4555  int proceed = 0;
4556  int ovector[30];
4557 
4558  if ((proceed = switch_regex_perform(dtstr, var, &re, ovector, sizeof(ovector) / sizeof(ovector[0])))) {
4559  int len = (strlen(dtstr) + strlen(var) + 10) * proceed;
4560  int i = 0;
4561  const char *replace = NULL;
4562 
4563  X = malloc(len);
4564 
4565  for (i = 0; i < proceed; i++) {
4566  if (pcre_get_substring(dtstr, ovector, proceed, i, &replace) >= 0) {
4567  if (replace) {
4568  switch_size_t plen = strlen(replace);
4569  memset(X, 'X', plen);
4570  *(X+plen) = '\0';
4571 
4572  switch_safe_free(substituted);
4573  substituted = switch_string_replace(substituted ? substituted : dtstr, replace, X);
4574 
4575  pcre_free_substring(replace);
4576  }
4577  }
4578  }
4579 
4580  if (!zstr(substituted)) {
4581  digit_string = substituted;
4582  }
4583  }
4584  }
4585 
4586  switch_channel_set_variable(channel, "digits_dialed", digit_string);
4588  switch_safe_free(substituted);
4589  switch_safe_free(X);
4590  } else {
4591  switch_channel_set_variable(channel, "digits_dialed", "none");
4592  }
4593 
4594  if (caller_profile->times) {
4595  switch_time_exp_t tm;
4596  switch_size_t retsize;
4597  const char *fmt = "%Y-%m-%d %T";
4598 
4599  switch_time_exp_lt(&tm, caller_profile->times->created);
4600  switch_strftime_nocheck(start, &retsize, sizeof(start), fmt, &tm);
4601  switch_channel_set_variable(channel, "start_stamp", start);
4602 
4603  switch_time_exp_lt(&tm, caller_profile->times->profile_created);
4604  switch_strftime_nocheck(profile_start, &retsize, sizeof(profile_start), fmt, &tm);
4605  switch_channel_set_variable(channel, "profile_start_stamp", profile_start);
4606 
4607  if (caller_profile->times->answered) {
4608  switch_time_exp_lt(&tm, caller_profile->times->answered);
4609  switch_strftime_nocheck(answer, &retsize, sizeof(answer), fmt, &tm);
4610  switch_channel_set_variable(channel, "answer_stamp", answer);
4611  }
4612 
4613  if (caller_profile->times->bridged) {
4614  switch_time_exp_lt(&tm, caller_profile->times->bridged);
4615  switch_strftime_nocheck(bridge, &retsize, sizeof(bridge), fmt, &tm);
4616  switch_channel_set_variable(channel, "bridge_stamp", bridge);
4617  }
4618 
4619  if (caller_profile->times->last_hold) {
4620  switch_time_exp_lt(&tm, caller_profile->times->last_hold);
4621  switch_strftime_nocheck(hold, &retsize, sizeof(hold), fmt, &tm);
4622  switch_channel_set_variable(channel, "hold_stamp", hold);
4623  }
4624 
4625  if (caller_profile->times->resurrected) {
4626  switch_time_exp_lt(&tm, caller_profile->times->resurrected);
4627  switch_strftime_nocheck(resurrect, &retsize, sizeof(resurrect), fmt, &tm);
4628  switch_channel_set_variable(channel, "resurrect_stamp", resurrect);
4629  }
4630 
4631  if (caller_profile->times->progress) {
4632  switch_time_exp_lt(&tm, caller_profile->times->progress);
4633  switch_strftime_nocheck(progress, &retsize, sizeof(progress), fmt, &tm);
4634  switch_channel_set_variable(channel, "progress_stamp", progress);
4635  }
4636 
4637  if (caller_profile->times->progress_media) {
4638  switch_time_exp_lt(&tm, caller_profile->times->progress_media);
4639  switch_strftime_nocheck(progress_media, &retsize, sizeof(progress_media), fmt, &tm);
4640  switch_channel_set_variable(channel, "progress_media_stamp", progress_media);
4641  }
4642 
4643  if (channel->hold_record) {
4645  switch_stream_handle_t stream = { 0 };
4646 
4647  SWITCH_STANDARD_STREAM(stream);
4648 
4649  stream.write_function(&stream, "{", SWITCH_VA_NONE);
4650 
4651  for (hr = channel->hold_record; hr; hr = hr->next) {
4652  stream.write_function(&stream, "{%"SWITCH_TIME_T_FMT",%"SWITCH_TIME_T_FMT"},", hr->on, hr->off);
4653  }
4654  end_of((char *)stream.data) = '}';
4655 
4656  switch_channel_set_variable(channel, "hold_events", (char *)stream.data);
4657  free(stream.data);
4658  }
4659 
4660  switch_time_exp_lt(&tm, caller_profile->times->hungup);
4661  switch_strftime_nocheck(end, &retsize, sizeof(end), fmt, &tm);
4662  switch_channel_set_variable(channel, "end_stamp", end);
4663 
4664  tt_created = (time_t) (caller_profile->times->created / 1000000);
4665  mtt_created = (time_t) (caller_profile->times->created / 1000);
4666  switch_snprintf(tmp, sizeof(tmp), "%" TIME_T_FMT, tt_created);
4667  switch_channel_set_variable(channel, "start_epoch", tmp);
4668  switch_snprintf(tmp, sizeof(tmp), "%" SWITCH_TIME_T_FMT, caller_profile->times->created);
4669  switch_channel_set_variable(channel, "start_uepoch", tmp);
4670 
4671  tt_prof_created = (time_t) (caller_profile->times->profile_created / 1000000);
4672  switch_snprintf(tmp, sizeof(tmp), "%" TIME_T_FMT, tt_prof_created);
4673  switch_channel_set_variable(channel, "profile_start_epoch", tmp);
4674  switch_snprintf(tmp, sizeof(tmp), "%" SWITCH_TIME_T_FMT, caller_profile->times->profile_created);
4675  switch_channel_set_variable(channel, "profile_start_uepoch", tmp);
4676 
4677  tt_answered = (time_t) (caller_profile->times->answered / 1000000);
4678  mtt_answered = (time_t) (caller_profile->times->answered / 1000);
4679  switch_snprintf(tmp, sizeof(tmp), "%" TIME_T_FMT, tt_answered);
4680  switch_channel_set_variable(channel, "answer_epoch", tmp);
4681  switch_snprintf(tmp, sizeof(tmp), "%" SWITCH_TIME_T_FMT, caller_profile->times->answered);
4682  switch_channel_set_variable(channel, "answer_uepoch", tmp);
4683 
4684  tt_bridged = (time_t) (caller_profile->times->bridged / 1000000);
4685  mtt_bridged = (time_t) (caller_profile->times->bridged / 1000);
4686  switch_snprintf(tmp, sizeof(tmp), "%" TIME_T_FMT, tt_bridged);
4687  switch_channel_set_variable(channel, "bridge_epoch", tmp);
4688  switch_snprintf(tmp, sizeof(tmp), "%" SWITCH_TIME_T_FMT, caller_profile->times->bridged);
4689  switch_channel_set_variable(channel, "bridge_uepoch", tmp);
4690 
4691  tt_last_hold = (time_t) (caller_profile->times->last_hold / 1000000);
4692  switch_snprintf(tmp, sizeof(tmp), "%" TIME_T_FMT, tt_last_hold);
4693  switch_channel_set_variable(channel, "last_hold_epoch", tmp);
4694  switch_snprintf(tmp, sizeof(tmp), "%" SWITCH_TIME_T_FMT, caller_profile->times->last_hold);
4695  switch_channel_set_variable(channel, "last_hold_uepoch", tmp);
4696 
4697  tt_hold_accum = (time_t) (caller_profile->times->hold_accum / 1000000);
4698  switch_snprintf(tmp, sizeof(tmp), "%" TIME_T_FMT, tt_hold_accum);
4699  switch_channel_set_variable(channel, "hold_accum_seconds", tmp);
4700  switch_snprintf(tmp, sizeof(tmp), "%" SWITCH_TIME_T_FMT, caller_profile->times->hold_accum);
4701  switch_channel_set_variable(channel, "hold_accum_usec", tmp);
4702  switch_snprintf(tmp, sizeof(tmp), "%" SWITCH_TIME_T_FMT, caller_profile->times->hold_accum / 1000);
4703  switch_channel_set_variable(channel, "hold_accum_ms", tmp);
4704 
4705  tt_resurrected = (time_t) (caller_profile->times->resurrected / 1000000);
4706  switch_snprintf(tmp, sizeof(tmp), "%" TIME_T_FMT, tt_resurrected);
4707  switch_channel_set_variable(channel, "resurrect_epoch", tmp);
4708  switch_snprintf(tmp, sizeof(tmp), "%" SWITCH_TIME_T_FMT, caller_profile->times->resurrected);
4709  switch_channel_set_variable(channel, "resurrect_uepoch", tmp);
4710 
4711  tt_progress = (time_t) (caller_profile->times->progress / 1000000);
4712  mtt_progress = (time_t) (caller_profile->times->progress / 1000);
4713  switch_snprintf(tmp, sizeof(tmp), "%" TIME_T_FMT, tt_progress);
4714  switch_channel_set_variable(channel, "progress_epoch", tmp);
4715  switch_snprintf(tmp, sizeof(tmp), "%" SWITCH_TIME_T_FMT, caller_profile->times->progress);
4716  switch_channel_set_variable(channel, "progress_uepoch", tmp);
4717 
4718  tt_progress_media = (time_t) (caller_profile->times->progress_media / 1000000);
4719  mtt_progress_media = (time_t) (caller_profile->times->progress_media / 1000);
4720  switch_snprintf(tmp, sizeof(tmp), "%" TIME_T_FMT, tt_progress_media);
4721  switch_channel_set_variable(channel, "progress_media_epoch", tmp);
4722  switch_snprintf(tmp, sizeof(tmp), "%" SWITCH_TIME_T_FMT, caller_profile->times->progress_media);
4723  switch_channel_set_variable(channel, "progress_media_uepoch", tmp);
4724 
4725  tt_hungup = (time_t) (caller_profile->times->hungup / 1000000);
4726  mtt_hungup = (time_t) (caller_profile->times->hungup / 1000);
4727  switch_snprintf(tmp, sizeof(tmp), "%" TIME_T_FMT, tt_hungup);
4728  switch_channel_set_variable(channel, "end_epoch", tmp);
4729  switch_snprintf(tmp, sizeof(tmp), "%" SWITCH_TIME_T_FMT, caller_profile->times->hungup);
4730  switch_channel_set_variable(channel, "end_uepoch", tmp);
4731 
4732  duration = (int32_t) (tt_hungup - tt_created);
4733  mduration = (int32_t) (mtt_hungup - mtt_created);
4734  uduration = caller_profile->times->hungup - caller_profile->times->created;
4735 
4736  if (caller_profile->times->bridged > caller_profile->times->created) {
4737  waitsec = (int32_t) (tt_bridged - tt_created);
4738  waitmsec = (int32_t) (mtt_bridged - mtt_created);
4739  waitusec = caller_profile->times->bridged - caller_profile->times->created;
4740  } else {
4741  waitsec = 0;
4742  waitmsec = 0;
4743  waitusec = 0;
4744  }
4745 
4746  if (caller_profile->times->answered) {
4747  billsec = (int32_t) (tt_hungup - tt_answered);
4748  billmsec = (int32_t) (mtt_hungup - mtt_answered);
4749  billusec = caller_profile->times->hungup - caller_profile->times->answered;
4750 
4751  legbillsec = (int32_t) (tt_hungup - tt_created);
4752  legbillmsec = (int32_t) (mtt_hungup - mtt_created);
4753  legbillusec = caller_profile->times->hungup - caller_profile->times->created;
4754 
4755  answersec = (int32_t) (tt_answered - tt_created);
4756  answermsec = (int32_t) (mtt_answered - mtt_created);
4757  answerusec = caller_profile->times->answered - caller_profile->times->created;
4758  }
4759 
4760  if (caller_profile->times->progress) {
4761  progresssec = (int32_t) (tt_progress - tt_created);
4762  progressmsec = (int32_t) (mtt_progress - mtt_created);
4763  progressusec = caller_profile->times->progress - caller_profile->times->created;
4764  }
4765 
4766  if (caller_profile->times->progress_media) {
4767  progress_mediasec = (int32_t) (tt_progress_media - tt_created);
4768  progress_mediamsec = (int32_t) (mtt_progress_media - mtt_created);
4769  progress_mediausec = caller_profile->times->progress_media - caller_profile->times->created;
4770  }
4771 
4772  }
4773 
4774  switch_channel_set_variable(channel, "last_app", last_app);
4775  switch_channel_set_variable(channel, "last_arg", last_arg);
4776  switch_channel_set_variable(channel, "caller_id", cid_buf);
4777 
4778  switch_snprintf(tmp, sizeof(tmp), "%d", duration);
4779  switch_channel_set_variable(channel, "duration", tmp);
4780 
4781  switch_snprintf(tmp, sizeof(tmp), "%d", billsec);
4782  switch_channel_set_variable(channel, "billsec", tmp);
4783 
4784  switch_snprintf(tmp, sizeof(tmp), "%"SWITCH_TIME_T_FMT, progresssec);
4785  switch_channel_set_variable(channel, "progresssec", tmp);
4786 
4787  switch_snprintf(tmp, sizeof(tmp), "%d", answersec);
4788  switch_channel_set_variable(channel, "answersec", tmp);
4789 
4790  switch_snprintf(tmp, sizeof(tmp), "%d", waitsec);
4791  switch_channel_set_variable(channel, "waitsec", tmp);
4792 
4793  switch_snprintf(tmp, sizeof(tmp), "%"SWITCH_TIME_T_FMT, progress_mediasec);
4794  switch_channel_set_variable(channel, "progress_mediasec", tmp);
4795 
4796  switch_snprintf(tmp, sizeof(tmp), "%d", legbillsec);
4797  switch_channel_set_variable(channel, "flow_billsec", tmp);
4798 
4799  switch_snprintf(tmp, sizeof(tmp), "%d", mduration);
4800  switch_channel_set_variable(channel, "mduration", tmp);
4801 
4802  switch_snprintf(tmp, sizeof(tmp), "%d", billmsec);
4803  switch_channel_set_variable(channel, "billmsec", tmp);
4804 
4805  switch_snprintf(tmp, sizeof(tmp), "%d", progressmsec);
4806  switch_channel_set_variable(channel, "progressmsec", tmp);
4807 
4808  switch_snprintf(tmp, sizeof(tmp), "%d", answermsec);
4809  switch_channel_set_variable(channel, "answermsec", tmp);
4810 
4811  switch_snprintf(tmp, sizeof(tmp), "%d", waitmsec);
4812  switch_channel_set_variable(channel, "waitmsec", tmp);
4813 
4814  switch_snprintf(tmp, sizeof(tmp), "%d", progress_mediamsec);
4815  switch_channel_set_variable(channel, "progress_mediamsec", tmp);
4816 
4817  switch_snprintf(tmp, sizeof(tmp), "%d", legbillmsec);
4818  switch_channel_set_variable(channel, "flow_billmsec", tmp);
4819 
4820  switch_snprintf(tmp, sizeof(tmp), "%" SWITCH_TIME_T_FMT, uduration);
4821  switch_channel_set_variable(channel, "uduration", tmp);
4822 
4823  switch_snprintf(tmp, sizeof(tmp), "%" SWITCH_TIME_T_FMT, billusec);
4824  switch_channel_set_variable(channel, "billusec", tmp);
4825 
4826  switch_snprintf(tmp, sizeof(tmp), "%" SWITCH_TIME_T_FMT, progressusec);
4827  switch_channel_set_variable(channel, "progressusec", tmp);
4828 
4829  switch_snprintf(tmp, sizeof(tmp), "%" SWITCH_TIME_T_FMT, answerusec);
4830  switch_channel_set_variable(channel, "answerusec", tmp);
4831 
4832  switch_snprintf(tmp, sizeof(tmp), "%" SWITCH_TIME_T_FMT, waitusec);
4833  switch_channel_set_variable(channel, "waitusec", tmp);
4834 
4835  switch_snprintf(tmp, sizeof(tmp), "%" SWITCH_TIME_T_FMT, progress_mediausec);
4836  switch_channel_set_variable(channel, "progress_mediausec", tmp);
4837 
4838  switch_snprintf(tmp, sizeof(tmp), "%" SWITCH_TIME_T_FMT, legbillusec);
4839  switch_channel_set_variable(channel, "flow_billusec", tmp);
4840 
4842 
4843  return status;
4844 }
4845 
4847 {
4848  const char *uuid = NULL;
4849 
4850  switch_mutex_lock(channel->profile_mutex);
4853  }
4854 
4855  if (uuid) {
4856  strncpy(buf, uuid, blen);
4857  uuid = (const char *) buf;
4858  }
4860 
4861  return uuid;
4862 }
4863 
4864 
4866 {
4867  const char *uuid = NULL;
4868 
4869  if (!(uuid = switch_channel_get_variable(channel, SWITCH_SIGNAL_BOND_VARIABLE))) {
4871  }
4872 
4873  return uuid;
4874 }
4875 
4877 {
4878  switch_core_session_t *session = channel->session;
4879  const char *transfer_on_fail = NULL;
4880  char *tof_data = NULL;
4881  char *tof_array[4] = { 0 };
4882  //int tof_arrayc = 0;
4883 
4884  if (!switch_channel_up_nosig(channel)) {
4885  return;
4886  }
4887 
4888  transfer_on_fail = switch_channel_get_variable(channel, "transfer_on_fail");
4889  tof_data = switch_core_session_strdup(session, transfer_on_fail);
4890  switch_split(tof_data, ' ', tof_array);
4891  transfer_on_fail = tof_array[0];
4892 
4893  /*
4894  if the variable continue_on_fail is set it can be:
4895  'true' to continue on all failures.
4896  'false' to not continue.
4897  A list of codes either names or numbers eg "user_busy,normal_temporary_failure,603"
4898  failure_causes acts as the opposite version
4899  EXCEPTION... ATTENDED_TRANSFER never is a reason to continue.......
4900  */
4901  if (cause != SWITCH_CAUSE_ATTENDED_TRANSFER) {
4902  const char *continue_on_fail = NULL, *failure_causes = NULL;
4903 
4904  continue_on_fail = switch_channel_get_variable(channel, "continue_on_fail");
4905  failure_causes = switch_channel_get_variable(channel, "failure_causes");
4906 
4907  if (continue_on_fail || failure_causes) {
4908  const char *cause_str;
4909  char cause_num[35] = "";
4910 
4911  cause_str = switch_channel_cause2str(cause);
4912  switch_snprintf(cause_num, sizeof(cause_num), "%u", cause);
4913 
4914  if (failure_causes) {
4915  char *lbuf = switch_core_session_strdup(session, failure_causes);
4916  char *argv[256] = { 0 };
4917  int argc = switch_separate_string(lbuf, ',', argv, (sizeof(argv) / sizeof(argv[0])));
4918  int i, x = 0;
4919 
4920  for (i = 0; i < argc; i++) {
4921  if (!strcasecmp(argv[i], cause_str) || !strcasecmp(argv[i], cause_num)) {
4922  x++;
4923  break;
4924  }
4925  }
4926  if (!x) {
4928  "Failure causes [%s]: Cause: %s\n", failure_causes, cause_str);
4929  return;
4930  }
4931  }
4932 
4933  if (continue_on_fail) {
4934  if (switch_true(continue_on_fail)) {
4935  return;
4936  } else {
4937  char *lbuf = switch_core_session_strdup(session, continue_on_fail);
4938  char *argv[256] = { 0 };
4939  int argc = switch_separate_string(lbuf, ',', argv, (sizeof(argv) / sizeof(argv[0])));
4940  int i;
4941 
4942  for (i = 0; i < argc; i++) {
4943  if (!strcasecmp(argv[i], cause_str) || !strcasecmp(argv[i], cause_num)) {
4945  "Continue on fail [%s]: Cause: %s\n", continue_on_fail, cause_str);
4946  return;
4947  }
4948  }
4949  }
4950  }
4951  } else {
4952  /* no answer is *always* a reason to continue */
4954  return;
4955  }
4956  }
4957 
4958  if (transfer_on_fail || failure_causes) {
4959  const char *cause_str;
4960  char cause_num[35] = "";
4961 
4962  cause_str = switch_channel_cause2str(cause);
4963  switch_snprintf(cause_num, sizeof(cause_num), "%u", cause);
4964 
4965  if ((tof_array[1] == NULL ) || (!strcasecmp(tof_array[1], "auto_cause"))){
4966  tof_array[1] = (char *) cause_str;
4967  }
4968 
4969  if (failure_causes) {
4970  char *lbuf = switch_core_session_strdup(session, failure_causes);
4971  char *argv[256] = { 0 };
4972  int argc = switch_separate_string(lbuf, ',', argv, (sizeof(argv) / sizeof(argv[0])));
4973  int i, x = 0;
4974 
4975  for (i = 0; i < argc; i++) {
4976  if (!strcasecmp(argv[i], cause_str) || !strcasecmp(argv[i], cause_num)) {
4977  x++;
4978  break;
4979  }
4980  }
4981  if (!x) {
4983  "Failure causes [%s]: Cause: %s\n", failure_causes, cause_str);
4984 
4985  switch_ivr_session_transfer(session, tof_array[1], tof_array[2], tof_array[3]);
4986  }
4987  }
4988 
4989  if (transfer_on_fail) {
4990  if (switch_true(transfer_on_fail)) {
4991  return;
4992  } else {
4993  char *lbuf = switch_core_session_strdup(session, transfer_on_fail);
4994  char *argv[256] = { 0 };
4995  int argc = switch_separate_string(lbuf, ',', argv, (sizeof(argv) / sizeof(argv[0])));
4996  int i;
4997 
4998  for (i = 0; i < argc; i++) {
4999  if (!strcasecmp(argv[i], cause_str) || !strcasecmp(argv[i], cause_num)) {
5001  "Transfer on fail [%s]: Cause: %s\n", transfer_on_fail, cause_str);
5002  switch_ivr_session_transfer(session, tof_array[1], tof_array[2], tof_array[3]);
5003  }
5004  }
5005  }
5006  }
5007  }
5008  }
5009 
5010 
5012  switch_channel_get_state(channel) != CS_ROUTING) {
5013  switch_channel_hangup(channel, cause);
5014  }
5015 }
5016 
5018 {
5019  memset(&globals, 0, sizeof(globals));
5020  globals.pool = pool;
5021 
5022  switch_mutex_init(&globals.device_mutex, SWITCH_MUTEX_NESTED, pool);
5023  switch_core_hash_init(&globals.device_hash);
5024 }
5025 
5027 {
5028  switch_core_hash_destroy(&globals.device_hash);
5029 }
5030 
5031 
5033 {
5035 
5036 
5037  memset(&drec->stats, 0, sizeof(switch_device_stats_t));
5038 
5039  switch_mutex_lock(drec->mutex);
5040  for(np = drec->uuid_list; np; np = np->next) {
5041  drec->stats.total++;
5043  drec->stats.total_in++;
5044  } else {
5045  drec->stats.total_out++;
5046  }
5047 
5048  if (!np->hup_profile) {
5049  drec->stats.offhook++;
5051  drec->stats.offhook_in++;
5052  } else {
5053  drec->stats.offhook_out++;
5054  }
5055 
5056  if (np->callstate == CCS_HELD) {
5057  drec->stats.held++;
5059  drec->stats.held_in++;
5060  } else {
5061  drec->stats.held_out++;
5062  }
5063  } else if (np->callstate == CCS_UNHELD) {
5064  drec->stats.unheld++;
5066  drec->stats.unheld_in++;
5067  } else {
5068  drec->stats.unheld_out++;
5069  }
5070  } else {
5071  if (np->callstate == CCS_EARLY) {
5072  drec->stats.early++;
5074  drec->stats.early_in++;
5075  } else {
5076  drec->stats.early_out++;
5077  }
5078  } else if (np->callstate == CCS_RINGING) {
5079  drec->stats.ringing++;
5081  drec->stats.ringing_in++;
5082  } else {
5083  drec->stats.ringing_out++;
5084  }
5085  } else if (np->callstate == CCS_RING_WAIT) {
5086  drec->stats.ring_wait++;
5087  } else if (np->callstate == CCS_HANGUP) {
5088  drec->stats.hup++;
5090  drec->stats.hup_in++;
5091  } else {
5092  drec->stats.hup_out++;
5093  }
5094  } else if (np->callstate != CCS_DOWN) {
5095  drec->stats.active++;
5097  drec->stats.active_in++;
5098  } else {
5099  drec->stats.active_out++;
5100  }
5101  }
5102  }
5103  } else {
5104  drec->stats.hup++;
5106  drec->stats.hup_in++;
5107  } else {
5108  drec->stats.hup_out++;
5109  }
5110  }
5111  }
5112  switch_mutex_unlock(drec->mutex);
5113 
5114 }
5115 
5117 {
5119  int sanity = 100;
5121  switch_event_t *event;
5122 
5123  if (!channel->device_node || !switch_channel_test_flag(channel, CF_FINAL_DEVICE_LEG)) {
5124  return;
5125  }
5126 
5127  while(--sanity && channel->device_node->parent->refs) {
5128  switch_yield(100000);
5129  }
5130 
5131  switch_log_printf(SWITCH_CHANNEL_CHANNEL_LOG(channel), SWITCH_LOG_DEBUG, "Destroying device cdr %s on device [%s]\n",
5132  channel->device_node->parent->uuid,
5133  channel->device_node->parent->device_id);
5134 
5136  int x = 0;
5137  char prefix[80] = "";
5138 
5139  switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "Type", "device");
5141 
5143  for(np = channel->device_node->parent->uuid_list; np; np = np->next) {
5144  switch_snprintf(prefix, sizeof(prefix), "Call-%d", ++x);
5146  }
5148 
5149  switch_event_fire(&event);
5150  }
5151 
5153  for(np = channel->device_node->parent->uuid_list; np; np = np->next) {
5154  if (np->xml_cdr) {
5155  switch_xml_free(np->xml_cdr);
5156  }
5157  if (np->event) {
5159  }
5160  }
5162 
5163  pool = channel->device_node->parent->pool;
5164 
5165  switch_mutex_lock(globals.device_mutex);
5167 
5168  switch_mutex_unlock(globals.device_mutex);
5169 
5170 
5171 }
5172 
5174 {
5175 
5176  switch_channel_check_device_state(channel, channel->callstate);
5177  process_device_hup(channel);
5178 
5179 }
5180 
5182 {
5183  switch_hold_record_t *hr, *newhr, *last = NULL;
5184  switch_device_record_t *drec = NULL;
5185  switch_device_node_t *node;
5186 
5187  if (!channel->device_node) {
5188  return;
5189  }
5190 
5191  switch_mutex_lock(globals.device_mutex);
5192  node = channel->device_node;
5193  drec = channel->device_node->parent;
5194 
5195  node->hup_profile = switch_caller_profile_dup(drec->pool, channel->caller_profile);
5196  fetch_device_stats(drec);
5197 
5198  switch_ivr_generate_xml_cdr(channel->session, &node->xml_cdr);
5201  }
5202 
5203  for (hr = channel->hold_record; hr; hr = hr->next) {
5204  newhr = switch_core_alloc(drec->pool, sizeof(*newhr));
5205  newhr->on = hr->on;
5206  newhr->off = hr->off;
5207 
5208  if (hr->uuid) {
5209  newhr->uuid = switch_core_strdup(drec->pool, hr->uuid);
5210  }
5211 
5212  if (!node->hold_record) {
5213  node->hold_record = newhr;
5214  } else if (last) {
5215  last->next = newhr;
5216  }
5217 
5218  last = newhr;
5219  }
5220 
5221  if (!drec->stats.offhook) { /* this is final call */
5222 
5223  switch_core_hash_delete(globals.device_hash, drec->device_id);
5224  switch_log_printf(SWITCH_CHANNEL_CHANNEL_LOG(channel), SWITCH_LOG_DEBUG, "Processing last call from device [%s]\n",
5225  drec->device_id);
5227  } else {
5228  channel->device_node = NULL;
5229  }
5230 
5231  drec->refs--;
5232 
5233  switch_mutex_unlock(globals.device_mutex);
5234 
5235 }
5236 
5238 {
5239  switch_device_record_t *drec = NULL;
5240  switch_device_state_binding_t *ptr = NULL;
5241  switch_event_t *event = NULL;
5242 
5243  if (!channel->device_node) {
5244  return;
5245  }
5246 
5247  drec = channel->device_node->parent;
5248 
5249  switch_mutex_lock(globals.device_mutex);
5250  switch_mutex_lock(drec->mutex);
5251 
5252  fetch_device_stats(drec);
5253 
5254  if (drec->state != SDS_HANGUP) {
5255  if (drec->stats.offhook == 0 || drec->stats.hup == drec->stats.total) {
5256  drec->state = SDS_HANGUP;
5257  } else {
5258  if (drec->stats.active == 0) {
5259  if ((drec->stats.ringing_out + drec->stats.early_out) > 0 || drec->stats.ring_wait > 0) {
5260  drec->state = SDS_RINGING;
5261  } else {
5262  if (drec->stats.held > 0) {
5263  drec->state = SDS_HELD;
5264  } else if (drec->stats.unheld > 0) {
5265  drec->state = SDS_UNHELD;
5266  } else {
5267  drec->state = SDS_DOWN;
5268  }
5269  }
5270  } else if (drec->stats.active == 1) {
5271  drec->state = SDS_ACTIVE;
5272  } else {
5273  drec->state = SDS_ACTIVE_MULTI;
5274  }
5275  }
5276  }
5277 
5278  if ((drec->state == SDS_DOWN && drec->last_state == SDS_DOWN) || (drec->state == SDS_HANGUP && drec->last_state == SDS_HANGUP)) {
5279  switch_mutex_unlock(drec->mutex);
5280  switch_mutex_unlock(globals.device_mutex);
5281  return;
5282  }
5283 
5284  if (!drec->call_start) {
5286  }
5287 
5288  switch(drec->state) {
5289  case SDS_RINGING:
5290  if (!drec->ring_start) {
5292  drec->ring_stop = 0;
5293  }
5294  break;
5295  case SDS_ACTIVE:
5296  case SDS_ACTIVE_MULTI:
5297  if (!drec->active_start) {
5299  drec->active_stop = 0;
5300  }
5301  break;
5302  case SDS_HELD:
5303  if (!drec->hold_start) {
5305  drec->hold_stop = 0;
5306  }
5307  break;
5308  default:
5309  break;
5310  }
5311 
5312  if (callstate != CCS_UNHELD && drec->active_start && drec->state != SDS_ACTIVE && drec->state != SDS_ACTIVE_MULTI) {
5314  }
5315 
5316  if (drec->ring_start && !drec->ring_stop && drec->state != SDS_RINGING) {
5317  drec->ring_stop = switch_micro_time_now();
5318  }
5319 
5320  if (drec->hold_start && !drec->hold_stop && drec->state != SDS_HELD) {
5321  drec->hold_stop = switch_micro_time_now();
5322  }
5323 
5324 
5326  switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "Device-ID", drec->device_id);
5329  switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "Device-Call-State", switch_channel_callstate2str(callstate));
5330  switch_event_add_header(event, SWITCH_STACK_BOTTOM, "Device-Total-Legs", "%u", drec->stats.total);
5331  switch_event_add_header(event, SWITCH_STACK_BOTTOM, "Device-Legs-Offhook", "%u", drec->stats.offhook);
5332  switch_event_add_header(event, SWITCH_STACK_BOTTOM, "Device-Legs-Ringing", "%u", drec->stats.ringing);
5333  switch_event_add_header(event, SWITCH_STACK_BOTTOM, "Device-Legs-Early", "%u", drec->stats.early);
5334  switch_event_add_header(event, SWITCH_STACK_BOTTOM, "Device-Legs-Active", "%u", drec->stats.active);
5335  switch_event_add_header(event, SWITCH_STACK_BOTTOM, "Device-Legs-Held", "%u", drec->stats.held);
5336  switch_event_add_header(event, SWITCH_STACK_BOTTOM, "Device-Legs-UnHeld", "%u", drec->stats.unheld);
5337  switch_event_add_header(event, SWITCH_STACK_BOTTOM, "Device-Legs-Hup", "%u", drec->stats.hup);
5338  switch_event_add_header(event, SWITCH_STACK_BOTTOM, "Device-Talk-Time-Start-Uepoch", "%"SWITCH_TIME_T_FMT, drec->active_start);
5339  if (drec->active_stop) {
5340  switch_event_add_header(event, SWITCH_STACK_BOTTOM, "Device-Talk-Time-Stop-Uepoch", "%"SWITCH_TIME_T_FMT, drec->active_stop);
5341  switch_event_add_header(event, SWITCH_STACK_BOTTOM, "Device-Talk-Time-Milliseconds", "%u", (uint32_t)(drec->active_stop - drec->active_start) / 1000);
5342  }
5343  }
5344 
5346  "%s device: %s\nState: %s Dev State: %s/%s Total:%u Offhook:%u "
5347  "Ringing:%u Early:%u Active:%u Held:%u Unheld:%u Hungup:%u Dur: %u Ringtime: %u Holdtime: %u %s\n",
5348  switch_channel_get_name(channel),
5349  drec->device_id,
5350  switch_channel_callstate2str(callstate),
5353  drec->stats.total,
5354  drec->stats.offhook,
5355  drec->stats.ringing,
5356  drec->stats.early,
5357  drec->stats.active,
5358  drec->stats.held,
5359  drec->stats.unheld,
5360  drec->stats.hup,
5361  drec->active_stop ? (uint32_t)(drec->active_stop - drec->active_start) / 1000 : 0,
5362  drec->ring_stop ? (uint32_t)(drec->ring_stop - drec->ring_start) / 1000 : 0,
5363  drec->hold_stop ? (uint32_t)(drec->hold_stop - drec->hold_start) / 1000 : 0,
5364  switch_channel_test_flag(channel, CF_FINAL_DEVICE_LEG) ? "FINAL LEG" : "");
5365 
5366  for (ptr = globals.device_bindings; ptr; ptr = ptr->next) {
5367  ptr->function(channel->session, callstate, drec);
5368  }
5369 
5370  drec->last_stats = drec->stats;
5371 
5372  if (drec->active_stop) {
5373  drec->active_start = drec->active_stop = 0;
5374  if (drec->state == SDS_ACTIVE || drec->state == SDS_ACTIVE_MULTI) {
5376  }
5377  }
5378 
5379  if (drec->hold_stop) {
5380  drec->hold_start = drec->hold_stop = 0;
5381  if (drec->state == SDS_HELD) {
5383  }
5384  }
5385 
5386  if (drec->ring_stop) {
5387  drec->ring_start = drec->ring_stop = 0;
5388  if (drec->state == SDS_RINGING) {
5390  }
5391  }
5392 
5394 
5395  drec->last_state = drec->state;
5396 
5397  switch_mutex_unlock(drec->mutex);
5398  switch_mutex_unlock(globals.device_mutex);
5399 
5400 
5401  if (event) {
5402  switch_event_fire(&event);
5403  }
5404 
5405 }
5406 
5407 /* assumed to be called under a lock */
5409 {
5410  switch_device_node_t *node;
5411 
5412  switch_assert(drec);
5413 
5415  node = switch_core_alloc(drec->pool, sizeof(*node));
5416 
5418  node->parent = drec;
5419  node->callstate = channel->callstate;
5421 
5422  channel->device_node = node;
5423 
5424  if (!drec->uuid_list) {
5425  drec->uuid_list = node;
5426  drec->uuid = node->uuid;
5427  } else {
5428  drec->uuid_tail->next = node;
5429  }
5430 
5431  drec->uuid_tail = node;
5432  drec->refs++;
5433 }
5434 
5435 static switch_status_t create_device_record(switch_device_record_t **drecp, const char *device_id)
5436 {
5437  switch_device_record_t *drec;
5439 
5440  switch_assert(drecp);
5441 
5443  drec = switch_core_alloc(pool, sizeof(*drec));
5444  drec->pool = pool;
5445  drec->device_id = switch_core_strdup(drec->pool, device_id);
5447 
5448  *drecp = drec;
5449 
5450  return SWITCH_STATUS_SUCCESS;
5451 }
5452 
5453 
5454 SWITCH_DECLARE(const char *) switch_channel_set_device_id(switch_channel_t *channel, const char *device_id)
5455 {
5456  switch_device_record_t *drec;
5457 
5458  if (channel->device_node) {
5459  return NULL;
5460  }
5461 
5462  channel->device_id = switch_core_session_strdup(channel->session, device_id);
5463 
5464  switch_mutex_lock(globals.device_mutex);
5465 
5466  if (!(drec = switch_core_hash_find(globals.device_hash, channel->device_id))) {
5467  create_device_record(&drec, channel->device_id);
5468  switch_core_hash_insert(globals.device_hash, drec->device_id, drec);
5469  }
5470 
5471  add_uuid(drec, channel);
5472 
5473  switch_mutex_unlock(globals.device_mutex);
5474 
5475  switch_log_printf(SWITCH_CHANNEL_CHANNEL_LOG(channel), SWITCH_LOG_DEBUG, "Setting DEVICE ID to [%s]\n", device_id);
5476 
5477  switch_channel_check_device_state(channel, channel->callstate);
5478 
5479  return device_id;
5480 }
5481 
5483 {
5484  if (channel->device_node) {
5486  return channel->device_node->parent;
5487  }
5488 
5489  return NULL;
5490 }
5491 
5493 {
5494  if (drecp && *drecp) {
5495  switch_mutex_unlock((*drecp)->mutex);
5496  *drecp = NULL;
5497  }
5498 }
5499 
5501 {
5502  switch_device_state_binding_t *binding = NULL, *ptr = NULL;
5503  assert(function != NULL);
5504 
5505  if (!(binding = (switch_device_state_binding_t *) switch_core_alloc(globals.pool, sizeof(*binding)))) {
5506  return SWITCH_STATUS_MEMERR;
5507  }
5508 
5509  binding->function = function;
5510  binding->user_data = user_data;
5511 
5512  switch_mutex_lock(globals.device_mutex);
5513  for (ptr = globals.device_bindings; ptr && ptr->next; ptr = ptr->next);
5514 
5515  if (ptr) {
5516  ptr->next = binding;
5517  } else {
5518  globals.device_bindings = binding;
5519  }
5520 
5521  switch_mutex_unlock(globals.device_mutex);
5522 
5523  return SWITCH_STATUS_SUCCESS;
5524 }
5525 
5527 {
5528  switch_device_state_binding_t *ptr, *last = NULL;
5530 
5531  switch_mutex_lock(globals.device_mutex);
5532  for (ptr = globals.device_bindings; ptr; ptr = ptr->next) {
5533  if (ptr->function == function) {
5534  status = SWITCH_STATUS_SUCCESS;
5535 
5536  if (last) {
5537  last->next = ptr->next;
5538  } else {
5539  globals.device_bindings = ptr->next;
5540  last = NULL;
5541  continue;
5542  }
5543  }
5544  last = ptr;
5545  }
5546  switch_mutex_unlock(globals.device_mutex);
5547 
5548  return status;
5549 }
5550 
5552 {
5554  char *use_sdp = (char *) sdp;
5555  char *patched_sdp = NULL;
5556 
5558  const char *var;
5559 
5560  if ((var = switch_channel_get_variable(from_channel, "bypass_media_sdp_filter"))) {
5561  if ((patched_sdp = switch_core_media_process_sdp_filter(use_sdp, var, from_channel->session))) {
5562  use_sdp = patched_sdp;
5563  }
5564  }
5565 
5566  switch_channel_set_variable(to_channel, SWITCH_B_SDP_VARIABLE, use_sdp);
5567  }
5568 
5569  switch_safe_free(patched_sdp);
5570 
5571  return status;
5572 }
5573 
5574 /* For Emacs:
5575  * Local Variables:
5576  * mode:c
5577  * indent-tabs-mode:t
5578  * tab-width:4
5579  * c-basic-offset:4
5580  * End:
5581  * For VIM:
5582  * vim:set softtabstop=4 shiftwidth=4 tabstop=4 noet:
5583  */
char * switch_channel_get_cap_string(switch_channel_t *channel)
switch_status_t switch_channel_set_variable_printf(switch_channel_t *channel, const char *varname, const char *fmt,...)
_Ret_opt_z_ char * switch_event_get_header_idx(switch_event_t *event, const char *header_name, int idx)
Definition: switch_event.c:846
void switch_channel_set_cap_value(switch_channel_t *channel, switch_channel_cap_t cap, uint32_t value)
switch_time_t switch_micro_time_now(void)
Get the current epoch time in microseconds.
Definition: switch_time.c:311
void switch_core_session_hangup_state(switch_core_session_t *session, switch_bool_t force)
void * switch_core_hash_find_locked(_In_ switch_hash_t *hash, _In_z_ const char *key, _In_ switch_mutex_t *mutex)
Retrieve data from a given hash.
switch_channel_state_t switch_channel_get_state(switch_channel_t *channel)
Get the current state of a channel in the state engine.
#define switch_event_fire(event)
Fire an event filling in most of the arguements with obvious values.
Definition: switch_event.h:413
#define switch_channel_hangup(channel, hangup_cause)
Hangup a channel flagging it&#39;s state machine to end.
#define switch_regex_safe_free(re)
Definition: switch_regex.h:79
void switch_xml_free(_In_opt_ switch_xml_t xml)
frees the memory allocated for an switch_xml structure
switch_xml_t xml_cdr
Definition: switch_core.h:83
unsigned int switch_queue_size(switch_queue_t *queue)
Definition: switch_apr.c:1238
void switch_channel_state_thread_unlock(switch_channel_t *channel)
switch_event_types_t event_id
Definition: switch_event.h:82
#define switch_core_new_memory_pool(p)
Create a new sub memory pool from the core&#39;s master pool.
Definition: switch_core.h:633
void switch_channel_set_flag_recursive(switch_channel_t *channel, switch_channel_flag_t flag)
switch_caller_extension_t * switch_channel_get_caller_extension(switch_channel_t *channel)
Retrieve caller extension from a given channel.
char * switch_core_session_sprintf(_In_ switch_core_session_t *session, _In_z_ _Printf_format_string_ const char *fmt,...)
printf-style style printing routine. The data is output to a string allocated from the session ...
switch_status_t switch_channel_set_profile_var(switch_channel_t *channel, const char *name, const char *val)
Set a variable on a given channel.
void switch_channel_flush_dtmf(switch_channel_t *channel)
switch_event_t * variables
#define SWITCH_SENSITIVE_DTMF_VARIABLE
Definition: switch_types.h:146
static switch_bool_t switch_true(const char *expr)
Evaluate the truthfullness of a string expression.
Definition: switch_utils.h:519
#define SWITCH_CHANNEL_SESSION_LOG(x)
An Abstract Representation of a dialplan extension.
switch_channel_callstate_t callstate
void switch_channel_handle_cause(switch_channel_t *channel, switch_call_cause_t cause)
switch_channel_state_t switch_channel_perform_set_state(switch_channel_t *channel, const char *file, const char *func, int line, switch_channel_state_t state)
Call Specific Data.
Definition: switch_caller.h:73
#define switch_set_flag(obj, flag)
Set a flag on an arbitrary object.
Definition: switch_utils.h:700
switch_call_direction_t logical_direction
#define SWITCH_ORIGINATOR_VARIABLE
Definition: switch_types.h:205
opaque_channel_flag_t
switch_core_session_message_types_t message_id
Definition: switch_core.h:183
#define SWITCH_CHANNEL_LOG
switch_status_t switch_channel_set_variable_name_printf(switch_channel_t *channel, const char *val, const char *fmt,...)
switch_status_t switch_mutex_trylock(switch_mutex_t *lock)
Definition: switch_apr.c:318
switch_status_t switch_core_hash_destroy(_Inout_ switch_hash_t **hash)
Destroy an existing hash table.
uint32_t switch_channel_test_flag_partner(switch_channel_t *channel, switch_channel_flag_t flag)
switch_status_t switch_channel_export_variable_var_check(switch_channel_t *channel, const char *varname, const char *val, const char *export_varname, switch_bool_t var_check)
void * switch_core_hash_find(_In_ switch_hash_t *hash, _In_z_ const char *key)
Retrieve data from a given hash.
struct switch_hold_record_s * next
Definition: switch_core.h:78
void switch_core_session_signal_state_change(_In_ switch_core_session_t *session)
struct switch_device_state_binding_s switch_device_state_binding_t
void switch_channel_set_hangup_time(switch_channel_t *channel)
switch_status_t switch_channel_perform_mark_pre_answered(switch_channel_t *channel, const char *file, const char *func, int line)
#define switch_event_del_header(_e, _h)
Definition: switch_event.h:212
#define switch_channel_export_variable(_channel, _varname, _value, _ev)
int switch_core_test_flag(int flag)
Definition: switch_core.c:1792
#define switch_core_hash_init(_hash)
Definition: switch_core.h:1431
#define switch_channel_set_state(channel, state)
Set the current state of a channel.
switch_status_t switch_channel_set_private(switch_channel_t *channel, const char *key, const void *private_info)
Set private data on channel.
void switch_channel_clear_state_flag(switch_channel_t *channel, switch_channel_flag_t flag)
void switch_channel_perform_video_sync(switch_channel_t *channel, const char *file, const char *func, int line)
switch_call_direction_t switch_channel_logical_direction(switch_channel_t *channel)
void switch_channel_event_set_basic_data(switch_channel_t *channel, switch_event_t *event)
switch_status_t switch_ivr_sleep(switch_core_session_t *session, uint32_t ms, switch_bool_t sync, switch_input_args_t *args)
Wait for time to pass for a specified number of milliseconds.
Definition: switch_ivr.c:127
switch_status_t switch_channel_dequeue_dtmf(switch_channel_t *channel, switch_dtmf_t *dtmf)
switch_event_t * api_list
switch_ring_ready_t
struct switch_caller_profile * originatee_caller_profile
switch_queue_t * dtmf_log_queue
switch_device_state_t state
Definition: switch_core.h:128
switch_event_types_t
Built-in Events.
void switch_channel_clear_device_record(switch_channel_t *channel)
#define SWITCH_CHANNEL_EXECUTE_ON_MEDIA_VARIABLE
Definition: switch_types.h:152
switch_device_state_t last_state
Definition: switch_core.h:129
switch_hash_t * app_flag_hash
char * switch_core_get_variable_pdup(_In_z_ const char *varname, switch_memory_pool_t *pool)
static int switch_string_has_escaped_data(const char *in)
Definition: switch_utils.h:384
void switch_channel_set_bridge_time(switch_channel_t *channel)
switch_call_direction_t
Definition: switch_types.h:307
switch_device_stats_t stats
Definition: switch_core.h:126
opaque_channel_flag_t opaque_flags
void switch_event_add_presence_data_cols(switch_channel_t *channel, switch_event_t *event, const char *prefix)
const char * switch_channel_callstate2str(switch_channel_callstate_t callstate)
cJSON *const to
switch_status_t switch_core_session_perform_receive_message(_In_ switch_core_session_t *session, _In_ switch_core_session_message_t *message, const char *file, const char *func, int line)
Receive a message on a given session.
switch_status_t switch_channel_queue_dtmf(switch_channel_t *channel, const switch_dtmf_t *dtmf)
switch_bool_t
Definition: switch_types.h:441
switch_mutex_t * switch_core_session_get_mutex(switch_core_session_t *session)
Signal a session&#39;s state machine thread that a state change has occured.
#define switch_channel_presence(_a, _b, _c, _d)
#define switch_core_strdup(_pool, _todup)
Copy a string using memory allocation from a given pool.
Definition: switch_core.h:733
#define switch_split(_data, _delim, _array)
Definition: switch_utils.h:375
const char * network_addr
Definition: switch_caller.h:93
#define SWITCH_PROTO_SPECIFIC_HANGUP_CAUSE_VARIABLE
Definition: switch_types.h:143
switch_time_t active_stop
Definition: switch_core.h:131
switch_status_t switch_api_execute(const char *cmd, const char *arg, switch_core_session_t *session, switch_stream_handle_t *stream)
Execute a registered API command.
#define SWITCH_CHANNEL_API_ON_PRE_ANSWER_VARIABLE
Definition: switch_types.h:163
Media Channel Interface.
switch_caller_profile_t * caller_profile
switch_channel_flag_t
Channel Flags.
#define switch_core_destroy_memory_pool(p)
Returns a subpool back to the main pool.
Definition: switch_core.h:642
switch_memory_pool_t * pool
switch_status_t switch_event_add_header(switch_event_t *event, switch_stack_t stack, const char *header_name, const char *fmt,...) PRINTF_FUNCTION(4
Add a header to an event.
Representation of an event.
Definition: switch_event.h:80
int switch_event_check_permission_list(switch_event_t *list, const char *name)
switch_caller_profile_t * hup_profile
Definition: switch_core.h:87
switch_hold_record_t * hold_record
#define switch_channel_ready(_channel)
const char * switch_channel_get_variable_strdup(switch_channel_t *channel, const char *varname)
Retrieve a copy of a variable from a given channel. switch_safe_free() call will be required...
An event Header.
Definition: switch_event.h:65
const char *const const char *const const cJSON *const value
An Abstract Representation of a dialplan Application.
switch_status_t switch_queue_trypop(switch_queue_t *queue, void **data)
Definition: switch_apr.c:1264
void * switch_channel_get_private_partner(switch_channel_t *channel, const char *key)
switch_core_session_message_types_t
Possible types of messages for inter-session communication.
switch_hash_t * private_hash
const char * name
switch_caller_profile_t * switch_caller_profile_clone(_In_ switch_core_session_t *session, _In_ switch_caller_profile_t *tocopy)
Clone an existing caller profile object.
switch_size_t switch_channel_dequeue_dtmf_string(switch_channel_t *channel, char *dtmf_str, switch_size_t len)
switch_status_t switch_channel_bind_device_state_handler(switch_device_state_function_t function, void *user_data)
void switch_channel_set_caller_profile(switch_channel_t *channel, switch_caller_profile_t *caller_profile)
Set the given channel&#39;s caller profile.
switch_channel_state_t switch_channel_get_running_state(switch_channel_t *channel)
void switch_channel_flip_cid(switch_channel_t *channel)
switch_status_t switch_channel_alloc(switch_channel_t **channel, switch_call_direction_t direction, switch_memory_pool_t *pool)
switch_originator_type_t
#define SWITCH_HOLD_MUSIC_VARIABLE
Definition: switch_types.h:193
switch_status_t switch_channel_set_variable_partner_var_check(switch_channel_t *channel, const char *varname, const char *value, switch_bool_t var_check)
switch_channel_cap_t
const char * switch_channel_get_partner_uuid_copy(switch_channel_t *channel, char *buf, switch_size_t blen)
switch_bool_t switch_is_number(const char *str)
switch_status_t switch_time_exp_lt(switch_time_exp_t *result, switch_time_t input)
Definition: switch_apr.c:346
struct switch_event * next
Definition: switch_event.h:101
const char * switch_core_session_get_external_id(_In_ switch_core_session_t *session)
Retrieve the unique external identifier from a session.
switch_time_t active_start
Definition: switch_core.h:130
switch_mutex_t * mutex
Definition: switch_core.h:140
#define end_of(_s)
Definition: switch_utils.h:685
struct switch_caller_profile * next
const char * switch_channel_get_variable_dup(switch_channel_t *channel, const char *varname, switch_bool_t dup, int idx)
Retrieve a variable from a given channel.
void switch_channel_set_state_flag(switch_channel_t *channel, switch_channel_flag_t flag)
Set given flag(s) on a given channel to be applied on the next state change.
switch_status_t switch_core_session_queue_message(_In_ switch_core_session_t *session, _In_ switch_core_session_message_t *message)
Queue a message on a session.
switch_device_state_binding_t * device_bindings
const char * switch_channel_device_state2str(switch_device_state_t device_state)
static const char * state_names[]
uint32_t duration
Definition: switch_types.h:302
int switch_channel_check_signal(switch_channel_t *channel, switch_bool_t in_thread_only)
struct real_pcre switch_regex_t
Definition: switch_regex.h:43
char * switch_channel_expand_variables_check(switch_channel_t *channel, const char *in, switch_event_t *var_list, switch_event_t *api_list, uint32_t recur)
Expand varaibles in a string based on the variables in a paticular channel.
const char * dialplan
Definition: switch_caller.h:77
switch_mutex_t * state_mutex
char * switch_url_encode(const char *url, char *buf, size_t len)
switch_status_t switch_event_dup(switch_event_t **event, switch_event_t *todup)
Duplicate an event.
switch_status_t switch_strftime_nocheck(char *s, switch_size_t *retsize, switch_size_t max, const char *format, switch_time_exp_t *tm)
Definition: switch_apr.c:202
switch_event_header_t * switch_channel_variable_first(switch_channel_t *channel)
Start iterating over the entries in the channel variable list.
#define SWITCH_MAX_STATE_HANDLERS
Definition: switch_types.h:596
switch_status_t switch_channel_perform_answer(switch_channel_t *channel, const char *file, const char *func, int line)
switch_bool_t switch_core_media_check_dtls(switch_core_session_t *session, switch_media_type_t type)
switch_status_t switch_core_session_execute_application_async(switch_core_session_t *session, const char *app, const char *arg)
char * switch_channel_get_flag_string(switch_channel_t *channel)
int switch_snprintf(_Out_z_cap_(len) char *buf, _In_ switch_size_t len, _In_z_ _Printf_format_string_ const char *format,...)
_Check_return_ _Ret_opt_z_ const char * switch_caller_get_field_by_name(_In_ switch_caller_profile_t *caller_profile, _In_z_ const char *name)
Get the value of a field in a caller profile based on it&#39;s name.
int switch_channel_test_private_flag(switch_channel_t *channel, uint32_t flags)
uint32_t switch_channel_test_flag(switch_channel_t *channel, switch_channel_flag_t flag)
Test for presence of given flag on a given channel.
void switch_channel_event_set_extended_data(switch_channel_t *channel, switch_event_t *event)
void switch_channel_perform_audio_sync(switch_channel_t *channel, const char *file, const char *func, int line)
void switch_channel_set_scope_variables(switch_channel_t *channel, switch_event_t **event)
const char * switch_channel_get_partner_uuid(switch_channel_t *channel)
switch_call_cause_t switch_channel_str2cause(const char *str)
switch_event_t * event
Definition: switch_core.h:84
switch_codec_t * switch_core_session_get_write_codec(_In_ switch_core_session_t *session)
Retrieve the write codec from a given session.
A message object designed to allow unlike technologies to exchange data.
Definition: switch_core.h:179
void switch_channel_set_origination_caller_profile(switch_channel_t *channel, switch_caller_profile_t *caller_profile)
Set the given channel&#39;s origination caller profile.
uint8_t switch_byte_t
Definition: switch_types.h:256
const char * username
Definition: switch_caller.h:75
#define SWITCH_ORIGINATE_SIGNAL_BOND_VARIABLE
Definition: switch_types.h:204
static void do_api_on(switch_channel_t *channel, const char *variable)
struct device_uuid_node_s * uuid_tail
Definition: switch_core.h:139
#define end_of_p(_s)
Definition: switch_utils.h:686
#define zstr(x)
Definition: switch_utils.h:314
switch_status_t switch_channel_get_scope_variables(switch_channel_t *channel, switch_event_t **event)
switch_codec_t * switch_core_session_get_read_codec(_In_ switch_core_session_t *session)
Retrieve the read codec from a given session.
switch_size_t switch_core_session_id(void)
Provide the current session_id.
int cJSON_bool fmt
Definition: switch_cJSON.h:150
switch_caller_profile_t * switch_channel_get_originatee_caller_profile(switch_channel_t *channel)
Retrieve the given channel&#39;s originatee caller profile.
uint32_t switch_core_max_dtmf_duration(uint32_t duration)
Definition: switch_core.c:1704
struct switch_caller_application * next
static switch_status_t send_ind(switch_channel_t *channel, switch_core_session_message_types_t msg_id, const char *file, const char *func, int line)
switch_channel_state_t switch_channel_perform_hangup(switch_channel_t *channel, const char *file, const char *func, int line, switch_call_cause_t hangup_cause)
static void process_device_hup(switch_channel_t *channel)
switch_status_t switch_ivr_session_transfer(_In_ switch_core_session_t *session, const char *extension, const char *dialplan, const char *context)
Transfer an existing session to another location.
switch_mutex_t * profile_mutex
#define is_dtmf(key)
determine if a character is a valid DTMF key
Definition: switch_utils.h:683
switch_time_t ring_stop
Definition: switch_core.h:134
void switch_channel_invert_cid(switch_channel_t *channel)
switch_status_t switch_mutex_unlock(switch_mutex_t *lock)
Definition: switch_apr.c:313
switch_status_t switch_core_session_get_read_impl(switch_core_session_t *session, switch_codec_implementation_t *impp)
void switch_channel_state_thread_lock(switch_channel_t *channel)
switch_event_t * scope_variables
#define switch_core_session_execute_application(_a, _b, _c)
Execute an application on a session.
Definition: switch_core.h:1129
_Ret_ switch_channel_t * switch_core_session_get_channel(_In_ switch_core_session_t *session)
Retrieve a pointer to the channel object associated with a given session.
void switch_core_recovery_track(switch_core_session_t *session)
switch_status_t switch_channel_queue_dtmf_string(switch_channel_t *channel, const char *dtmf_string)
switch_status_t switch_core_media_check_autoadj(switch_core_session_t *session)
void switch_channel_clear_app_flag_key(const char *key, switch_channel_t *channel, uint32_t flags)
void switch_channel_mark_hold(switch_channel_t *channel, switch_bool_t on)
#define SWITCH_MUTEX_NESTED
Definition: switch_apr.h:318
uint32_t switch_core_min_dtmf_duration(uint32_t duration)
Definition: switch_core.c:1744
switch_caller_extension_t * queued_extension
void switch_channel_wait_for_state(switch_channel_t *channel, switch_channel_t *other_channel, switch_channel_state_t want_state)
switch_status_t switch_channel_perform_pre_answer(switch_channel_t *channel, const char *file, const char *func, int line)
void switch_caller_profile_event_set_data(_In_ switch_caller_profile_t *caller_profile, _In_opt_z_ const char *prefix, _In_ switch_event_t *event)
Add headers to an existing event in regards to a specific profile.
void switch_core_session_enable_heartbeat(switch_core_session_t *session, uint32_t seconds)
switch_channel_state_t running_state
int64_t switch_time_t
Definition: switch_apr.h:188
switch_status_t switch_channel_perform_mark_ring_ready_value(switch_channel_t *channel, switch_ring_ready_t rv, const char *file, const char *func, int line)
void switch_channel_event_set_data(switch_channel_t *channel, switch_event_t *event)
switch_byte_t switch_byte_t * buf
switch_event_t * app_list
switch_time_t on
Definition: switch_core.h:75
void switch_channel_set_private_flag(switch_channel_t *channel, uint32_t flags)
switch_channel_timetable_t * switch_channel_get_timetable(switch_channel_t *channel)
if((uint32_t)(unpack->cur - unpack->buf) > unpack->buflen)
const char * switch_channel_state_name(switch_channel_state_t state)
switch_status_t switch_core_get_variables(switch_event_t **event)
Definition: switch_core.c:386
#define switch_yield(ms)
Wait a desired number of microseconds and yield the CPU.
Definition: switch_utils.h:998
switch_device_state_function_t function
struct switch_caller_extension * next
#define switch_channel_audio_sync(_c)
unsigned int switch_separate_string(_In_ char *buf, char delim, _Post_count_(return) char **array, unsigned int arraylen)
Separate a string into an array based on a character delimiter.
#define resize(l)
switch_status_t switch_core_hash_insert_locked(_In_ switch_hash_t *hash, _In_z_ const char *key, _In_opt_ const void *data, _In_opt_ switch_mutex_t *mutex)
Insert data into a hash.
switch_event_t * log_tags
switch_call_direction_t logical_direction
void switch_channel_global_init(switch_memory_pool_t *pool)
const char * switch_channel_get_hold_music(switch_channel_t *channel)
switch_byte_t in
void switch_channel_sort_cid(switch_channel_t *channel)
switch_time_t hold_start
Definition: switch_core.h:135
const char *const string
Definition: switch_cJSON.h:162
void switch_event_merge(switch_event_t *event, switch_event_t *tomerge)
switch_status_t switch_mutex_lock(switch_mutex_t *lock)
Definition: switch_apr.c:308
const char * callee_id_number
Definition: switch_caller.h:89
void switch_channel_process_export(switch_channel_t *channel, switch_channel_t *peer_channel, switch_event_t *var_event, const char *export_varname)
char * switch_core_media_process_sdp_filter(const char *sdp, const char *cmd_buf, switch_core_session_t *session)
int switch_channel_test_ready(switch_channel_t *channel, switch_bool_t check_ready, switch_bool_t check_media)
Determine if a channel is ready for io.
switch_status_t switch_channel_set_log_tag(switch_channel_t *channel, const char *tagname, const char *tagvalue)
switch_status_t switch_channel_add_variable_var_check(switch_channel_t *channel, const char *varname, const char *value, switch_bool_t var_check, switch_stack_t stack)
static void fetch_device_stats(switch_device_record_t *drec)
switch_queue_t * dtmf_queue
#define switch_core_session_request_video_refresh(_s)
Definition: switch_core.h:2892
uint32_t switch_channel_test_cap_partner(switch_channel_t *channel, switch_channel_cap_t cap)
#define switch_core_alloc(_pool, _mem)
Allocate memory directly from a memory pool.
Definition: switch_core.h:684
#define SWITCH_ENDPOINT_DISPOSITION_VARIABLE
Definition: switch_types.h:192
void switch_channel_clear_state_handler(switch_channel_t *channel, const switch_state_handler_table_t *state_handler)
clear a state handler table from a given channel
switch_call_direction_t direction
Definition: switch_core.h:88
#define switch_channel_get_variable(_c, _v)
int index
Definition: switch_cJSON.h:160
switch_status_t switch_event_add_header_string(switch_event_t *event, switch_stack_t stack, const char *header_name, const char *data)
Add a string header to an event.
switch_status_t switch_channel_caller_extension_masquerade(switch_channel_t *orig_channel, switch_channel_t *new_channel, uint32_t offset)
#define switch_zmalloc(ptr, len)
switch_call_cause_t * switch_channel_get_cause_ptr(switch_channel_t *channel)
const char * caller_id_name
Definition: switch_caller.h:79
#define switch_channel_set_variable_partner(_channel, _var, _val)
#define switch_safe_free(it)
Free a pointer and set it to NULL unless it already is NULL.
Definition: switch_utils.h:885
uint32_t private_flags
switch_originator_type_t last_profile_type
switch_call_cause_t switch_channel_get_cause(switch_channel_t *channel)
switch_status_t switch_channel_get_variables(switch_channel_t *channel, switch_event_t **event)
void switch_channel_process_device_hangup(switch_channel_t *channel)
switch_status_t switch_mutex_init(switch_mutex_t **lock, unsigned int flags, switch_memory_pool_t *pool)
Definition: switch_apr.c:293
#define SWITCH_BLANK_STRING
Definition: switch_types.h:48
switch_time_t ring_start
Definition: switch_core.h:133
void switch_core_media_break(switch_core_session_t *session, switch_media_type_t type)
void switch_channel_set_flag_value(switch_channel_t *channel, switch_channel_flag_t flag, uint32_t value)
Set given flag(s) on a given channel.
#define SWITCH_CHANNEL_EXECUTE_ON_ANSWER_VARIABLE
Definition: switch_types.h:150
switch_status_t switch_channel_transfer_variable_prefix(switch_channel_t *orig_channel, switch_channel_t *new_channel, const char *prefix)
static struct switch_cause_table CAUSE_CHART[]
#define SWITCH_TIME_T_FMT
switch_status_t switch_channel_export_variable_printf(switch_channel_t *channel, const char *varname, const char *export_varname, const char *fmt,...)
#define switch_channel_down_nosig(_channel)
switch_status_t switch_core_session_recv_dtmf(_In_ switch_core_session_t *session, const switch_dtmf_t *dtmf)
RECV DTMF on a session.
static struct @1 globals
struct profile_node_s * next
Definition: switch_caller.h:66
void switch_channel_set_caller_extension(switch_channel_t *channel, switch_caller_extension_t *caller_extension)
Assign a caller extension to a given channel.
struct switch_caller_profile * origination_caller_profile
switch_time_t profile_created
switch_call_direction_t direction
static void switch_channel_check_device_state(switch_channel_t *channel, switch_channel_callstate_t callstate)
uintptr_t switch_size_t
void switch_channel_set_presence_data_vals(switch_channel_t *channel, const char *presence_data_cols)
Takes presence_data_cols as a parameter or as a channel variable and copies them to channel profile v...
int switch_vasprintf(_Out_opt_ char **buf, _In_z_ _Printf_format_string_ const char *format, _In_ va_list ap)
const switch_state_handler_table_t * state_handlers[SWITCH_MAX_STATE_HANDLERS]
void bridge(CoreSession &session_a, CoreSession &session_b)
bridge the audio of session_b into session_a
switch_bool_t switch_channel_set_flag_partner(switch_channel_t *channel, switch_channel_flag_t flag)
Set given flag(s) on a given channel&#39;s bridge partner.
switch_status_t switch_channel_pass_sdp(switch_channel_t *from_channel, switch_channel_t *to_channel, const char *sdp)
#define TIME_T_FMT
void switch_caller_extension_add_application(_In_ switch_core_session_t *session, _In_ switch_caller_extension_t *caller_extension, _In_z_ const char *application_name, _In_z_ const char *extra_data)
Add an application (instruction) to the given extension.
switch_status_t switch_channel_get_variable_buf(switch_channel_t *channel, const char *varname, char *buf, switch_size_t buflen)
Retrieve a variable from a given channel to a pre-allocated buffer without using a memory pool...
switch_status_t switch_channel_wait_for_app_flag(switch_channel_t *channel, uint32_t app_flag, const char *key, switch_bool_t pres, uint32_t to)
void switch_channel_set_originator_caller_profile(switch_channel_t *channel, switch_caller_profile_t *caller_profile)
Set the given channel&#39;s originator caller profile.
uint8_t destination_number_numplan
struct device_uuid_node_s * next
Definition: switch_core.h:90
uint32_t flags[CF_FLAG_MAX]
static switch_bool_t switch_string_var_check_const(const char *s)
Definition: switch_utils.h:788
switch_byte_t switch_byte_t uint32_t buflen
switch_mutex_t * flag_mutex
const char * profile_index
void switch_cond_next(void)
Definition: switch_time.c:658
#define switch_core_session_get_partner(_session, _partner)
Definition: switch_core.h:1028
switch_time_t call_start
Definition: switch_core.h:137
switch_size_t switch_channel_has_dtmf(switch_channel_t *channel)
switch_status_t switch_core_session_get_write_impl(switch_core_session_t *session, switch_codec_implementation_t *impp)
struct switch_caller_profile * originator_caller_profile
#define SWITCH_STANDARD_STREAM(s)
void switch_channel_global_uninit(void)
const char * switch_channel_get_hold_music_partner(switch_channel_t *channel)
int switch_regex_perform(const char *field, const char *expression, switch_regex_t **new_re, int *ovector, uint32_t olen)
Definition: switch_regex.c:55
void switch_channel_wait_for_state_timeout(switch_channel_t *channel, switch_channel_state_t want_state, uint32_t timeout)
switch_channel_callstate_t
void switch_channel_restart(switch_channel_t *channel)
char * switch_copy_string(_Out_z_cap_(dst_size) char *dst, _In_z_ const char *src, _In_ switch_size_t dst_size)
uint32_t switch_core_session_count(void)
Provide the total number of sessions.
switch_status_t switch_caller_extension_clone(switch_caller_extension_t **new_ext, switch_caller_extension_t *orig, switch_memory_pool_t *pool)
switch_status_t switch_core_session_queue_event(_In_ switch_core_session_t *session, _Inout_ switch_event_t **event)
Queue an event on a given session.
switch_status_t switch_event_base_add_header(switch_event_t *event, switch_stack_t stack, const char *header_name, char *data)
void switch_channel_set_hunt_caller_profile(switch_channel_t *channel, switch_caller_profile_t *caller_profile)
switch_call_cause_t
static switch_status_t switch_event_create_plain(switch_event_t **event, switch_event_types_t event_id)
Definition: switch_event.h:386
char * switch_channel_get_uuid(switch_channel_t *channel)
Retrieve the given channel&#39;s unique id.
switch_caller_extension_t * switch_caller_extension_new(_In_ switch_core_session_t *session, _In_z_ const char *extension_name, _In_z_ const char *extension_number)
Create a new extension with desired parameters.
void * switch_core_hash_delete(_In_ switch_hash_t *hash, _In_z_ const char *key)
Delete data from a hash based on desired key.
switch_device_state_t
void switch_core_session_rwunlock(_In_ switch_core_session_t *session)
Unlock a read or write lock on as given session.
switch_status_t switch_channel_dtmf_lock(switch_channel_t *channel)
switch_status_t switch_channel_get_variables_prefix(switch_channel_t *channel, const char *prefix, switch_event_t **event)
char * switch_core_session_get_uuid(_In_ switch_core_session_t *session)
Retrieve the unique identifier from a session.
switch_channel_callstate_t callstate
Definition: switch_core.h:85
switch_mutex_t * mutex
switch_status_t switch_channel_set_variable_var_check(switch_channel_t *channel, const char *varname, const char *value, switch_bool_t var_check)
uint32_t switch_channel_test_cap(switch_channel_t *channel, switch_channel_cap_t cap)
switch_channel_state_t switch_channel_perform_set_running_state(switch_channel_t *channel, switch_channel_state_t state, const char *file, const char *func, int line)
void switch_channel_set_app_flag_key(const char *key, switch_channel_t *channel, uint32_t flags)
#define SWITCH_CHANNEL_EXECUTE_ON_RING_VARIABLE
Definition: switch_types.h:153
void switch_channel_perform_set_callstate(switch_channel_t *channel, switch_channel_callstate_t callstate, const char *file, const char *func, int line)
#define switch_str_nil(s)
Make a null string a blank string instead.
Definition: switch_utils.h:993
switch_status_t switch_channel_unbind_device_state_handler(switch_device_state_function_t function)
struct fspr_thread_mutex_t switch_mutex_t
Definition: switch_apr.h:314
switch_device_state_t device_state
unsigned int switch_core_session_running(_In_ switch_core_session_t *session)
determine if the session&#39;s state machine is running
const char * switch_channel_get_variable_partner(switch_channel_t *channel, const char *varname)
uint32_t switch_core_default_dtmf_duration(uint32_t duration)
Definition: switch_core.c:1721
switch_channel_state_t
Channel States (these are the defaults, CS_SOFT_EXECUTE, CS_EXCHANGE_MEDIA, and CS_CONSUME_MEDIA are ...
#define switch_channel_expand_variables(_channel, _in)
void switch_channel_clear_cap(switch_channel_t *channel, switch_channel_cap_t cap)
switch_stream_handle_write_function_t write_function
static void add_uuid(switch_device_record_t *drec, switch_channel_t *channel)
#define switch_channel_down(_channel)
const char * switch_channel_set_device_id(switch_channel_t *channel, const char *device_id)
#define switch_channel_set_callstate(channel, state)
#define SWITCH_DTMF_LOG_LEN
Definition: switch_types.h:237
void switch_channel_step_caller_profile(switch_channel_t *channel)
#define SWITCH_PASSTHRU_PTIME_MISMATCH_VARIABLE
Definition: switch_types.h:135
void switch_channel_clear_private_flag(switch_channel_t *channel, uint32_t flags)
#define SWITCH_SIZE_T_FMT
void switch_channel_uninit(switch_channel_t *channel)
Uninitalize a channel.
switch_status_t
Common return values.
void switch_core_session_wake_video_thread(switch_core_session_t *session)
#define MESSAGE_STAMP_FFL(_m)
Definition: switch_core.h:175
switch_memory_pool_t * pool
Definition: switch_core.h:141
char * switch_string_replace(const char *string, const char *search, const char *replace)
void * switch_channel_get_private(switch_channel_t *channel, const char *key)
Retrieve private from a given channel.
static struct switch_callstate_table CALLSTATE_CHART[]
static switch_status_t create_device_record(switch_device_record_t **drecp, const char *device_id)
switch_core_session_t * switch_channel_get_session(switch_channel_t *channel)
#define SWITCH_ENABLE_HEARTBEAT_EVENTS_VARIABLE
Definition: switch_types.h:136
struct switch_event_header * next
Definition: switch_event.h:76
switch_call_direction_t direction
#define SWITCH_SIGNAL_BOND_VARIABLE
Definition: switch_types.h:203
uint32_t caps[CC_FLAG_MAX]
#define check_media(session)
struct switch_device_record_s * parent
Definition: switch_core.h:89
const switch_state_handler_table_t * switch_channel_get_state_handler(switch_channel_t *channel, int index)
Retrieve an state handler tablefrom a given channel at given index level.
void switch_channel_variable_last(switch_channel_t *channel)
Stop iterating over channel variables.
void switch_channel_set_originatee_caller_profile(switch_channel_t *channel, switch_caller_profile_t *caller_profile)
Set the given channel&#39;s originatee caller profile.
switch_device_record_t * switch_channel_get_device_record(switch_channel_t *channel)
switch_mutex_t * dtmf_mutex
switch_status_t switch_core_session_start_video_thread(switch_core_session_t *session)
#define switch_core_hash_insert(_h, _k, _d)
Definition: switch_core.h:1479
static struct switch_device_state_table DEVICE_STATE_CHART[]
#define switch_core_session_locate(uuid_str)
Locate a session based on it&#39;s uuid.
Definition: switch_core.h:932
int switch_channel_state_change_pending(switch_channel_t *channel)
switch_channel_state_t state
Main Library Header.
struct switch_caller_profile * hunt_caller_profile
#define SWITCH_B_SDP_VARIABLE
Definition: switch_types.h:199
switch_call_cause_t switch_channel_get_cause_q850(switch_channel_t *channel)
#define switch_event_create(event, id)
Create a new event assuming it will not be custom event and therefore hiding the unused parameters...
Definition: switch_event.h:384
switch_status_t switch_ivr_parse_signal_data(switch_core_session_t *session, switch_bool_t all, switch_bool_t only_session_thread)
Definition: switch_ivr.c:869
uint32_t switch_channel_del_variable_prefix(switch_channel_t *channel, const char *prefix)
#define SWITCH_DECLARE(type)
#define SWITCH_TEMP_HOLD_MUSIC_VARIABLE
Definition: switch_types.h:194
#define switch_event_get_header(_e, _h)
Definition: switch_event.h:172
#define switch_channel_set_flag(_c, _f)
switch_status_t switch_channel_set_name(switch_channel_t *channel, const char *name)
Assign a name to a given channel.
#define SWITCH_CHANNEL_NAME_VARIABLE
Definition: switch_types.h:183
switch_call_cause_t cause
struct device_uuid_node_s * uuid_list
Definition: switch_core.h:138
switch_call_direction_t switch_channel_direction(switch_channel_t *channel)
switch_app_log_t * switch_core_session_get_app_log(_In_ switch_core_session_t *session)
switch_mutex_t * thread_mutex
switch_call_cause_t switch_channel_cause_q850(switch_call_cause_t cause)
switch_status_t switch_queue_trypush(switch_queue_t *queue, void *data)
Definition: switch_apr.c:1279
struct switch_channel_timetable * times
static void careful_set(switch_channel_t *channel, switch_channel_state_t *state, switch_channel_state_t val)
switch_caller_profile_t * switch_caller_profile_dup(_In_ switch_memory_pool_t *pool, _In_ switch_caller_profile_t *tocopy)
Duplicate an existing caller profile object.
char * switch_channel_build_param_string(switch_channel_t *channel, switch_caller_profile_t *caller_profile, const char *prefix)
#define SWITCH_CHANNEL_EXECUTE_ON_PRE_ANSWER_VARIABLE
Definition: switch_types.h:151
const char * caller_id_number
Definition: switch_caller.h:81
switch_mutex_t * device_mutex
int switch_channel_add_state_handler(switch_channel_t *channel, const switch_state_handler_table_t *state_handler)
add a state handler table to a given channel
char * key
Definition: switch_msrp.c:64
#define switch_core_session_alloc(_session, _memory)
Allocate memory from a session&#39;s pool.
Definition: switch_core.h:696
switch_stack_t
Expression of how to stack a list.
#define switch_test_flag(obj, flag)
Test for the existance of a flag on an arbitary object.
Definition: switch_utils.h:693
switch_dtmf_source_t source
Definition: switch_types.h:304
void switch_channel_clear_flag_recursive(switch_channel_t *channel, switch_channel_flag_t flag)
void switch_log_printf(_In_ switch_text_channel_t channel, _In_z_ const char *file, _In_z_ const char *func, _In_ int line, _In_opt_z_ const char *userdata, _In_ switch_log_level_t level, _In_z_ _Printf_format_string_ const char *fmt,...) PRINTF_FUNCTION(7
Write log data to the logging engine.
#define SWITCH_VA_NONE
profile_node_t * soft
A table of settings and callbacks that define a paticular implementation of a codec.
void(* switch_device_state_function_t)(switch_core_session_t *session, switch_channel_callstate_t callstate, switch_device_record_t *drec)
Definition: switch_core.h:145
unsigned int switch_core_session_started(_In_ switch_core_session_t *session)
switch_device_node_t * device_node
switch_channel_callstate_t switch_channel_get_callstate(switch_channel_t *channel)
switch_channel_state_t switch_channel_name_state(const char *name)
switch_status_t switch_ivr_transfer_recordings(switch_core_session_t *orig_session, switch_core_session_t *new_session)
switch_hold_record_t * hold_record
Definition: switch_core.h:86
switch_device_stats_t last_stats
Definition: switch_core.h:127
switch_caller_profile_t * switch_channel_get_origination_caller_profile(switch_channel_t *channel)
Retrieve the given channel&#39;s origination caller profile.
int count
Definition: switch_cJSON.h:204
struct fspr_pool_t switch_memory_pool_t
#define switch_channel_up_nosig(_channel)
switch_status_t switch_channel_perform_ring_ready_value(switch_channel_t *channel, switch_ring_ready_t rv, const char *file, const char *func, int line)
#define SWITCH_CHANNEL_API_ON_MEDIA_VARIABLE
Definition: switch_types.h:164
switch_status_t switch_queue_create(switch_queue_t **queue, unsigned int queue_capacity, switch_memory_pool_t *pool)
Definition: switch_apr.c:1233
#define switch_core_session_strdup(_session, _todup)
Copy a string using memory allocation from a session&#39;s pool.
Definition: switch_core.h:719
switch_hold_record_t * switch_channel_get_hold_record(switch_channel_t *channel)
void switch_event_destroy(switch_event_t **event)
Destroy an event.
switch_channel_callstate_t callstate
switch_time_t last_call_time
Definition: switch_core.h:132
switch_status_t switch_channel_try_dtmf_lock(switch_channel_t *channel)
void switch_channel_set_direction(switch_channel_t *channel, switch_call_direction_t direction)
switch_caller_extension_t * switch_channel_get_queued_extension(switch_channel_t *channel)
switch_status_t switch_channel_get_log_tags(switch_channel_t *channel, switch_event_t **log_tags)
int32_t switch_core_session_ctl(switch_session_ctl_t cmd, void *val)
send a control message to the core
Definition: switch_core.c:2651
switch_status_t switch_channel_wait_for_flag(switch_channel_t *channel, switch_channel_flag_t want_flag, switch_bool_t pres, uint32_t to, switch_channel_t *super_channel)
void switch_channel_clear_flag(switch_channel_t *channel, switch_channel_flag_t flag)
Clear given flag(s) from a channel.
int switch_channel_test_app_flag_key(const char *key, switch_channel_t *channel, uint32_t flags)
char * switch_core_sprintf(_In_ switch_memory_pool_t *pool, _In_z_ _Printf_format_string_ const char *fmt,...)
printf-style style printing routine. The data is output to a string allocated from the pool ...
void switch_channel_transfer_to_extension(switch_channel_t *channel, switch_caller_extension_t *caller_extension)
#define switch_assert(expr)
uint8_t state_flags[CF_FLAG_MAX]
#define switch_channel_set_variable(_channel, _var, _val)
switch_time_t switch_time_now(void)
Definition: switch_apr.c:325
switch_status_t switch_channel_init(switch_channel_t *channel, switch_core_session_t *session, switch_channel_state_t state, switch_channel_flag_t flag)
Connect a newly allocated channel to a session object and setup it&#39;s initial state.
switch_status_t switch_ivr_generate_xml_cdr(switch_core_session_t *session, switch_xml_t *xml_cdr)
Generate an XML CDR report.
Definition: switch_ivr.c:2862
switch_caller_profile_t * switch_channel_get_caller_profile(switch_channel_t *channel)
Retrieve the given channel&#39;s caller profile.
switch_hash_t * device_hash
#define switch_core_session_kill_channel(session, sig)
Send a signal to a channel.
Definition: switch_core.h:1393
void switch_channel_release_device_record(switch_device_record_t **drecp)
char * switch_channel_get_name(switch_channel_t *channel)
Retrieve the name of a given channel.
switch_core_session_t * session
switch_status_t switch_channel_pass_callee_id(switch_channel_t *channel, switch_channel_t *other_channel)
switch_event_t * var_list
SWITCH_BEGIN_EXTERN_C char * switch_mprintf(const char *zFormat,...)
switch_status_t switch_channel_state_thread_trylock(switch_channel_t *channel)
memset(buf, 0, buflen)
#define switch_channel_media_up(_channel)
#define SWITCH_CHANNEL_API_ON_RING_VARIABLE
Definition: switch_types.h:165
#define SWITCH_CHANNEL_CHANNEL_LOG(x)
const char * callee_id_name
Definition: switch_caller.h:87
switch_status_t switch_channel_execute_on_value(switch_channel_t *channel, const char *variable_value)
switch_status_t switch_channel_set_variable_strip_quotes_var_check(switch_channel_t *channel, const char *varname, const char *value, switch_bool_t var_check)
switch_time_t off
Definition: switch_core.h:76
switch_bool_t switch_channel_clear_flag_partner(switch_channel_t *channel, switch_channel_flag_t flag)
Clears given flag(s) on a given channel&#39;s bridge partner.
switch_bool_t switch_core_session_in_thread(switch_core_session_t *session)
switch_call_cause_t hangup_cause
switch_memory_pool_t * switch_core_session_get_pool(_In_ switch_core_session_t *session)
Retrieve the memory pool from a session.
switch_time_t progress_media
void switch_channel_perform_presence(switch_channel_t *channel, const char *rpid, const char *status, const char *id, const char *file, const char *func, int line)
Fire A presence event for the channel.
switch_status_t switch_channel_perform_mark_answered(switch_channel_t *channel, const char *file, const char *func, int line)
struct switch_device_state_binding_s * next
switch_event_header_t * headers
Definition: switch_event.h:90
switch_status_t switch_channel_api_on(switch_channel_t *channel, const char *variable_prefix)
switch_status_t switch_channel_execute_on(switch_channel_t *channel, const char *variable_prefix)
struct switch_app_log * next
Definition: switch_core.h:63
struct switch_caller_extension * caller_extension
switch_status_t switch_channel_dtmf_unlock(switch_channel_t *channel)
#define SWITCH_CHANNEL_API_ON_ANSWER_VARIABLE
Definition: switch_types.h:162
switch_channel_callstate_t switch_channel_str2callstate(const char *str)
switch_status_t switch_core_session_thread_launch(_In_ switch_core_session_t *session)
Launch the session thread (state machine) on a given session.
switch_caller_profile_t * switch_channel_get_originator_caller_profile(switch_channel_t *channel)
Retrieve the given channel&#39;s originator caller profile.
switch_status_t switch_channel_set_timestamps(switch_channel_t *channel)
switch_memory_pool_t * pool
switch_status_t switch_channel_perform_acknowledge_call(switch_channel_t *channel, const char *file, const char *func, int line)
switch_time_t hold_stop
Definition: switch_core.h:136
const char * switch_channel_cause2str(switch_call_cause_t cause)