Asterisk - The Open Source Telephony Project GIT-master-7e7a603
Data Structures | Macros | Functions | Variables
audiohook.c File Reference

Audiohooks Architecture. More...

#include "asterisk.h"
#include <signal.h>
#include "asterisk/channel.h"
#include "asterisk/utils.h"
#include "asterisk/lock.h"
#include "asterisk/audiohook.h"
#include "asterisk/slinfactory.h"
#include "asterisk/frame.h"
#include "asterisk/translate.h"
#include "asterisk/format_cache.h"
#include "asterisk/test.h"
Include dependency graph for audiohook.c:

Go to the source code of this file.

Data Structures

struct  ast_audiohook_list
 
struct  ast_audiohook_translate
 
struct  audiohook_volume
 Audiohook volume adjustment structure. More...
 

Macros

#define AST_AUDIOHOOK_LONG_QUEUE_TOLERANCE   500
 
#define AST_AUDIOHOOK_SMALL_QUEUE_TOLERANCE   100
 
#define AST_AUDIOHOOK_SYNC_TOLERANCE   100
 
#define DEFAULT_INTERNAL_SAMPLE_RATE   8000
 
#define SHOULD_MUTE(hook, dir)
 

Functions

int ast_audiohook_attach (struct ast_channel *chan, struct ast_audiohook *audiohook)
 Attach audiohook to channel. More...
 
int ast_audiohook_destroy (struct ast_audiohook *audiohook)
 Destroys an audiohook structure. More...
 
int ast_audiohook_detach (struct ast_audiohook *audiohook)
 Detach audiohook from channel. More...
 
void ast_audiohook_detach_list (struct ast_audiohook_list *audiohook_list)
 Detach audiohooks from list and destroy said list. More...
 
int ast_audiohook_detach_source (struct ast_channel *chan, const char *source)
 Detach specified source audiohook from channel. More...
 
int ast_audiohook_init (struct ast_audiohook *audiohook, enum ast_audiohook_type type, const char *source, enum ast_audiohook_init_flags init_flags)
 Initialize an audiohook structure. More...
 
void ast_audiohook_move_all (struct ast_channel *old_chan, struct ast_channel *new_chan)
 Move all audiohooks from one channel to another. More...
 
void ast_audiohook_move_by_source (struct ast_channel *old_chan, struct ast_channel *new_chan, const char *source)
 Move an audiohook from one channel to a new one. More...
 
struct ast_frameast_audiohook_read_frame (struct ast_audiohook *audiohook, size_t samples, enum ast_audiohook_direction direction, struct ast_format *format)
 Reads a frame in from the audiohook structure. More...
 
struct ast_frameast_audiohook_read_frame_all (struct ast_audiohook *audiohook, size_t samples, struct ast_format *format, struct ast_frame **read_frame, struct ast_frame **write_frame)
 Reads a frame in from the audiohook structure in mixed audio mode and copies read and write frame data to provided arguments. More...
 
int ast_audiohook_remove (struct ast_channel *chan, struct ast_audiohook *audiohook)
 Remove an audiohook from a specified channel. More...
 
int ast_audiohook_set_frame_feed_direction (struct ast_audiohook *audiohook, enum ast_audiohook_direction direction)
 Sets direction on audiohook. More...
 
int ast_audiohook_set_mute (struct ast_channel *chan, const char *source, enum ast_audiohook_flags flag, int clear)
 Mute frames read from or written to a channel. More...
 
int ast_audiohook_set_mute_all (struct ast_channel *chan, const char *source, enum ast_audiohook_flags flag, int clearmute)
 Mute frames read from or written for all audiohooks on a channel. More...
 
void ast_audiohook_trigger_wait (struct ast_audiohook *audiohook)
 Wait for audiohook trigger to be triggered. More...
 
void ast_audiohook_update_status (struct ast_audiohook *audiohook, enum ast_audiohook_status status)
 Update audiohook's status. More...
 
int ast_audiohook_volume_adjust (struct ast_channel *chan, enum ast_audiohook_direction direction, int volume)
 Adjust the volume on frames read from or written to a channel. More...
 
int ast_audiohook_volume_get (struct ast_channel *chan, enum ast_audiohook_direction direction)
 Retrieve the volume adjustment value on frames read from or written to a channel. More...
 
int ast_audiohook_volume_set (struct ast_channel *chan, enum ast_audiohook_direction direction, int volume)
 Adjust the volume on frames read from or written to a channel. More...
 
int ast_audiohook_write_frame (struct ast_audiohook *audiohook, enum ast_audiohook_direction direction, struct ast_frame *frame)
 Writes a frame into the audiohook structure. More...
 
struct ast_frameast_audiohook_write_list (struct ast_channel *chan, struct ast_audiohook_list *audiohook_list, enum ast_audiohook_direction direction, struct ast_frame *frame)
 Pass a frame off to be handled by the audiohook core. More...
 
int ast_audiohook_write_list_empty (struct ast_audiohook_list *audiohook_list)
 Determine if a audiohook_list is empty or not. More...
 
int ast_channel_audiohook_count_by_source (struct ast_channel *chan, const char *source, enum ast_audiohook_type type)
 Find out how many audiohooks from a certain source exist on a given channel, regardless of status. More...
 
int ast_channel_audiohook_count_by_source_running (struct ast_channel *chan, const char *source, enum ast_audiohook_type type)
 Find out how many spies of a certain type exist on a given channel, and are in state running. More...
 
static struct ast_frameaudio_audiohook_write_list (struct ast_channel *chan, struct ast_audiohook_list *audiohook_list, enum ast_audiohook_direction direction, struct ast_frame *frame)
 Pass an AUDIO frame off to be handled by the audiohook core. More...
 
static void audiohook_list_set_hook_rate (struct ast_audiohook_list *audiohook_list, struct ast_audiohook *audiohook, int *rate)
 Set the audiohook's internal sample rate to the audiohook_list's rate, but only when native slin compatibility is turned on. More...
 
static void audiohook_list_set_samplerate_compatibility (struct ast_audiohook_list *audiohook_list)
 
static struct ast_frameaudiohook_list_translate_to_native (struct ast_audiohook_list *audiohook_list, enum ast_audiohook_direction direction, struct ast_frame *slin_frame, struct ast_format *outformat)
 
static struct ast_frameaudiohook_list_translate_to_slin (struct ast_audiohook_list *audiohook_list, enum ast_audiohook_direction direction, struct ast_frame *frame)
 
static void audiohook_move (struct ast_channel *old_chan, struct ast_channel *new_chan, struct ast_audiohook *audiohook)
 
static struct ast_frameaudiohook_read_frame_both (struct ast_audiohook *audiohook, size_t samples, struct ast_frame **read_reference, struct ast_frame **write_reference)
 
static struct ast_frameaudiohook_read_frame_helper (struct ast_audiohook *audiohook, size_t samples, enum ast_audiohook_direction direction, struct ast_format *format, struct ast_frame **read_reference, struct ast_frame **write_reference)
 
static struct ast_frameaudiohook_read_frame_single (struct ast_audiohook *audiohook, size_t samples, enum ast_audiohook_direction direction)
 
static int audiohook_set_internal_rate (struct ast_audiohook *audiohook, int rate, int reset)
 
static int audiohook_volume_callback (struct ast_audiohook *audiohook, struct ast_channel *chan, struct ast_frame *frame, enum ast_audiohook_direction direction)
 Helper function which actually gets called by audiohooks to perform the adjustment. More...
 
static void audiohook_volume_destroy (void *data)
 Callback used to destroy the audiohook volume datastore. More...
 
static struct audiohook_volumeaudiohook_volume_get (struct ast_channel *chan, int create)
 Helper function which finds and optionally creates an audiohook_volume_datastore datastore on a channel. More...
 
static struct ast_framedtmf_audiohook_write_list (struct ast_channel *chan, struct ast_audiohook_list *audiohook_list, enum ast_audiohook_direction direction, struct ast_frame *frame)
 Pass a DTMF frame off to be handled by the audiohook core. More...
 
static struct ast_audiohookfind_audiohook_by_source (struct ast_audiohook_list *audiohook_list, const char *source)
 find an audiohook based on its source More...
 

Variables

static const struct ast_datastore_info audiohook_volume_datastore
 Datastore used to store audiohook volume information. More...
 

Detailed Description

Audiohooks Architecture.

Author
Joshua Colp jcolp.nosp@m.@dig.nosp@m.ium.c.nosp@m.om

Definition in file audiohook.c.

Macro Definition Documentation

◆ AST_AUDIOHOOK_LONG_QUEUE_TOLERANCE

#define AST_AUDIOHOOK_LONG_QUEUE_TOLERANCE   500

Otheriwise we still don't want the queue to grow indefinitely

Definition at line 46 of file audiohook.c.

◆ AST_AUDIOHOOK_SMALL_QUEUE_TOLERANCE

#define AST_AUDIOHOOK_SMALL_QUEUE_TOLERANCE   100

When small queue is enabled, this is the maximum amount of audio that can remain queued at a time.

Definition at line 45 of file audiohook.c.

◆ AST_AUDIOHOOK_SYNC_TOLERANCE

#define AST_AUDIOHOOK_SYNC_TOLERANCE   100

Tolerance in milliseconds for audiohooks synchronization

Definition at line 44 of file audiohook.c.

◆ DEFAULT_INTERNAL_SAMPLE_RATE

#define DEFAULT_INTERNAL_SAMPLE_RATE   8000

Definition at line 48 of file audiohook.c.

◆ SHOULD_MUTE

#define SHOULD_MUTE (   hook,
  dir 
)
Value:
@ AST_AUDIOHOOK_DIRECTION_READ
Definition: audiohook.h:49
@ AST_AUDIOHOOK_DIRECTION_WRITE
Definition: audiohook.h:50
@ AST_AUDIOHOOK_MUTE_READ
Definition: audiohook.h:64
@ AST_AUDIOHOOK_MUTE_WRITE
Definition: audiohook.h:65
#define ast_test_flag(p, flag)
Definition: utils.h:63

Definition at line 162 of file audiohook.c.

Function Documentation

◆ ast_audiohook_attach()

int ast_audiohook_attach ( struct ast_channel chan,
struct ast_audiohook audiohook 
)

Attach audiohook to channel.

Parameters
chan
audiohook
Returns
0 on success
Return values
-1on failure

Definition at line 484 of file audiohook.c.

485{
486 ast_channel_lock(chan);
487
488 /* Don't allow an audiohook to be attached to a channel that is already hung up.
489 * The hang up process is what actually notifies the audiohook that it should
490 * stop.
491 */
493 ast_channel_unlock(chan);
494 return -1;
495 }
496
497 if (!ast_channel_audiohooks(chan)) {
498 struct ast_audiohook_list *ahlist;
499 /* Whoops... allocate a new structure */
500 if (!(ahlist = ast_calloc(1, sizeof(*ahlist)))) {
501 ast_channel_unlock(chan);
502 return -1;
503 }
504 ast_channel_audiohooks_set(chan, ahlist);
508 /* This sample rate will adjust as necessary when writing to the list. */
510 }
511
512 /* Drop into respective list */
513 if (audiohook->type == AST_AUDIOHOOK_TYPE_SPY) {
514 AST_LIST_INSERT_TAIL(&ast_channel_audiohooks(chan)->spy_list, audiohook, list);
515 } else if (audiohook->type == AST_AUDIOHOOK_TYPE_WHISPER) {
517 } else if (audiohook->type == AST_AUDIOHOOK_TYPE_MANIPULATE) {
519 }
520
521 /*
522 * Initialize the audiohook's rate to the default. If it needs to be,
523 * it will get updated later.
524 */
527
528 /* Change status over to running since it is now attached */
530
531 if (ast_channel_is_bridged(chan)) {
533 }
534
535 ast_channel_unlock(chan);
536
537 return 0;
538}
#define ast_calloc(num, len)
A wrapper for calloc()
Definition: astmm.h:202
static void audiohook_list_set_samplerate_compatibility(struct ast_audiohook_list *audiohook_list)
Definition: audiohook.c:456
void ast_audiohook_update_status(struct ast_audiohook *audiohook, enum ast_audiohook_status status)
Update audiohook's status.
Definition: audiohook.c:540
static int audiohook_set_internal_rate(struct ast_audiohook *audiohook, int rate, int reset)
Definition: audiohook.c:70
#define DEFAULT_INTERNAL_SAMPLE_RATE
Definition: audiohook.c:48
@ AST_AUDIOHOOK_TYPE_MANIPULATE
Definition: audiohook.h:38
@ AST_AUDIOHOOK_TYPE_SPY
Definition: audiohook.h:36
@ AST_AUDIOHOOK_TYPE_WHISPER
Definition: audiohook.h:37
@ AST_AUDIOHOOK_STATUS_RUNNING
Definition: audiohook.h:43
void ast_channel_set_unbridged_nolock(struct ast_channel *chan, int value)
Variant of ast_channel_set_unbridged. Use this if the channel is already locked prior to calling.
#define ast_channel_lock(chan)
Definition: channel.h:2922
struct ast_flags * ast_channel_flags(struct ast_channel *chan)
void ast_channel_audiohooks_set(struct ast_channel *chan, struct ast_audiohook_list *value)
struct ast_audiohook_list * ast_channel_audiohooks(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:10545
@ AST_FLAG_ZOMBIE
Definition: channel.h:987
#define ast_channel_unlock(chan)
Definition: channel.h:2923
#define AST_LIST_HEAD_INIT_NOLOCK(head)
Initializes a list head structure.
Definition: linkedlists.h:681
#define AST_LIST_INSERT_TAIL(head, elm, field)
Appends a list entry to the tail of a list.
Definition: linkedlists.h:731
int list_internal_samp_rate
Definition: audiohook.c:61
struct ast_audiohook_list::@302 spy_list
struct ast_audiohook_list::@304 manipulate_list
struct ast_audiohook_list::@303 whisper_list
enum ast_audiohook_type type
Definition: audiohook.h:107

References AST_AUDIOHOOK_STATUS_RUNNING, AST_AUDIOHOOK_TYPE_MANIPULATE, AST_AUDIOHOOK_TYPE_SPY, AST_AUDIOHOOK_TYPE_WHISPER, ast_audiohook_update_status(), ast_calloc, ast_channel_audiohooks(), ast_channel_audiohooks_set(), ast_channel_flags(), ast_channel_is_bridged(), ast_channel_lock, ast_channel_set_unbridged_nolock(), ast_channel_unlock, AST_FLAG_ZOMBIE, AST_LIST_HEAD_INIT_NOLOCK, AST_LIST_INSERT_TAIL, ast_test_flag, audiohook_list_set_samplerate_compatibility(), audiohook_set_internal_rate(), DEFAULT_INTERNAL_SAMPLE_RATE, ast_audiohook_list::list_internal_samp_rate, ast_audiohook_list::manipulate_list, ast_audiohook_list::spy_list, ast_audiohook::type, and ast_audiohook_list::whisper_list.

Referenced by audiohook_move(), audiohook_volume_get(), detect_write(), enable_jack_hook(), init_hook(), pitchshift_helper(), scramble_write(), set_talk_detect(), snoop_setup_audiohook(), speex_write(), start_spying(), startmon(), and volume_write().

◆ ast_audiohook_destroy()

int ast_audiohook_destroy ( struct ast_audiohook audiohook)

Destroys an audiohook structure.

Parameters
audiohook
Return values
0on success
-1on failure

Definition at line 124 of file audiohook.c.

125{
126 /* Drop the factories used by this audiohook type */
127 switch (audiohook->type) {
132 break;
133 default:
134 break;
135 }
136
137 /* Destroy translation path if present */
138 if (audiohook->trans_pvt)
140
141 ao2_cleanup(audiohook->format);
142
143 /* Lock and trigger be gone! */
144 ast_cond_destroy(&audiohook->trigger);
145 ast_mutex_destroy(&audiohook->lock);
146
147 return 0;
148}
#define ao2_cleanup(obj)
Definition: astobj2.h:1934
#define ast_cond_destroy(cond)
Definition: lock.h:202
#define ast_mutex_destroy(a)
Definition: lock.h:188
void ast_slinfactory_destroy(struct ast_slinfactory *sf)
Destroy the contents of a slinfactory.
Definition: slinfactory.c:58
ast_cond_t trigger
Definition: audiohook.h:106
struct ast_slinfactory read_factory
Definition: audiohook.h:112
struct ast_trans_pvt * trans_pvt
Definition: audiohook.h:117
struct ast_format * format
Definition: audiohook.h:116
ast_mutex_t lock
Definition: audiohook.h:105
struct ast_slinfactory write_factory
Definition: audiohook.h:113
void ast_translator_free_path(struct ast_trans_pvt *tr)
Frees a translator path Frees the given translator path structure.
Definition: translate.c:476

References ao2_cleanup, AST_AUDIOHOOK_TYPE_SPY, AST_AUDIOHOOK_TYPE_WHISPER, ast_cond_destroy, ast_mutex_destroy, ast_slinfactory_destroy(), ast_translator_free_path(), ast_audiohook::format, ast_audiohook::lock, ast_audiohook::read_factory, ast_audiohook::trans_pvt, ast_audiohook::trigger, ast_audiohook::type, and ast_audiohook::write_factory.

Referenced by audiohook_volume_destroy(), channel_spy(), datastore_destroy_cb(), destroy_callback(), destroy_jack_data(), destroy_monitor_audiohook(), hook_datastore_destroy_callback(), launch_monitor_thread(), multi_autochan_free(), and snoop_destroy().

◆ ast_audiohook_detach()

int ast_audiohook_detach ( struct ast_audiohook audiohook)

Detach audiohook from channel.

Parameters
audiohook
Returns
Returns 0 on success, -1 on failure

Definition at line 550 of file audiohook.c.

551{
552 if (audiohook->status == AST_AUDIOHOOK_STATUS_NEW || audiohook->status == AST_AUDIOHOOK_STATUS_DONE) {
553 return 0;
554 }
555
557
558 while (audiohook->status != AST_AUDIOHOOK_STATUS_DONE) {
560 }
561
562 return 0;
563}
void ast_audiohook_trigger_wait(struct ast_audiohook *audiohook)
Wait for audiohook trigger to be triggered.
Definition: audiohook.c:1094
@ AST_AUDIOHOOK_STATUS_DONE
Definition: audiohook.h:45
@ AST_AUDIOHOOK_STATUS_NEW
Definition: audiohook.h:42
@ AST_AUDIOHOOK_STATUS_SHUTDOWN
Definition: audiohook.h:44
enum ast_audiohook_status status
Definition: audiohook.h:108

References AST_AUDIOHOOK_STATUS_DONE, AST_AUDIOHOOK_STATUS_NEW, AST_AUDIOHOOK_STATUS_SHUTDOWN, ast_audiohook_trigger_wait(), ast_audiohook_update_status(), and ast_audiohook::status.

Referenced by channel_spy(), destroy_callback(), destroy_monitor_audiohook(), disable_jack_hook(), hook_datastore_destroy_callback(), multi_autochan_free(), snoop_hangup(), and speex_write().

◆ ast_audiohook_detach_list()

void ast_audiohook_detach_list ( struct ast_audiohook_list audiohook_list)

Detach audiohooks from list and destroy said list.

Parameters
audiohook_listList of audiohooks (NULL tolerant)

Definition at line 565 of file audiohook.c.

566{
567 int i;
568 struct ast_audiohook *audiohook;
569
570 if (!audiohook_list) {
571 return;
572 }
573
574 /* Drop any spies */
575 while ((audiohook = AST_LIST_REMOVE_HEAD(&audiohook_list->spy_list, list))) {
577 }
578
579 /* Drop any whispering sources */
580 while ((audiohook = AST_LIST_REMOVE_HEAD(&audiohook_list->whisper_list, list))) {
582 }
583
584 /* Drop any manipulators */
585 while ((audiohook = AST_LIST_REMOVE_HEAD(&audiohook_list->manipulate_list, list))) {
587 audiohook->manipulate_callback(audiohook, NULL, NULL, 0);
588 }
589
590 /* Drop translation paths if present */
591 for (i = 0; i < 2; i++) {
592 if (audiohook_list->in_translate[i].trans_pvt) {
594 ao2_cleanup(audiohook_list->in_translate[i].format);
595 }
596 if (audiohook_list->out_translate[i].trans_pvt) {
598 ao2_cleanup(audiohook_list->in_translate[i].format);
599 }
600 }
601
602 /* Free ourselves */
603 ast_free(audiohook_list);
604}
#define ast_free(a)
Definition: astmm.h:180
#define AST_LIST_REMOVE_HEAD(head, field)
Removes and returns the head entry from a list.
Definition: linkedlists.h:833
#define NULL
Definition: resample.c:96
struct ast_audiohook_translate out_translate[2]
Definition: audiohook.c:64
struct ast_audiohook_translate in_translate[2]
Definition: audiohook.c:63
struct ast_trans_pvt * trans_pvt
Definition: audiohook.c:51
struct ast_format * format
Definition: audiohook.c:52
ast_audiohook_manipulate_callback manipulate_callback
Definition: audiohook.h:118
struct ast_audiohook::@187 list

References ao2_cleanup, AST_AUDIOHOOK_STATUS_DONE, ast_audiohook_update_status(), ast_free, AST_LIST_REMOVE_HEAD, ast_translator_free_path(), ast_audiohook_translate::format, ast_audiohook_list::in_translate, ast_audiohook::list, ast_audiohook::manipulate_callback, ast_audiohook_list::manipulate_list, NULL, ast_audiohook_list::out_translate, ast_audiohook_list::spy_list, ast_audiohook_translate::trans_pvt, and ast_audiohook_list::whisper_list.

Referenced by __ast_read(), ast_write_stream(), and destroy_hooks().

◆ ast_audiohook_detach_source()

int ast_audiohook_detach_source ( struct ast_channel chan,
const char *  source 
)

Detach specified source audiohook from channel.

Parameters
chanChannel to detach from
sourceName of source to detach
Return values
0on success
-1on failure
Note
The channel does not need to be locked before calling this function.

Definition at line 698 of file audiohook.c.

699{
700 struct ast_audiohook *audiohook = NULL;
701
702 ast_channel_lock(chan);
703
704 /* Ensure the channel has audiohooks on it */
705 if (!ast_channel_audiohooks(chan)) {
706 ast_channel_unlock(chan);
707 return -1;
708 }
709
711
712 ast_channel_unlock(chan);
713
714 if (audiohook && audiohook->status != AST_AUDIOHOOK_STATUS_DONE) {
716 }
717
718 return (audiohook ? 0 : -1);
719}
static struct ast_audiohook * find_audiohook_by_source(struct ast_audiohook_list *audiohook_list, const char *source)
find an audiohook based on its source
Definition: audiohook.c:612
const char * source
Definition: audiohook.h:110

References AST_AUDIOHOOK_STATUS_DONE, AST_AUDIOHOOK_STATUS_SHUTDOWN, ast_audiohook_update_status(), ast_channel_audiohooks(), ast_channel_lock, ast_channel_unlock, find_audiohook_by_source(), NULL, ast_audiohook::source, and ast_audiohook::status.

◆ ast_audiohook_init()

int ast_audiohook_init ( struct ast_audiohook audiohook,
enum ast_audiohook_type  type,
const char *  source,
enum ast_audiohook_init_flags  flags 
)

Initialize an audiohook structure.

Parameters
audiohook
typeType of audiohook to initialize this as
sourceWho is initializing this audiohook
flags
Return values
0on success
-1on failure

Definition at line 100 of file audiohook.c.

101{
102 /* Need to keep the type and source */
103 audiohook->type = type;
104 audiohook->source = source;
105
106 /* Initialize lock that protects our audiohook */
107 ast_mutex_init(&audiohook->lock);
108 ast_cond_init(&audiohook->trigger, NULL);
109
110 audiohook->init_flags = init_flags;
111
112 /* Set direction to BOTH so that we feed frames in both directions */
114
115 /* initialize internal rate at 8khz, this will adjust if necessary */
117
118 /* Since we are just starting out... this audiohook is new */
120
121 return 0;
122}
@ AST_AUDIOHOOK_DIRECTION_BOTH
Definition: audiohook.h:51
static const char type[]
Definition: chan_ooh323.c:109
#define ast_cond_init(cond, attr)
Definition: lock.h:201
#define ast_mutex_init(pmutex)
Definition: lock.h:186
enum ast_audiohook_init_flags init_flags
Definition: audiohook.h:109
enum ast_audiohook_direction direction
Definition: audiohook.h:121

References AST_AUDIOHOOK_DIRECTION_BOTH, AST_AUDIOHOOK_STATUS_NEW, ast_audiohook_update_status(), ast_cond_init, ast_mutex_init, audiohook_set_internal_rate(), DEFAULT_INTERNAL_SAMPLE_RATE, ast_audiohook::direction, ast_audiohook::init_flags, ast_audiohook::lock, NULL, ast_audiohook::source, ast_audiohook::trigger, type, and ast_audiohook::type.

Referenced by attach_barge(), audiohook_volume_get(), channel_spy(), detect_write(), do_broadcast(), enable_jack_hook(), hook_state_alloc(), launch_monitor_thread(), pitchshift_helper(), scramble_write(), set_talk_detect(), snoop_setup_audiohook(), speex_write(), and volume_write().

◆ ast_audiohook_move_all()

void ast_audiohook_move_all ( struct ast_channel old_chan,
struct ast_channel new_chan 
)

Move all audiohooks from one channel to another.

Note
It is required that both old_chan and new_chan are locked prior to calling this function. Besides needing to protect the data within the channels, not locking these channels can lead to a potential deadlock.
Parameters
old_chanThe source of the audiohooks being moved
new_chanThe destination channel for the audiohooks to be moved to

Definition at line 672 of file audiohook.c.

673{
674 struct ast_audiohook *audiohook;
675 struct ast_audiohook_list *audiohook_list;
676
677 audiohook_list = ast_channel_audiohooks(old_chan);
678 if (!audiohook_list) {
679 return;
680 }
681
682 AST_LIST_TRAVERSE_SAFE_BEGIN(&audiohook_list->spy_list, audiohook, list) {
683 audiohook_move(old_chan, new_chan, audiohook);
684 }
686
687 AST_LIST_TRAVERSE_SAFE_BEGIN(&audiohook_list->whisper_list, audiohook, list) {
688 audiohook_move(old_chan, new_chan, audiohook);
689 }
691
692 AST_LIST_TRAVERSE_SAFE_BEGIN(&audiohook_list->manipulate_list, audiohook, list) {
693 audiohook_move(old_chan, new_chan, audiohook);
694 }
696}
static void audiohook_move(struct ast_channel *old_chan, struct ast_channel *new_chan, struct ast_audiohook *audiohook)
Definition: audiohook.c:637
#define AST_LIST_TRAVERSE_SAFE_END
Closes a safe loop traversal block.
Definition: linkedlists.h:615
#define AST_LIST_TRAVERSE_SAFE_BEGIN(head, var, field)
Loops safely over (traverses) the entries in a list.
Definition: linkedlists.h:529

References ast_channel_audiohooks(), AST_LIST_TRAVERSE_SAFE_BEGIN, AST_LIST_TRAVERSE_SAFE_END, audiohook_move(), ast_audiohook_list::manipulate_list, ast_audiohook_list::spy_list, and ast_audiohook_list::whisper_list.

Referenced by channel_do_masquerade().

◆ ast_audiohook_move_by_source()

void ast_audiohook_move_by_source ( struct ast_channel old_chan,
struct ast_channel new_chan,
const char *  source 
)

Move an audiohook from one channel to a new one.

Todo:
Currently only the first audiohook of a specific source found will be moved. We should add the capability to move multiple audiohooks from a single source as well.
Note
It is required that both old_chan and new_chan are locked prior to calling this function. Besides needing to protect the data within the channels, not locking these channels can lead to a potential deadlock
Parameters
old_chanThe source of the audiohook to move
new_chanThe destination to which we want the audiohook to move
sourceThe source of the audiohook we want to move

Definition at line 656 of file audiohook.c.

657{
658 struct ast_audiohook *audiohook;
659
660 if (!ast_channel_audiohooks(old_chan)) {
661 return;
662 }
663
665 if (!audiohook) {
666 return;
667 }
668
669 audiohook_move(old_chan, new_chan, audiohook);
670}

References ast_channel_audiohooks(), audiohook_move(), find_audiohook_by_source(), and ast_audiohook::source.

◆ ast_audiohook_read_frame()

struct ast_frame * ast_audiohook_read_frame ( struct ast_audiohook audiohook,
size_t  samples,
enum ast_audiohook_direction  direction,
struct ast_format format 
)

Reads a frame in from the audiohook structure.

Parameters
audiohook
samplesNumber of samples wanted
directionDirection the audio frame came from
formatFormat of frame remote side wants back
Returns
frame on success
Return values
NULLon failure

Definition at line 446 of file audiohook.c.

447{
448 return audiohook_read_frame_helper(audiohook, samples, direction, format, NULL, NULL);
449}
static struct ast_frame * audiohook_read_frame_helper(struct ast_audiohook *audiohook, size_t samples, enum ast_audiohook_direction direction, struct ast_format *format, struct ast_frame **read_reference, struct ast_frame **write_reference)
Definition: audiohook.c:384
direction

References audiohook_read_frame_helper(), NULL, and ast_frame::samples.

Referenced by snoop_read(), and spy_generate().

◆ ast_audiohook_read_frame_all()

struct ast_frame * ast_audiohook_read_frame_all ( struct ast_audiohook audiohook,
size_t  samples,
struct ast_format format,
struct ast_frame **  read_frame,
struct ast_frame **  write_frame 
)

Reads a frame in from the audiohook structure in mixed audio mode and copies read and write frame data to provided arguments.

Parameters
audiohook
samplesNumber of samples wanted
formatFormat of frame remote side wants back
read_frameif available, we'll copy the read buffer to this.
write_frameif available, we'll copy the write buffer to this.
Returns
frame on success
Return values
NULLon failure

Definition at line 451 of file audiohook.c.

452{
453 return audiohook_read_frame_helper(audiohook, samples, AST_AUDIOHOOK_DIRECTION_BOTH, format, read_frame, write_frame);
454}
static struct ast_frame * read_frame(struct ast_filestream *s, int *whennext)
Definition: file.c:911

References AST_AUDIOHOOK_DIRECTION_BOTH, audiohook_read_frame_helper(), read_frame(), and ast_frame::samples.

Referenced by mixmonitor_thread(), and spy_generate().

◆ ast_audiohook_remove()

int ast_audiohook_remove ( struct ast_channel chan,
struct ast_audiohook audiohook 
)

Remove an audiohook from a specified channel.

Parameters
chanChannel to remove from
audiohookAudiohook to remove
Return values
0on success
-1on failure
Note
The channel does not need to be locked before calling this function

Definition at line 721 of file audiohook.c.

722{
723 ast_channel_lock(chan);
724
725 if (!ast_channel_audiohooks(chan)) {
726 ast_channel_unlock(chan);
727 return -1;
728 }
729
730 if (audiohook->type == AST_AUDIOHOOK_TYPE_SPY) {
731 AST_LIST_REMOVE(&ast_channel_audiohooks(chan)->spy_list, audiohook, list);
732 } else if (audiohook->type == AST_AUDIOHOOK_TYPE_WHISPER) {
733 AST_LIST_REMOVE(&ast_channel_audiohooks(chan)->whisper_list, audiohook, list);
734 } else if (audiohook->type == AST_AUDIOHOOK_TYPE_MANIPULATE) {
735 AST_LIST_REMOVE(&ast_channel_audiohooks(chan)->manipulate_list, audiohook, list);
736 }
737
740
741 if (ast_channel_is_bridged(chan)) {
743 }
744
745 ast_channel_unlock(chan);
746
747 return 0;
748}
#define AST_LIST_REMOVE(head, elm, field)
Removes a specific entry from a list.
Definition: linkedlists.h:856

References AST_AUDIOHOOK_STATUS_DONE, AST_AUDIOHOOK_TYPE_MANIPULATE, AST_AUDIOHOOK_TYPE_SPY, AST_AUDIOHOOK_TYPE_WHISPER, ast_audiohook_update_status(), ast_channel_audiohooks(), ast_channel_is_bridged(), ast_channel_lock, ast_channel_set_unbridged_nolock(), ast_channel_unlock, AST_LIST_REMOVE, audiohook_list_set_samplerate_compatibility(), ast_audiohook::list, and ast_audiohook::type.

Referenced by audiohook_move(), remove_detect(), remove_scrambler(), remove_talk_detect(), and speex_write().

◆ ast_audiohook_set_frame_feed_direction()

int ast_audiohook_set_frame_feed_direction ( struct ast_audiohook audiohook,
enum ast_audiohook_direction  direction 
)

Sets direction on audiohook.

Parameters
audiohook
directionIn which direction should the audiohook feed frames, ie if we are snooping 'in', set direction to READ so that only the 'in' frames are fed to the slin factory
Return values
0on success
-1on failure due to audiohook already in use or in shutdown. Can only set direction on NEW audiohooks

Definition at line 150 of file audiohook.c.

151{
152 /* Only set the direction on new audiohooks */
153 if (audiohook->status != AST_AUDIOHOOK_STATUS_NEW) {
154 ast_debug(3, "Can not set direction on attached Audiohook %p\n", audiohook);
155 return -1;
156 }
157
158 audiohook->direction = direction;
159 return 0;
160}
#define ast_debug(level,...)
Log a DEBUG message.

References AST_AUDIOHOOK_STATUS_NEW, ast_debug, ast_audiohook::direction, and ast_audiohook::status.

Referenced by snoop_setup_audiohook().

◆ ast_audiohook_set_mute()

int ast_audiohook_set_mute ( struct ast_channel chan,
const char *  source,
enum ast_audiohook_flags  flag,
int  clear 
)

Mute frames read from or written to a channel.

Parameters
chanChannel to muck with
sourceType of audiohook
flagwhich direction to set / clear
clearset or clear muted frames on direction based on flag parameter
Return values
0success
-1failure

Definition at line 1353 of file audiohook.c.

1354{
1355 struct ast_audiohook *audiohook = NULL;
1356
1357 ast_channel_lock(chan);
1358
1359 /* Ensure the channel has audiohooks on it */
1360 if (!ast_channel_audiohooks(chan)) {
1361 ast_channel_unlock(chan);
1362 return -1;
1363 }
1364
1366
1367 if (audiohook) {
1368 if (clear) {
1369 ast_clear_flag(audiohook, flag);
1370 } else {
1371 ast_set_flag(audiohook, flag);
1372 }
1373 }
1374
1375 ast_channel_unlock(chan);
1376
1377 return (audiohook ? 0 : -1);
1378}
long int flag
Definition: f2c.h:83
#define ast_clear_flag(p, flag)
Definition: utils.h:77
#define ast_set_flag(p, flag)
Definition: utils.h:70

References ast_channel_audiohooks(), ast_channel_lock, ast_channel_unlock, ast_clear_flag, ast_set_flag, find_audiohook_by_source(), NULL, and ast_audiohook::source.

◆ ast_audiohook_set_mute_all()

int ast_audiohook_set_mute_all ( struct ast_channel chan,
const char *  source,
enum ast_audiohook_flags  flag,
int  clear 
)

Mute frames read from or written for all audiohooks on a channel.

Parameters
chanChannel to muck with
sourceType of audiohooks
flagwhich direction to set / clear
clearset or clear muted frames on direction based on flag parameter
Return values
>=0number of muted audiohooks
-1failure

Definition at line 1380 of file audiohook.c.

1381{
1382 struct ast_audiohook *audiohook = NULL;
1383 int count = 0;
1384
1385 ast_channel_lock(chan);
1386
1387 if (!ast_channel_audiohooks(chan)) {
1388 ast_channel_unlock(chan);
1389 return -1;
1390 }
1391
1392 AST_LIST_TRAVERSE(&ast_channel_audiohooks(chan)->spy_list, audiohook, list) {
1393 if (!strcasecmp(audiohook->source, source)) {
1394 count++;
1395 if (clearmute) {
1396 ast_clear_flag(audiohook, flag);
1397 } else {
1398 ast_set_flag(audiohook, flag);
1399 }
1400 }
1401 }
1402
1403 ast_test_suite_event_notify("AUDIOHOOK_GROUP_MUTE_TOGGLE", "Channel: %s\r\nSource: %s\r\nCount: %d\r\n",
1404 ast_channel_name(chan), source, count);
1405
1406 ast_channel_unlock(chan);
1407
1408 return count;
1409}
const char * ast_channel_name(const struct ast_channel *chan)
#define AST_LIST_TRAVERSE(head, var, field)
Loops over (traverses) the entries in a list.
Definition: linkedlists.h:491
#define ast_test_suite_event_notify(s, f,...)
Definition: test.h:189

References ast_channel_audiohooks(), ast_channel_lock, ast_channel_name(), ast_channel_unlock, ast_clear_flag, AST_LIST_TRAVERSE, ast_set_flag, ast_test_suite_event_notify, ast_audiohook::list, NULL, and ast_audiohook::source.

Referenced by manager_mute_mixmonitor().

◆ ast_audiohook_trigger_wait()

void ast_audiohook_trigger_wait ( struct ast_audiohook audiohook)

Wait for audiohook trigger to be triggered.

Parameters
audiohookAudiohook to wait on

Definition at line 1094 of file audiohook.c.

1095{
1096 struct timeval wait;
1097 struct timespec ts;
1098
1099 wait = ast_tvadd(ast_tvnow(), ast_samp2tv(50000, 1000));
1100 ts.tv_sec = wait.tv_sec;
1101 ts.tv_nsec = wait.tv_usec * 1000;
1102
1103 ast_cond_timedwait(&audiohook->trigger, &audiohook->lock, &ts);
1104
1105 return;
1106}
#define ast_cond_timedwait(cond, mutex, time)
Definition: lock.h:206
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
struct timeval ast_tvnow(void)
Returns current timeval. Meant to replace calls to gettimeofday().
Definition: time.h:159

References ast_cond_timedwait, ast_samp2tv(), ast_tvadd(), ast_tvnow(), ast_audiohook::lock, and ast_audiohook::trigger.

Referenced by ast_audiohook_detach(), and mixmonitor_thread().

◆ ast_audiohook_update_status()

void ast_audiohook_update_status ( struct ast_audiohook audiohook,
enum ast_audiohook_status  status 
)

Update audiohook's status.

Parameters
audiohook
status
Note
once status is updated to DONE, this function can not be used to set the status back to any other setting. Setting DONE effectively locks the status as such.

Definition at line 540 of file audiohook.c.

541{
542 ast_audiohook_lock(audiohook);
543 if (audiohook->status != AST_AUDIOHOOK_STATUS_DONE) {
544 audiohook->status = status;
545 ast_cond_signal(&audiohook->trigger);
546 }
547 ast_audiohook_unlock(audiohook);
548}
jack_status_t status
Definition: app_jack.c:146
#define ast_audiohook_lock(ah)
Lock an audiohook.
Definition: audiohook.h:313
#define ast_audiohook_unlock(ah)
Unlock an audiohook.
Definition: audiohook.h:318
#define ast_cond_signal(cond)
Definition: lock.h:203

References ast_audiohook_lock, AST_AUDIOHOOK_STATUS_DONE, ast_audiohook_unlock, ast_cond_signal, status, ast_audiohook::status, and ast_audiohook::trigger.

Referenced by ast_audiohook_attach(), ast_audiohook_detach(), ast_audiohook_detach_list(), ast_audiohook_detach_source(), ast_audiohook_init(), ast_audiohook_remove(), audio_audiohook_write_list(), dtmf_audiohook_write_list(), and stop_mixmonitor_full().

◆ ast_audiohook_volume_adjust()

int ast_audiohook_volume_adjust ( struct ast_channel chan,
enum ast_audiohook_direction  direction,
int  volume 
)

Adjust the volume on frames read from or written to a channel.

Parameters
chanChannel to muck with
directionDirection to increase
volumeValue to adjust the adjustment by
Return values
0on success
-1on failure
Since
1.6.1

Definition at line 1333 of file audiohook.c.

1334{
1336
1337 /* Attempt to find the audiohook volume information, and create an audiohook if none exists */
1338 if (!(audiohook_volume = audiohook_volume_get(chan, 1))) {
1339 return -1;
1340 }
1341
1342 /* Based on the direction change the specific adjustment value */
1345 }
1348 }
1349
1350 return 0;
1351}
static struct audiohook_volume * audiohook_volume_get(struct ast_channel *chan, int create)
Helper function which finds and optionally creates an audiohook_volume_datastore datastore on a chann...
Definition: audiohook.c:1258
Audiohook volume adjustment structure.
Definition: audiohook.c:1183

References AST_AUDIOHOOK_DIRECTION_BOTH, AST_AUDIOHOOK_DIRECTION_READ, AST_AUDIOHOOK_DIRECTION_WRITE, audiohook_volume_get(), NULL, audiohook_volume::read_adjustment, and audiohook_volume::write_adjustment.

Referenced by execute_menu_entry().

◆ ast_audiohook_volume_get()

int ast_audiohook_volume_get ( struct ast_channel chan,
enum ast_audiohook_direction  direction 
)

Retrieve the volume adjustment value on frames read from or written to a channel.

Parameters
chanChannel to retrieve volume adjustment from
directionDirection to retrieve
Returns
adjustment value
Since
1.6.1

Definition at line 1313 of file audiohook.c.

1314{
1316 int adjustment = 0;
1317
1318 /* Attempt to find the audiohook volume information, but do not create it as we only want to look at the values */
1319 if (!(audiohook_volume = audiohook_volume_get(chan, 0))) {
1320 return 0;
1321 }
1322
1323 /* Grab the adjustment value based on direction given */
1325 adjustment = audiohook_volume->read_adjustment;
1327 adjustment = audiohook_volume->write_adjustment;
1328 }
1329
1330 return adjustment;
1331}

References AST_AUDIOHOOK_DIRECTION_READ, AST_AUDIOHOOK_DIRECTION_WRITE, audiohook_volume_get(), NULL, audiohook_volume::read_adjustment, and audiohook_volume::write_adjustment.

Referenced by confbridge_exec().

◆ ast_audiohook_volume_set()

int ast_audiohook_volume_set ( struct ast_channel chan,
enum ast_audiohook_direction  direction,
int  volume 
)

Adjust the volume on frames read from or written to a channel.

Parameters
chanChannel to muck with
directionDirection to set on
volumeValue to adjust the volume by
Return values
0on success
-1on failure
Since
1.6.1

Definition at line 1293 of file audiohook.c.

1294{
1296
1297 /* Attempt to find the audiohook volume information, but only create it if we are not setting the adjustment value to zero */
1298 if (!(audiohook_volume = audiohook_volume_get(chan, (volume ? 1 : 0)))) {
1299 return -1;
1300 }
1301
1302 /* Now based on the direction set the proper value */
1305 }
1308 }
1309
1310 return 0;
1311}

References AST_AUDIOHOOK_DIRECTION_BOTH, AST_AUDIOHOOK_DIRECTION_READ, AST_AUDIOHOOK_DIRECTION_WRITE, audiohook_volume_get(), NULL, audiohook_volume::read_adjustment, and audiohook_volume::write_adjustment.

Referenced by confbridge_exec(), and execute_menu_entry().

◆ ast_audiohook_write_frame()

int ast_audiohook_write_frame ( struct ast_audiohook audiohook,
enum ast_audiohook_direction  direction,
struct ast_frame frame 
)

Writes a frame into the audiohook structure.

Parameters
audiohook
directionDirection the audio frame came from
frameFrame to write in
Return values
0on success
-1on failure

Definition at line 167 of file audiohook.c.

168{
169 struct ast_slinfactory *factory = (direction == AST_AUDIOHOOK_DIRECTION_READ ? &audiohook->read_factory : &audiohook->write_factory);
170 struct ast_slinfactory *other_factory = (direction == AST_AUDIOHOOK_DIRECTION_READ ? &audiohook->write_factory : &audiohook->read_factory);
171 struct timeval *rwtime = (direction == AST_AUDIOHOOK_DIRECTION_READ ? &audiohook->read_time : &audiohook->write_time), previous_time = *rwtime;
172 int our_factory_samples;
173 int our_factory_ms;
174 int other_factory_samples;
175 int other_factory_ms;
176
177 /* Don't feed the frame if we are set to read and this is a write frame or if set to
178 write and this is a read frame as we don't want it. Plus, it can cause mis-resampling
179 if the READ and WRITE frames have different bitrates */
180 if (audiohook->direction != AST_AUDIOHOOK_DIRECTION_BOTH && audiohook->direction != direction) {
181 return 0;
182 }
183
184 /* Update last feeding time to be current */
185 *rwtime = ast_tvnow();
186
187 our_factory_samples = ast_slinfactory_available(factory);
188 our_factory_ms = ast_tvdiff_ms(*rwtime, previous_time) + (our_factory_samples / (audiohook->hook_internal_samp_rate / 1000));
189 other_factory_samples = ast_slinfactory_available(other_factory);
190 other_factory_ms = other_factory_samples / (audiohook->hook_internal_samp_rate / 1000);
191
192 if (ast_test_flag(audiohook, AST_AUDIOHOOK_TRIGGER_SYNC) && (our_factory_ms - other_factory_ms > AST_AUDIOHOOK_SYNC_TOLERANCE)) {
193 ast_debug(4, "Flushing audiohook %p so it remains in sync\n", audiohook);
194 ast_slinfactory_flush(factory);
195 ast_slinfactory_flush(other_factory);
196 }
197
198 if (ast_test_flag(audiohook, AST_AUDIOHOOK_SMALL_QUEUE) && ((our_factory_ms > AST_AUDIOHOOK_SMALL_QUEUE_TOLERANCE) || (other_factory_ms > AST_AUDIOHOOK_SMALL_QUEUE_TOLERANCE))) {
199 ast_debug(4, "Audiohook %p has stale audio in its factories. Flushing them both\n", audiohook);
200 ast_slinfactory_flush(factory);
201 ast_slinfactory_flush(other_factory);
202 } else if ((our_factory_ms > AST_AUDIOHOOK_LONG_QUEUE_TOLERANCE) || (other_factory_ms > AST_AUDIOHOOK_LONG_QUEUE_TOLERANCE)) {
203 ast_debug(4, "Audiohook %p has stale audio in its factories. Flushing them both\n", audiohook);
204 ast_slinfactory_flush(factory);
205 ast_slinfactory_flush(other_factory);
206 }
207
208 /* Write frame out to respective factory */
209 ast_slinfactory_feed(factory, frame);
210
211 /* If we need to notify the respective handler of this audiohook, do so */
213 ast_cond_signal(&audiohook->trigger);
215 ast_cond_signal(&audiohook->trigger);
216 } else if (ast_test_flag(audiohook, AST_AUDIOHOOK_TRIGGER_SYNC)) {
217 ast_cond_signal(&audiohook->trigger);
218 }
219
220 return 0;
221}
#define AST_AUDIOHOOK_LONG_QUEUE_TOLERANCE
Definition: audiohook.c:46
#define AST_AUDIOHOOK_SYNC_TOLERANCE
Definition: audiohook.c:44
#define AST_AUDIOHOOK_SMALL_QUEUE_TOLERANCE
Definition: audiohook.c:45
@ AST_AUDIOHOOK_TRIGGER_MODE
Definition: audiohook.h:55
@ AST_AUDIOHOOK_SMALL_QUEUE
Definition: audiohook.h:63
@ AST_AUDIOHOOK_TRIGGER_READ
Definition: audiohook.h:56
@ AST_AUDIOHOOK_TRIGGER_WRITE
Definition: audiohook.h:57
@ AST_AUDIOHOOK_TRIGGER_SYNC
Definition: audiohook.h:59
unsigned int ast_slinfactory_available(const struct ast_slinfactory *sf)
Retrieve number of samples currently in a slinfactory.
Definition: slinfactory.c:199
void ast_slinfactory_flush(struct ast_slinfactory *sf)
Flush the contents of a slinfactory.
Definition: slinfactory.c:204
int ast_slinfactory_feed(struct ast_slinfactory *sf, struct ast_frame *f)
Feed audio into a slinfactory.
Definition: slinfactory.c:77
struct timeval write_time
Definition: audiohook.h:115
struct timeval read_time
Definition: audiohook.h:114
unsigned int hook_internal_samp_rate
Definition: audiohook.h:120
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

References AST_AUDIOHOOK_DIRECTION_BOTH, AST_AUDIOHOOK_DIRECTION_READ, AST_AUDIOHOOK_DIRECTION_WRITE, AST_AUDIOHOOK_LONG_QUEUE_TOLERANCE, AST_AUDIOHOOK_SMALL_QUEUE, AST_AUDIOHOOK_SMALL_QUEUE_TOLERANCE, AST_AUDIOHOOK_SYNC_TOLERANCE, AST_AUDIOHOOK_TRIGGER_MODE, AST_AUDIOHOOK_TRIGGER_READ, AST_AUDIOHOOK_TRIGGER_SYNC, AST_AUDIOHOOK_TRIGGER_WRITE, ast_cond_signal, ast_debug, ast_slinfactory_available(), ast_slinfactory_feed(), ast_slinfactory_flush(), ast_test_flag, ast_tvdiff_ms(), ast_tvnow(), ast_audiohook::direction, ast_audiohook::hook_internal_samp_rate, ast_audiohook::read_factory, ast_audiohook::read_time, ast_audiohook::trigger, ast_audiohook::write_factory, and ast_audiohook::write_time.

Referenced by audio_audiohook_write_list(), channel_spy(), do_broadcast(), and snoop_write().

◆ ast_audiohook_write_list()

struct ast_frame * ast_audiohook_write_list ( struct ast_channel chan,
struct ast_audiohook_list audiohook_list,
enum ast_audiohook_direction  direction,
struct ast_frame frame 
)

Pass a frame off to be handled by the audiohook core.

Parameters
chanChannel that the list is coming off of
audiohook_listList of audiohooks
directionDirection frame is coming in from
frameThe frame itself
Returns
frame on success
Return values
NULLon failure

Definition at line 1079 of file audiohook.c.

1080{
1081 /* Pass off frame to it's respective list write function */
1082 if (frame->frametype == AST_FRAME_VOICE) {
1083 return audio_audiohook_write_list(chan, audiohook_list, direction, frame);
1084 } else if (frame->frametype == AST_FRAME_DTMF) {
1085 return dtmf_audiohook_write_list(chan, audiohook_list, direction, frame);
1086 } else {
1087 return frame;
1088 }
1089}
static struct ast_frame * audio_audiohook_write_list(struct ast_channel *chan, struct ast_audiohook_list *audiohook_list, enum ast_audiohook_direction direction, struct ast_frame *frame)
Pass an AUDIO frame off to be handled by the audiohook core.
Definition: audiohook.c:914
static struct ast_frame * dtmf_audiohook_write_list(struct ast_channel *chan, struct ast_audiohook_list *audiohook_list, enum ast_audiohook_direction direction, struct ast_frame *frame)
Pass a DTMF frame off to be handled by the audiohook core.
Definition: audiohook.c:758
#define AST_FRAME_DTMF
@ AST_FRAME_VOICE
enum ast_frame_type frametype

References AST_FRAME_DTMF, AST_FRAME_VOICE, audio_audiohook_write_list(), dtmf_audiohook_write_list(), and ast_frame::frametype.

Referenced by __ast_read(), and ast_write_stream().

◆ ast_audiohook_write_list_empty()

int ast_audiohook_write_list_empty ( struct ast_audiohook_list audiohook_list)

Determine if a audiohook_list is empty or not.

Parameters
audiohook_listAudiohook to check. (NULL also means empty)
Return values
0false
1true

Definition at line 1071 of file audiohook.c.

1072{
1073 return !audiohook_list
1074 || (AST_LIST_EMPTY(&audiohook_list->spy_list)
1075 && AST_LIST_EMPTY(&audiohook_list->whisper_list)
1076 && AST_LIST_EMPTY(&audiohook_list->manipulate_list));
1077}
#define AST_LIST_EMPTY(head)
Checks whether the specified list contains any entries.
Definition: linkedlists.h:450

References AST_LIST_EMPTY, ast_audiohook_list::manipulate_list, ast_audiohook_list::spy_list, and ast_audiohook_list::whisper_list.

Referenced by __ast_read(), ast_channel_has_audio_frame_or_monitor(), ast_channel_has_hook_requiring_audio(), and ast_write_stream().

◆ ast_channel_audiohook_count_by_source()

int ast_channel_audiohook_count_by_source ( struct ast_channel chan,
const char *  source,
enum ast_audiohook_type  type 
)

Find out how many audiohooks from a certain source exist on a given channel, regardless of status.

Parameters
chanThe channel on which to find the spies
sourceThe audiohook's source
typeThe type of audiohook
Returns
number of audiohooks which are from the source specified

Note: Function performs nlocking.

Definition at line 1109 of file audiohook.c.

1110{
1111 int count = 0;
1112 struct ast_audiohook *ah = NULL;
1113
1114 if (!ast_channel_audiohooks(chan)) {
1115 return -1;
1116 }
1117
1118 switch (type) {
1120 AST_LIST_TRAVERSE(&ast_channel_audiohooks(chan)->spy_list, ah, list) {
1121 if (!strcmp(ah->source, source)) {
1122 count++;
1123 }
1124 }
1125 break;
1127 AST_LIST_TRAVERSE(&ast_channel_audiohooks(chan)->whisper_list, ah, list) {
1128 if (!strcmp(ah->source, source)) {
1129 count++;
1130 }
1131 }
1132 break;
1134 AST_LIST_TRAVERSE(&ast_channel_audiohooks(chan)->manipulate_list, ah, list) {
1135 if (!strcmp(ah->source, source)) {
1136 count++;
1137 }
1138 }
1139 break;
1140 default:
1141 ast_debug(1, "Invalid audiohook type supplied, (%u)\n", type);
1142 return -1;
1143 }
1144
1145 return count;
1146}

References AST_AUDIOHOOK_TYPE_MANIPULATE, AST_AUDIOHOOK_TYPE_SPY, AST_AUDIOHOOK_TYPE_WHISPER, ast_channel_audiohooks(), ast_debug, AST_LIST_TRAVERSE, ast_audiohook::list, NULL, ast_audiohook::source, and type.

Referenced by feature_automixmonitor().

◆ ast_channel_audiohook_count_by_source_running()

int ast_channel_audiohook_count_by_source_running ( struct ast_channel chan,
const char *  source,
enum ast_audiohook_type  type 
)

Find out how many spies of a certain type exist on a given channel, and are in state running.

Parameters
chanThe channel on which to find the spies
sourceThe source of the audiohook
typeThe type of spy to look for
Returns
number of running audiohooks which are from the source specified

Note: Function performs no locking.

Definition at line 1149 of file audiohook.c.

1150{
1151 int count = 0;
1152 struct ast_audiohook *ah = NULL;
1153 if (!ast_channel_audiohooks(chan))
1154 return -1;
1155
1156 switch (type) {
1158 AST_LIST_TRAVERSE(&ast_channel_audiohooks(chan)->spy_list, ah, list) {
1159 if ((!strcmp(ah->source, source)) && (ah->status == AST_AUDIOHOOK_STATUS_RUNNING))
1160 count++;
1161 }
1162 break;
1164 AST_LIST_TRAVERSE(&ast_channel_audiohooks(chan)->whisper_list, ah, list) {
1165 if ((!strcmp(ah->source, source)) && (ah->status == AST_AUDIOHOOK_STATUS_RUNNING))
1166 count++;
1167 }
1168 break;
1170 AST_LIST_TRAVERSE(&ast_channel_audiohooks(chan)->manipulate_list, ah, list) {
1171 if ((!strcmp(ah->source, source)) && (ah->status == AST_AUDIOHOOK_STATUS_RUNNING))
1172 count++;
1173 }
1174 break;
1175 default:
1176 ast_debug(1, "Invalid audiohook type supplied, (%u)\n", type);
1177 return -1;
1178 }
1179 return count;
1180}

References AST_AUDIOHOOK_STATUS_RUNNING, AST_AUDIOHOOK_TYPE_MANIPULATE, AST_AUDIOHOOK_TYPE_SPY, AST_AUDIOHOOK_TYPE_WHISPER, ast_channel_audiohooks(), ast_debug, AST_LIST_TRAVERSE, ast_audiohook::list, NULL, ast_audiohook::source, ast_audiohook::status, and type.

◆ audio_audiohook_write_list()

static struct ast_frame * audio_audiohook_write_list ( struct ast_channel chan,
struct ast_audiohook_list audiohook_list,
enum ast_audiohook_direction  direction,
struct ast_frame frame 
)
static

Pass an AUDIO frame off to be handled by the audiohook core.

This function has 3 ast_frames and 3 parts to handle each. At the beginning of this function all 3 frames, start_frame, middle_frame, and end_frame point to the initial input frame.

Part_1: Translate the start_frame into SLINEAR audio if it is not already in that format. The result of this part is middle_frame is guaranteed to be in SLINEAR format for Part_2. Part_2: Send middle_frame off to spies and manipulators. At this point middle_frame is either a new frame as result of the translation, or points directly to the start_frame because no translation to SLINEAR audio was required. Part_3: Translate end_frame's audio back into the format of start frame if necessary. This is only necessary if manipulation of middle_frame occurred.

Parameters
chanChannel that the list is coming off of
audiohook_listList of audiohooks
directionDirection frame is coming in from
frameThe frame itself
Returns
frame on success
Return values
NULLon failure

Definition at line 914 of file audiohook.c.

915{
916 struct ast_frame *start_frame = frame, *middle_frame = frame, *end_frame = frame;
917 struct ast_audiohook *audiohook = NULL;
918 int samples;
919 int middle_frame_manipulated = 0;
920 int removed = 0;
921 int internal_sample_rate;
922
923 /* ---Part_1. translate start_frame to SLINEAR if necessary. */
924 if (!(middle_frame = audiohook_list_translate_to_slin(audiohook_list, direction, start_frame))) {
925 return frame;
926 }
927
928 /* If the translation resulted in an interpolated frame then immediately return as audiohooks
929 * rely on actual media being present to do things.
930 */
931 if (!middle_frame->data.ptr) {
932 if (middle_frame != start_frame) {
933 ast_frfree(middle_frame);
934 }
935 return start_frame;
936 }
937
938 samples = middle_frame->samples;
939
940 /*
941 * While processing each audiohook check to see if the internal sample rate needs
942 * to be adjusted (it should be the highest rate specified between formats and
943 * hooks). The given audiohook_list's internal sample rate is then set to the
944 * updated value before returning.
945 *
946 * If slin compatibility mode is turned on then an audiohook's internal sample
947 * rate is set to its audiohook_list's rate. If an audiohook_list's rate is
948 * adjusted during this pass then the change is picked up by the audiohooks
949 * on the next pass.
950 */
951 internal_sample_rate = audiohook_list->list_internal_samp_rate;
952
953 /* ---Part_2: Send middle_frame to spy and manipulator lists. middle_frame is guaranteed to be SLINEAR here.*/
954 /* Queue up signed linear frame to each spy */
955 AST_LIST_TRAVERSE_SAFE_BEGIN(&audiohook_list->spy_list, audiohook, list) {
956 ast_audiohook_lock(audiohook);
957 if (audiohook->status != AST_AUDIOHOOK_STATUS_RUNNING) {
959 removed = 1;
961 ast_audiohook_unlock(audiohook);
962 if (ast_channel_is_bridged(chan)) {
964 }
965 continue;
966 }
967 audiohook_list_set_hook_rate(audiohook_list, audiohook, &internal_sample_rate);
968 ast_audiohook_write_frame(audiohook, direction, middle_frame);
969 ast_audiohook_unlock(audiohook);
970 }
972
973 /* If this frame is being written out to the channel then we need to use whisper sources */
974 if (!AST_LIST_EMPTY(&audiohook_list->whisper_list)) {
975 int i = 0;
976 short read_buf[samples], combine_buf[samples], *data1 = NULL, *data2 = NULL;
977 memset(&combine_buf, 0, sizeof(combine_buf));
978 AST_LIST_TRAVERSE_SAFE_BEGIN(&audiohook_list->whisper_list, audiohook, list) {
979 struct ast_slinfactory *factory = (direction == AST_AUDIOHOOK_DIRECTION_READ ? &audiohook->read_factory : &audiohook->write_factory);
980 ast_audiohook_lock(audiohook);
981 if (audiohook->status != AST_AUDIOHOOK_STATUS_RUNNING) {
983 removed = 1;
985 ast_audiohook_unlock(audiohook);
986 if (ast_channel_is_bridged(chan)) {
988 }
989 continue;
990 }
991 audiohook_list_set_hook_rate(audiohook_list, audiohook, &internal_sample_rate);
992 if (ast_slinfactory_available(factory) >= samples && ast_slinfactory_read(factory, read_buf, samples)) {
993 /* Take audio from this whisper source and combine it into our main buffer */
994 for (i = 0, data1 = combine_buf, data2 = read_buf; i < samples; i++, data1++, data2++) {
995 ast_slinear_saturated_add(data1, data2);
996 }
997 }
998 ast_audiohook_unlock(audiohook);
999 }
1001 /* We take all of the combined whisper sources and combine them into the audio being written out */
1002 for (i = 0, data1 = middle_frame->data.ptr, data2 = combine_buf; i < samples; i++, data1++, data2++) {
1003 ast_slinear_saturated_add(data1, data2);
1004 }
1005 middle_frame_manipulated = 1;
1006 }
1007
1008 /* Pass off frame to manipulate audiohooks */
1009 if (!AST_LIST_EMPTY(&audiohook_list->manipulate_list)) {
1010 AST_LIST_TRAVERSE_SAFE_BEGIN(&audiohook_list->manipulate_list, audiohook, list) {
1011 ast_audiohook_lock(audiohook);
1012 if (audiohook->status != AST_AUDIOHOOK_STATUS_RUNNING) {
1014 removed = 1;
1016 ast_audiohook_unlock(audiohook);
1017 /* We basically drop all of our links to the manipulate audiohook and prod it to do it's own destructive things */
1018 audiohook->manipulate_callback(audiohook, chan, NULL, direction);
1019 if (ast_channel_is_bridged(chan)) {
1021 }
1022 continue;
1023 }
1024 audiohook_list_set_hook_rate(audiohook_list, audiohook, &internal_sample_rate);
1025 /*
1026 * Feed in frame to manipulation.
1027 */
1028 if (!audiohook->manipulate_callback(audiohook, chan, middle_frame, direction)) {
1029 /*
1030 * XXX FAILURES ARE IGNORED XXX
1031 * If the manipulation fails then the frame will be returned in its original state.
1032 * Since there are potentially more manipulator callbacks in the list, no action should
1033 * be taken here to exit early.
1034 */
1035 middle_frame_manipulated = 1;
1036 }
1037 ast_audiohook_unlock(audiohook);
1038 }
1040 }
1041
1042 /* ---Part_3: Decide what to do with the end_frame (whether to transcode or not) */
1043 if (middle_frame_manipulated) {
1044 if (!(end_frame = audiohook_list_translate_to_native(audiohook_list, direction, middle_frame, start_frame->subclass.format))) {
1045 /* translation failed, so just pass back the input frame */
1046 end_frame = start_frame;
1047 }
1048 } else {
1049 end_frame = start_frame;
1050 }
1051 /* clean up our middle_frame if required */
1052 if (middle_frame != end_frame) {
1053 ast_frfree(middle_frame);
1054 middle_frame = NULL;
1055 }
1056
1057 /* Before returning, if an audiohook got removed, reset samplerate compatibility */
1058 if (removed) {
1060 } else {
1061 /*
1062 * Set the audiohook_list's rate to the updated rate. Note that if a hook
1063 * was removed then the list's internal rate is reset to the default.
1064 */
1065 audiohook_list->list_internal_samp_rate = internal_sample_rate;
1066 }
1067
1068 return end_frame;
1069}
int ast_audiohook_write_frame(struct ast_audiohook *audiohook, enum ast_audiohook_direction direction, struct ast_frame *frame)
Writes a frame into the audiohook structure.
Definition: audiohook.c:167
static struct ast_frame * audiohook_list_translate_to_native(struct ast_audiohook_list *audiohook_list, enum ast_audiohook_direction direction, struct ast_frame *slin_frame, struct ast_format *outformat)
Definition: audiohook.c:836
static struct ast_frame * audiohook_list_translate_to_slin(struct ast_audiohook_list *audiohook_list, enum ast_audiohook_direction direction, struct ast_frame *frame)
Definition: audiohook.c:790
static void audiohook_list_set_hook_rate(struct ast_audiohook_list *audiohook_list, struct ast_audiohook *audiohook, int *rate)
Set the audiohook's internal sample rate to the audiohook_list's rate, but only when native slin comp...
Definition: audiohook.c:868
#define ast_frfree(fr)
#define AST_LIST_REMOVE_CURRENT(field)
Removes the current entry from a list during a traversal.
Definition: linkedlists.h:557
int ast_slinfactory_read(struct ast_slinfactory *sf, short *buf, size_t samples)
Read samples from a slinfactory.
Definition: slinfactory.c:145
struct ast_format * format
Data structure associated with a single frame of data.
struct ast_frame_subclass subclass
static force_inline void ast_slinear_saturated_add(short *input, short *value)
Definition: utils.h:450

References AST_AUDIOHOOK_DIRECTION_READ, ast_audiohook_lock, AST_AUDIOHOOK_STATUS_DONE, AST_AUDIOHOOK_STATUS_RUNNING, ast_audiohook_unlock, ast_audiohook_update_status(), ast_audiohook_write_frame(), ast_channel_is_bridged(), ast_channel_set_unbridged_nolock(), ast_frfree, AST_LIST_EMPTY, AST_LIST_REMOVE_CURRENT, AST_LIST_TRAVERSE_SAFE_BEGIN, AST_LIST_TRAVERSE_SAFE_END, ast_slinear_saturated_add(), ast_slinfactory_available(), ast_slinfactory_read(), audiohook_list_set_hook_rate(), audiohook_list_set_samplerate_compatibility(), audiohook_list_translate_to_native(), audiohook_list_translate_to_slin(), ast_frame_subclass::format, ast_audiohook::list, ast_audiohook_list::list_internal_samp_rate, ast_audiohook::manipulate_callback, ast_audiohook_list::manipulate_list, NULL, ast_audiohook::read_factory, ast_frame::samples, ast_audiohook_list::spy_list, ast_audiohook::status, ast_frame::subclass, ast_audiohook_list::whisper_list, and ast_audiohook::write_factory.

Referenced by ast_audiohook_write_list().

◆ audiohook_list_set_hook_rate()

static void audiohook_list_set_hook_rate ( struct ast_audiohook_list audiohook_list,
struct ast_audiohook audiohook,
int *  rate 
)
static

Set the audiohook's internal sample rate to the audiohook_list's rate, but only when native slin compatibility is turned on.

Parameters
audiohook_listaudiohook_list data object
audiohookthe audiohook to update
ratethe current max internal sample rate

Definition at line 868 of file audiohook.c.

870{
871 /* The rate should always be the max between itself and the hook */
872 if (audiohook->hook_internal_samp_rate > *rate) {
873 *rate = audiohook->hook_internal_samp_rate;
874 }
875
876 /*
877 * If native slin compatibility is turned on then update the audiohook
878 * with the audiohook_list's current rate. Note, the audiohook's rate is
879 * set to the audiohook_list's rate and not the given rate. If there is
880 * a change in rate the hook's rate is changed on its next check.
881 */
882 if (audiohook_list->native_slin_compatible) {
884 audiohook_set_internal_rate(audiohook, audiohook_list->list_internal_samp_rate, 1);
885 } else {
887 }
888}
@ AST_AUDIOHOOK_COMPATIBLE
Definition: audiohook.h:66
int native_slin_compatible
Definition: audiohook.c:60

References AST_AUDIOHOOK_COMPATIBLE, ast_clear_flag, ast_set_flag, audiohook_set_internal_rate(), ast_audiohook::hook_internal_samp_rate, ast_audiohook_list::list_internal_samp_rate, and ast_audiohook_list::native_slin_compatible.

Referenced by audio_audiohook_write_list().

◆ audiohook_list_set_samplerate_compatibility()

static void audiohook_list_set_samplerate_compatibility ( struct ast_audiohook_list audiohook_list)
static

Definition at line 456 of file audiohook.c.

457{
458 struct ast_audiohook *ah = NULL;
459
460 /*
461 * Anytime the samplerate compatibility is set (attach/remove an audiohook) the
462 * list's internal sample rate needs to be reset so that the next time processing
463 * through write_list, if needed, it will get updated to the correct rate.
464 *
465 * A list's internal rate always chooses the higher between its own rate and a
466 * given rate. If the current rate is being driven by an audiohook that wanted a
467 * higher rate then when this audiohook is removed the list's rate would remain
468 * at that level when it should be lower, and with no way to lower it since any
469 * rate compared against it would be lower.
470 *
471 * By setting it back to the lowest rate it can recalulate the new highest rate.
472 */
474
475 audiohook_list->native_slin_compatible = 1;
476 AST_LIST_TRAVERSE(&audiohook_list->manipulate_list, ah, list) {
478 audiohook_list->native_slin_compatible = 0;
479 return;
480 }
481 }
482}
@ AST_AUDIOHOOK_MANIPULATE_ALL_RATES
Definition: audiohook.h:75

References AST_AUDIOHOOK_MANIPULATE_ALL_RATES, AST_LIST_TRAVERSE, DEFAULT_INTERNAL_SAMPLE_RATE, ast_audiohook::init_flags, ast_audiohook::list, ast_audiohook_list::list_internal_samp_rate, ast_audiohook_list::manipulate_list, ast_audiohook_list::native_slin_compatible, and NULL.

Referenced by ast_audiohook_attach(), ast_audiohook_remove(), audio_audiohook_write_list(), and dtmf_audiohook_write_list().

◆ audiohook_list_translate_to_native()

static struct ast_frame * audiohook_list_translate_to_native ( struct ast_audiohook_list audiohook_list,
enum ast_audiohook_direction  direction,
struct ast_frame slin_frame,
struct ast_format outformat 
)
static

Definition at line 836 of file audiohook.c.

838{
839 struct ast_audiohook_translate *out_translate = (direction == AST_AUDIOHOOK_DIRECTION_READ ? &audiohook_list->out_translate[0] : &audiohook_list->out_translate[1]);
840 struct ast_frame *outframe = NULL;
841 if (ast_format_cmp(slin_frame->subclass.format, outformat) == AST_FORMAT_CMP_NOT_EQUAL) {
842 /* rebuild translators if necessary */
843 if (ast_format_cmp(out_translate->format, outformat) == AST_FORMAT_CMP_NOT_EQUAL) {
844 if (out_translate->trans_pvt) {
845 ast_translator_free_path(out_translate->trans_pvt);
846 }
847 if (!(out_translate->trans_pvt = ast_translator_build_path(outformat, slin_frame->subclass.format))) {
848 return NULL;
849 }
850 ao2_replace(out_translate->format, outformat);
851 }
852 /* translate back to the format the frame came in as. */
853 if (!(outframe = ast_translate(out_translate->trans_pvt, slin_frame, 0))) {
854 return NULL;
855 }
856 }
857 return outframe;
858}
#define ao2_replace(dst, src)
Replace one object reference with another cleaning up the original.
Definition: astobj2.h:501
enum ast_format_cmp_res ast_format_cmp(const struct ast_format *format1, const struct ast_format *format2)
Compare two formats.
Definition: format.c:201
@ AST_FORMAT_CMP_NOT_EQUAL
Definition: format.h:38
struct ast_frame * ast_translate(struct ast_trans_pvt *tr, struct ast_frame *f, int consume)
translates one or more frames Apply an input frame into the translator and receive zero or one output...
Definition: translate.c:566
struct ast_trans_pvt * ast_translator_build_path(struct ast_format *dest, struct ast_format *source)
Builds a translator path Build a path (possibly NULL) from source to dest.
Definition: translate.c:486

References ao2_replace, AST_AUDIOHOOK_DIRECTION_READ, ast_format_cmp(), AST_FORMAT_CMP_NOT_EQUAL, ast_translate(), ast_translator_build_path(), ast_translator_free_path(), ast_frame_subclass::format, ast_audiohook_translate::format, NULL, ast_audiohook_list::out_translate, ast_frame::subclass, and ast_audiohook_translate::trans_pvt.

Referenced by audio_audiohook_write_list().

◆ audiohook_list_translate_to_slin()

static struct ast_frame * audiohook_list_translate_to_slin ( struct ast_audiohook_list audiohook_list,
enum ast_audiohook_direction  direction,
struct ast_frame frame 
)
static

Definition at line 790 of file audiohook.c.

792{
794 &audiohook_list->in_translate[0] : &audiohook_list->in_translate[1]);
795 struct ast_frame *new_frame = frame;
796 struct ast_format *slin;
797
798 /*
799 * If we are capable of sample rates other that 8khz, update the internal
800 * audiohook_list's rate and higher sample rate audio arrives. If native
801 * slin compatibility is turned on all audiohooks in the list will be
802 * updated as well during read/write processing.
803 */
804 audiohook_list->list_internal_samp_rate =
806
808 if (ast_format_cmp(frame->subclass.format, slin) == AST_FORMAT_CMP_EQUAL) {
809 return new_frame;
810 }
811
812 if (!in_translate->format ||
813 ast_format_cmp(frame->subclass.format, in_translate->format) != AST_FORMAT_CMP_EQUAL) {
814 struct ast_trans_pvt *new_trans;
815
816 new_trans = ast_translator_build_path(slin, frame->subclass.format);
817 if (!new_trans) {
818 return NULL;
819 }
820
821 if (in_translate->trans_pvt) {
822 ast_translator_free_path(in_translate->trans_pvt);
823 }
824 in_translate->trans_pvt = new_trans;
825
826 ao2_replace(in_translate->format, frame->subclass.format);
827 }
828
829 if (!(new_frame = ast_translate(in_translate->trans_pvt, frame, 0))) {
830 return NULL;
831 }
832
833 return new_frame;
834}
unsigned int ast_format_get_sample_rate(const struct ast_format *format)
Get the sample rate of a media format.
Definition: format.c:379
@ AST_FORMAT_CMP_EQUAL
Definition: format.h:36
struct ast_format * ast_format_cache_get_slin_by_rate(unsigned int rate)
Retrieve the best signed linear format given a sample rate.
Definition: format_cache.c:512
Definition of a media format.
Definition: format.c:43
Default structure for translators, with the basic fields and buffers, all allocated as part of the sa...
Definition: translate.h:213
#define MAX(a, b)
Definition: utils.h:233

References ao2_replace, AST_AUDIOHOOK_DIRECTION_READ, ast_format_cache_get_slin_by_rate(), ast_format_cmp(), AST_FORMAT_CMP_EQUAL, ast_format_get_sample_rate(), ast_translate(), ast_translator_build_path(), ast_translator_free_path(), ast_frame_subclass::format, ast_audiohook_translate::format, ast_audiohook_list::in_translate, ast_audiohook_list::list_internal_samp_rate, MAX, NULL, ast_frame::subclass, and ast_audiohook_translate::trans_pvt.

Referenced by audio_audiohook_write_list().

◆ audiohook_move()

static void audiohook_move ( struct ast_channel old_chan,
struct ast_channel new_chan,
struct ast_audiohook audiohook 
)
static

Definition at line 637 of file audiohook.c.

638{
639 enum ast_audiohook_status oldstatus;
640
641 /* By locking both channels and the audiohook, we can assure that
642 * another thread will not have a chance to read the audiohook's status
643 * as done, even though ast_audiohook_remove signals the trigger
644 * condition.
645 */
646 ast_audiohook_lock(audiohook);
647 oldstatus = audiohook->status;
648
649 ast_audiohook_remove(old_chan, audiohook);
650 ast_audiohook_attach(new_chan, audiohook);
651
652 audiohook->status = oldstatus;
653 ast_audiohook_unlock(audiohook);
654}
int ast_audiohook_remove(struct ast_channel *chan, struct ast_audiohook *audiohook)
Remove an audiohook from a specified channel.
Definition: audiohook.c:721
int ast_audiohook_attach(struct ast_channel *chan, struct ast_audiohook *audiohook)
Attach audiohook to channel.
Definition: audiohook.c:484
ast_audiohook_status
Definition: audiohook.h:41

References ast_audiohook_attach(), ast_audiohook_lock, ast_audiohook_remove(), ast_audiohook_unlock, and ast_audiohook::status.

Referenced by ast_audiohook_move_all(), and ast_audiohook_move_by_source().

◆ audiohook_read_frame_both()

static struct ast_frame * audiohook_read_frame_both ( struct ast_audiohook audiohook,
size_t  samples,
struct ast_frame **  read_reference,
struct ast_frame **  write_reference 
)
static

Definition at line 257 of file audiohook.c.

258{
259 int count;
260 int usable_read;
261 int usable_write;
262 short adjust_value;
263 short buf1[samples];
264 short buf2[samples];
265 short *read_buf = NULL;
266 short *write_buf = NULL;
267 struct ast_frame frame = {
269 .datalen = sizeof(buf1),
270 .samples = samples,
271 };
272
273 /* Make sure both factories have the required samples */
274 usable_read = (ast_slinfactory_available(&audiohook->read_factory) >= samples ? 1 : 0);
275 usable_write = (ast_slinfactory_available(&audiohook->write_factory) >= samples ? 1 : 0);
276
277 if (!usable_read && !usable_write) {
278 /* If both factories are unusable bail out */
279 ast_debug(3, "Read factory %p and write factory %p both fail to provide %zu samples\n", &audiohook->read_factory, &audiohook->write_factory, samples);
280 return NULL;
281 }
282
283 /* If we want to provide only a read factory make sure we aren't waiting for other audio */
284 if (usable_read && !usable_write && (ast_tvdiff_ms(ast_tvnow(), audiohook->write_time) < (samples/8)*2)) {
285 ast_debug(3, "Write factory %p was pretty quick last time, waiting for them.\n", &audiohook->write_factory);
286 return NULL;
287 }
288
289 /* If we want to provide only a write factory make sure we aren't waiting for other audio */
290 if (usable_write && !usable_read && (ast_tvdiff_ms(ast_tvnow(), audiohook->read_time) < (samples/8)*2)) {
291 ast_debug(3, "Read factory %p was pretty quick last time, waiting for them.\n", &audiohook->read_factory);
292 return NULL;
293 }
294
295 /* Start with the read factory... if there are enough samples, read them in */
296 if (usable_read) {
297 if (ast_slinfactory_read(&audiohook->read_factory, buf1, samples)) {
298 read_buf = buf1;
299
300 if ((ast_test_flag(audiohook, AST_AUDIOHOOK_MUTE_READ))) {
301 /* Clear the frame data if we are muting */
302 memset(buf1, 0, sizeof(buf1));
303 } else if (audiohook->options.read_volume) {
304 /* Adjust read volume if need be */
305 adjust_value = abs(audiohook->options.read_volume);
306 for (count = 0; count < samples; count++) {
307 if (audiohook->options.read_volume > 0) {
308 ast_slinear_saturated_multiply(&buf1[count], &adjust_value);
309 } else if (audiohook->options.read_volume < 0) {
310 ast_slinear_saturated_divide(&buf1[count], &adjust_value);
311 }
312 }
313 }
314 }
315 } else {
316 ast_debug(1, "Failed to get %d samples from read factory %p\n", (int)samples, &audiohook->read_factory);
317 }
318
319 /* Move on to the write factory... if there are enough samples, read them in */
320 if (usable_write) {
321 if (ast_slinfactory_read(&audiohook->write_factory, buf2, samples)) {
322 write_buf = buf2;
323
324 if ((ast_test_flag(audiohook, AST_AUDIOHOOK_MUTE_WRITE))) {
325 /* Clear the frame data if we are muting */
326 memset(buf2, 0, sizeof(buf2));
327 } else if (audiohook->options.write_volume) {
328 /* Adjust write volume if need be */
329 adjust_value = abs(audiohook->options.write_volume);
330 for (count = 0; count < samples; count++) {
331 if (audiohook->options.write_volume > 0) {
332 ast_slinear_saturated_multiply(&buf2[count], &adjust_value);
333 } else if (audiohook->options.write_volume < 0) {
334 ast_slinear_saturated_divide(&buf2[count], &adjust_value);
335 }
336 }
337 }
338 }
339 } else {
340 ast_debug(3, "Failed to get %d samples from write factory %p\n", (int)samples, &audiohook->write_factory);
341 }
342
344
345 /* Should we substitute silence if one side lacks audio? */
347 if (read_reference && !read_buf && write_buf) {
348 read_buf = buf1;
349 memset(buf1, 0, sizeof(buf1));
350 } else if (write_reference && read_buf && !write_buf) {
351 write_buf = buf2;
352 memset(buf2, 0, sizeof(buf2));
353 }
354 }
355
356 /* Basically we figure out which buffer to use... and if mixing can be done here */
357 if (read_buf && read_reference) {
358 frame.data.ptr = read_buf;
359 *read_reference = ast_frdup(&frame);
360 }
361 if (write_buf && write_reference) {
362 frame.data.ptr = write_buf;
363 *write_reference = ast_frdup(&frame);
364 }
365
366 /* Make the correct buffer part of the built frame, so it gets duplicated. */
367 if (read_buf) {
368 frame.data.ptr = read_buf;
369 if (write_buf) {
370 for (count = 0; count < samples; count++) {
372 }
373 }
374 } else if (write_buf) {
375 frame.data.ptr = write_buf;
376 } else {
377 return NULL;
378 }
379
380 /* Yahoo, a combined copy of the audio! */
381 return ast_frdup(&frame);
382}
@ AST_AUDIOHOOK_SUBSTITUTE_SILENCE
Definition: audiohook.h:68
void write_buf(int file, char *buffer, int num)
Definition: eagi_proxy.c:312
#define abs(x)
Definition: f2c.h:195
static struct ast_threadstorage buf2
static struct ast_threadstorage buf1
#define ast_frdup(fr)
Copies a frame.
struct ast_audiohook_options options
Definition: audiohook.h:119
union ast_frame::@226 data
static force_inline void ast_slinear_saturated_multiply(short *input, short *value)
Definition: utils.h:476
static force_inline void ast_slinear_saturated_divide(short *input, short *value)
Definition: utils.h:502

References abs, AST_AUDIOHOOK_MUTE_READ, AST_AUDIOHOOK_MUTE_WRITE, AST_AUDIOHOOK_SUBSTITUTE_SILENCE, ast_debug, ast_format_cache_get_slin_by_rate(), AST_FRAME_VOICE, ast_frdup, ast_slinear_saturated_add(), ast_slinear_saturated_divide(), ast_slinear_saturated_multiply(), ast_slinfactory_available(), ast_slinfactory_read(), ast_test_flag, ast_tvdiff_ms(), ast_tvnow(), buf1, buf2, ast_frame::data, ast_frame_subclass::format, ast_frame::frametype, ast_audiohook::hook_internal_samp_rate, NULL, ast_audiohook::options, ast_frame::ptr, ast_audiohook::read_factory, ast_audiohook::read_time, ast_audiohook_options::read_volume, ast_frame::samples, ast_frame::subclass, write_buf(), ast_audiohook::write_factory, ast_audiohook::write_time, and ast_audiohook_options::write_volume.

Referenced by audiohook_read_frame_helper().

◆ audiohook_read_frame_helper()

static struct ast_frame * audiohook_read_frame_helper ( struct ast_audiohook audiohook,
size_t  samples,
enum ast_audiohook_direction  direction,
struct ast_format format,
struct ast_frame **  read_reference,
struct ast_frame **  write_reference 
)
static

Definition at line 384 of file audiohook.c.

385{
386 struct ast_frame *read_frame = NULL, *final_frame = NULL;
387 struct ast_format *slin;
388
389 /*
390 * Update the rate if compatibility mode is turned off or if it is
391 * turned on and the format rate is higher than the current rate.
392 *
393 * This makes it so any unnecessary rate switching/resetting does
394 * not take place and also any associated audiohook_list's internal
395 * sample rate maintains the highest sample rate between hooks.
396 */
397 if (!ast_test_flag(audiohook, AST_AUDIOHOOK_COMPATIBLE) ||
401 }
402
403 /* If the sample rate of the requested format differs from that of the underlying audiohook
404 * sample rate determine how many samples we actually need to get from the audiohook. This
405 * needs to occur as the signed linear factory stores them at the rate of the audiohook.
406 * We do this by determining the duration of audio they've requested and then determining
407 * how many samples that would be in the audiohook format.
408 */
409 if (ast_format_get_sample_rate(format) != audiohook->hook_internal_samp_rate) {
410 samples = (audiohook->hook_internal_samp_rate / 1000) * (samples / (ast_format_get_sample_rate(format) / 1000));
411 }
412
414 audiohook_read_frame_both(audiohook, samples, read_reference, write_reference) :
415 audiohook_read_frame_single(audiohook, samples, direction)))) {
416 return NULL;
417 }
418
420
421 /* If they don't want signed linear back out, we'll have to send it through the translation path */
422 if (ast_format_cmp(format, slin) != AST_FORMAT_CMP_EQUAL) {
423 /* Rebuild translation path if different format then previously */
424 if (ast_format_cmp(format, audiohook->format) == AST_FORMAT_CMP_NOT_EQUAL) {
425 if (audiohook->trans_pvt) {
427 audiohook->trans_pvt = NULL;
428 }
429
430 /* Setup new translation path for this format... if we fail we can't very well return signed linear so free the frame and return nothing */
431 if (!(audiohook->trans_pvt = ast_translator_build_path(format, slin))) {
433 return NULL;
434 }
435 ao2_replace(audiohook->format, format);
436 }
437 /* Convert to requested format, and allow the read in frame to be freed */
438 final_frame = ast_translate(audiohook->trans_pvt, read_frame, 1);
439 } else {
440 final_frame = read_frame;
441 }
442
443 return final_frame;
444}
static struct ast_frame * audiohook_read_frame_single(struct ast_audiohook *audiohook, size_t samples, enum ast_audiohook_direction direction)
Definition: audiohook.c:223
static struct ast_frame * audiohook_read_frame_both(struct ast_audiohook *audiohook, size_t samples, struct ast_frame **read_reference, struct ast_frame **write_reference)
Definition: audiohook.c:257

References ao2_replace, AST_AUDIOHOOK_COMPATIBLE, AST_AUDIOHOOK_DIRECTION_BOTH, ast_format_cache_get_slin_by_rate(), ast_format_cmp(), AST_FORMAT_CMP_EQUAL, AST_FORMAT_CMP_NOT_EQUAL, ast_format_get_sample_rate(), ast_frfree, ast_test_flag, ast_translate(), ast_translator_build_path(), ast_translator_free_path(), audiohook_read_frame_both(), audiohook_read_frame_single(), audiohook_set_internal_rate(), ast_audiohook::format, ast_audiohook::hook_internal_samp_rate, NULL, read_frame(), and ast_audiohook::trans_pvt.

Referenced by ast_audiohook_read_frame(), and ast_audiohook_read_frame_all().

◆ audiohook_read_frame_single()

static struct ast_frame * audiohook_read_frame_single ( struct ast_audiohook audiohook,
size_t  samples,
enum ast_audiohook_direction  direction 
)
static

Definition at line 223 of file audiohook.c.

224{
225 struct ast_slinfactory *factory = (direction == AST_AUDIOHOOK_DIRECTION_READ ? &audiohook->read_factory : &audiohook->write_factory);
226 int vol = (direction == AST_AUDIOHOOK_DIRECTION_READ ? audiohook->options.read_volume : audiohook->options.write_volume);
227 short buf[samples];
228 struct ast_frame frame = {
230 .subclass.format = ast_format_cache_get_slin_by_rate(audiohook->hook_internal_samp_rate),
231 .data.ptr = buf,
232 .datalen = sizeof(buf),
233 .samples = samples,
234 };
235
236 /* Ensure the factory is able to give us the samples we want */
237 if (samples > ast_slinfactory_available(factory)) {
238 return NULL;
239 }
240
241 /* Read data in from factory */
242 if (!ast_slinfactory_read(factory, buf, samples)) {
243 return NULL;
244 }
245
246 if (SHOULD_MUTE(audiohook, direction)) {
247 /* Swap frame data for zeros if mute is required */
248 ast_frame_clear(&frame);
249 } else if (vol) {
250 /* If a volume adjustment needs to be applied apply it */
251 ast_frame_adjust_volume(&frame, vol);
252 }
253
254 return ast_frdup(&frame);
255}
#define SHOULD_MUTE(hook, dir)
Definition: audiohook.c:162
char buf[BUFSIZE]
Definition: eagi_proxy.c:66
int ast_frame_clear(struct ast_frame *frame)
Clear all audio samples from an ast_frame. The frame must be AST_FRAME_VOICE and AST_FORMAT_SLINEAR.
Definition: main/frame.c:859
int ast_frame_adjust_volume(struct ast_frame *f, int adjustment)
Adjusts the volume of the audio samples contained in a frame.
Definition: main/frame.c:787

References AST_AUDIOHOOK_DIRECTION_READ, ast_format_cache_get_slin_by_rate(), ast_frame_adjust_volume(), ast_frame_clear(), AST_FRAME_VOICE, ast_frdup, ast_slinfactory_available(), ast_slinfactory_read(), buf, ast_frame::frametype, ast_audiohook::hook_internal_samp_rate, NULL, ast_audiohook::options, ast_audiohook::read_factory, ast_audiohook_options::read_volume, ast_frame::samples, SHOULD_MUTE, ast_audiohook::write_factory, and ast_audiohook_options::write_volume.

Referenced by audiohook_read_frame_helper().

◆ audiohook_set_internal_rate()

static int audiohook_set_internal_rate ( struct ast_audiohook audiohook,
int  rate,
int  reset 
)
static

Definition at line 70 of file audiohook.c.

71{
72 struct ast_format *slin;
73
74 if (audiohook->hook_internal_samp_rate == rate) {
75 return 0;
76 }
77
78 audiohook->hook_internal_samp_rate = rate;
79
81
82 /* Setup the factories that are needed for this audiohook type */
83 switch (audiohook->type) {
86 if (reset) {
89 }
92 break;
93 default:
94 break;
95 }
96
97 return 0;
98}
int ast_slinfactory_init_with_format(struct ast_slinfactory *sf, struct ast_format *slin_out)
Initialize a slinfactory.
Definition: slinfactory.c:46

References AST_AUDIOHOOK_TYPE_SPY, AST_AUDIOHOOK_TYPE_WHISPER, ast_format_cache_get_slin_by_rate(), ast_slinfactory_destroy(), ast_slinfactory_init_with_format(), ast_audiohook::hook_internal_samp_rate, ast_audiohook::read_factory, ast_audiohook::type, and ast_audiohook::write_factory.

Referenced by ast_audiohook_attach(), ast_audiohook_init(), audiohook_list_set_hook_rate(), and audiohook_read_frame_helper().

◆ audiohook_volume_callback()

static int audiohook_volume_callback ( struct ast_audiohook audiohook,
struct ast_channel chan,
struct ast_frame frame,
enum ast_audiohook_direction  direction 
)
static

Helper function which actually gets called by audiohooks to perform the adjustment.

Parameters
audiohookAudiohook attached to the channel
chanChannel we are attached to
frameFrame of audio we want to manipulate
directionDirection the audio came in from
Return values
0on success
-1on failure

Definition at line 1219 of file audiohook.c.

1220{
1221 struct ast_datastore *datastore = NULL;
1223 int *gain = NULL;
1224
1225 /* If the audiohook is shutting down don't even bother */
1227 return 0;
1228 }
1229
1230 /* Try to find the datastore containg adjustment information, if we can't just bail out */
1231 if (!(datastore = ast_channel_datastore_find(chan, &audiohook_volume_datastore, NULL))) {
1232 return 0;
1233 }
1234
1235 audiohook_volume = datastore->data;
1236
1237 /* Based on direction grab the appropriate adjustment value */
1242 }
1243
1244 /* If an adjustment value is present modify the frame */
1245 if (gain && *gain) {
1246 ast_frame_adjust_volume(frame, *gain);
1247 }
1248
1249 return 0;
1250}
static const struct ast_datastore_info audiohook_volume_datastore
Datastore used to store audiohook volume information.
Definition: audiohook.c:1206
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:2399
Structure for a data store object.
Definition: datastore.h:64
void * data
Definition: datastore.h:66
struct ast_audiohook audiohook
Definition: audiohook.c:1184

References AST_AUDIOHOOK_DIRECTION_READ, AST_AUDIOHOOK_DIRECTION_WRITE, AST_AUDIOHOOK_STATUS_DONE, ast_channel_datastore_find(), ast_frame_adjust_volume(), audiohook_volume::audiohook, audiohook_volume_datastore, ast_datastore::data, NULL, audiohook_volume::read_adjustment, ast_audiohook::status, and audiohook_volume::write_adjustment.

Referenced by audiohook_volume_get().

◆ audiohook_volume_destroy()

static void audiohook_volume_destroy ( void *  data)
static

Callback used to destroy the audiohook volume datastore.

Parameters
dataVolume information structure

Definition at line 1192 of file audiohook.c.

1193{
1194 struct audiohook_volume *audiohook_volume = data;
1195
1196 /* Destroy the audiohook as it is no longer in use */
1198
1199 /* Finally free ourselves, we are of no more use */
1201
1202 return;
1203}
int ast_audiohook_destroy(struct ast_audiohook *audiohook)
Destroys an audiohook structure.
Definition: audiohook.c:124

References ast_audiohook_destroy(), ast_free, and audiohook_volume::audiohook.

◆ audiohook_volume_get()

static struct audiohook_volume * audiohook_volume_get ( struct ast_channel chan,
int  create 
)
static

Helper function which finds and optionally creates an audiohook_volume_datastore datastore on a channel.

Parameters
chanChannel to look on
createWhether to create the datastore if not found
Returns
audiohook_volume structure on success
Return values
NULLon failure

Definition at line 1258 of file audiohook.c.

1259{
1260 struct ast_datastore *datastore = NULL;
1262
1263 /* If we are able to find the datastore return the contents (which is actually an audiohook_volume structure) */
1264 if ((datastore = ast_channel_datastore_find(chan, &audiohook_volume_datastore, NULL))) {
1265 return datastore->data;
1266 }
1267
1268 /* If we are not allowed to create a datastore or if we fail to create a datastore, bail out now as we have nothing for them */
1269 if (!create || !(datastore = ast_datastore_alloc(&audiohook_volume_datastore, NULL))) {
1270 return NULL;
1271 }
1272
1273 /* Create a new audiohook_volume structure to contain our adjustments and audiohook */
1274 if (!(audiohook_volume = ast_calloc(1, sizeof(*audiohook_volume)))) {
1275 ast_datastore_free(datastore);
1276 return NULL;
1277 }
1278
1279 /* Setup our audiohook structure so we can manipulate the audio */
1282
1283 /* Attach the audiohook_volume blob to the datastore and attach to the channel */
1284 datastore->data = audiohook_volume;
1285 ast_channel_datastore_add(chan, datastore);
1286
1287 /* All is well... put the audiohook into motion */
1289
1290 return audiohook_volume;
1291}
static int audiohook_volume_callback(struct ast_audiohook *audiohook, struct ast_channel *chan, struct ast_frame *frame, enum ast_audiohook_direction direction)
Helper function which actually gets called by audiohooks to perform the adjustment.
Definition: audiohook.c:1219
int ast_audiohook_init(struct ast_audiohook *audiohook, enum ast_audiohook_type type, const char *source, enum ast_audiohook_init_flags init_flags)
Initialize an audiohook structure.
Definition: audiohook.c:100
int ast_channel_datastore_add(struct ast_channel *chan, struct ast_datastore *datastore)
Add a datastore to a channel.
Definition: channel.c:2385
#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

References ast_audiohook_attach(), ast_audiohook_init(), AST_AUDIOHOOK_MANIPULATE_ALL_RATES, AST_AUDIOHOOK_TYPE_MANIPULATE, ast_calloc, ast_channel_datastore_add(), ast_channel_datastore_find(), ast_datastore_alloc, ast_datastore_free(), audiohook_volume::audiohook, audiohook_volume_callback(), audiohook_volume_datastore, ast_datastore::data, ast_audiohook::manipulate_callback, and NULL.

Referenced by ast_audiohook_volume_adjust(), ast_audiohook_volume_get(), and ast_audiohook_volume_set().

◆ dtmf_audiohook_write_list()

static struct ast_frame * dtmf_audiohook_write_list ( struct ast_channel chan,
struct ast_audiohook_list audiohook_list,
enum ast_audiohook_direction  direction,
struct ast_frame frame 
)
static

Pass a DTMF frame off to be handled by the audiohook core.

Parameters
chanChannel that the list is coming off of
audiohook_listList of audiohooks
directionDirection frame is coming in from
frameThe frame itself
Returns
frame on success
Return values
NULLon failure

Definition at line 758 of file audiohook.c.

759{
760 struct ast_audiohook *audiohook = NULL;
761 int removed = 0;
762
763 AST_LIST_TRAVERSE_SAFE_BEGIN(&audiohook_list->manipulate_list, audiohook, list) {
764 ast_audiohook_lock(audiohook);
765 if (audiohook->status != AST_AUDIOHOOK_STATUS_RUNNING) {
767 removed = 1;
769 ast_audiohook_unlock(audiohook);
770 audiohook->manipulate_callback(audiohook, NULL, NULL, 0);
771 if (ast_channel_is_bridged(chan)) {
773 }
774 continue;
775 }
776 if (ast_test_flag(audiohook, AST_AUDIOHOOK_WANTS_DTMF)) {
777 audiohook->manipulate_callback(audiohook, chan, frame, direction);
778 }
779 ast_audiohook_unlock(audiohook);
780 }
782
783 /* if an audiohook got removed, reset samplerate compatibility */
784 if (removed) {
786 }
787 return frame;
788}
@ AST_AUDIOHOOK_WANTS_DTMF
Definition: audiohook.h:58

References ast_audiohook_lock, AST_AUDIOHOOK_STATUS_DONE, AST_AUDIOHOOK_STATUS_RUNNING, ast_audiohook_unlock, ast_audiohook_update_status(), AST_AUDIOHOOK_WANTS_DTMF, ast_channel_is_bridged(), ast_channel_set_unbridged_nolock(), AST_LIST_REMOVE_CURRENT, AST_LIST_TRAVERSE_SAFE_BEGIN, AST_LIST_TRAVERSE_SAFE_END, ast_test_flag, audiohook_list_set_samplerate_compatibility(), ast_audiohook::list, ast_audiohook::manipulate_callback, ast_audiohook_list::manipulate_list, NULL, and ast_audiohook::status.

Referenced by ast_audiohook_write_list().

◆ find_audiohook_by_source()

static struct ast_audiohook * find_audiohook_by_source ( struct ast_audiohook_list audiohook_list,
const char *  source 
)
static

find an audiohook based on its source

Parameters
audiohook_listThe list of audiohooks to search in
sourceThe source of the audiohook we wish to find
Returns
corresponding audiohook
Return values
NULLif it cannot be found

Definition at line 612 of file audiohook.c.

613{
614 struct ast_audiohook *audiohook = NULL;
615
616 AST_LIST_TRAVERSE(&audiohook_list->spy_list, audiohook, list) {
617 if (!strcasecmp(audiohook->source, source)) {
618 return audiohook;
619 }
620 }
621
622 AST_LIST_TRAVERSE(&audiohook_list->whisper_list, audiohook, list) {
623 if (!strcasecmp(audiohook->source, source)) {
624 return audiohook;
625 }
626 }
627
628 AST_LIST_TRAVERSE(&audiohook_list->manipulate_list, audiohook, list) {
629 if (!strcasecmp(audiohook->source, source)) {
630 return audiohook;
631 }
632 }
633
634 return NULL;
635}

References AST_LIST_TRAVERSE, ast_audiohook::list, ast_audiohook_list::manipulate_list, NULL, ast_audiohook::source, ast_audiohook_list::spy_list, and ast_audiohook_list::whisper_list.

Referenced by ast_audiohook_detach_source(), ast_audiohook_move_by_source(), and ast_audiohook_set_mute().

Variable Documentation

◆ audiohook_volume_datastore

const struct ast_datastore_info audiohook_volume_datastore
static
Initial value:
= {
.type = "Volume",
}
static void audiohook_volume_destroy(void *data)
Callback used to destroy the audiohook volume datastore.
Definition: audiohook.c:1192

Datastore used to store audiohook volume information.

Definition at line 1206 of file audiohook.c.

Referenced by audiohook_volume_callback(), and audiohook_volume_get().