Asterisk - The Open Source Telephony Project GIT-master-754dea3
All Data Structures Namespaces Files Functions Variables Typedefs Enumerations Enumerator Properties Macros Modules Pages
control.c
Go to the documentation of this file.
1/*
2 * Asterisk -- An open source telephony toolkit.
3 *
4 * Copyright (C) 2013, Digium, Inc.
5 *
6 * David M. Lee, II <dlee@digium.com>
7 *
8 * See http://www.asterisk.org for more information about
9 * the Asterisk project. Please do not directly contact
10 * any of the maintainers of this project for assistance;
11 * the project provides a web site, mailing lists and IRC
12 * channels for your use.
13 *
14 * This program is free software, distributed under the terms of
15 * the GNU General Public License Version 2. See the LICENSE file
16 * at the top of the source tree.
17 */
18
19/*! \file
20 *
21 * \brief Stasis application control support.
22 *
23 * \author David M. Lee, II <dlee@digium.com>
24 */
25
26#include "asterisk.h"
27
29#include "asterisk/stasis_app.h"
30#include "asterisk/causes.h"
31
32#include "command.h"
33#include "control.h"
34#include "app.h"
35#include "asterisk/dial.h"
36#include "asterisk/bridge.h"
40#include "asterisk/frame.h"
41#include "asterisk/pbx.h"
43#include "asterisk/app.h"
44
45#include <signal.h>
46
48
49/*!
50 * \brief Indicates if the Stasis app internals are being shut down
51 */
52static int shutting_down;
53
56 /*! Queue of commands to dispatch on the channel */
58 /*!
59 * The associated channel.
60 * Be very careful with the threading associated w/ manipulating
61 * the channel.
62 */
64 /*!
65 * When a channel is in a bridge, the bridge that it is in.
66 */
68 /*!
69 * Bridge features which should be applied to the channel when it enters the next bridge. These only apply to the next bridge and will be emptied thereafter.
70 */
72 /*!
73 * Holding place for channel's PBX while imparted to a bridge.
74 */
75 struct ast_pbx *pbx;
76 /*!
77 * A list of rules to check before adding a channel to a bridge.
78 */
80 /*!
81 * A list of rules to check before removing a channel from a bridge.
82 */
84 /*!
85 * Silence generator, when silence is being generated.
86 */
88 /*!
89 * The app for which this control is currently controlling.
90 * This can change through the use of the /channels/{channelId}/move
91 * command.
92 */
93 struct stasis_app *app;
94 /*!
95 * The name of the next Stasis application to move to.
96 */
97 char *next_app;
98 /*!
99 * The list of arguments to pass to StasisStart when moving to another app.
100 */
102 /*!
103 * When set, /c app_stasis should exit and continue in the dialplan.
104 */
105 unsigned int is_done:1;
106 /*!
107 * When set, /c app_stasis should exit indicating failure and continue
108 * in the dialplan.
109 */
110 unsigned int failed:1;
111};
112
113static void control_dtor(void *obj)
114{
115 struct stasis_app_control *control = obj;
116
117 ao2_cleanup(control->command_queue);
118
120 ao2_cleanup(control->app);
121
122 control_move_cleanup(control);
123
124 ast_cond_destroy(&control->wait_cond);
128
129}
130
132{
133 struct stasis_app_control *control;
134 int res;
135
136 control = ao2_alloc(sizeof(*control), control_dtor);
137 if (!control) {
138 return NULL;
139 }
140
141 AST_LIST_HEAD_INIT(&control->add_rules);
143
144 res = ast_cond_init(&control->wait_cond, NULL);
145 if (res != 0) {
146 ast_log(LOG_ERROR, "Error initializing ast_cond_t: %s\n",
147 strerror(errno));
148 ao2_ref(control, -1);
149 return NULL;
150 }
151
152 control->app = ao2_bump(app);
153
155 control->channel = channel;
156
159 if (!control->command_queue) {
160 ao2_ref(control, -1);
161 return NULL;
162 }
163
164 control->next_app = NULL;
165 AST_VECTOR_INIT(&control->next_app_args, 0);
166
167 return control;
168}
169
171 struct stasis_app_control *control,
172 struct app_control_rules *list, struct stasis_app_control_rule *obj)
173{
174 ao2_lock(control->command_queue);
175 AST_LIST_INSERT_TAIL(list, obj, next);
176 ao2_unlock(control->command_queue);
177}
178
180 struct stasis_app_control *control,
181 struct app_control_rules *list, struct stasis_app_control_rule *obj)
182{
184
185 ao2_lock(control->command_queue);
187 if (rule == obj) {
189 break;
190 }
191 }
193 ao2_unlock(control->command_queue);
194}
195
196/*!
197 * \internal
198 * \brief Checks to make sure each rule in the given list passes.
199 *
200 * \details Loops over a list of rules checking for rejections or failures.
201 * If one rule fails its resulting error code is returned.
202 *
203 * \note Command queue should be locked before calling this function.
204 *
205 * \param control The stasis application control
206 * \param list The list of rules to check
207 *
208 * \retval 0 if all rules pass
209 * \retval non-zero error code if a rule fails
210 */
212 const struct stasis_app_control *control,
213 struct app_control_rules *list)
214{
215 int res = 0;
217 AST_LIST_TRAVERSE(list, rule, next) {
218 if ((res = rule->check_rule(control))) {
219 return res;
220 }
221 }
222 return res;
223}
224
226 struct stasis_app_control *control,
228{
229 return app_control_register_rule(control, &control->add_rules, rule);
230}
231
233 struct stasis_app_control *control,
235{
236 app_control_unregister_rule(control, &control->add_rules, rule);
237}
238
240 struct stasis_app_control *control,
242{
243 return app_control_register_rule(control, &control->remove_rules, rule);
244}
245
247 struct stasis_app_control *control,
249{
250 app_control_unregister_rule(control, &control->remove_rules, rule);
251}
252
254 struct stasis_app_control *control)
255{
256 return app_control_check_rules(control, &control->add_rules);
257}
258
260 struct stasis_app_control *control)
261{
262 return app_control_check_rules(control, &control->remove_rules);
263}
264
265static int noop_cb(struct stasis_app_control *control,
266 struct ast_channel *chan, void *data)
267{
268 return 0;
269}
270
271/*! Callback type to see if the command can execute
272 note: command_queue is locked during callback */
273typedef int (*app_command_can_exec_cb)(struct stasis_app_control *control);
274
276 struct stasis_app_control *control, stasis_app_command_cb command_fn,
278 app_command_can_exec_cb can_exec_fn)
279{
280 int retval;
281 struct stasis_app_command *command;
282
283 command_fn = command_fn ? : noop_cb;
284
285 command = command_create(command_fn, data, data_destructor);
286 if (!command) {
287 return NULL;
288 }
289
290 ao2_lock(control->command_queue);
291 if (control->is_done) {
292 ao2_unlock(control->command_queue);
293 ao2_ref(command, -1);
294 return NULL;
295 }
296 if (can_exec_fn && (retval = can_exec_fn(control))) {
297 ao2_unlock(control->command_queue);
298 command_complete(command, retval);
299 return command;
300 }
301
302 ao2_link_flags(control->command_queue, command, OBJ_NOLOCK);
303 ast_cond_signal(&control->wait_cond);
304
305 if (control->channel) {
306 /* Queue a null frame so that if and when the channel is waited on,
307 return immediately to process the new command */
309 }
310
311 ao2_unlock(control->command_queue);
312
313 return command;
314}
315
317 struct stasis_app_control *control, stasis_app_command_cb command_fn,
319{
320 return exec_command_on_condition(control, command_fn, data, data_destructor, NULL);
321}
322
323static int app_control_add_role(struct stasis_app_control *control,
324 struct ast_channel *chan, void *data)
325{
326 char *role = data;
327
328 return ast_channel_add_bridge_role(chan, role);
329}
330
331int stasis_app_control_add_role(struct stasis_app_control *control, const char *role)
332{
333 char *role_dup;
334
335 role_dup = ast_strdup(role);
336 if (!role_dup) {
337 return -1;
338 }
339
341
342 return 0;
343}
344
346 struct ast_channel *chan, void *data)
347{
349
350 return 0;
351}
352
354{
356}
357
359{
360 return ao2_container_count(control->command_queue);
361}
362
364{
365 /* Called from stasis_app_exec thread; no lock needed */
366 return control->is_done;
367}
368
370{
371 /* Locking necessary to sync with other threads adding commands to the queue. */
372 ao2_lock(control->command_queue);
373 control->is_done = 1;
374 ao2_unlock(control->command_queue);
375}
376
378{
379 control->failed = 1;
380}
381
383{
384 return control->failed;
385}
386
387
392};
393
394static int app_control_continue(struct stasis_app_control *control,
395 struct ast_channel *chan, void *data)
396{
397 struct stasis_app_control_continue_data *continue_data = data;
398
399 ast_assert(control->channel != NULL);
400
401 /* If we're in a Stasis bridge, depart it before going back to the
402 * dialplan */
403 if (stasis_app_get_bridge(control)) {
404 ast_bridge_depart(control->channel);
405 }
406
407 /* Called from stasis_app_exec thread; no lock needed */
408 ast_explicit_goto(control->channel, continue_data->context, continue_data->extension, continue_data->priority);
409
410 control_mark_done(control);
411
412 return 0;
413}
414
415int stasis_app_control_continue(struct stasis_app_control *control, const char *context, const char *extension, int priority)
416{
417 struct stasis_app_control_continue_data *continue_data;
418
419 if (!(continue_data = ast_calloc(1, sizeof(*continue_data)))) {
420 return -1;
421 }
422 ast_copy_string(continue_data->context, S_OR(context, ""), sizeof(continue_data->context));
423 ast_copy_string(continue_data->extension, S_OR(extension, ""), sizeof(continue_data->extension));
424 if (priority > 0) {
425 continue_data->priority = priority;
426 } else {
427 continue_data->priority = -1;
428 }
429
431
432 return 0;
433}
434
436 char *app_name;
437 char *app_args;
438};
439
440static int app_control_move(struct stasis_app_control *control,
441 struct ast_channel *chan, void *data)
442{
443 struct stasis_app_control_move_data *move_data = data;
444
445 control->next_app = ast_strdup(move_data->app_name);
446 if (!control->next_app) {
447 ast_log(LOG_ERROR, "Allocation failed for next app\n");
448 return -1;
449 }
450
451 if (move_data->app_args) {
452 char *token;
453
454 while ((token = strtok_r(move_data->app_args, ",", &move_data->app_args))) {
455 int res;
456 char *arg;
457
458 if (!(arg = ast_strdup(token))) {
459 ast_log(LOG_ERROR, "Allocation failed for next app arg\n");
460 control_move_cleanup(control);
461 return -1;
462 }
463
464 res = AST_VECTOR_APPEND(&control->next_app_args, arg);
465 if (res) {
466 ast_log(LOG_ERROR, "Failed to append arg to next app args\n");
467 ast_free(arg);
468 control_move_cleanup(control);
469 return -1;
470 }
471 }
472 }
473
474 return 0;
475}
476
477int stasis_app_control_move(struct stasis_app_control *control, const char *app_name, const char *app_args)
478{
479 struct stasis_app_control_move_data *move_data;
480 size_t size;
481
482 size = sizeof(*move_data) + strlen(app_name) + 1;
483 if (app_args) {
484 /* Application arguments are optional */
485 size += strlen(app_args) + 1;
486 }
487
488 if (!(move_data = ast_calloc(1, size))) {
489 return -1;
490 }
491
492 move_data->app_name = (char *)move_data + sizeof(*move_data);
493 strcpy(move_data->app_name, app_name); /* Safe */
494
495 if (app_args) {
496 move_data->app_args = move_data->app_name + strlen(app_name) + 1;
497 strcpy(move_data->app_args, app_args); /* Safe */
498 } else {
499 move_data->app_args = NULL;
500 }
501
503
504 return 0;
505}
506
507static int app_control_redirect(struct stasis_app_control *control,
508 struct ast_channel *chan, void *data)
509{
510 char *endpoint = data;
511 int res;
512
513 ast_assert(control->channel != NULL);
514 ast_assert(endpoint != NULL);
515
516 res = ast_transfer(control->channel, endpoint);
517 if (!res) {
518 ast_log(LOG_NOTICE, "Unsupported transfer requested on channel '%s'\n",
519 ast_channel_name(control->channel));
520 return 0;
521 }
522
523 return 0;
524}
525
526int stasis_app_control_redirect(struct stasis_app_control *control, const char *endpoint)
527{
528 char *endpoint_data = ast_strdup(endpoint);
529
530 if (!endpoint_data) {
531 return -1;
532 }
533
535
536 return 0;
537}
538
542 unsigned int duration;
543 int after;
544 char dtmf[];
545};
546
547static void dtmf_in_bridge(struct ast_channel *chan, struct stasis_app_control_dtmf_data *dtmf_data)
548{
549 if (dtmf_data->before) {
550 usleep(dtmf_data->before * 1000);
551 }
552
553 ast_dtmf_stream_external(chan, dtmf_data->dtmf, dtmf_data->between, dtmf_data->duration);
554
555 if (dtmf_data->after) {
556 usleep(dtmf_data->after * 1000);
557 }
558}
559
560static void dtmf_no_bridge(struct ast_channel *chan, struct stasis_app_control_dtmf_data *dtmf_data)
561{
562 if (dtmf_data->before) {
563 ast_safe_sleep(chan, dtmf_data->before);
564 }
565
566 ast_dtmf_stream(chan, NULL, dtmf_data->dtmf, dtmf_data->between, dtmf_data->duration);
567
568 if (dtmf_data->after) {
569 ast_safe_sleep(chan, dtmf_data->after);
570 }
571}
572
573static int app_control_dtmf(struct stasis_app_control *control,
574 struct ast_channel *chan, void *data)
575{
576 struct stasis_app_control_dtmf_data *dtmf_data = data;
577
578 if (ast_channel_state(chan) != AST_STATE_UP) {
580 }
581
582 if (stasis_app_get_bridge(control)) {
583 dtmf_in_bridge(chan, dtmf_data);
584 } else {
585 dtmf_no_bridge(chan, dtmf_data);
586 }
587
588 return 0;
589}
590
591int stasis_app_control_dtmf(struct stasis_app_control *control, const char *dtmf, int before, int between, unsigned int duration, int after)
592{
593 struct stasis_app_control_dtmf_data *dtmf_data;
594
595 if (!(dtmf_data = ast_calloc(1, sizeof(*dtmf_data) + strlen(dtmf) + 1))) {
596 return -1;
597 }
598
599 dtmf_data->before = before;
600 dtmf_data->between = between;
601 dtmf_data->duration = duration;
602 dtmf_data->after = after;
603 strcpy(dtmf_data->dtmf, dtmf);
604
606
607 return 0;
608}
609
610static int app_control_ring(struct stasis_app_control *control,
611 struct ast_channel *chan, void *data)
612{
614
615 return 0;
616}
617
619{
621
622 return 0;
623}
624
625static int app_control_ring_stop(struct stasis_app_control *control,
626 struct ast_channel *chan, void *data)
627{
628 ast_indicate(control->channel, -1);
629
630 return 0;
631}
632
634{
636
637 return 0;
638}
639
642 unsigned int direction;
643};
644
645static int app_control_mute(struct stasis_app_control *control,
646 struct ast_channel *chan, void *data)
647{
648 struct stasis_app_control_mute_data *mute_data = data;
649
650 ast_channel_lock(chan);
651 ast_channel_suppress(control->channel, mute_data->direction, mute_data->frametype);
652 ast_channel_unlock(chan);
653
654 return 0;
655}
656
658{
659 struct stasis_app_control_mute_data *mute_data;
660
661 if (!(mute_data = ast_calloc(1, sizeof(*mute_data)))) {
662 return -1;
663 }
664
665 mute_data->direction = direction;
666 mute_data->frametype = frametype;
667
669
670 return 0;
671}
672
673static int app_control_unmute(struct stasis_app_control *control,
674 struct ast_channel *chan, void *data)
675{
676 struct stasis_app_control_mute_data *mute_data = data;
677
678 ast_channel_lock(chan);
679 ast_channel_unsuppress(control->channel, mute_data->direction, mute_data->frametype);
680 ast_channel_unlock(chan);
681
682 return 0;
683}
684
686{
687 struct stasis_app_control_mute_data *mute_data;
688
689 if (!(mute_data = ast_calloc(1, sizeof(*mute_data)))) {
690 return -1;
691 }
692
693 mute_data->direction = direction;
694 mute_data->frametype = frametype;
695
697
698 return 0;
699}
700
701/*!
702 * \brief structure for queuing ARI channel variable setting
703 *
704 * It may seem weird to define this custom structure given that we already have
705 * ast_var_t and ast_variable defined elsewhere. The problem with those is that
706 * they are not tolerant of NULL channel variable value pointers. In fact, in both
707 * cases, the best they could do is to have a zero-length variable value. However,
708 * when un-setting a channel variable, it is important to pass a NULL value, not
709 * a zero-length string.
710 */
711struct chanvar {
712 /*! Name of variable to set/unset */
713 char *name;
714 /*! Value of variable to set. If unsetting, this will be NULL */
715 char *value;
716};
717
718static void free_chanvar(void *data)
719{
720 struct chanvar *var = data;
721
722 ast_free(var->name);
723 ast_free(var->value);
724 ast_free(var);
725}
726
728 struct ast_channel *chan, void *data)
729{
730 struct chanvar *var = data;
731
732 pbx_builtin_setvar_helper(control->channel, var->name, var->value);
733
734 return 0;
735}
736
737int stasis_app_control_set_channel_var(struct stasis_app_control *control, const char *variable, const char *value)
738{
739 struct chanvar *var;
740
741 var = ast_calloc(1, sizeof(*var));
742 if (!var) {
743 return -1;
744 }
745
746 var->name = ast_strdup(variable);
747 if (!var->name) {
749 return -1;
750 }
751
752 /* It's kosher for value to be NULL. It means the variable is being unset */
753 if (value) {
754 var->value = ast_strdup(value);
755 if (!var->value) {
757 return -1;
758 }
759 }
760
762
763 return 0;
764}
765
766static int app_control_hold(struct stasis_app_control *control,
767 struct ast_channel *chan, void *data)
768{
770
771 return 0;
772}
773
775{
777}
778
779static int app_control_unhold(struct stasis_app_control *control,
780 struct ast_channel *chan, void *data)
781{
783
784 return 0;
785}
786
788{
790}
791
792static int app_control_moh_start(struct stasis_app_control *control,
793 struct ast_channel *chan, void *data)
794{
795 char *moh_class = data;
796
797 if (ast_channel_state(chan) != AST_STATE_UP) {
799 }
800
801 ast_moh_start(chan, moh_class, NULL);
802
803 return 0;
804}
805
806void stasis_app_control_moh_start(struct stasis_app_control *control, const char *moh_class)
807{
808 char *data = NULL;
809
810 if (!ast_strlen_zero(moh_class)) {
811 data = ast_strdup(moh_class);
812 }
813
815}
816
817static int app_control_moh_stop(struct stasis_app_control *control,
818 struct ast_channel *chan, void *data)
819{
820 ast_moh_stop(chan);
821 return 0;
822}
823
825{
827}
828
830 struct ast_channel *chan, void *data)
831{
832 if (ast_channel_state(chan) != AST_STATE_UP) {
834 }
835
836 if (control->silgen) {
837 /* We have a silence generator, but it may have been implicitly
838 * disabled by media actions (music on hold, playing media,
839 * etc.) Just stop it and restart a new one.
840 */
842 control->channel, control->silgen);
843 }
844
845 ast_debug(3, "%s: Starting silence generator\n",
848
849 if (!control->silgen) {
851 "%s: Failed to start silence generator.\n",
853 }
854
855 return 0;
856}
857
859{
861}
862
864{
865 if (control->silgen) {
866 ast_debug(3, "%s: Stopping silence generator\n",
869 control->channel, control->silgen);
870 control->silgen = NULL;
871 }
872}
873
875 struct ast_channel *chan, void *data)
876{
878 return 0;
879}
880
882{
884}
885
887 const struct stasis_app_control *control)
888{
890}
891
893 stasis_app_command_cb command_fn, void *data,
894 command_data_destructor_fn data_destructor,
895 app_command_can_exec_cb can_exec_fn)
896{
897 int ret;
898 struct stasis_app_command *command;
899
900 if (control == NULL || control->is_done) {
901 /* If exec_command_on_condition fails, it calls the data_destructor.
902 * In order to provide consistent behavior, we'll also call the data_destructor
903 * on this error path. This way, callers never have to call the
904 * data_destructor themselves.
905 */
906 if (data_destructor) {
908 }
909 return -1;
910 }
911
913 control, command_fn, data, data_destructor, can_exec_fn);
914 if (!command) {
915 return -1;
916 }
917
918 ret = command_join(command);
919 ao2_ref(command, -1);
920
921 return ret;
922}
923
926{
927 return app_send_command_on_condition(control, command_fn, data, data_destructor, NULL);
928}
929
931 stasis_app_command_cb command_fn, void *data,
933{
934 struct stasis_app_command *command;
935
936 if (control == NULL || control->is_done) {
937 /* If exec_command fails, it calls the data_destructor. In order to
938 * provide consistent behavior, we'll also call the data_destructor
939 * on this error path. This way, callers never have to call the
940 * data_destructor themselves.
941 */
942 if (data_destructor) {
944 }
945 return -1;
946 }
947
948 command = exec_command(control, command_fn, data, data_destructor);
949 if (!command) {
950 return -1;
951 }
952 ao2_ref(command, -1);
953
954 return 0;
955}
956
958{
959 struct ast_bridge *ret;
960
961 if (!control) {
962 return NULL;
963 }
964
965 ao2_lock(control);
966 ret = control->bridge;
967 ao2_unlock(control);
968
969 return ret;
970}
971
972/*!
973 * \brief Singleton dial bridge
974 *
975 * The dial bridge is a holding bridge used to hold all
976 * outbound dialed channels that are not in any "real" ARI-created
977 * bridge. The dial bridge is invisible, meaning that it does not
978 * show up in channel snapshots, AMI or ARI output, and no events
979 * get raised for it.
980 *
981 * This is used to keep dialed channels confined to the bridging system
982 * and unify the threading model used for dialing outbound channels.
983 */
984static struct ast_bridge *dial_bridge;
986
987/*!
988 * \brief Retrieve a reference to the dial bridge.
989 *
990 * If the dial bridge has not been created yet, it will
991 * be created, otherwise, a reference to the existing bridge
992 * will be returned.
993 *
994 * The caller will need to unreference the dial bridge once
995 * they are finished with it.
996 *
997 * \retval NULL Unable to find/create the dial bridge
998 * \retval non-NULL A reference to the dial bridge
999 */
1000static struct ast_bridge *get_dial_bridge(void)
1001{
1002 struct ast_bridge *ret_bridge = NULL;
1003
1005
1006 if (shutting_down) {
1007 goto end;
1008 }
1009
1010 if (dial_bridge) {
1011 ret_bridge = ao2_bump(dial_bridge);
1012 goto end;
1013 }
1014
1015 dial_bridge = stasis_app_bridge_create_invisible("holding", "dial_bridge", NULL);
1016 if (!dial_bridge) {
1017 goto end;
1018 }
1019 ret_bridge = ao2_bump(dial_bridge);
1020
1021end:
1023 return ret_bridge;
1024}
1025
1026static int bridge_channel_depart(struct stasis_app_control *control,
1027 struct ast_channel *chan, void *data);
1028
1029/*!
1030 * \brief after bridge callback for the dial bridge
1031 *
1032 * The only purpose of this callback is to ensure that the control structure's
1033 * bridge pointer is NULLed
1034 */
1035static void dial_bridge_after_cb(struct ast_channel *chan, void *data)
1036{
1037 struct stasis_app_control *control = data;
1038 struct ast_bridge_channel *bridge_channel;
1039
1041 bridge_channel = ast_channel_get_bridge_channel(chan);
1043
1044 ast_debug(3, "Channel: <%s> Reason: %d\n", ast_channel_name(control->channel), ast_channel_hangupcause(chan));
1045
1047
1048 control->bridge = NULL;
1049}
1050
1052{
1053 struct stasis_app_control *control = data;
1054
1055 ast_debug(3, "Channel: <%s> Reason: %d\n", ast_channel_name(control->channel), reason);
1056 dial_bridge_after_cb(control->channel, data);
1057}
1058
1059/*!
1060 * \brief Add a channel to the singleton dial bridge.
1061 *
1062 * \param control The Stasis control structure
1063 * \param chan The channel to add to the bridge
1064 * \retval -1 Failed
1065 * \retval 0 Success
1066 */
1067static int add_to_dial_bridge(struct stasis_app_control *control, struct ast_channel *chan)
1068{
1069 struct ast_bridge *bridge;
1070
1071 bridge = get_dial_bridge();
1072 if (!bridge) {
1073 return -1;
1074 }
1075
1076 control->bridge = bridge;
1079 control->bridge = NULL;
1080 ao2_ref(bridge, -1);
1081 return -1;
1082 }
1083
1084 ao2_ref(bridge, -1);
1085
1086 return 0;
1087}
1088
1089/*!
1090 * \brief Depart a channel from a bridge, and potentially add it back to the dial bridge
1091 *
1092 * \param control Take a guess
1093 * \param chan Take another guess
1094 */
1095static int depart_channel(struct stasis_app_control *control, struct ast_channel *chan)
1096{
1097 ast_bridge_depart(chan);
1098
1099 /* Channels which have a PBX are not ones that have been created and dialed from ARI. They
1100 * have externally come in from the dialplan, and thus should not be placed into the dial
1101 * bridge. Only channels which are created and dialed in ARI should go into the dial bridge.
1102 */
1103 if (!ast_check_hangup(chan) && ast_channel_state(chan) != AST_STATE_UP && !ast_channel_pbx(chan)) {
1104 /* Channel is still being dialed, so put it back in the dialing bridge */
1105 add_to_dial_bridge(control, chan);
1106 }
1107
1108 return 0;
1109}
1110
1112 struct ast_channel *chan, void *data)
1113{
1114 struct ast_bridge_channel *bridge_channel;
1115
1117 bridge_channel = ast_channel_internal_bridge_channel(chan);
1119
1120 if (bridge_channel != data) {
1121 ast_debug(3, "%s: Channel is no longer in departable state\n",
1123 return -1;
1124 }
1125
1126 ast_debug(3, "%s: Channel departing bridge\n",
1128
1129 depart_channel(control, chan);
1130
1131 return 0;
1132}
1133
1134static void internal_bridge_after_cb(struct ast_channel *chan, void *data,
1135 enum ast_bridge_after_cb_reason reason)
1136{
1137 struct stasis_app_control *control = data;
1138 struct ast_bridge_channel *bridge_channel;
1139
1140 ao2_lock(control);
1141 ast_debug(3, "%s, %s: %s\n",
1142 ast_channel_uniqueid(chan), control->bridge ? control->bridge->uniqueid : "unknown",
1144
1146 /* The impart actually failed so control->bridge isn't valid. */
1147 control->bridge = NULL;
1148 }
1149
1150 ast_assert(chan == control->channel);
1151
1152 /* Restore the channel's PBX */
1153 ast_channel_pbx_set(control->channel, control->pbx);
1154 control->pbx = NULL;
1155
1156 if (control->bridge) {
1157 app_unsubscribe_bridge(control->app, control->bridge);
1158
1159 /* No longer in the bridge */
1160 control->bridge = NULL;
1161
1162 /* Get the bridge channel so we don't depart from the wrong bridge */
1164 bridge_channel = ast_channel_get_bridge_channel(chan);
1166
1167 /* Depart this channel from the bridge using the command queue if possible */
1169 }
1170
1172 /* The channel has had a StasisEnd published on it, but until now had remained in
1173 * the bridging system. This means that the channel moved from a Stasis bridge to a
1174 * non-Stasis bridge and is now exiting the bridging system. Because of this, the
1175 * channel needs to exit the Stasis application and go to wherever the non-Stasis
1176 * bridge has directed it to go. If the non-Stasis bridge has not set up an after
1177 * bridge destination, then the channel should be hung up.
1178 */
1179 int hangup_flag;
1180
1183 ast_softhangup_nolock(chan, hangup_flag);
1185 }
1186 ao2_unlock(control);
1187}
1188
1189static void bridge_after_cb(struct ast_channel *chan, void *data)
1190{
1191 struct stasis_app_control *control = data;
1192
1194}
1195
1197 void *data)
1198{
1199 struct stasis_app_control *control = data;
1200
1201 internal_bridge_after_cb(control->channel, data, reason);
1202
1203 ast_debug(3, " reason: %s\n",
1205}
1206
1207/*!
1208 * \brief Dial timeout datastore
1209 *
1210 * A datastore is used because a channel may change
1211 * bridges during the course of a dial attempt. This
1212 * may be because the channel changes from the dial bridge
1213 * to a standard bridge, or it may move between standard
1214 * bridges. In order to keep the dial timeout, we need
1215 * to keep the timeout information local to the channel.
1216 * That is what this datastore is for
1217 */
1219 .type = "ARI dial timeout",
1220};
1221
1222static int hangup_channel(struct stasis_app_control *control,
1223 struct ast_channel *chan, void *data)
1224{
1225 /* Set cause code to No Answer to be consistent with other dial timeout operations */
1226 ast_channel_lock(chan);
1228 ast_channel_unlock(chan);
1230 return 0;
1231}
1232
1233/*!
1234 * \brief Dial timeout
1235 *
1236 * This is a bridge interval hook callback. The interval hook triggering
1237 * means that the dial timeout has been reached. If the channel has not
1238 * been answered by the time this callback is called, then the channel
1239 * is hung up
1240 *
1241 * \param bridge_channel Bridge channel on which interval hook has been called
1242 * \param ignore Ignored
1243 * \return -1 (i.e. remove the interval hook)
1244 */
1245static int bridge_timeout(struct ast_bridge_channel *bridge_channel, void *ignore)
1246{
1247 struct ast_datastore *datastore;
1248 RAII_VAR(struct stasis_app_control *, control, NULL, ao2_cleanup);
1249
1250 control = stasis_app_control_find_by_channel(bridge_channel->chan);
1251
1252 ast_channel_lock(bridge_channel->chan);
1253 if (ast_channel_state(bridge_channel->chan) != AST_STATE_UP) {
1254 /* Don't bother removing the datastore because it will happen when the channel is hung up */
1255 ast_channel_unlock(bridge_channel->chan);
1257 return -1;
1258 }
1259
1260 datastore = ast_channel_datastore_find(bridge_channel->chan, &timeout_datastore, NULL);
1261 if (!datastore) {
1262 ast_channel_unlock(bridge_channel->chan);
1263 return -1;
1264 }
1265 ast_channel_datastore_remove(bridge_channel->chan, datastore);
1266 ast_channel_unlock(bridge_channel->chan);
1267 ast_datastore_free(datastore);
1268
1269 return -1;
1270}
1271
1272/*!
1273 * \brief Set a dial timeout interval hook on the channel.
1274 *
1275 * The absolute time that the timeout should occur is stored on
1276 * a datastore on the channel. This time is converted into a relative
1277 * number of milliseconds in the future. Then an interval hook is set
1278 * to trigger in that number of milliseconds.
1279 *
1280 * \pre chan is locked
1281 *
1282 * \param chan The channel on which to set the interval hook
1283 */
1284static void set_interval_hook(struct ast_channel *chan)
1285{
1286 struct ast_datastore *datastore;
1287 struct timeval *hangup_time;
1288 int64_t ms;
1289 struct ast_bridge_channel *bridge_channel;
1290
1292 if (!datastore) {
1293 return;
1294 }
1295
1296 hangup_time = datastore->data;
1297
1298 ms = ast_tvdiff_ms(*hangup_time, ast_tvnow());
1299 bridge_channel = ast_channel_get_bridge_channel(chan);
1300 if (!bridge_channel) {
1301 return;
1302 }
1303
1304 if (ast_bridge_interval_hook(bridge_channel->features, 0, ms > 0 ? ms : 1,
1305 bridge_timeout, NULL, NULL, 0)) {
1306 ao2_ref(bridge_channel, -1);
1307 return;
1308 }
1309
1310 ast_queue_frame(bridge_channel->chan, &ast_null_frame);
1311 ao2_ref(bridge_channel, -1);
1312}
1313
1315{
1316 int res;
1317 struct ast_bridge_features *features;
1319
1320 if (!control || !bridge) {
1321 return -1;
1322 }
1323
1324 ast_debug(3, "%s: Adding to bridge %s\n",
1326 bridge->uniqueid);
1327
1328 ast_assert(chan != NULL);
1329
1330 /* Depart whatever Stasis bridge we're currently in. */
1331 if (stasis_app_get_bridge(control)) {
1332 /* Note that it looks like there's a race condition here, since
1333 * we don't have control locked. But this happens from the
1334 * control callback thread, so there won't be any other
1335 * concurrent attempts to bridge.
1336 */
1337 ast_bridge_depart(chan);
1338 }
1339
1340
1342 bridge_after_cb_failed, control);
1343 if (res != 0) {
1344 ast_log(LOG_ERROR, "Error setting after-bridge callback\n");
1345 return -1;
1346 }
1347
1348 ao2_lock(control);
1349
1350 /* Ensure the controlling application is subscribed early enough
1351 * to receive the ChannelEnteredBridge message. This works in concert
1352 * with the subscription handled in the Stasis application execution
1353 * loop */
1354 app_subscribe_bridge(control->app, bridge);
1355
1356 /* Save off the channel's PBX */
1357 ast_assert(control->pbx == NULL);
1358 if (!control->pbx) {
1359 control->pbx = ast_channel_pbx(chan);
1361 }
1362
1363 /* Pull bridge features from the control */
1364 features = control->bridge_features;
1365 control->bridge_features = NULL;
1366 if (features && features->inhibit_colp) {
1368 }
1369
1371 /* We need to set control->bridge here since bridge_after_cb may be run
1372 * before ast_bridge_impart returns. bridge_after_cb gets a reason
1373 * code so it can tell if the bridge is actually valid or not.
1374 */
1375 control->bridge = bridge;
1376
1377 /* We can't be holding the control lock while impart is running
1378 * or we could create a deadlock with bridge_after_cb which also
1379 * tries to lock control.
1380 */
1381 ao2_unlock(control);
1382 res = ast_bridge_impart(bridge,
1383 chan,
1384 swap,
1385 features, /* features */
1386 flags);
1387 if (res != 0) {
1388 /* ast_bridge_impart failed before it could spawn the depart
1389 * thread. The callbacks aren't called in this case.
1390 * The impart could still fail even if ast_bridge_impart returned
1391 * ok but that's handled by bridge_after_cb.
1392 */
1393 ast_log(LOG_ERROR, "Error adding channel to bridge\n");
1394 ao2_lock(control);
1395 ast_channel_pbx_set(chan, control->pbx);
1396 control->pbx = NULL;
1397 control->bridge = NULL;
1398 ao2_unlock(control);
1399 } else {
1400 ast_channel_lock(chan);
1401 set_interval_hook(chan);
1402 ast_channel_unlock(chan);
1403 }
1404
1405 return res;
1406}
1407
1408int control_add_channel_to_bridge(struct stasis_app_control *control, struct ast_channel *chan, void *data)
1409{
1410 return control_swap_channel_in_bridge(control, data, chan, NULL);
1411}
1412
1414 struct stasis_app_control *control, struct ast_bridge *bridge)
1415{
1416 ast_debug(3, "%s: Sending channel add_to_bridge command\n",
1418
1420 control, control_add_channel_to_bridge, bridge, NULL,
1422}
1423
1425 struct stasis_app_control *control,
1426 struct ast_channel *chan, void *data)
1427{
1428 struct ast_bridge *bridge = data;
1429
1430 if (!control) {
1431 return -1;
1432 }
1433
1434 /* We should only depart from our own bridge */
1435 ast_debug(3, "%s: Departing bridge %s\n",
1437 bridge->uniqueid);
1438
1439 if (bridge != stasis_app_get_bridge(control)) {
1440 ast_log(LOG_WARNING, "%s: Not in bridge %s; not removing\n",
1442 bridge->uniqueid);
1443 return -1;
1444 }
1445
1446 depart_channel(control, chan);
1447 return 0;
1448}
1449
1451 struct stasis_app_control *control, struct ast_bridge *bridge)
1452{
1453 ast_debug(3, "%s: Sending channel remove_from_bridge command\n",
1458}
1459
1461 const struct stasis_app_control *control)
1462{
1463 return ast_channel_uniqueid(control->channel);
1464}
1465
1467 struct stasis_app_control *control, struct stasis_message *message)
1468{
1469 if (!control || !control->channel || !message) {
1470 return;
1471 }
1473}
1474
1477{
1478 return ast_queue_control(control->channel, frame_type);
1479}
1480
1482 struct stasis_app_control *control)
1483{
1484 struct ast_bridge_features *features;
1485
1486 features = ast_bridge_features_new();
1487 if (!features) {
1488 return 1;
1489 }
1490 control->bridge_features = features;
1491 return 0;
1492}
1493
1495 struct stasis_app_control *control, int absorb)
1496{
1497 control->bridge_features->dtmf_passthrough = !absorb;
1498}
1499
1501 struct stasis_app_control *control, int mute)
1502{
1503 control->bridge_features->mute = mute;
1504}
1505
1507 struct stasis_app_control *control, int inhibit_colp)
1508{
1510}
1511
1513{
1514 struct ao2_iterator iter;
1515 struct stasis_app_command *command;
1516
1518 while ((command = ao2_iterator_next(&iter))) {
1519 command_complete(command, -1);
1520 ao2_ref(command, -1);
1521 }
1522 ao2_iterator_destroy(&iter);
1523}
1524
1526 struct ast_channel *chan)
1527{
1528 int count = 0;
1529 struct ao2_iterator iter;
1530 struct stasis_app_command *command;
1531
1532 ast_assert(control->channel == chan);
1533
1535 while ((command = ao2_iterator_next(&iter))) {
1536 command_invoke(command, control, chan);
1537 ao2_ref(command, -1);
1538 ++count;
1539 }
1540 ao2_iterator_destroy(&iter);
1541
1542 return count;
1543}
1544
1546{
1547 if (!control) {
1548 return;
1549 }
1550
1551 ast_assert(control->command_queue != NULL);
1552
1553 ao2_lock(control->command_queue);
1554 while (ao2_container_count(control->command_queue) == 0) {
1555 int res = ast_cond_wait(&control->wait_cond,
1557 if (res < 0) {
1558 ast_log(LOG_ERROR, "Error waiting on command queue\n");
1559 break;
1560 }
1561 }
1562 ao2_unlock(control->command_queue);
1563}
1564
1566 struct ast_channel *chan)
1567{
1568 struct ao2_container *command_queue;
1569 int count = 0;
1570 struct ao2_iterator iter;
1571 struct stasis_app_command *command;
1572
1573 ast_channel_lock(chan);
1574 command_queue = command_prestart_get_container(chan);
1575 ast_channel_unlock(chan);
1576 if (!command_queue) {
1577 return 0;
1578 }
1579
1580 iter = ao2_iterator_init(command_queue, AO2_ITERATOR_UNLINK);
1581
1582 while ((command = ao2_iterator_next(&iter))) {
1583 command_invoke(command, control, chan);
1584 ao2_cleanup(command);
1585 ++count;
1586 }
1587
1588 ao2_iterator_destroy(&iter);
1589 ao2_cleanup(command_queue);
1590 return count;
1591}
1592
1594{
1595 return control->app;
1596}
1597
1599 unsigned int timeout;
1600 char dialstring[0];
1601};
1602
1604 unsigned int timeout)
1605{
1606 struct control_dial_args *args;
1607
1608 args = ast_malloc(sizeof(*args) + strlen(dialstring) + 1);
1609 if (!args) {
1610 return NULL;
1611 }
1612
1613 args->timeout = timeout;
1614 /* Safe */
1615 strcpy(args->dialstring, dialstring);
1616
1617 return args;
1618}
1619
1620static void control_dial_args_destroy(void *data)
1621{
1622 struct control_dial_args *args = data;
1623
1624 ast_free(args);
1625}
1626
1627/*!
1628 * \brief Set dial timeout on a channel to be dialed.
1629 *
1630 * \param chan The channel on which to set the dial timeout
1631 * \param timeout The timeout in seconds
1632 */
1633static int set_timeout(struct ast_channel *chan, unsigned int timeout)
1634{
1635 struct ast_datastore *datastore;
1636 struct timeval *hangup_time;
1637
1638 hangup_time = ast_malloc(sizeof(struct timeval));
1639
1641 if (!datastore) {
1642 return -1;
1643 }
1644 *hangup_time = ast_tvadd(ast_tvnow(), ast_samp2tv(timeout, 1));
1645 datastore->data = hangup_time;
1646
1647 ast_channel_lock(chan);
1648 ast_channel_datastore_add(chan, datastore);
1649
1650 if (ast_channel_is_bridged(chan)) {
1651 set_interval_hook(chan);
1652 }
1653 ast_channel_unlock(chan);
1654
1655 return 0;
1656}
1657
1658static int app_control_dial(struct stasis_app_control *control,
1659 struct ast_channel *chan, void *data)
1660{
1661 struct control_dial_args *args = data;
1662 int bridged;
1663
1664 ast_channel_lock(chan);
1665 bridged = ast_channel_is_bridged(chan);
1666 ast_channel_unlock(chan);
1667
1668 if (!bridged && add_to_dial_bridge(control, chan)) {
1669 return -1;
1670 }
1671
1672 if (args->timeout && set_timeout(chan, args->timeout)) {
1673 return -1;
1674 }
1675
1676 if (ast_call(chan, args->dialstring, 0)) {
1677 /* If call fails normally this channel would then just be normally hung up and destroyed.
1678 * In this case though the channel is being handled by the ARI control thread and dial
1679 * bridge which needs to be notified that the channel should be hung up. To do this we
1680 * queue a soft hangup which will cause each to wake up, see that the channel has been
1681 * hung up, and then destroy it.
1682 */
1683 int hangup_flag;
1685 ast_channel_lock(chan);
1686 ast_softhangup_nolock(chan, hangup_flag);
1687 ast_channel_unlock(chan);
1688 return -1;
1689 }
1690
1691 ast_channel_publish_dial(NULL, chan, args->dialstring, NULL);
1692
1693 return 0;
1694}
1695
1697 const char *dialstring, unsigned int timeout)
1698{
1699 struct control_dial_args *args;
1700
1702 if (!args) {
1703 return -1;
1704 }
1705
1708}
1709
1711{
1713 shutting_down = 1;
1714 if (dial_bridge) {
1716 dial_bridge = NULL;
1717 }
1719}
1720
1721void control_set_app(struct stasis_app_control *control, struct stasis_app *app)
1722{
1723 ao2_cleanup(control->app);
1724 control->app = ao2_bump(app);
1725}
1726
1728{
1729 return control->next_app;
1730}
1731
1733{
1734 ast_free(control->next_app);
1735 control->next_app = NULL;
1736
1738}
1739
1741{
1742 return AST_VECTOR_STEAL_ELEMENTS(&control->next_app_args);
1743}
1744
1746{
1747 return AST_VECTOR_SIZE(&control->next_app_args);
1748}
static const char app[]
Definition: app_adsiprog.c:56
#define var
Definition: ast_expr2f.c:605
Asterisk main include file. File version handling, generic pbx functions.
#define ast_free(a)
Definition: astmm.h:180
#define ast_strdup(str)
A wrapper for strdup()
Definition: astmm.h:241
void ast_free_ptr(void *ptr)
free() wrapper
Definition: astmm.c:1739
#define ast_calloc(num, len)
A wrapper for calloc()
Definition: astmm.h:202
#define ast_malloc(len)
A wrapper for malloc()
Definition: astmm.h:191
#define ast_log
Definition: astobj2.c:42
#define ao2_iterator_next(iter)
Definition: astobj2.h:1911
@ AO2_ALLOC_OPT_LOCK_MUTEX
Definition: astobj2.h:363
int ao2_container_count(struct ao2_container *c)
Returns the number of elements in a container.
#define ao2_cleanup(obj)
Definition: astobj2.h:1934
@ AO2_ITERATOR_UNLINK
Definition: astobj2.h:1863
#define ao2_link_flags(container, obj, flags)
Add an object to a container.
Definition: astobj2.h:1554
struct ao2_iterator ao2_iterator_init(struct ao2_container *c, int flags) attribute_warn_unused_result
Create an iterator for a container.
#define ao2_unlock(a)
Definition: astobj2.h:729
#define ao2_lock(a)
Definition: astobj2.h:717
#define ao2_ref(o, delta)
Reference/unreference an object and return the old refcount.
Definition: astobj2.h:459
void __ao2_cleanup(void *obj)
Definition: astobj2.c:677
void * ao2_object_get_lockaddr(void *obj)
Return the mutex lock address of an object.
Definition: astobj2.c:476
#define ao2_bump(obj)
Bump refcount on an AO2 object by one, returning the object.
Definition: astobj2.h:480
void ao2_iterator_destroy(struct ao2_iterator *iter)
Destroy a container iterator.
@ OBJ_NOLOCK
Assume that the ao2_container is already locked.
Definition: astobj2.h:1063
#define ao2_container_alloc_list(ao2_options, container_options, sort_fn, cmp_fn)
Allocate and initialize a list container.
Definition: astobj2.h:1327
#define ao2_alloc(data_size, destructor_fn)
Definition: astobj2.h:409
Bridging API.
int ast_bridge_depart(struct ast_channel *chan)
Depart a channel from a bridge.
Definition: bridge.c:1975
int ast_bridge_destroy(struct ast_bridge *bridge, int cause)
Destroy a bridge.
Definition: bridge.c:1009
int ast_bridge_impart(struct ast_bridge *bridge, struct ast_channel *chan, struct ast_channel *swap, struct ast_bridge_features *features, enum ast_bridge_impart_flags flags) attribute_warn_unused_result
Impart a channel to a bridge (non-blocking)
Definition: bridge.c:1947
@ AST_BRIDGE_IMPART_CHAN_DEPARTABLE
Definition: bridge.h:592
@ AST_BRIDGE_IMPART_INHIBIT_JOIN_COLP
Definition: bridge.h:596
After Bridge Execution API.
ast_bridge_after_cb_reason
Definition: bridge_after.h:37
@ AST_BRIDGE_AFTER_CB_REASON_IMPART_FAILED
Definition: bridge_after.h:49
@ AST_BRIDGE_AFTER_CB_REASON_DEPART
Definition: bridge_after.h:45
int ast_bridge_setup_after_goto(struct ast_channel *chan)
Setup any after bridge goto location to begin execution.
Definition: bridge_after.c:435
int ast_bridge_set_after_callback(struct ast_channel *chan, ast_bridge_after_cb callback, ast_bridge_after_cb_failed failed, void *data)
Setup an after bridge callback for when the channel leaves the bridging system.
Definition: bridge_after.c:251
const char * ast_bridge_after_cb_reason_string(enum ast_bridge_after_cb_reason reason)
Get a string representation of an after bridge callback reason.
Definition: bridge_after.c:288
Basic bridge subclass API.
Channel Bridging API.
int ast_bridge_interval_hook(struct ast_bridge_features *features, enum ast_bridge_hook_timer_option flags, unsigned int interval, ast_bridge_hook_callback callback, void *hook_pvt, ast_bridge_hook_pvt_destructor destructor, enum ast_bridge_hook_remove_flags remove_flags)
Attach an interval hook to a bridge features structure.
Definition: bridge.c:3388
struct ast_bridge_features * ast_bridge_features_new(void)
Allocate a new bridge features struct.
Definition: bridge.c:3752
void ast_bridge_features_destroy(struct ast_bridge_features *features)
Destroy an allocated bridge features struct.
Definition: bridge.c:3743
int ast_channel_add_bridge_role(struct ast_channel *chan, const char *role_name)
Adds a bridge role to a channel.
Definition: bridge_roles.c:313
void ast_channel_clear_bridge_roles(struct ast_channel *chan)
Removes all bridge roles currently on a channel.
Definition: bridge_roles.c:356
Internal Asterisk hangup causes.
#define AST_CAUSE_NO_ANSWER
Definition: causes.h:109
static int priority
const char * ast_channel_name(const struct ast_channel *chan)
int ast_call(struct ast_channel *chan, const char *addr, int timeout)
Make a call.
Definition: channel.c:6478
int ast_channel_datastore_add(struct ast_channel *chan, struct ast_datastore *datastore)
Add a datastore to a channel.
Definition: channel.c:2414
int ast_channel_datastore_remove(struct ast_channel *chan, struct ast_datastore *datastore)
Remove a datastore from a channel.
Definition: channel.c:2423
struct stasis_topic * ast_channel_topic(struct ast_channel *chan)
A topic which publishes the events for a particular channel.
#define ast_channel_lock(chan)
Definition: channel.h:2970
struct ast_silence_generator * ast_channel_start_silence_generator(struct ast_channel *chan)
Starts a silence generator on the given channel.
Definition: channel.c:8190
int ast_queue_control(struct ast_channel *chan, enum ast_control_frame_type control)
Queue a control frame without payload.
Definition: channel.c:1258
void ast_channel_stop_silence_generator(struct ast_channel *chan, struct ast_silence_generator *state)
Stops a previously-started silence generator on the given channel.
Definition: channel.c:8236
#define ast_channel_ref(c)
Increase channel reference count.
Definition: channel.h:2995
int ast_queue_frame(struct ast_channel *chan, struct ast_frame *f)
Queue one or more frames to a channel's frame queue.
Definition: channel.c:1158
const char * ast_channel_uniqueid(const struct ast_channel *chan)
int ast_channel_suppress(struct ast_channel *chan, unsigned int direction, enum ast_frame_type frametype)
Suppress passing of a frame type on a channel.
Definition: channel.c:10803
struct ast_bridge_channel * ast_channel_internal_bridge_channel(const struct ast_channel *chan)
int ast_check_hangup(struct ast_channel *chan)
Check to see if a channel is needing hang up.
Definition: channel.c:445
int ast_channel_hangupcause(const struct ast_channel *chan)
int ast_channel_is_bridged(const struct ast_channel *chan)
Determine if a channel is in a bridge.
Definition: channel.c:10571
int ast_softhangup(struct ast_channel *chan, int cause)
Softly hangup up a channel.
Definition: channel.c:2500
#define AST_MAX_CONTEXT
Definition: channel.h:135
struct ast_bridge_channel * ast_channel_get_bridge_channel(struct ast_channel *chan)
Get a reference to the channel's bridge pointer.
Definition: channel.c:10608
int ast_transfer(struct ast_channel *chan, char *dest)
Transfer a channel (if supported).
Definition: channel.c:6502
int ast_channel_unsuppress(struct ast_channel *chan, unsigned int direction, enum ast_frame_type frametype)
Stop suppressing of a frame type on a channel.
Definition: channel.c:10865
int ast_softhangup_nolock(struct ast_channel *chan, int cause)
Softly hangup up a channel (no channel lock)
Definition: channel.c:2487
struct ast_pbx * ast_channel_pbx(const struct ast_channel *chan)
void ast_channel_hangupcause_set(struct ast_channel *chan, int value)
#define ast_channel_cleanup(c)
Cleanup a channel reference.
Definition: channel.h:3017
int ast_indicate(struct ast_channel *chan, int condition)
Indicates condition of channel.
Definition: channel.c:4294
int ast_safe_sleep(struct ast_channel *chan, int ms)
Wait for a specified amount of time, looking for hangups.
Definition: channel.c:1601
#define ast_channel_unlock(chan)
Definition: channel.h:2971
#define AST_MAX_EXTENSION
Definition: channel.h:134
void ast_channel_pbx_set(struct ast_channel *chan, struct ast_pbx *value)
struct ast_datastore * ast_channel_datastore_find(struct ast_channel *chan, const struct ast_datastore_info *info, const char *uid)
Find a datastore on a channel.
Definition: channel.c:2428
@ AST_SOFTHANGUP_ASYNCGOTO
Definition: channel.h:1146
@ AST_SOFTHANGUP_EXPLICIT
Definition: channel.h:1168
@ AST_SOFTHANGUP_DEV
Definition: channel.h:1141
ast_channel_state
ast_channel states
Definition: channelstate.h:35
@ AST_STATE_UP
Definition: channelstate.h:42
frame_type
Definition: codec_builtin.c:44
struct ao2_container * command_prestart_get_container(struct ast_channel *chan)
Get the Stasis() prestart commands for a channel.
Definition: command.c:160
struct stasis_app_command * command_create(stasis_app_command_cb callback, void *data, command_data_destructor_fn data_destructor)
Definition: command.c:55
void command_invoke(struct stasis_app_command *command, struct stasis_app_control *control, struct ast_channel *chan)
Definition: command.c:101
void command_complete(struct stasis_app_command *command, int retval)
Definition: command.c:77
int command_join(struct stasis_app_command *command)
Definition: command.c:86
Internal API for the Stasis application commands.
void stasis_app_control_mute_in_bridge(struct stasis_app_control *control, int mute)
Set whether audio from the channel is muted instead of passing through to the bridge.
Definition: control.c:1500
static void app_control_unregister_rule(struct stasis_app_control *control, struct app_control_rules *list, struct stasis_app_control_rule *obj)
Definition: control.c:179
void stasis_app_control_register_remove_rule(struct stasis_app_control *control, struct stasis_app_control_rule *rule)
Registers a remove channel from bridge rule.
Definition: control.c:239
static void set_interval_hook(struct ast_channel *chan)
Set a dial timeout interval hook on the channel.
Definition: control.c:1284
static int app_control_ring(struct stasis_app_control *control, struct ast_channel *chan, void *data)
Definition: control.c:610
struct stasis_app_control * control_create(struct ast_channel *channel, struct stasis_app *app)
Create a control object.
Definition: control.c:131
static void dtmf_no_bridge(struct ast_channel *chan, struct stasis_app_control_dtmf_data *dtmf_data)
Definition: control.c:560
void stasis_app_control_moh_start(struct stasis_app_control *control, const char *moh_class)
Play music on hold to a channel (does not affect hold status)
Definition: control.c:806
const char * stasis_app_control_get_channel_id(const struct stasis_app_control *control)
Returns the uniqueid of the channel associated with this control.
Definition: control.c:1460
static int app_control_continue(struct stasis_app_control *control, struct ast_channel *chan, void *data)
Definition: control.c:394
static int set_timeout(struct ast_channel *chan, unsigned int timeout)
Set dial timeout on a channel to be dialed.
Definition: control.c:1633
static int add_to_dial_bridge(struct stasis_app_control *control, struct ast_channel *chan)
Add a channel to the singleton dial bridge.
Definition: control.c:1067
int stasis_app_control_queue_control(struct stasis_app_control *control, enum ast_control_frame_type frame_type)
Queue a control frame without payload.
Definition: control.c:1475
static int app_control_set_channel_var(struct stasis_app_control *control, struct ast_channel *chan, void *data)
Definition: control.c:727
void stasis_app_control_mark_failed(struct stasis_app_control *control)
Set the failed flag on a control structure.
Definition: control.c:377
int control_prestart_dispatch_all(struct stasis_app_control *control, struct ast_channel *chan)
Dispatch all queued prestart commands.
Definition: control.c:1565
static int app_control_clear_roles(struct stasis_app_control *control, struct ast_channel *chan, void *data)
Definition: control.c:345
void stasis_app_control_unregister_remove_rule(struct stasis_app_control *control, struct stasis_app_control_rule *rule)
Unregisters a remove channel from bridge rule.
Definition: control.c:246
int control_swap_channel_in_bridge(struct stasis_app_control *control, struct ast_bridge *bridge, struct ast_channel *chan, struct ast_channel *swap)
Command for swapping a channel in a bridge.
Definition: control.c:1314
int stasis_app_control_unmute(struct stasis_app_control *control, unsigned int direction, enum ast_frame_type frametype)
Unmute the channel associated with this control.
Definition: control.c:685
void stasis_app_control_unhold(struct stasis_app_control *control)
Remove the channel associated with the control from hold.
Definition: control.c:787
static struct stasis_app_command * exec_command_on_condition(struct stasis_app_control *control, stasis_app_command_cb command_fn, void *data, command_data_destructor_fn data_destructor, app_command_can_exec_cb can_exec_fn)
Definition: control.c:275
int stasis_app_control_add_channel_to_bridge(struct stasis_app_control *control, struct ast_bridge *bridge)
Add a channel to the bridge.
Definition: control.c:1413
void control_wait(struct stasis_app_control *control)
Blocks until control's command queue has a command available.
Definition: control.c:1545
int stasis_app_control_is_failed(const struct stasis_app_control *control)
Check if a control object is marked as "failed".
Definition: control.c:382
char ** control_next_app_args(struct stasis_app_control *control)
Returns the list of arguments to pass to the application we are moving to.
Definition: control.c:1740
static void bridge_after_cb(struct ast_channel *chan, void *data)
Definition: control.c:1189
int stasis_app_send_command(struct stasis_app_control *control, stasis_app_command_cb command_fn, void *data, command_data_destructor_fn data_destructor)
Invokes a command on a control's channel.
Definition: control.c:924
void stasis_app_control_silence_start(struct stasis_app_control *control)
Start playing silence to a channel.
Definition: control.c:858
void stasis_app_control_silence_stop(struct stasis_app_control *control)
Stop playing silence to a channel.
Definition: control.c:881
int control_add_channel_to_bridge(struct stasis_app_control *control, struct ast_channel *chan, void *data)
Command callback for adding a channel to a bridge.
Definition: control.c:1408
static void dial_bridge_after_cb(struct ast_channel *chan, void *data)
after bridge callback for the dial bridge
Definition: control.c:1035
void stasis_app_control_clear_roles(struct stasis_app_control *control)
Clear bridge roles currently applied to a channel controlled by a stasis app control.
Definition: control.c:353
int stasis_app_control_continue(struct stasis_app_control *control, const char *context, const char *extension, int priority)
Exit res_stasis and continue execution in the dialplan.
Definition: control.c:415
static int app_control_ring_stop(struct stasis_app_control *control, struct ast_channel *chan, void *data)
Definition: control.c:625
static int app_control_hold(struct stasis_app_control *control, struct ast_channel *chan, void *data)
Definition: control.c:766
static enum stasis_app_control_channel_result app_control_check_rules(const struct stasis_app_control *control, struct app_control_rules *list)
Definition: control.c:211
static int app_control_unhold(struct stasis_app_control *control, struct ast_channel *chan, void *data)
Definition: control.c:779
int control_is_done(struct stasis_app_control *control)
Returns true if control_continue() has been called on this control.
Definition: control.c:363
static void free_chanvar(void *data)
Definition: control.c:718
void stasis_app_control_publish(struct stasis_app_control *control, struct stasis_message *message)
Publish a message to the control's channel's topic.
Definition: control.c:1466
int stasis_app_control_redirect(struct stasis_app_control *control, const char *endpoint)
Redirect a channel in res_stasis to a particular endpoint.
Definition: control.c:526
static int app_control_remove_channel_from_bridge(struct stasis_app_control *control, struct ast_channel *chan, void *data)
Definition: control.c:1424
void stasis_app_control_moh_stop(struct stasis_app_control *control)
Stop playing music on hold to a channel (does not affect hold status)
Definition: control.c:824
static struct control_dial_args * control_dial_args_alloc(const char *dialstring, unsigned int timeout)
Definition: control.c:1603
int stasis_app_control_move(struct stasis_app_control *control, const char *app_name, const char *app_args)
Exit res_stasis and move to another Stasis application.
Definition: control.c:477
int stasis_app_control_mute(struct stasis_app_control *control, unsigned int direction, enum ast_frame_type frametype)
Mute the channel associated with this control.
Definition: control.c:657
static struct ast_bridge * dial_bridge
Singleton dial bridge.
Definition: control.c:984
void stasis_app_control_absorb_dtmf_in_bridge(struct stasis_app_control *control, int absorb)
Set whether DTMF from the channel is absorbed instead of passing through to the bridge.
Definition: control.c:1494
struct ast_datastore_info timeout_datastore
Dial timeout datastore.
Definition: control.c:1218
static int app_control_moh_stop(struct stasis_app_control *control, struct ast_channel *chan, void *data)
Definition: control.c:817
static void bridge_after_cb_failed(enum ast_bridge_after_cb_reason reason, void *data)
Definition: control.c:1196
int(* app_command_can_exec_cb)(struct stasis_app_control *control)
Definition: control.c:273
static int noop_cb(struct stasis_app_control *control, struct ast_channel *chan, void *data)
Definition: control.c:265
int stasis_app_control_dtmf(struct stasis_app_control *control, const char *dtmf, int before, int between, unsigned int duration, int after)
Send DTMF to the channel associated with this control.
Definition: control.c:591
static int shutting_down
Indicates if the Stasis app internals are being shut down.
Definition: control.c:52
void control_flush_queue(struct stasis_app_control *control)
Flush the control command queue.
Definition: control.c:1512
int stasis_app_control_add_role(struct stasis_app_control *control, const char *role)
Apply a bridge role to a channel controlled by a stasis app control.
Definition: control.c:331
static int app_control_add_role(struct stasis_app_control *control, struct ast_channel *chan, void *data)
Definition: control.c:323
void stasis_app_control_inhibit_colp_in_bridge(struct stasis_app_control *control, int inhibit_colp)
Set whether COLP frames should be generated when joining the bridge.
Definition: control.c:1506
static int app_send_command_on_condition(struct stasis_app_control *control, stasis_app_command_cb command_fn, void *data, command_data_destructor_fn data_destructor, app_command_can_exec_cb can_exec_fn)
Definition: control.c:892
int stasis_app_control_remove_channel_from_bridge(struct stasis_app_control *control, struct ast_bridge *bridge)
Remove a channel from the bridge.
Definition: control.c:1450
struct ast_bridge * stasis_app_get_bridge(struct stasis_app_control *control)
Gets the bridge currently associated with a control object.
Definition: control.c:957
int stasis_app_send_command_async(struct stasis_app_control *control, stasis_app_command_cb command_fn, void *data, command_data_destructor_fn data_destructor)
Asynchronous version of stasis_app_send_command().
Definition: control.c:930
int stasis_app_control_bridge_features_init(struct stasis_app_control *control)
Initialize bridge features into a channel control.
Definition: control.c:1481
static int app_control_move(struct stasis_app_control *control, struct ast_channel *chan, void *data)
Definition: control.c:440
static int hangup_channel(struct stasis_app_control *control, struct ast_channel *chan, void *data)
Definition: control.c:1222
static int depart_channel(struct stasis_app_control *control, struct ast_channel *chan)
Depart a channel from a bridge, and potentially add it back to the dial bridge.
Definition: control.c:1095
void stasis_app_control_hold(struct stasis_app_control *control)
Place the channel associated with the control on hold.
Definition: control.c:774
int control_dispatch_all(struct stasis_app_control *control, struct ast_channel *chan)
Dispatch all commands enqueued to this control.
Definition: control.c:1525
int control_next_app_args_size(struct stasis_app_control *control)
Returns the number of arguments to be passed to the application we are moving to.
Definition: control.c:1745
void stasis_app_control_unregister_add_rule(struct stasis_app_control *control, struct stasis_app_control_rule *rule)
UnRegister an add channel to bridge rule.
Definition: control.c:232
void control_set_app(struct stasis_app_control *control, struct stasis_app *app)
Set the application the control object belongs to.
Definition: control.c:1721
static void dtmf_in_bridge(struct ast_channel *chan, struct stasis_app_control_dtmf_data *dtmf_data)
Definition: control.c:547
static int app_control_unmute(struct stasis_app_control *control, struct ast_channel *chan, void *data)
Definition: control.c:673
static void app_control_register_rule(struct stasis_app_control *control, struct app_control_rules *list, struct stasis_app_control_rule *obj)
Definition: control.c:170
int stasis_app_control_ring_stop(struct stasis_app_control *control)
Stop locally generated ringing on the channel associated with this control.
Definition: control.c:633
static int app_control_redirect(struct stasis_app_control *control, struct ast_channel *chan, void *data)
Definition: control.c:507
int stasis_app_control_dial(struct stasis_app_control *control, const char *dialstring, unsigned int timeout)
Dial a channel.
Definition: control.c:1696
static int bridge_timeout(struct ast_bridge_channel *bridge_channel, void *ignore)
Dial timeout.
Definition: control.c:1245
struct ast_channel_snapshot * stasis_app_control_get_snapshot(const struct stasis_app_control *control)
Returns the most recent snapshot for the associated channel.
Definition: control.c:886
int control_command_count(struct stasis_app_control *control)
Returns the count of items in a control's command queue.
Definition: control.c:358
static int app_control_can_remove_channel_from_bridge(struct stasis_app_control *control)
Definition: control.c:259
static void control_dial_args_destroy(void *data)
Definition: control.c:1620
void stasis_app_control_register_add_rule(struct stasis_app_control *control, struct stasis_app_control_rule *rule)
Registers an add channel to bridge rule.
Definition: control.c:225
void control_silence_stop_now(struct stasis_app_control *control)
Stop playing silence to a channel right now.
Definition: control.c:863
static int app_control_silence_start(struct stasis_app_control *control, struct ast_channel *chan, void *data)
Definition: control.c:829
static int app_control_can_add_channel_to_bridge(struct stasis_app_control *control)
Definition: control.c:253
static struct stasis_app_command * exec_command(struct stasis_app_control *control, stasis_app_command_cb command_fn, void *data, command_data_destructor_fn data_destructor)
Definition: control.c:316
static int app_control_dial(struct stasis_app_control *control, struct ast_channel *chan, void *data)
Definition: control.c:1658
static int app_control_moh_start(struct stasis_app_control *control, struct ast_channel *chan, void *data)
Definition: control.c:792
static ast_mutex_t dial_bridge_lock
Definition: control.c:985
static int bridge_channel_depart(struct stasis_app_control *control, struct ast_channel *chan, void *data)
Definition: control.c:1111
static struct ast_bridge * get_dial_bridge(void)
Retrieve a reference to the dial bridge.
Definition: control.c:1000
char * control_next_app(struct stasis_app_control *control)
Returns the name of the application we are moving to.
Definition: control.c:1727
static void internal_bridge_after_cb(struct ast_channel *chan, void *data, enum ast_bridge_after_cb_reason reason)
Definition: control.c:1134
static int app_control_dtmf(struct stasis_app_control *control, struct ast_channel *chan, void *data)
Definition: control.c:573
struct stasis_app * control_app(struct stasis_app_control *control)
Returns the pointer (non-reffed) to the app associated with this control.
Definition: control.c:1593
int stasis_app_control_set_channel_var(struct stasis_app_control *control, const char *variable, const char *value)
Set a variable on the channel associated with this control to value.
Definition: control.c:737
static int app_control_silence_stop(struct stasis_app_control *control, struct ast_channel *chan, void *data)
Definition: control.c:874
static int app_control_mute(struct stasis_app_control *control, struct ast_channel *chan, void *data)
Definition: control.c:645
static void dial_bridge_after_cb_failed(enum ast_bridge_after_cb_reason reason, void *data)
Definition: control.c:1051
int stasis_app_control_ring(struct stasis_app_control *control)
Indicate ringing to the channel associated with this control.
Definition: control.c:618
void control_move_cleanup(struct stasis_app_control *control)
Free any memory that was allocated for switching applications via /channels/{channelId}/move.
Definition: control.c:1732
void stasis_app_control_shutdown(void)
Let Stasis app internals shut down.
Definition: control.c:1710
static void control_dtor(void *obj)
Definition: control.c:113
void control_mark_done(struct stasis_app_control *control)
Definition: control.c:369
Internal API for the Stasis application controller.
#define ast_datastore_alloc(info, uid)
Definition: datastore.h:85
int ast_datastore_free(struct ast_datastore *datastore)
Free a data store object.
Definition: datastore.c:68
Dialing API.
char * end
Definition: eagi_proxy.c:73
direction
struct ast_channel_snapshot * ast_channel_snapshot_get_latest(const char *uniqueid)
Obtain the latest ast_channel_snapshot from the Stasis Message Bus API cache. This is an ao2 object,...
void ast_channel_publish_dial(struct ast_channel *caller, struct ast_channel *peer, const char *dialstring, const char *dialstatus)
Publish in the ast_channel_topic or ast_channel_topic_all topics a stasis message for the channels in...
Application convenience functions, designed to give consistent look and feel to Asterisk apps.
void ast_dtmf_stream_external(struct ast_channel *chan, const char *digits, int between, unsigned int duration)
Send a string of DTMF digits to a channel from an external thread.
Definition: main/app.c:1142
int ast_dtmf_stream(struct ast_channel *chan, struct ast_channel *peer, const char *digits, int between, unsigned int duration)
Send a string of DTMF digits to a channel.
Definition: main/app.c:1127
Asterisk internal frame definitions.
ast_frame_type
Frame types.
ast_control_frame_type
Internal control frame subtype field values.
@ AST_CONTROL_PROGRESS
@ AST_CONTROL_UNHOLD
@ AST_CONTROL_RINGING
@ AST_CONTROL_HOLD
struct ast_frame ast_null_frame
Definition: main/frame.c:79
#define ast_debug(level,...)
Log a DEBUG message.
#define LOG_ERROR
#define LOG_NOTICE
#define LOG_WARNING
#define AST_RWLIST_REMOVE_CURRENT
Definition: linkedlists.h:570
#define AST_RWLIST_TRAVERSE_SAFE_BEGIN
Definition: linkedlists.h:545
#define AST_LIST_TRAVERSE(head, var, field)
Loops over (traverses) the entries in a list.
Definition: linkedlists.h:491
#define AST_LIST_HEAD_DESTROY(head)
Destroys a list head structure.
Definition: linkedlists.h:653
#define AST_LIST_INSERT_TAIL(head, elm, field)
Appends a list entry to the tail of a list.
Definition: linkedlists.h:731
#define AST_RWLIST_TRAVERSE_SAFE_END
Definition: linkedlists.h:617
#define AST_LIST_HEAD_INIT(head)
Initializes a list head structure.
Definition: linkedlists.h:626
#define AST_LIST_HEAD(name, type)
Defines a structure to be used to hold a list of specified type.
Definition: linkedlists.h:173
#define ast_cond_destroy(cond)
Definition: lock.h:206
#define ast_cond_wait(cond, mutex)
Definition: lock.h:209
#define ast_cond_init(cond, attr)
Definition: lock.h:205
#define ast_mutex_unlock(a)
Definition: lock.h:194
pthread_cond_t ast_cond_t
Definition: lock.h:182
#define ast_mutex_lock(a)
Definition: lock.h:193
#define AST_MUTEX_DEFINE_STATIC(mutex)
Definition: lock.h:524
#define ast_cond_signal(cond)
Definition: lock.h:207
int errno
Music on hold handling.
int ast_moh_start(struct ast_channel *chan, const char *mclass, const char *interpclass)
Turn on music on hold on a given channel.
Definition: channel.c:7787
void ast_moh_stop(struct ast_channel *chan)
Turn off music on hold on a given channel.
Definition: channel.c:7797
def ignore(key=None, val=None, section=None, pjsip=None, nmapped=None, type='endpoint')
Definition: sip_to_pjsip.py:48
Core PBX routines and definitions.
int pbx_builtin_setvar_helper(struct ast_channel *chan, const char *name, const char *value)
Add a variable to the channel variable stack, removing the most recently set value for the same name.
int ast_explicit_goto(struct ast_channel *chan, const char *context, const char *exten, int priority)
Definition: pbx.c:6960
const char * app_name(struct ast_app *app)
Definition: pbx_app.c:463
int app_subscribe_bridge(struct stasis_app *app, struct ast_bridge *bridge)
Add a bridge subscription to an existing channel subscription.
int app_unsubscribe_bridge(struct stasis_app *app, struct ast_bridge *bridge)
Cancel the bridge subscription for an application.
Internal API for the Stasis application controller.
#define NULL
Definition: resample.c:96
void stasis_publish(struct stasis_topic *topic, struct stasis_message *message)
Publish a message to a topic's subscribers.
Definition: stasis.c:1538
Stasis Application API. See Stasis Application API for detailed documentation.
struct ast_bridge * stasis_app_bridge_create_invisible(const char *type, const char *name, const char *id)
Create an invisible bridge of the specified type.
Definition: res_stasis.c:896
stasis_app_control_channel_result
Result codes used when adding/removing channels to/from bridges.
Definition: stasis_app.h:800
int stasis_app_channel_is_stasis_end_published(struct ast_channel *chan)
Has this channel had a StasisEnd published on it?
Definition: res_stasis.c:1344
struct stasis_app_control * stasis_app_control_find_by_channel(const struct ast_channel *chan)
Returns the handler for the given channel.
Definition: res_stasis.c:338
void(* command_data_destructor_fn)(void *data)
Typedef for data destructor for stasis app commands.
int(* stasis_app_command_cb)(struct stasis_app_control *control, struct ast_channel *chan, void *data)
#define S_OR(a, b)
returns the equivalent of logic or for strings: first one if not empty, otherwise second one.
Definition: strings.h:80
static force_inline int attribute_pure ast_strlen_zero(const char *s)
Definition: strings.h:65
void ast_copy_string(char *dst, const char *src, size_t size)
Size-limited null-terminating string copy.
Definition: strings.h:425
Generic container type.
When we need to walk through a container, we use an ao2_iterator to keep track of the current positio...
Definition: astobj2.h:1821
Structure that contains information regarding a channel in a bridge.
struct ast_channel * swap
struct ast_bridge * bridge
Bridge this channel is participating in.
struct ast_bridge_features * features
struct ast_channel * chan
Structure that contains features information.
unsigned int dtmf_passthrough
unsigned int inhibit_colp
Structure that contains information about a bridge.
Definition: bridge.h:353
const ast_string_field uniqueid
Definition: bridge.h:405
Structure representing a snapshot of channel state.
Main Channel structure associated with a channel.
Structure for a data store type.
Definition: datastore.h:31
const char * type
Definition: datastore.h:32
Structure for a data store object.
Definition: datastore.h:64
void * data
Definition: datastore.h:66
Definition: pbx.h:215
structure for queuing ARI channel variable setting
Definition: control.c:711
char * value
Definition: control.c:715
char * name
Definition: control.c:713
char dialstring[0]
Definition: control.c:1600
unsigned int timeout
Definition: control.c:1599
structure to hold extensions
command_data_destructor_fn data_destructor
Definition: command.c:38
char extension[AST_MAX_EXTENSION]
Definition: control.c:390
char context[AST_MAX_CONTEXT]
Definition: control.c:389
enum ast_frame_type frametype
Definition: control.c:641
Rule to check to see if an operation is allowed.
Definition: stasis_app.h:355
struct stasis_app_control_rule * next
Definition: stasis_app.h:365
struct stasis_app_control::@492 next_app_args
struct stasis_app * app
Definition: control.c:93
struct ast_bridge_features * bridge_features
Definition: control.c:71
struct ast_channel * channel
Definition: control.c:63
struct ast_bridge * bridge
Definition: control.c:67
struct ao2_container * command_queue
Definition: control.c:57
unsigned int is_done
Definition: control.c:105
struct app_control_rules remove_rules
Definition: control.c:83
struct ast_silence_generator * silgen
Definition: control.c:87
char * next_app
Definition: control.c:97
struct app_control_rules add_rules
Definition: control.c:79
unsigned int failed
Definition: control.c:110
ast_cond_t wait_cond
Definition: control.c:55
struct ast_pbx * pbx
Definition: control.c:75
int value
Definition: syslog.c:37
const char * args
struct timeval ast_samp2tv(unsigned int _nsamp, unsigned int _rate)
Returns a timeval corresponding to the duration of n samples at rate r. Useful to convert samples to ...
Definition: time.h:282
struct timeval ast_tvadd(struct timeval a, struct timeval b)
Returns the sum of two timevals a + b.
Definition: extconf.c:2282
int64_t ast_tvdiff_ms(struct timeval end, struct timeval start)
Computes the difference (in milliseconds) between two struct timeval instances.
Definition: time.h:107
struct timeval ast_tvnow(void)
Returns current timeval. Meant to replace calls to gettimeofday().
Definition: time.h:159
#define RAII_VAR(vartype, varname, initval, dtor)
Declare a variable that will call a destructor function when it goes out of scope.
Definition: utils.h:941
#define ast_assert(a)
Definition: utils.h:739
#define AST_VECTOR_RESET(vec, cleanup)
Reset vector.
Definition: vector.h:625
#define AST_VECTOR_SIZE(vec)
Get the number of elements in a vector.
Definition: vector.h:609
#define AST_VECTOR_STEAL_ELEMENTS(vec)
Steal the elements from a vector and reinitialize.
Definition: vector.h:140
#define AST_VECTOR_INIT(vec, size)
Initialize a vector.
Definition: vector.h:113
#define AST_VECTOR_APPEND(vec, elem)
Append an element to a vector, growing the vector if needed.
Definition: vector.h:256
#define AST_VECTOR(name, type)
Define a vector structure.
Definition: vector.h:44