RTS API Documentation  1.10.11
switch_core_state_machine.c
Go to the documentation of this file.
1 /*
2  * FreeSWITCH Modular Media Switching Software Library / Soft-Switch Application
3  * Copyright (C) 2005-2014, 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  * Paul D. Tinsley <pdt at jackhammer.org>
29  *
30  *
31  * switch_core_state_machine.c -- Main Core Library (state machine)
32  *
33  */
34 
35 #include <switch.h>
37 
39 {
41 
44  } else {
47  } else {
49  }
50  }
51 
53 }
54 
56 {
57  switch_caller_extension_t *extension;
58 
59  switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_DEBUG, "%s Standard HANGUP, cause: %s\n",
61 
62  if (switch_true(switch_channel_get_variable(session->channel, "log_audio_stats_on_hangup"))) {
63  switch_rtp_stats_t *audio_stats = NULL;
64 
67  if (audio_stats) {
70  "%s Call statistics:\n"
71  "in_raw_bytes: %d\n"
72  "in_media_bytes: %d\n"
73  "in_packet_count: %d\n"
74  "in_media_packet_count: %d\n"
75  "in_skip_packet_count: %d\n"
76  "in_jitter_packet_count: %d\n"
77  "in_dtmf_packet_count: %d\n"
78  "in_cng_packet_count: %d\n"
79  "in_flush_packet_count: %d\n"
80  "in_largest_jb_size: %d\n\n"
81  "in_jitter_min_variance: %lf\n"
82  "in_jitter_max_variance: %lf\n"
83  "in_jitter_loss_rate: %lf\n"
84  "in_jitter_burst_rate: %lf\n"
85  "in_mean_interval: %lf\n\n"
86  "in_flaw_total: %d\n"
87  "in_quality_percentage: %lf\n"
88  "in_mos: %lf\n\n"
89  "out_raw_bytes: %d\n"
90  "out_media_bytes: %d\n"
91  "out_packet_count: %d\n"
92  "out_media_packet_count: %d\n"
93  "out_skip_packet_count: %d\n"
94  "out_dtmf_packet_count: %d\n"
95  "out_cng_packet_count: %d\n\n"
96  "rtcp_packet_count: %d\n"
97  "rtcp_octet_count: %d\n",
99  (int)audio_stats->inbound.raw_bytes,
100  (int)audio_stats->inbound.media_bytes,
101  (int)audio_stats->inbound.packet_count,
102  (int)audio_stats->inbound.media_packet_count,
103  (int)audio_stats->inbound.skip_packet_count,
104  (int)audio_stats->inbound.jb_packet_count,
105  (int)audio_stats->inbound.dtmf_packet_count,
106  (int)audio_stats->inbound.cng_packet_count,
107  (int)audio_stats->inbound.flush_packet_count,
108  (int)audio_stats->inbound.largest_jb_size,
109  audio_stats->inbound.min_variance,
110  audio_stats->inbound.max_variance,
111  audio_stats->inbound.lossrate,
112  audio_stats->inbound.burstrate,
113  audio_stats->inbound.mean_interval,
114  (int)audio_stats->inbound.flaws,
115  audio_stats->inbound.R,
116  audio_stats->inbound.mos,
117  (int)audio_stats->outbound.raw_bytes,
118  (int)audio_stats->outbound.media_bytes,
119  (int)audio_stats->outbound.packet_count,
120  (int)audio_stats->outbound.media_packet_count,
121  (int)audio_stats->outbound.skip_packet_count,
122  (int)audio_stats->outbound.dtmf_packet_count,
123  (int)audio_stats->outbound.cng_packet_count,
124  (int)audio_stats->rtcp.packet_count,
125  (int)audio_stats->rtcp.octet_count
126  );
127  } else {
128  switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_WARNING, "%s Missing call statistics!\n",
129  switch_channel_get_name(session->channel));
130  }
131  }
132 
135 
137  return;
138  }
139 
140  if ((extension = switch_channel_get_caller_extension(session->channel)) == 0) {
141  return;
142  }
143 
144  while(extension->current_application) {
145  switch_caller_application_t *current_application = extension->current_application;
146  switch_status_t status;
147 
148  extension->current_application = extension->current_application->next;
149 
151  current_application->application_name, current_application->application_data);
152 
153 
154  if (status != SWITCH_STATUS_SUCCESS && status != SWITCH_STATUS_IGNORE) {
155  return;
156  }
157  }
158 
159 
160 
161 
162 
163 }
164 
166 {
167 
168  switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_DEBUG, "%s Standard REPORTING, cause: %s\n",
170 }
171 
173 {
174 
176 }
177 
179 {
180  switch_channel_set_variable(session->channel, "call_uuid", switch_core_session_get_uuid(session));
181 
183 
185  switch_core_session_t *other_session = NULL;
186  const char *uuid = switch_core_session_get_uuid(session);
187 
189  const char *other_uuid = switch_channel_get_partner_uuid(session->channel);
190  int x = 0;
191 
192  if (other_uuid) {
193  for (x = 0; other_session == NULL && x < 20; x++) {
194  if (!switch_channel_up(session->channel)) {
195  break;
196  }
197  other_session = switch_core_session_locate(other_uuid);
198  switch_yield(100000);
199  }
200  }
201 
202  if (other_session) {
203  switch_channel_t *other_channel = switch_core_session_get_channel(other_session);
205  switch_channel_wait_for_state_timeout(other_channel, CS_RESET, 5000);
206  switch_channel_wait_for_flag(other_channel, CF_MEDIA_ACK, SWITCH_TRUE, 2000, NULL);
207 
209  switch_ivr_signal_bridge(session, other_session);
210  } else {
211  switch_ivr_uuid_bridge(uuid, other_uuid);
212  }
213  switch_core_session_rwunlock(other_session);
214  }
215  }
216 
218  }
219 
220 }
221 
223 {
224  switch_dialplan_interface_t *dialplan_interface = NULL;
225  switch_caller_profile_t *caller_profile;
226  switch_caller_extension_t *extension = NULL;
227  char *expanded = NULL;
228  char *dpstr = NULL;
229 
231 
232  switch_channel_set_variable(session->channel, "call_uuid", switch_core_session_get_uuid(session));
233 
234  if ((switch_channel_test_flag(session->channel, CF_ANSWERED) ||
238  }
239 
240  if ((caller_profile = switch_channel_get_caller_profile(session->channel)) == 0) {
241  switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_ERROR, "Can't get profile!\n");
243  return;
244  } else {
245  char *dp[25];
246  int argc, x, count = 0;
247 
248  if ((extension = switch_channel_get_queued_extension(session->channel))) {
249  switch_channel_set_caller_extension(session->channel, extension);
251  goto end;
252  }
253 
254  if (!zstr(caller_profile->dialplan)) {
255  if ((dpstr = switch_core_session_strdup(session, caller_profile->dialplan))) {
256  expanded = switch_channel_expand_variables(session->channel, dpstr);
257  argc = switch_separate_string(expanded, ',', dp, (sizeof(dp) / sizeof(dp[0])));
258  for (x = 0; x < argc; x++) {
259  char *dpname = dp[x];
260  char *dparg = NULL;
261 
262  if (dpname) {
263  if ((dparg = strchr(dpname, ':'))) {
264  *dparg++ = '\0';
265  }
266  } else {
267  continue;
268  }
269  if (!(dialplan_interface = switch_loadable_module_get_dialplan_interface(dpname))) {
270  switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_ERROR, "Dialplan [%s] not found, skipping\n", dpname);
271  continue;
272  }
273 
274  count++;
275 
276  extension = dialplan_interface->hunt_function(session, dparg, NULL);
277  UNPROTECT_INTERFACE(dialplan_interface);
278 
279  if (extension) {
280  switch_channel_set_caller_extension(session->channel, extension);
282  goto end;
283  }
284  }
285  }
286  }
287 
288  if (!count) {
292  "No Dialplan on answered channel, changing state to HANGUP\n");
294  } else {
295  switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_DEBUG, "No Dialplan, changing state to CONSUME_MEDIA\n");
297  }
298  goto end;
299  }
300  }
301  }
302 
303  if (!extension) {
304 
306  switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_INFO, "No Route, Aborting\n");
308  }
309  }
310 
311  end:
312 
313  if (expanded && dpstr && expanded != dpstr) {
314  free(expanded);
315  }
316 }
317 
319 {
320  switch_caller_extension_t *extension;
321  const char *uuid;
322 
324 
326 
327  switch_channel_set_variable(session->channel, "call_uuid", switch_core_session_get_uuid(session));
328 
329  if (switch_channel_get_variable(session->channel, "recovered") && !switch_channel_test_flag(session->channel, CF_RECOVERED)) {
331  }
332 
333  top:
335 
339 
340  if ((extension = switch_channel_get_caller_extension(session->channel)) == 0) {
342  return;
343  }
344 
345  while (switch_channel_get_state(session->channel) == CS_EXECUTE && extension->current_application) {
346  switch_caller_application_t *current_application = extension->current_application;
347 
348  extension->current_application = extension->current_application->next;
349 
351  current_application->application_name,
352  current_application->application_data) != SWITCH_STATUS_SUCCESS) {
353  return;
354  }
355 
356  if (switch_channel_test_flag(session->channel, CF_RESET)) {
357  goto top;
358  }
359 
360  }
361 
364  (uuid = switch_channel_get_variable(session->channel, "blind_transfer_uuid"))) {
365  switch_core_session_t *other_session;
366 
367  if ((other_session = switch_core_session_locate(uuid))) {
368  switch_core_session_message_t msg = { 0 };
370  msg.from = __FILE__;
371  msg.numeric_arg = 0;
372  switch_core_session_receive_message(other_session, &msg);
373  switch_core_session_rwunlock(other_session);
374 
375  switch_channel_set_variable(session->channel, "park_timeout", "10:blind_transfer");
378  }
379  }
380 
382  switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_NOTICE, "%s has executed the last dialplan instruction, hanging up.\n",
383  switch_channel_get_name(session->channel));
385  }
386 }
387 
389 {
390 
391  switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_DEBUG, "%s Standard EXCHANGE_MEDIA\n", switch_channel_get_name(session->channel));
392 }
393 
395 {
396  switch_assert(session != NULL);
397  switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_DEBUG, "%s Standard SOFT_EXECUTE\n", switch_channel_get_name(session->channel));
398 }
399 
401 {
402  switch_assert(session != NULL);
405  switch_ivr_park(session, NULL);
406 }
407 
409 {
410  switch_assert(session != NULL);
411  switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_DEBUG, "%s Standard CONSUME_MEDIA\n", switch_channel_get_name(session->channel));
412 }
413 
415 {
416  switch_assert(session != NULL);
417  switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_DEBUG, "%s Standard HIBERNATE\n", switch_channel_get_name(session->channel));
418 }
419 
421 {
422  return;
423 }
424 
425 #define STATE_MACRO(__STATE, __STATE_STR) do { \
426  midstate = state; \
427  switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_DEBUG, "(%s) State %s\n", switch_channel_get_name(session->channel), __STATE_STR); \
428  if (state < CS_HANGUP && switch_channel_get_callstate(session->channel) == CCS_UNHELD) { \
429  switch_channel_set_callstate(session->channel, CCS_ACTIVE); \
430  } \
431  switch_core_session_request_video_refresh(session); \
432  switch_core_media_gen_key_frame(session); \
433  proceed = 1; \
434  while (do_extra_handlers && (application_state_handler = switch_channel_get_state_handler(session->channel, index++)) != 0) { \
435  if (!switch_test_flag(application_state_handler, SSH_FLAG_PRE_EXEC)) {\
436  continue; \
437  } \
438  if (!application_state_handler->on_##__STATE \
439  || (application_state_handler->on_##__STATE \
440  && application_state_handler->on_##__STATE(session) == SWITCH_STATUS_SUCCESS \
441  )) { \
442  proceed++; \
443  continue; \
444  } else { \
445  proceed = 0; \
446  break; \
447  } \
448  } \
449  index = 0; \
450  if (!proceed) global_proceed = 0; \
451  proceed = 1; \
452  while (do_extra_handlers && proceed && (application_state_handler = switch_core_get_state_handler(index++)) != 0) { \
453  if (!switch_test_flag(application_state_handler, SSH_FLAG_PRE_EXEC)) { \
454  continue; \
455  } \
456  if (!application_state_handler->on_##__STATE || \
457  (application_state_handler->on_##__STATE && \
458  application_state_handler->on_##__STATE(session) == SWITCH_STATUS_SUCCESS \
459  )) { \
460  proceed++; \
461  continue; \
462  } else { \
463  proceed = 0; \
464  break; \
465  } \
466  } \
467  index = 0; \
468  if (!proceed) global_proceed = 0; \
469  if (!driver_state_handler->on_##__STATE || (driver_state_handler->on_##__STATE(session) == SWITCH_STATUS_SUCCESS )) { \
470  while (do_extra_handlers && (application_state_handler = switch_channel_get_state_handler(session->channel, index++)) != 0) { \
471  if (switch_test_flag(application_state_handler, SSH_FLAG_PRE_EXEC)) { \
472  continue; \
473  } \
474  if (!application_state_handler->on_##__STATE \
475  || (application_state_handler->on_##__STATE \
476  && application_state_handler->on_##__STATE(session) == SWITCH_STATUS_SUCCESS \
477  )) { \
478  proceed++; \
479  continue; \
480  } else { \
481  proceed = 0; \
482  break; \
483  } \
484  } \
485  index = 0; \
486  if (!proceed) global_proceed = 0; \
487  proceed = 1; \
488  while (do_extra_handlers && proceed && (application_state_handler = switch_core_get_state_handler(index++)) != 0) { \
489  if (switch_test_flag(application_state_handler, SSH_FLAG_PRE_EXEC)) { \
490  continue; \
491  } \
492  if (!application_state_handler->on_##__STATE || \
493  (application_state_handler->on_##__STATE && \
494  application_state_handler->on_##__STATE(session) == SWITCH_STATUS_SUCCESS \
495  )) { \
496  proceed++; \
497  continue; \
498  } else { \
499  proceed = 0; \
500  break; \
501  } \
502  } \
503  if (!proceed || midstate != switch_channel_get_state(session->channel)) global_proceed = 0; \
504  if (global_proceed) { \
505  switch_core_standard_on_##__STATE(session); \
506  } \
507  } \
508  switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_DEBUG, "(%s) State %s going to sleep\n", switch_channel_get_name(session->channel), __STATE_STR); \
509  } while (silly)
510 
511 
513 {
515 
516  if (state == CS_ROUTING || state == CS_HANGUP) {
518  switch_channel_presence(session->channel, "unknown", "cancelled", NULL);
519  switch_channel_set_variable(session->channel, "presence_call_info", NULL);
520  } else {
521  switch_channel_presence(session->channel, "unknown", switch_channel_state_name(state), NULL);
522  }
523  }
524 }
525 
526 
527 
529 {
530  switch_channel_state_t state = CS_NEW, midstate = CS_DESTROY, endstate;
531  const switch_endpoint_interface_t *endpoint_interface;
532  const switch_state_handler_table_t *driver_state_handler = NULL;
533  const switch_state_handler_table_t *application_state_handler = NULL;
534  int silly = 0;
535  uint32_t new_loops = 500;
536 
537  /*
538  Life of the channel. you have channel and pool in your session
539  everywhere you go you use the session to malloc with
540  switch_core_session_alloc(session, <size>)
541 
542  The endpoint module gets the first crack at implementing the state
543  if it wants to, it can cancel the default behavior by returning SWITCH_STATUS_FALSE
544 
545  Next comes the channel's event handler table that can be set by an application
546  which also can veto the next behavior in line by returning SWITCH_STATUS_FALSE
547 
548  Finally the default state behavior is called.
549 
550 
551  */
552  switch_assert(session != NULL);
553 
555  endpoint_interface = session->endpoint_interface;
556  switch_assert(endpoint_interface != NULL);
557 
558  driver_state_handler = endpoint_interface->state_handler;
559  switch_assert(driver_state_handler != NULL);
560 
561  switch_mutex_lock(session->mutex);
562 
563  while ((state = switch_channel_get_state(session->channel)) != CS_DESTROY) {
564 
567  if ((state = switch_channel_get_state(session->channel)) == CS_DESTROY) {
568  break;
569  }
570  }
571 
572  midstate = state;
573  if (state != switch_channel_get_running_state(session->channel) || state >= CS_HANGUP) {
574  int index = 0;
575  int proceed = 1;
576  int global_proceed = 1;
577  int do_extra_handlers = 1;
580 
581  switch_channel_set_running_state(session->channel, state);
585 
586  if (session->endpoint_interface->io_routines->state_run) {
587  rstatus = session->endpoint_interface->io_routines->state_run(session);
588  }
589 
590  if (rstatus == SWITCH_STATUS_SUCCESS) {
591  for (ptr = session->event_hooks.state_run; ptr; ptr = ptr->next) {
592  if (ptr->state_run(session) != SWITCH_STATUS_SUCCESS) {
593  break;
594  }
595  }
596  }
597 
598  switch (state) {
599  case CS_NEW: /* Just created, Waiting for first instructions */
601  break;
602  case CS_DESTROY:
603  goto done;
604  case CS_REPORTING: /* Call Detail */
605  {
608  }
609  goto done;
610  case CS_HANGUP: /* Deactivate and end the thread */
611  {
613  if (switch_channel_test_flag(session->channel, CF_VIDEO)) {
615  }
617  }
618 
619  break;
620  case CS_INIT: /* Basic setup tasks */
621  {
623 
624  STATE_MACRO(init, "INIT");
625 
627  switch_channel_event_set_data(session->channel, event);
628  switch_event_fire(&event);
629  }
630 
633  switch_channel_event_set_data(session->channel, event);
634  switch_event_fire(&event);
635  }
636  }
637  }
638  break;
639  case CS_ROUTING: /* Look for a dialplan and find something to do */
640  STATE_MACRO(routing, "ROUTING");
641  break;
642  case CS_RESET: /* Reset */
643  STATE_MACRO(reset, "RESET");
644  break;
645  /* These other states are intended for prolonged durations so we do not signal lock for them */
646  case CS_EXECUTE: /* Execute an Operation */
647  STATE_MACRO(execute, "EXECUTE");
648  break;
649  case CS_EXCHANGE_MEDIA: /* loop all data back to source */
650  STATE_MACRO(exchange_media, "EXCHANGE_MEDIA");
651  break;
652  case CS_SOFT_EXECUTE: /* send/recieve data to/from another channel */
653  STATE_MACRO(soft_execute, "SOFT_EXECUTE");
654  break;
655  case CS_PARK: /* wait in limbo */
656  STATE_MACRO(park, "PARK");
657  break;
658  case CS_CONSUME_MEDIA: /* wait in limbo */
659  STATE_MACRO(consume_media, "CONSUME_MEDIA");
660  break;
661  case CS_HIBERNATE: /* sleep */
662  STATE_MACRO(hibernate, "HIBERNATE");
663  break;
664  case CS_NONE:
665  abort();
666  break;
667  }
668 
669  check_presence(session);
670 
671  if (midstate == CS_DESTROY) {
672  break;
673  }
674 
675  }
676 
677  endstate = switch_channel_get_state(session->channel);
678 
679  if (endstate == switch_channel_get_running_state(session->channel)) {
680  if (endstate == CS_NEW) {
681  switch_yield(20000);
683  if (!--new_loops) {
684  switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_WARNING, "%s %s Abandoned\n",
685  session->uuid_str, switch_core_session_get_name(session));
688  }
689  } else {
691 
693 
696  } else if (switch_channel_get_state(session->channel) == switch_channel_get_running_state(session->channel)) {
698  switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_DEBUG1, "%s session thread sleep state: %s!\n",
701  switch_thread_cond_wait(session->cond, session->mutex);
702  switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_DEBUG1, "%s session thread wake state: %s!\n",
706  }
707 
709 
711  }
712  }
713  }
714  done:
715  switch_mutex_unlock(session->mutex);
716 
718 }
719 
721 {
723  const switch_endpoint_interface_t *endpoint_interface;
724  const switch_state_handler_table_t *driver_state_handler = NULL;
725  const switch_state_handler_table_t *application_state_handler = NULL;
726  int proceed = 1;
727  int global_proceed = 1;
728  int do_extra_handlers = 1;
729  int silly = 0;
730  int index = 0;
731 
733 
734  switch_assert(session != NULL);
738 
739  endpoint_interface = session->endpoint_interface;
740  switch_assert(endpoint_interface != NULL);
741 
742  driver_state_handler = endpoint_interface->state_handler;
743  switch_assert(driver_state_handler != NULL);
744 
745  STATE_MACRO(destroy, "DESTROY");
746 
748 
749  return;
750 }
751 
752 static void api_hook(switch_core_session_t *session, const char *hook_var, int use_session)
753 {
754  if (!zstr(hook_var)) {
755  switch_stream_handle_t stream = { 0 };
756  char *cmd = strdup(hook_var);
757  char *arg = NULL;
758  char *expanded = NULL;
759 
760  if ((arg = strchr(cmd, ':')) && *(arg + 1) == ':') {
761  *arg++ = '\0';
762  *arg++ = '\0';
763  } else {
764  if ((arg = strchr(cmd, ' '))) {
765  *arg++ = '\0';
766  }
767  }
768 
769  SWITCH_STANDARD_STREAM(stream);
770 
773  expanded = switch_event_expand_headers(stream.param_event, arg);
774 
775  switch_api_execute(cmd, expanded, use_session ? session : NULL, &stream);
776 
777  switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_DEBUG, "Hangup Command %s %s(%s):\n%s\n",
778  use_session ? "with Session" : "with no Session", cmd, switch_str_nil(expanded),
779  switch_str_nil((char *) stream.data) );
780 
781  if (expanded != arg) {
782  switch_safe_free(expanded);
783  }
784 
785  switch_safe_free(cmd);
786 
787  switch_safe_free(stream.data);
788  }
789 }
790 
791 
792 
794 {
797  int proceed = 1;
798  int global_proceed = 1;
799  int do_extra_handlers = 1;
800  int silly = 0;
801  int index = 0;
803  const switch_endpoint_interface_t *endpoint_interface;
804  const switch_state_handler_table_t *driver_state_handler = NULL;
805  const switch_state_handler_table_t *application_state_handler = NULL;
806  const char *hook_var;
807  int use_session = 0;
808 
809  if (!force) {
811  return;
812  }
813 
814  if (switch_thread_self() != session->thread_id) {
815  switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_DEBUG10, "%s thread mismatch skipping state handler.\n",
816  switch_channel_get_name(session->channel));
817  return;
818  }
819  }
820 
821  if (switch_test_flag(session, SSF_HANGUP)) {
822  switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_DEBUG10, "%s handler already called, skipping state handler.\n",
823  switch_channel_get_name(session->channel));
824  return;
825  }
826 
827  endpoint_interface = session->endpoint_interface;
828  switch_assert(endpoint_interface != NULL);
829 
830  driver_state_handler = endpoint_interface->state_handler;
831  switch_assert(driver_state_handler != NULL);
832 
834 
836 
838 
839  switch_channel_set_variable(session->channel, "hangup_cause", switch_channel_cause2str(cause));
840  switch_channel_set_variable_printf(session->channel, "hangup_cause_q850", "%d", cause_q850);
841  //switch_channel_presence(session->channel, "unknown", switch_channel_cause2str(cause), NULL);
842 
845 
846  STATE_MACRO(hangup, "HANGUP");
847 
849 
851 
853  use_session = 1;
854  }
855 
856  api_hook(session, hook_var, use_session);
857  }
858 
860 
861  switch_set_flag(session, SSF_HANGUP);
862 
863 }
864 
866 {
868  const switch_endpoint_interface_t *endpoint_interface;
869  const switch_state_handler_table_t *driver_state_handler = NULL;
870  const switch_state_handler_table_t *application_state_handler = NULL;
871  int proceed = 1;
872  int global_proceed = 1;
873  int do_extra_handlers = 1;
874  int silly = 0;
875  int index = 0;
877  const char *skip_var = switch_channel_get_variable(session->channel, SWITCH_SKIP_CDR_CAUSES_VARIABLE);
878  const char *hook_var;
879  int use_session = 0;
882 
884  return;
885  }
886 
888 
889  switch_assert(session != NULL);
890 
891  endpoint_interface = session->endpoint_interface;
892  switch_assert(endpoint_interface != NULL);
893 
894  driver_state_handler = endpoint_interface->state_handler;
895  switch_assert(driver_state_handler != NULL);
896 
897  if (!zstr(var)) {
898  if (!strcasecmp(var, "a_only")) {
900  do_extra_handlers = 0;
901  }
902  } else if (!strcasecmp(var, "b_only")) {
904  do_extra_handlers = 0;
905  }
906  } else if (!switch_true(var)) {
907  do_extra_handlers = 0;
908  }
909  }
910 
911 
912  if (!zstr(skip_var)) {
913  int x, ttl = 0;
914  char *list[128] = { 0 };
915  char *dup = switch_core_session_strdup(session, skip_var);
916 
917  ttl = switch_split(dup, '|', list);
918 
919  for(x = 0; x < ttl; x++) {
920  if (switch_channel_str2cause(list[x]) == cause) {
921  do_extra_handlers = 0;
922  break;
923  }
924  }
925  }
926 
927  if (switch_channel_test_flag(session->channel, CF_NO_CDR)) {
928  do_extra_handlers = 0;
929  }
930 
931 
932  STATE_MACRO(reporting, "REPORTING");
933 
935 
937  use_session = 1;
938  }
939 
940  api_hook(session, hook_var, use_session);
941  }
942 
945  switch_channel_event_set_data(session->channel, event);
946  if (switch_true(switch_channel_get_variable(session->channel, "hangup_complete_with_xml"))) {
947  switch_xml_t cdr = NULL;
948  char *xml_cdr_text;
949 
950  if (switch_ivr_generate_xml_cdr(session, &cdr) == SWITCH_STATUS_SUCCESS) {
951  xml_cdr_text = switch_xml_toxml(cdr, SWITCH_FALSE);
952  switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "CDR-Attached", "xml");
953  switch_event_add_body(event, "%s", xml_cdr_text);
954  switch_xml_free(cdr);
955  switch_safe_free(xml_cdr_text);
956  }
957  }
958  switch_event_fire(&event);
959  }
960 
961 
962 
963  return;
964 }
965 
966 /* For Emacs:
967  * Local Variables:
968  * mode:c
969  * indent-tabs-mode:t
970  * tab-width:4
971  * c-basic-offset:4
972  * End:
973  * For VIM:
974  * vim:set softtabstop=4 shiftwidth=4 tabstop=4 noet:
975  */
switch_status_t switch_channel_set_variable_printf(switch_channel_t *channel, const char *varname, const char *fmt,...)
void switch_core_session_hangup_state(switch_core_session_t *session, switch_bool_t force)
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.
void switch_xml_free(_In_opt_ switch_xml_t xml)
frees the memory allocated for an switch_xml structure
switch_size_t flaws
Definition: switch_types.h:723
switch_caller_extension_t * switch_channel_get_caller_extension(switch_channel_t *channel)
Retrieve caller extension from a given channel.
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)
#define switch_event_expand_headers(_event, _in)
Definition: switch_event.h:355
An Abstract Representation of a dialplan extension.
void switch_core_session_reporting_state(switch_core_session_t *session)
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_rtp_numbers_t inbound
Definition: switch_types.h:771
switch_io_state_run_t state_run
switch_rtp_stats_t * switch_core_media_get_stats(switch_core_session_t *session, switch_media_type_t type, switch_memory_pool_t *pool)
switch_status_t switch_channel_set_timestamps(_In_ switch_channel_t *channel)
switch_core_session_message_types_t message_id
Definition: switch_core.h:183
void switch_core_session_reset(_In_ switch_core_session_t *session, switch_bool_t flush_dtmf, switch_bool_t reset_read_codec)
Reset the buffers and resampler on a session.
void switch_channel_set_hangup_time(switch_channel_t *channel)
Abstraction of an module endpoint interface This is the glue between the abstract idea of a "channel"...
const char * switch_channel_get_partner_uuid(switch_channel_t *channel)
#define switch_channel_set_state(channel, state)
Set the current state of a channel.
const char * switch_channel_cause2str(_In_ switch_call_cause_t cause)
return a cause string for a given cause
static void switch_core_standard_on_hibernate(switch_core_session_t *session)
#define switch_channel_stop_broadcast(_channel)
#define switch_channel_up(_channel)
void switch_channel_state_thread_lock(switch_channel_t *channel)
void switch_channel_event_set_data(_In_ switch_channel_t *channel, _In_ switch_event_t *event)
Add information about a given channel to an event object.
switch_bool_t
Definition: switch_types.h:441
#define switch_channel_presence(_a, _b, _c, _d)
static void switch_core_standard_on_destroy(switch_core_session_t *session)
switch_size_t largest_jb_size
Definition: switch_types.h:703
#define switch_split(_data, _delim, _array)
Definition: switch_utils.h:375
static void check_presence(switch_core_session_t *session)
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.
switch_memory_pool_t * pool
Representation of an event.
Definition: switch_event.h:80
#define switch_channel_ready(_channel)
switch_status_t switch_event_add_body(switch_event_t *event, const char *fmt,...) PRINTF_FUNCTION(2
Add a body to an event.
switch_io_routines_t * io_routines
An Abstract Representation of a dialplan Application.
static void switch_core_standard_on_execute(switch_core_session_t *session)
switch_status_t switch_ivr_park(switch_core_session_t *session, switch_input_args_t *args)
Definition: switch_ivr.c:946
switch_channel_state_t switch_channel_get_running_state(switch_channel_t *channel)
A representation of an XML tree.
Definition: switch_xml.h:79
void switch_channel_clear_device_record(switch_channel_t *channel)
Abstract interface to a dialplan module.
switch_status_t switch_thread_cond_wait(switch_thread_cond_t *cond, switch_mutex_t *mutex)
Definition: switch_apr.c:378
const char * switch_channel_state_name(_In_ switch_channel_state_t state)
Render the name of the provided state enum.
#define switch_core_session_get_name(_s)
Definition: switch_core.h:265
static void switch_core_standard_on_exchange_media(switch_core_session_t *session)
switch_size_t media_bytes
Definition: switch_types.h:694
const char * dialplan
Definition: switch_caller.h:77
switch_status_t switch_ivr_parse_all_events(switch_core_session_t *session)
Parse all commands from an event.
Definition: switch_ivr.c:913
void switch_channel_process_device_hangup(switch_channel_t *channel)
switch_status_t switch_ivr_media(const char *uuid, switch_media_flag_t flags)
Signal a session to request direct media access to it&#39;s remote end.
Definition: switch_ivr.c:1773
switch_size_t dtmf_packet_count
Definition: switch_types.h:700
struct switch_runtime runtime
Definition: switch_core.c:86
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.
struct switch_io_event_hook_state_run * next
#define SWITCH_SKIP_CDR_CAUSES_VARIABLE
Definition: switch_types.h:180
A message object designed to allow unlike technologies to exchange data.
Definition: switch_core.h:179
#define zstr(x)
Definition: switch_utils.h:314
switch_caller_profile_t * switch_channel_get_originatee_caller_profile(switch_channel_t *channel)
Retrieve the given channel&#39;s originatee caller profile.
struct switch_caller_application * next
switch_io_event_hooks_t event_hooks
switch_status_t switch_mutex_unlock(switch_mutex_t *lock)
Definition: switch_apr.c:313
#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.
#define UNPROTECT_INTERFACE(_it)
switch_state_handler_table_t * state_handler
#define switch_clear_flag(obj, flag)
Clear a flag on an arbitrary object while locked.
Definition: switch_utils.h:724
switch_status_t switch_ivr_blind_transfer_ack(switch_core_session_t *session, switch_bool_t success)
Definition: switch_ivr.c:4418
static void switch_core_standard_on_consume_media(switch_core_session_t *session)
switch_rtcp_numbers_t rtcp
Definition: switch_types.h:773
#define SWITCH_PROCESS_CDR_VARIABLE
Definition: switch_types.h:179
#define switch_yield(ms)
Wait a desired number of microseconds and yield the CPU.
Definition: switch_utils.h:998
#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.
switch_channel_t * channel
static void switch_core_standard_on_init(switch_core_session_t *session)
switch_size_t raw_bytes
Definition: switch_types.h:693
switch_dialplan_hunt_function_t hunt_function
void switch_core_recovery_untrack(switch_core_session_t *session, switch_bool_t force)
switch_status_t switch_mutex_lock(switch_mutex_t *lock)
Definition: switch_apr.c:308
#define SWITCH_API_REPORTING_HOOK_VARIABLE
Definition: switch_types.h:177
#define switch_channel_get_variable(_c, _v)
int index
Definition: switch_cJSON.h:160
#define switch_channel_video_sync(_c)
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.
#define switch_channel_set_running_state(channel, state)
#define switch_safe_free(it)
Free a pointer and set it to NULL unless it already is NULL.
Definition: switch_utils.h:885
switch_rtp_numbers_t outbound
Definition: switch_types.h:772
switch_status_t switch_ivr_uuid_bridge(const char *originator_uuid, const char *originatee_uuid)
Bridge two existing sessions.
switch_status_t switch_channel_get_variables(switch_channel_t *channel, switch_event_t **event)
#define SWITCH_SESSION_IN_HANGUP_HOOK_VARIABLE
Definition: switch_types.h:178
Node in which to store state run callback hooks.
static void switch_core_standard_on_park(switch_core_session_t *session)
switch_call_cause_t switch_channel_str2cause(_In_ const char *str)
return a cause code for a given string
void switch_channel_set_caller_extension(switch_channel_t *channel, switch_caller_extension_t *caller_extension)
Assign a caller extension to a given channel.
switch_thread_cond_t * cond
char uuid_str[SWITCH_UUID_FORMATTED_LENGTH+1]
#define SWITCH_STANDARD_STREAM(s)
void switch_channel_wait_for_state_timeout(switch_channel_t *other_channel, switch_channel_state_t want_state, uint32_t timeout)
switch_size_t packet_count
Definition: switch_types.h:695
switch_call_cause_t
switch_caller_application_t * current_application
#define switch_core_session_receive_message(_session, _message)
Definition: switch_core.h:1247
void switch_core_session_rwunlock(_In_ switch_core_session_t *session)
Unlock a read or write lock on as given session.
void switch_core_session_video_reset(switch_core_session_t *session)
char * switch_core_session_get_uuid(_In_ switch_core_session_t *session)
Retrieve the unique identifier from a session.
switch_size_t jb_packet_count
Definition: switch_types.h:699
static void switch_core_standard_on_soft_execute(switch_core_session_t *session)
switch_size_t skip_packet_count
Definition: switch_types.h:698
void switch_core_media_set_stats(switch_core_session_t *session)
#define switch_str_nil(s)
Make a null string a blank string instead.
Definition: switch_utils.h:993
static void switch_core_standard_on_reset(switch_core_session_t *session)
switch_caller_extension_t * switch_channel_get_queued_extension(switch_channel_t *channel)
static void switch_core_standard_on_reporting(switch_core_session_t *session)
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_core_session_run(switch_core_session_t *session)
#define switch_channel_set_callstate(channel, state)
switch_status_t
Common return values.
void switch_core_session_wake_video_thread(switch_core_session_t *session)
switch_dialplan_interface_t * switch_loadable_module_get_dialplan_interface(const char *name)
Retrieve the dialplan interface by it&#39;s registered name.
#define switch_core_session_locate(uuid_str)
Locate a session based on it&#39;s uuid.
Definition: switch_core.h:932
Main Library Header.
#define switch_core_media_bug_remove_all(_s)
Definition: switch_core.h:405
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_size_t flush_packet_count
Definition: switch_types.h:702
#define SWITCH_DECLARE(type)
#define switch_channel_set_flag(_c, _f)
switch_size_t media_packet_count
Definition: switch_types.h:697
switch_call_direction_t switch_channel_direction(switch_channel_t *channel)
static void switch_core_standard_on_hangup(switch_core_session_t *session)
static void switch_core_standard_on_routing(switch_core_session_t *session)
switch_mutex_t * mutex
static void api_hook(switch_core_session_t *session, const char *hook_var, int use_session)
#define STATE_MACRO(__STATE, __STATE_STR)
#define switch_test_flag(obj, flag)
Test for the existance of a flag on an arbitary object.
Definition: switch_utils.h:693
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.
switch_endpoint_interface_t * endpoint_interface
int count
Definition: switch_cJSON.h:204
struct fspr_pool_t switch_memory_pool_t
#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_thread_id_t switch_thread_self(void)
Definition: switch_apr.c:102
void switch_channel_state_thread_unlock(switch_channel_t *channel)
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.
switch_thread_id_t thread_id
switch_status_t switch_ivr_signal_bridge(switch_core_session_t *session, switch_core_session_t *peer_session)
Bridge Signalling from one session to another.
#define switch_assert(expr)
#define switch_channel_set_variable(_channel, _var, _val)
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
#define SWITCH_API_HANGUP_HOOK_VARIABLE
Definition: switch_types.h:176
switch_caller_profile_t * switch_channel_get_caller_profile(switch_channel_t *channel)
Retrieve the given channel&#39;s caller profile.
char * switch_channel_get_name(switch_channel_t *channel)
Retrieve the name of a given channel.
switch_memory_pool_t * switch_core_session_get_pool(_In_ switch_core_session_t *session)
Retrieve the memory pool from a session.
void switch_core_session_destroy_state(switch_core_session_t *session)
switch_call_cause_t switch_channel_get_cause(_In_ switch_channel_t *channel)
return the cause code for a given channel
#define switch_xml_toxml(xml, prn_header)
Converts an switch_xml structure back to xml in html format. Returns a string of html data that \ mus...
Definition: switch_xml.h:224
switch_caller_profile_t * switch_channel_get_originator_caller_profile(switch_channel_t *channel)
Retrieve the given channel&#39;s originator caller profile.
void switch_core_state_machine_init(switch_memory_pool_t *pool)
switch_size_t cng_packet_count
Definition: switch_types.h:701
switch_status_t switch_ivr_parse_all_messages(switch_core_session_t *session)
Definition: switch_ivr.c:847