doubango/tinyMEDIA/src/tmedia_producer.c
c732d49e
 #if HAVE_CRT
 #define _CRTDBG_MAP_ALLOC 
 #include <stdlib.h> 
 #include <crtdbg.h>
 #endif //HAVE_CRT
 /*
 * Copyright (C) 2010-2014 Mamadou DIOP
 *	
 * This file is part of Open Source Doubango Framework.
 *
 * DOUBANGO is free software: you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation, either version 3 of the License, or
 * (at your option) any later version.
 *	
 * DOUBANGO is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *	
 * You should have received a copy of the GNU General Public License
 * along with DOUBANGO.
 *
 */
 
 /**@file tmedia_producer.c
  * @brief Base producer object.
  */
 #include "tinymedia/tmedia_producer.h"
 #include "tinymedia/tmedia_defaults.h"
 
 #include "tsk_debug.h"
 
 /**@defgroup tmedia_producer_group Producers
 */
 
 /* pointer to all registered producers */
 const tmedia_producer_plugin_def_t* __tmedia_producer_plugins[TMED_PRODUCER_MAX_PLUGINS] = {0};
 
 /**@ingroup tmedia_producer_group
 * Initialize the producer.
 * @param self The producer to initialize
 * @retval Zero if succeed and non-zero error code otherwise.
 *
 * @sa @ref tmedia_producer_deinit
 */
 int tmedia_producer_init(tmedia_producer_t* self)
 {
 	if(!self){
 		TSK_DEBUG_ERROR("Invalid parameter");
 		return -1;
 	}
 
 	self->video.chroma = TMEDIA_PRODUCER_CHROMA_DEFAULT;
 	self->audio.bits_per_sample = TMEDIA_PRODUCER_BITS_PER_SAMPLE_DEFAULT;
 	self->audio.channels = TMEDIA_PRODUCER_CHANNELS_DEFAULT;
 	self->audio.rate = TMEDIA_PRODUCER_RATE_DEFAULT;
 	self->audio.volume = tmedia_defaults_get_volume();
 
 	return 0;
 }
 
 /**@ingroup tmedia_producer_group
 * callback to encode and send() data
 * select callback for processing buffer data media
 */
 int tmedia_producer_set_enc_callback(tmedia_producer_t *self, tmedia_producer_enc_cb_f callback, const void* callback_data)
 {
 	if(!self){
 		TSK_DEBUG_ERROR("Invalid parameter");
 		return -1;
 	}
 
 	self->enc_cb.callback = callback;
 	self->enc_cb.callback_data = callback_data;
 
 	return 0;
 }
 
 /**@ingroup tmedia_producer_group
 * callback to send() data "as is"
 */
 int tmedia_producer_set_raw_callback(tmedia_producer_t *self, tmedia_producer_raw_cb_f callback, const void* callback_data)
 {
 	if(!self){
 		TSK_DEBUG_ERROR("Invalid parameter");
 		return -1;
 	}
 
 	self->raw_cb.callback = callback;
 	self->raw_cb.chunck_curr.usr_data = callback_data;
 
 	return 0;
 }
 
 /**@ingroup tmedia_producer_group
 * @retval Zero if succeed and non-zero error code otherwise
 */
 int tmedia_producer_set(tmedia_producer_t* self, const tmedia_param_t* param)
 {
 	if(!self || !self->plugin || !self->plugin->set || !param){
 		TSK_DEBUG_ERROR("Invalid parameter");
 		return -1;
 	}
 	return self->plugin->set(self, param);
 }
 
 /**@ingroup tmedia_producer_group
 * Alert the producer to be prepared to start.
 * @param self the producer to prepare
 * @param codec The codec to use to prepare the producer
 * @retval Zero if succeed and non-zero error code otherwise
 */
 int tmedia_producer_prepare(tmedia_producer_t *self, const tmedia_codec_t* codec)
 {
 	int ret;
 	if (!self || !self->plugin || !self->plugin->prepare || !codec) {
 		TSK_DEBUG_ERROR("Invalid parameter");
 		return -1;
 	}
 	if ((ret = self->plugin->prepare(self, codec)) == 0) {
 		self->is_prepared = tsk_true;
 	}
 	return ret;
 }
 
 /**@ingroup tmedia_producer_group
 * Starts the producer
 * @param self The producer to start
 * @retval Zero if succeed and non-zero error code otherwise
 */
 int tmedia_producer_start(tmedia_producer_t *self)
 {
 	int ret;
 	if (!self || !self->plugin || !self->plugin->start) {
 		TSK_DEBUG_ERROR("Invalid parameter");
 		return -1;
 	}
 	if ((ret = self->plugin->start(self)) == 0) {
 		self->is_started = tsk_true;
 	}
 	return ret;
 }
 
 
 /**@ingroup tmedia_producer_group
 * Pauses the producer
 * @param self The producer to pause
 * @retval Zero if succeed and non-zero error code otherwise
 */
 int tmedia_producer_pause(tmedia_producer_t *self)
 {
 	if (!self || !self->plugin || !self->plugin->pause) {
 		TSK_DEBUG_ERROR("Invalid parameter");
 		return -1;
 	}
 	return self->plugin->pause(self);
 }
 
 
 /**@ingroup tmedia_producer_group
 * Stops the producer
 * @param self The producer to stop
 * @retval Zero if succeed and non-zero error code otherwise
 */
 int tmedia_producer_stop(tmedia_producer_t *self)
 {
 	int ret;
 	if (!self || !self->plugin || !self->plugin->stop) {
 		TSK_DEBUG_ERROR("Invalid parameter");
 		return -1;
 	}
 	if ((ret = self->plugin->stop(self)) == 0) {
 		self->is_started = tsk_false;
 	}
 	return ret;
 }
 
 
 /**@ingroup tmedia_producer_group
 * DeInitialize the producer.
 * @param self The producer to deinitialize
 * @retval Zero if succeed and non-zero error code otherwise.
 *
 * @sa @ref tmedia_producer_deinit
 */
 int tmedia_producer_deinit(tmedia_producer_t* self)
 {
 	if(!self){
 		TSK_DEBUG_ERROR("Invalid parameter");
 		return -1;
 	}
 	return 0;
 }
 
 /**@ingroup tmedia_producer_group
 * Creates a new producer using an already registered plugin.
 * @param type The type of the producer to create
 * @param session_id
 * @sa @ref tmedia_producer_plugin_register()
 */
 tmedia_producer_t* tmedia_producer_create(tmedia_type_t type, uint64_t session_id)
 {
 	tmedia_producer_t* producer = tsk_null;
 	const tmedia_producer_plugin_def_t* plugin;
 	tsk_size_t i = 0;
 
 	while((i < TMED_PRODUCER_MAX_PLUGINS) && (plugin = __tmedia_producer_plugins[i++])){
 		if(plugin->objdef && plugin->type == type){
 			if((producer = tsk_object_new(plugin->objdef))){
 				/* initialize the newly created producer */
 				producer->plugin = plugin;
 				producer->session_id = session_id;
 				break;
 			}
 		}
 	}
 
 	return producer;
 }
 
 /**@ingroup tmedia_producer_group
 * Registers a producer plugin.
 * @param plugin the definition of the plugin.
 * @retval Zero if succeed and non-zero error code otherwise.
 * @sa @ref tmedia_producer_create()
 */
 int tmedia_producer_plugin_register(const tmedia_producer_plugin_def_t* plugin)
 {
 	tsk_size_t i;
 	if(!plugin){
 		TSK_DEBUG_ERROR("Invalid parameter");
 		return -1;
 	}
 
 	/* add or replace the plugin */
 	for(i = 0; i<TMED_PRODUCER_MAX_PLUGINS; i++){
 		if(!__tmedia_producer_plugins[i] || (__tmedia_producer_plugins[i] == plugin)){
 			__tmedia_producer_plugins[i] = plugin;
 			return 0;
 		}
 	}
 	
 	TSK_DEBUG_ERROR("There are already %d plugins.", TMED_PRODUCER_MAX_PLUGINS);
 	return -2;
 }
 
 /**@ingroup tmedia_producer_group
 * UnRegisters a producer plugin.
 * @param plugin the definition of the plugin.
 * @retval Zero if succeed and non-zero error code otherwise.
 */
 int tmedia_producer_plugin_unregister(const tmedia_producer_plugin_def_t* plugin)
 {
 	tsk_size_t i;
 	tsk_bool_t found = tsk_false;
 	if(!plugin){
 		TSK_DEBUG_ERROR("Invalid Parameter");
 		return -1;
 	}
 
 	/* find the plugin to unregister */
 	for(i = 0; i<TMED_PRODUCER_MAX_PLUGINS && __tmedia_producer_plugins[i]; i++){
 		if(__tmedia_producer_plugins[i] == plugin){
 			__tmedia_producer_plugins[i] = tsk_null;
 			found = tsk_true;
 			break;
 		}
 	}
 
 	/* compact */
 	if(found){
 		for(; i<(TMED_PRODUCER_MAX_PLUGINS - 1); i++){
 			if(__tmedia_producer_plugins[i+1]){
 				__tmedia_producer_plugins[i] = __tmedia_producer_plugins[i+1];
 			}
 			else{
 				break;
 			}
 		}
 		__tmedia_producer_plugins[i] = tsk_null;
 	}
 	return (found ? 0 : -2);
 }
 
 /**@ingroup tmedia_producer_group
 * UnRegisters all producers matching the given type.
 * @param type the type of the plugins to unregister.
 * @retval Zero if succeed and non-zero error code otherwise.
 */
 int tmedia_producer_plugin_unregister_by_type(tmedia_type_t type)
 {
 	int i, j;
 	
 	/* find the plugin to unregister */
 	for (i = 0; i < TMED_PRODUCER_MAX_PLUGINS && __tmedia_producer_plugins[i]; ) {
 		if ((__tmedia_producer_plugins[i]->type & type) == __tmedia_producer_plugins[i]->type) {
 			__tmedia_producer_plugins[i] = tsk_null;
 			/* compact */
 			for (j = i; j < (TMED_PRODUCER_MAX_PLUGINS - 1) && __tmedia_producer_plugins[j + 1]; ++j) {
 				__tmedia_producer_plugins[j] = __tmedia_producer_plugins[j + 1];
 			}
 			__tmedia_producer_plugins[j] = tsk_null;
 		}
 		else {
 			++i;
 		}
 	}
 	return 0;
 }