summaryrefslogtreecommitdiff
path: root/main/framehook.c
diff options
context:
space:
mode:
authorJoshua Colp <jcolp@digium.com>2013-12-17 18:26:09 +0000
committerJoshua Colp <jcolp@digium.com>2013-12-17 18:26:09 +0000
commit9fc2cc178a8685dcb3afb9ed9d820f17d39dc4cd (patch)
tree26a0e93982a39c206dd4db4f8f2cd82a684b3e48 /main/framehook.c
parent27f37f6e3df6de123a8af8096303d71bebcb180d (diff)
framehooks: Re-iterate if framehook provides different frame.
Framehooks can be used in a reactive manner to execute specific logic when a frame is received with a certain type and payload. Since it is possible for framehooks to provide frames it was possible for this reactive framehook to be unaware of frames it is looking for. This change makes it so that when framehooks return a modified frame the code will now re-iterate (from the beginning) and call any previous framehooks that have not provided a modified frame themselves. Review: https://reviewboard.asterisk.org/r/3046/ ........ Merged revisions 404027 from http://svn.asterisk.org/svn/asterisk/branches/12 git-svn-id: https://origsvn.digium.com/svn/asterisk/trunk@404028 65c4cc65-6c06-0410-ace0-fbb531ad65f3
Diffstat (limited to 'main/framehook.c')
-rw-r--r--main/framehook.c48
1 files changed, 40 insertions, 8 deletions
diff --git a/main/framehook.c b/main/framehook.c
index f85a4c298..0d353cf36 100644
--- a/main/framehook.c
+++ b/main/framehook.c
@@ -49,6 +49,9 @@ struct ast_framehook {
};
struct ast_framehook_list {
+ /*! the number of hooks currently present */
+ unsigned int count;
+ /*! id for next framehook added */
unsigned int id_count;
AST_LIST_HEAD_NOLOCK(, ast_framehook) list;
};
@@ -73,21 +76,49 @@ static void framehook_detach_and_destroy(struct ast_framehook *framehook)
static struct ast_frame *framehook_list_push_event(struct ast_framehook_list *framehooks, struct ast_frame *frame, enum ast_framehook_event event)
{
struct ast_framehook *framehook;
+ struct ast_frame *original_frame;
+ int *skip;
+ size_t skip_size;
if (!framehooks) {
return frame;
}
- AST_LIST_TRAVERSE_SAFE_BEGIN(&framehooks->list, framehook, list) {
- if (framehook->detach_and_destroy_me) {
- /* this guy is signaled for destruction */
- AST_LIST_REMOVE_CURRENT(list);
- framehook_detach_and_destroy(framehook);
- } else {
+ skip_size = sizeof(int) * framehooks->count;
+ skip = alloca(skip_size);
+ memset(skip, 0, skip_size);
+
+ do {
+ unsigned int num = 0;
+ original_frame = frame;
+
+ AST_LIST_TRAVERSE_SAFE_BEGIN(&framehooks->list, framehook, list) {
+ if (framehook->detach_and_destroy_me) {
+ /* this guy is signaled for destruction */
+ AST_LIST_REMOVE_CURRENT(list);
+ framehook_detach_and_destroy(framehook);
+ continue;
+ }
+
+ /* If this framehook has been marked as needing to be skipped, do so */
+ if (skip[num]) {
+ num++;
+ continue;
+ }
+
frame = framehook->i.event_cb(framehook->chan, frame, event, framehook->i.data);
+
+ if (frame != original_frame) {
+ /* To prevent looping we skip any framehooks that have already provided a modified frame */
+ skip[num] = 1;
+ break;
+ }
+
+ num++;
}
- }
- AST_LIST_TRAVERSE_SAFE_END;
+ AST_LIST_TRAVERSE_SAFE_END;
+ } while (frame != original_frame);
+
return frame;
}
@@ -116,6 +147,7 @@ int ast_framehook_attach(struct ast_channel *chan, struct ast_framehook_interfac
ast_channel_framehooks_set(chan, fh_list);
}
+ ast_channel_framehooks(chan)->count++;
framehook->id = ++ast_channel_framehooks(chan)->id_count;
AST_LIST_INSERT_TAIL(&ast_channel_framehooks(chan)->list, framehook, list);