[Freeswitch-trunk] [commit] r13851 - freeswitch/trunk/src/mod/formats/mod_shell_stream

FreeSWITCH SVN anthm at freeswitch.org
Thu Jun 18 15:36:51 PDT 2009


Author: anthm
Date: Thu Jun 18 17:36:51 2009
New Revision: 13851

Log:
add mod_shell_stream

Added:
   freeswitch/trunk/src/mod/formats/mod_shell_stream/
   freeswitch/trunk/src/mod/formats/mod_shell_stream/mod_shell_stream.c

Added: freeswitch/trunk/src/mod/formats/mod_shell_stream/mod_shell_stream.c
==============================================================================
--- (empty file)
+++ freeswitch/trunk/src/mod/formats/mod_shell_stream/mod_shell_stream.c	Thu Jun 18 17:36:51 2009
@@ -0,0 +1,224 @@
+/* 
+ * FreeSWITCH Modular Media Switching Software Library / Soft-Switch Application
+ * Copyright (C) 2005-2009, Anthony Minessale II <anthm at freeswitch.org>
+ *
+ * Version: MPL 1.1
+ *
+ * The contents of this file are subject to the Mozilla Public License Version
+ * 1.1 (the "License"); you may not use this file except in compliance with
+ * the License. You may obtain a copy of the License at
+ * http://www.mozilla.org/MPL/
+ *
+ * Software distributed under the License is distributed on an "AS IS" basis,
+ * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
+ * for the specific language governing rights and limitations under the
+ * License.
+ *
+ * The Original Code is FreeSWITCH Modular Media Switching Software Library / Soft-Switch Application
+ *
+ * The Initial Developer of the Original Code is
+ * Anthony Minessale II <anthm at freeswitch.org>
+ * Portions created by the Initial Developer are Copyright (C)
+ * the Initial Developer. All Rights Reserved.
+ *
+ * Contributor(s):
+ * 
+ * Anthony Minessale II <anthm at freeswitch.org>
+ *
+ * mod_shell_stream.c -- Local Streaming Audio
+ *
+ */
+#include <switch.h>
+
+#define MY_BUF_LEN 1024 * 32
+#define MY_BLOCK_SIZE MY_BUF_LEN
+
+
+SWITCH_MODULE_LOAD_FUNCTION(mod_shell_stream_load);
+
+SWITCH_MODULE_DEFINITION(mod_shell_stream, mod_shell_stream_load, NULL, NULL);
+
+struct shell_stream_context {
+	int fds[2];
+	int pid;
+	char *command;
+	switch_buffer_t *audio_buffer;
+	switch_mutex_t *mutex;	
+	switch_thread_rwlock_t *rwlock;
+	int running;
+};
+
+typedef struct shell_stream_context shell_stream_context_t;
+
+
+static void *SWITCH_THREAD_FUNC buffer_thread_run(switch_thread_t *thread, void *obj)
+{
+	shell_stream_context_t *context = (shell_stream_context_t *) obj;
+	switch_byte_t data[MY_BUF_LEN];
+	ssize_t rlen;
+
+	context->running = 1;
+
+	if (switch_thread_rwlock_tryrdlock(context->rwlock) != SWITCH_STATUS_SUCCESS) {
+		switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CRIT, "Read Lock Fail\n");
+		goto end;
+	}
+	
+	while(context->running) {
+		
+		rlen = read(context->fds[0], data, MY_BUF_LEN);
+
+		if (rlen <= 3) {
+			break;
+		}
+		
+		switch_mutex_lock(context->mutex);
+		switch_buffer_write(context->audio_buffer, data, rlen);
+		switch_mutex_unlock(context->mutex);
+	}
+
+	switch_thread_rwlock_unlock(context->rwlock);
+
+ end:
+
+	context->running = 0;
+
+	return NULL;
+}
+
+static switch_status_t shell_stream_file_open(switch_file_handle_t *handle, const char *path)
+{
+	shell_stream_context_t *context;
+	switch_status_t status = SWITCH_STATUS_SUCCESS;
+	switch_thread_t *thread;
+	switch_threadattr_t *thd_attr = NULL;
+	
+	if (switch_test_flag(handle, SWITCH_FILE_FLAG_WRITE)) {
+		switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "This format does not support writing!\n");
+		return SWITCH_STATUS_FALSE;
+	}
+	
+	context = switch_core_alloc(handle->memory_pool, sizeof(*context));
+	
+	context->fds[0] = -1;
+	context->fds[1] = -1;
+	context->command = switch_core_strdup(handle->memory_pool, path);
+
+	if (pipe(context->fds)) {
+		goto error;
+	} else { /* good to go*/
+		context->pid = fork();
+		
+		if (context->pid < 0) { /* ok maybe not */
+			goto error;
+		} else if (context->pid) { /* parent */
+			handle->private_info = context;
+			status = SWITCH_STATUS_SUCCESS;
+			close(context->fds[1]);
+			context->fds[1] = -1;
+
+			if (switch_buffer_create_dynamic(&context->audio_buffer, MY_BLOCK_SIZE, MY_BUF_LEN, 0) != SWITCH_STATUS_SUCCESS) {
+				switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CRIT, "Write Buffer Failed!\n");
+				goto error;
+			}
+
+			switch_thread_rwlock_create(&context->rwlock, handle->memory_pool);
+			switch_mutex_init(&context->mutex, SWITCH_MUTEX_NESTED, handle->memory_pool);
+			
+			switch_threadattr_create(&thd_attr, handle->memory_pool);
+			switch_threadattr_detach_set(thd_attr, 1);
+			switch_threadattr_stacksize_set(thd_attr, SWITCH_THREAD_STACKSIZE);
+			switch_thread_create(&thread, thd_attr, buffer_thread_run, context, handle->memory_pool);
+			context->running = 2;
+
+			while(context->running == 2) {
+				switch_cond_next();
+			}
+
+			goto end;
+		} else { /*  child */
+			close(context->fds[0]);
+			dup2(context->fds[1], STDOUT_FILENO);
+			switch_system(context->command, SWITCH_TRUE);
+			printf("EOF");
+			close(context->fds[1]);
+			exit(0);
+		}
+	}
+
+ error:
+
+	close(context->fds[0]);
+	close(context->fds[1]);
+	status = SWITCH_STATUS_FALSE;
+		
+
+ end:
+		
+	return status;
+}
+
+static switch_status_t shell_stream_file_close(switch_file_handle_t *handle)
+{
+	shell_stream_context_t *context = handle->private_info;
+	
+	context->running = 0;
+
+	if (context->fds[0] > -1) {
+		close(context->fds[0]);
+	}
+
+	switch_thread_rwlock_wrlock(context->rwlock);
+	switch_thread_rwlock_unlock(context->rwlock);
+
+	return SWITCH_STATUS_SUCCESS;
+}
+
+static switch_status_t shell_stream_file_read(switch_file_handle_t *handle, void *data, size_t *len)
+{
+	shell_stream_context_t *context = handle->private_info;
+	switch_status_t status = SWITCH_STATUS_SUCCESS;
+	switch_size_t rlen = *len * 2;
+
+	while(context->running && switch_buffer_inuse(context->audio_buffer) < rlen) {
+		switch_cond_next();
+	}
+	
+	switch_mutex_lock(context->mutex);
+	*len = switch_buffer_read(context->audio_buffer, data, rlen) / 2;
+	switch_mutex_unlock(context->mutex);
+
+	return status;
+}
+
+/* Registration */
+
+static char *supported_formats[SWITCH_MAX_CODECS] = { 0 };
+
+SWITCH_MODULE_LOAD_FUNCTION(mod_shell_stream_load)
+{
+	switch_file_interface_t *file_interface;
+	supported_formats[0] = "shell_stream";
+
+	*module_interface = switch_loadable_module_create_module_interface(pool, modname);
+	file_interface = switch_loadable_module_create_interface(*module_interface, SWITCH_FILE_INTERFACE);
+	file_interface->interface_name = modname;
+	file_interface->extens = supported_formats;
+	file_interface->file_open = shell_stream_file_open;
+	file_interface->file_close = shell_stream_file_close;
+	file_interface->file_read = shell_stream_file_read;
+
+	/* indicate that the module should continue to be loaded */
+	return SWITCH_STATUS_SUCCESS;
+}
+
+/* For Emacs:
+ * Local Variables:
+ * mode:c
+ * indent-tabs-mode:t
+ * tab-width:4
+ * c-basic-offset:4
+ * End:
+ * For VIM:
+ * vim:set softtabstop=4 shiftwidth=4 tabstop=4:
+ */



More information about the Freeswitch-trunk mailing list