mirror of
				https://github.com/asterisk/asterisk.git
				synced 2025-11-04 05:15:22 +00:00 
			
		
		
		
	
		
			
	
	
		
			185 lines
		
	
	
		
			5.1 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
		
		
			
		
	
	
			185 lines
		
	
	
		
			5.1 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| 
								 | 
							
								/*
							 | 
						||
| 
								 | 
							
								 * Asterisk -- An open source telephony toolkit.
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Copyright (C) 2010, Digium, Inc.
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * David Vossel <dvossel@digium.com>
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * See http://www.asterisk.org for more information about
							 | 
						||
| 
								 | 
							
								 * the Asterisk project. Please do not directly contact
							 | 
						||
| 
								 | 
							
								 * any of the maintainers of this project for assistance;
							 | 
						||
| 
								 | 
							
								 * the project provides a web site, mailing lists and IRC
							 | 
						||
| 
								 | 
							
								 * channels for your use.
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * This program is free software, distributed under the terms of
							 | 
						||
| 
								 | 
							
								 * the GNU General Public License Version 2. See the LICENSE file
							 | 
						||
| 
								 | 
							
								 * at the top of the source tree.
							 | 
						||
| 
								 | 
							
								 */
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/*! \file
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * \brief FrameHooks Architecture
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * \author David Vossel <dvossel@digium.com>
							 | 
						||
| 
								 | 
							
								 */
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#include "asterisk.h"
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								ASTERISK_FILE_VERSION(__FILE__, "$Revision$")
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#include "asterisk/channel.h"
							 | 
						||
| 
								 | 
							
								#include "asterisk/linkedlists.h"
							 | 
						||
| 
								 | 
							
								#include "asterisk/framehook.h"
							 | 
						||
| 
								 | 
							
								#include "asterisk/frame.h"
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								struct ast_framehook {
							 | 
						||
| 
								 | 
							
									struct ast_framehook_interface i;
							 | 
						||
| 
								 | 
							
									/*! This pointer to ast_channel the framehook is attached to. */
							 | 
						||
| 
								 | 
							
									struct ast_channel *chan;
							 | 
						||
| 
								 | 
							
									/*! the id representing this framehook on a channel */
							 | 
						||
| 
								 | 
							
									unsigned int id;
							 | 
						||
| 
								 | 
							
									/*! when set, this signals the read and write function to detach the hook */
							 | 
						||
| 
								 | 
							
									int detach_and_destroy_me;
							 | 
						||
| 
								 | 
							
									/*! list entry for ast_framehook_list object */
							 | 
						||
| 
								 | 
							
									AST_LIST_ENTRY(ast_framehook) list;
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								struct ast_framehook_list {
							 | 
						||
| 
								 | 
							
									unsigned int id_count;
							 | 
						||
| 
								 | 
							
									AST_LIST_HEAD_NOLOCK(, ast_framehook) list;
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								static void framehook_detach_and_destroy(struct ast_framehook *framehook)
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
									struct ast_frame *frame;
							 | 
						||
| 
								 | 
							
									frame = framehook->i.event_cb(framehook->chan, NULL, AST_FRAMEHOOK_EVENT_DETACHED, framehook->i.data);
							 | 
						||
| 
								 | 
							
									/* never assume anything about this function. If you can return a frame during
							 | 
						||
| 
								 | 
							
									 * the detached event, then assume someone will. */
							 | 
						||
| 
								 | 
							
									if (frame) {
							 | 
						||
| 
								 | 
							
										ast_frfree(frame);
							 | 
						||
| 
								 | 
							
									}
							 | 
						||
| 
								 | 
							
									framehook->chan = NULL;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									if (framehook->i.destroy_cb) {
							 | 
						||
| 
								 | 
							
										framehook->i.destroy_cb(framehook->i.data);
							 | 
						||
| 
								 | 
							
									}
							 | 
						||
| 
								 | 
							
									ast_free(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;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									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 {
							 | 
						||
| 
								 | 
							
											frame = framehook->i.event_cb(framehook->chan, frame, event, framehook->i.data);
							 | 
						||
| 
								 | 
							
										}
							 | 
						||
| 
								 | 
							
									}
							 | 
						||
| 
								 | 
							
									AST_LIST_TRAVERSE_SAFE_END;
							 | 
						||
| 
								 | 
							
									return frame;
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								int ast_framehook_attach(struct ast_channel *chan, struct ast_framehook_interface *i)
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
									struct ast_framehook *framehook;
							 | 
						||
| 
								 | 
							
									struct ast_frame *frame;
							 | 
						||
| 
								 | 
							
									if (i->version != AST_FRAMEHOOK_INTERFACE_VERSION) {
							 | 
						||
| 
								 | 
							
										ast_log(LOG_ERROR, "Version '%hu' of framehook interface not what we compiled against (%hu)\n",
							 | 
						||
| 
								 | 
							
											i->version, AST_FRAMEHOOK_INTERFACE_VERSION);
							 | 
						||
| 
								 | 
							
										return -1;
							 | 
						||
| 
								 | 
							
									}
							 | 
						||
| 
								 | 
							
									if (!i->event_cb || !(framehook = ast_calloc(1, sizeof(*framehook)))) {
							 | 
						||
| 
								 | 
							
										return -1;
							 | 
						||
| 
								 | 
							
									}
							 | 
						||
| 
								 | 
							
									framehook->i = *i;
							 | 
						||
| 
								 | 
							
									framehook->chan = chan;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									/* create the framehook list if it didn't already exist */
							 | 
						||
| 
								 | 
							
									if (!chan->framehooks && !(chan->framehooks = ast_calloc(1, sizeof(*chan->framehooks)))) {
							 | 
						||
| 
								 | 
							
										ast_free(framehook);
							 | 
						||
| 
								 | 
							
										return -1;
							 | 
						||
| 
								 | 
							
									}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									framehook->id = ++chan->framehooks->id_count;
							 | 
						||
| 
								 | 
							
									AST_LIST_INSERT_TAIL(&chan->framehooks->list, framehook, list);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									/* Tell the event callback we're live and rocking */
							 | 
						||
| 
								 | 
							
									frame = framehook->i.event_cb(framehook->chan, NULL, AST_FRAMEHOOK_EVENT_ATTACHED, framehook->i.data);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									/* Never assume anything about this function. If you can return a frame during
							 | 
						||
| 
								 | 
							
									 * the attached event, then assume someone will. */
							 | 
						||
| 
								 | 
							
									if (frame) {
							 | 
						||
| 
								 | 
							
										ast_frfree(frame);
							 | 
						||
| 
								 | 
							
									}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									return framehook->id;
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								int ast_framehook_detach(struct ast_channel *chan, int id)
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
									struct ast_framehook *framehook;
							 | 
						||
| 
								 | 
							
									int res = -1;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									if (!chan->framehooks) {
							 | 
						||
| 
								 | 
							
										return res;
							 | 
						||
| 
								 | 
							
									}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									AST_LIST_TRAVERSE_SAFE_BEGIN(&chan->framehooks->list, framehook, list) {
							 | 
						||
| 
								 | 
							
										if (framehook->id == id) {
							 | 
						||
| 
								 | 
							
											/* we mark for detachment rather than doing explicitly here because
							 | 
						||
| 
								 | 
							
											 * it needs to be safe for this function to be called within the
							 | 
						||
| 
								 | 
							
											 * event callback.  If we allowed the hook to actually be destroyed
							 | 
						||
| 
								 | 
							
											 * immediately here, the event callback would crash on exit. */
							 | 
						||
| 
								 | 
							
											framehook->detach_and_destroy_me = 1;
							 | 
						||
| 
								 | 
							
											res = 0;
							 | 
						||
| 
								 | 
							
											break;
							 | 
						||
| 
								 | 
							
										}
							 | 
						||
| 
								 | 
							
									}
							 | 
						||
| 
								 | 
							
									AST_LIST_TRAVERSE_SAFE_END;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									return res;
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								int ast_framehook_list_destroy(struct ast_channel *chan)
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
									struct ast_framehook *framehook;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									if (!chan->framehooks) {
							 | 
						||
| 
								 | 
							
										return 0;
							 | 
						||
| 
								 | 
							
									}
							 | 
						||
| 
								 | 
							
									AST_LIST_TRAVERSE_SAFE_BEGIN(&chan->framehooks->list, framehook, list) {
							 | 
						||
| 
								 | 
							
										AST_LIST_REMOVE_CURRENT(list);
							 | 
						||
| 
								 | 
							
										framehook_detach_and_destroy(framehook);
							 | 
						||
| 
								 | 
							
									}
							 | 
						||
| 
								 | 
							
									AST_LIST_TRAVERSE_SAFE_END;
							 | 
						||
| 
								 | 
							
									ast_free(chan->framehooks);
							 | 
						||
| 
								 | 
							
									chan->framehooks = NULL;
							 | 
						||
| 
								 | 
							
									return 0;
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								int ast_framehook_list_is_empty(struct ast_framehook_list *framehooks)
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
									if (!framehooks) {
							 | 
						||
| 
								 | 
							
										return 1;
							 | 
						||
| 
								 | 
							
									}
							 | 
						||
| 
								 | 
							
									return AST_LIST_EMPTY(&framehooks->list) ? 1 : 0;
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								struct ast_frame *ast_framehook_list_write_event(struct ast_framehook_list *framehooks, struct ast_frame *frame)
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
									return framehook_list_push_event(framehooks, frame, AST_FRAMEHOOK_EVENT_WRITE);
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								struct ast_frame *ast_framehook_list_read_event(struct ast_framehook_list *framehooks, struct ast_frame *frame)
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
									return framehook_list_push_event(framehooks, frame, AST_FRAMEHOOK_EVENT_READ);
							 | 
						||
| 
								 | 
							
								}
							 |