2012-02-23 16:16:55 -08:00
|
|
|
/*
|
|
|
|
* FreeSWITCH Modular Media Switching Software Library / Soft-Switch Application
|
2014-02-05 15:02:28 -06:00
|
|
|
* Copyright (C) 2005-2014, Anthony Minessale II <anthm@freeswitch.org>
|
2012-02-23 16:16:55 -08:00
|
|
|
*
|
|
|
|
* 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@freeswitch.org>
|
|
|
|
* Portions created by the Initial Developer are Copyright (C)
|
|
|
|
* the Initial Developer. All Rights Reserved.
|
|
|
|
*
|
|
|
|
* Contributor(s):
|
|
|
|
*
|
|
|
|
* William King william.king@quentustech.com
|
|
|
|
*
|
|
|
|
* mod_vlc.c -- VLC streaming module
|
|
|
|
*
|
|
|
|
* Examples:
|
|
|
|
*
|
2012-03-24 14:09:57 -07:00
|
|
|
* To playback from an audio source into a file:
|
2012-02-23 16:16:55 -08:00
|
|
|
* File: vlc:///path/to/file
|
|
|
|
* Stream: http://path.to.file.com:port/file.pls
|
|
|
|
* Stream: vlc://ftp://path.to.file.com:port/file.mp3
|
|
|
|
*
|
2012-03-24 14:09:57 -07:00
|
|
|
* To stream from a call(channel) out to a remote destination:
|
|
|
|
* vlc://#transcode{acodec=vorb,channels=1,samplerate=16000}:standard{access=http,mux=ogg,dst=:8080/thing.ogg}
|
|
|
|
*
|
2012-02-23 16:16:55 -08:00
|
|
|
* Notes:
|
|
|
|
*
|
|
|
|
* Requires at least libvlc version 1.2
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
#include <switch.h>
|
|
|
|
#include <vlc/vlc.h>
|
|
|
|
#include <vlc/libvlc_media_player.h>
|
2012-11-01 11:47:38 -07:00
|
|
|
#include <vlc/libvlc_events.h>
|
2012-02-23 16:16:55 -08:00
|
|
|
|
2012-03-24 14:09:57 -07:00
|
|
|
#define VLC_BUFFER_SIZE 65536
|
2012-02-23 16:16:55 -08:00
|
|
|
|
|
|
|
static char *vlc_file_supported_formats[SWITCH_MAX_CODECS] = { 0 };
|
|
|
|
|
2012-03-24 14:09:57 -07:00
|
|
|
typedef int (*imem_get_t)(void *data, const char *cookie,
|
|
|
|
int64_t *dts, int64_t *pts, unsigned *flags,
|
|
|
|
size_t *, void **);
|
|
|
|
typedef void (*imem_release_t)(void *data, const char *cookie, size_t, void *);
|
|
|
|
|
2012-02-24 15:41:12 -08:00
|
|
|
/* Change valud to -vvv for vlc related debug. Be careful since vlc is at least as verbose as FS about logging */
|
|
|
|
const char *vlc_args = "";
|
|
|
|
|
2012-03-24 14:09:24 -07:00
|
|
|
libvlc_instance_t *read_inst;
|
2012-02-23 16:16:55 -08:00
|
|
|
|
|
|
|
struct vlc_file_context {
|
|
|
|
libvlc_media_player_t *mp;
|
|
|
|
libvlc_media_t *m;
|
|
|
|
switch_file_handle_t fh;
|
|
|
|
switch_memory_pool_t *pool;
|
|
|
|
switch_buffer_t *audio_buffer;
|
|
|
|
switch_mutex_t *audio_mutex;
|
2012-03-24 14:09:24 -07:00
|
|
|
switch_thread_cond_t *started;
|
2012-02-23 16:16:55 -08:00
|
|
|
char *path;
|
|
|
|
int samples;
|
|
|
|
int playing;
|
2012-03-24 14:09:57 -07:00
|
|
|
int samplerate;
|
2014-06-12 22:06:33 +05:00
|
|
|
int channels;
|
2012-02-23 16:16:55 -08:00
|
|
|
int err;
|
2012-03-24 14:09:24 -07:00
|
|
|
int pts;
|
2012-03-24 14:09:57 -07:00
|
|
|
libvlc_instance_t *inst_out;
|
2012-02-23 16:16:55 -08:00
|
|
|
};
|
|
|
|
|
|
|
|
typedef struct vlc_file_context vlc_file_context_t;
|
|
|
|
|
|
|
|
SWITCH_MODULE_SHUTDOWN_FUNCTION(mod_vlc_shutdown);
|
|
|
|
SWITCH_MODULE_LOAD_FUNCTION(mod_vlc_load);
|
|
|
|
SWITCH_MODULE_DEFINITION(mod_vlc, mod_vlc_load, mod_vlc_shutdown, NULL);
|
|
|
|
|
2012-11-01 11:47:38 -07:00
|
|
|
static void vlc_mediaplayer_error_callback(const libvlc_event_t * event, void * data)
|
|
|
|
{
|
|
|
|
vlc_file_context_t *context = (vlc_file_context_t *) data;
|
|
|
|
int status = libvlc_media_get_state(context->m);
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_DEBUG, "Got a libvlc_MediaPlayerEncounteredError callback. mediaPlayer Status: %d\n", status);
|
|
|
|
if (status == libvlc_Error) {
|
|
|
|
context->err = 1;
|
|
|
|
switch_thread_cond_signal(context->started);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
static void vlc_media_state_callback(const libvlc_event_t * event, void * data)
|
|
|
|
{
|
|
|
|
vlc_file_context_t *context = (vlc_file_context_t *) data;
|
|
|
|
int new_state = event->u.media_state_changed.new_state;
|
|
|
|
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_DEBUG, "Got a libvlc_MediaStateChanged callback. New state: %d\n", new_state);
|
|
|
|
if (new_state == libvlc_Ended || new_state == libvlc_Error) {
|
|
|
|
switch_thread_cond_signal(context->started);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2012-02-23 16:16:55 -08:00
|
|
|
void vlc_auto_play_callback(void *data, const void *samples, unsigned count, int64_t pts) {
|
2012-03-24 14:09:57 -07:00
|
|
|
|
2012-02-23 16:16:55 -08:00
|
|
|
vlc_file_context_t *context = (vlc_file_context_t *) data;
|
|
|
|
|
|
|
|
switch_mutex_lock(context->audio_mutex);
|
|
|
|
if (context->audio_buffer) {
|
2014-06-12 22:06:33 +05:00
|
|
|
if (!switch_buffer_write(context->audio_buffer, samples, count * 2 * context->channels)) {
|
2012-02-23 16:16:55 -08:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Buffer error\n");
|
|
|
|
}
|
|
|
|
}
|
2012-03-24 14:09:24 -07:00
|
|
|
|
|
|
|
if(! context->playing ) {
|
|
|
|
context->playing = 1;
|
|
|
|
switch_thread_cond_signal(context->started);
|
|
|
|
}
|
2012-02-23 16:16:55 -08:00
|
|
|
switch_mutex_unlock(context->audio_mutex);
|
2012-03-24 14:09:57 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
int vlc_imem_get_callback(void *data, const char *cookie, int64_t *dts, int64_t *pts, unsigned *flags, size_t *size, void **output)
|
|
|
|
{
|
|
|
|
vlc_file_context_t *context = (vlc_file_context_t *) data;
|
|
|
|
int samples = 0;
|
|
|
|
int bytes = 0;
|
|
|
|
|
|
|
|
switch_mutex_lock(context->audio_mutex);
|
|
|
|
|
|
|
|
/* If the stream should no longer be sending audio */
|
|
|
|
/* then pretend we have less than one sample of audio */
|
|
|
|
/* so that libvlc will close the client connections */
|
|
|
|
if ( context->playing == 0 && switch_buffer_inuse(context->audio_buffer) == 0 ) {
|
|
|
|
switch_mutex_unlock(context->audio_mutex);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
samples = context->samples;
|
|
|
|
context->samples = 0;
|
|
|
|
|
|
|
|
if ( samples ) {
|
2014-06-12 22:06:33 +05:00
|
|
|
bytes = samples * 2 * context->channels;
|
2012-03-24 14:09:57 -07:00
|
|
|
*output = malloc(bytes);
|
|
|
|
bytes = switch_buffer_read(context->audio_buffer, *output, bytes);
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_DEBUG, "VLC imem samples: %d\n", samples);
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_DEBUG, "VLC imem bytes: %d\n", bytes);
|
|
|
|
} else {
|
|
|
|
bytes = 128;
|
|
|
|
*output = malloc(bytes);
|
|
|
|
memset(*output, 0, bytes);
|
|
|
|
}
|
|
|
|
switch_mutex_unlock(context->audio_mutex);
|
|
|
|
|
|
|
|
*size = (size_t) bytes;
|
|
|
|
return 0;
|
|
|
|
}
|
2012-02-23 16:16:55 -08:00
|
|
|
|
2012-03-24 14:09:57 -07:00
|
|
|
void vlc_imem_release_callback(void *data, const char *cookie, size_t size, void *unknown)
|
|
|
|
{
|
|
|
|
free(unknown);
|
2012-02-23 16:16:55 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
static switch_status_t vlc_file_open(switch_file_handle_t *handle, const char *path)
|
|
|
|
{
|
|
|
|
vlc_file_context_t *context;
|
2012-11-01 11:47:38 -07:00
|
|
|
libvlc_event_manager_t *mp_event_manager, *m_event_manager;
|
2012-02-23 16:16:55 -08:00
|
|
|
|
|
|
|
context = switch_core_alloc(handle->memory_pool, sizeof(*context));
|
|
|
|
context->pool = handle->memory_pool;
|
|
|
|
|
|
|
|
context->path = switch_core_strdup(context->pool, path);
|
2012-03-24 14:09:57 -07:00
|
|
|
|
|
|
|
switch_buffer_create_dynamic(&(context->audio_buffer), VLC_BUFFER_SIZE, VLC_BUFFER_SIZE * 8, 0);
|
2012-03-24 14:09:24 -07:00
|
|
|
switch_mutex_init(&context->audio_mutex, SWITCH_MUTEX_NESTED, context->pool);
|
|
|
|
switch_thread_cond_create(&(context->started), context->pool);
|
|
|
|
|
|
|
|
if (switch_test_flag(handle, SWITCH_FILE_FLAG_READ)) {
|
|
|
|
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_DEBUG, "VLC open %s for reading\n", path);
|
|
|
|
|
|
|
|
/* Determine if this is a url or a path */
|
|
|
|
/* TODO: Change this so that it tries local files first, and then if it fails try location. */
|
|
|
|
if(! strncmp(context->path, "http", 4)){
|
|
|
|
context->m = libvlc_media_new_location(read_inst, context->path);
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_NOTICE, "VLC Path is http %s\n", context->path);
|
|
|
|
} else if (! strncmp(context->path, "mms", 3)){
|
|
|
|
context->m = libvlc_media_new_path(read_inst, context->path);
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_NOTICE, "VLC Path is mms %s\n", context->path);
|
|
|
|
} else if (! strncmp(context->path, "/", 1)){
|
|
|
|
context->m = libvlc_media_new_path(read_inst, context->path);
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_NOTICE, "VLC Path is file %s\n", context->path);
|
|
|
|
} else {
|
|
|
|
context->m = libvlc_media_new_location(read_inst, context->path);
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_NOTICE, "VLC Path is unknown type %s\n", context->path);
|
|
|
|
}
|
2012-04-26 09:05:56 -07:00
|
|
|
|
2012-04-26 10:13:07 -07:00
|
|
|
if ( context->m == NULL ) {
|
2012-04-26 09:05:56 -07:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "VLC error opening %s for reading\n", path);
|
|
|
|
return SWITCH_STATUS_GENERR;
|
|
|
|
}
|
2012-03-24 14:09:24 -07:00
|
|
|
|
|
|
|
context->playing = 0;
|
|
|
|
context->err = 0;
|
|
|
|
|
|
|
|
context->mp = libvlc_media_player_new_from_media(context->m);
|
|
|
|
|
2014-06-12 22:06:33 +05:00
|
|
|
if (!handle->samplerate) {
|
2012-03-24 14:09:24 -07:00
|
|
|
handle->samplerate = 16000;
|
2014-06-12 22:06:33 +05:00
|
|
|
}
|
2012-03-24 14:09:57 -07:00
|
|
|
|
|
|
|
context->samplerate = handle->samplerate;
|
2014-06-12 22:06:33 +05:00
|
|
|
context->channels = handle->channels;
|
|
|
|
|
|
|
|
libvlc_audio_set_format(context->mp, "S16N", context->samplerate, handle->channels);
|
2012-03-24 14:09:24 -07:00
|
|
|
|
2012-11-01 11:47:38 -07:00
|
|
|
m_event_manager = libvlc_media_event_manager(context->m);
|
|
|
|
libvlc_event_attach(m_event_manager, libvlc_MediaStateChanged, vlc_media_state_callback, (void *) context);
|
|
|
|
|
|
|
|
mp_event_manager = libvlc_media_player_event_manager(context->mp);
|
|
|
|
libvlc_event_attach(mp_event_manager, libvlc_MediaPlayerEncounteredError, vlc_mediaplayer_error_callback, (void *) context);
|
|
|
|
|
2012-03-24 14:09:24 -07:00
|
|
|
libvlc_audio_set_callbacks(context->mp, vlc_auto_play_callback, NULL,NULL,NULL,NULL, (void *) context);
|
|
|
|
|
|
|
|
libvlc_media_player_play(context->mp);
|
|
|
|
|
|
|
|
} else if (switch_test_flag(handle, SWITCH_FILE_FLAG_WRITE)) {
|
2012-03-24 14:09:57 -07:00
|
|
|
const char * opts[10] = {
|
|
|
|
vlc_args,
|
|
|
|
switch_mprintf("--sout=%s", path)
|
|
|
|
};
|
|
|
|
int opts_count = 10;
|
|
|
|
|
|
|
|
if ( !handle->samplerate)
|
|
|
|
handle->samplerate = 16000;
|
|
|
|
|
|
|
|
context->samplerate = handle->samplerate;
|
|
|
|
|
|
|
|
opts[2] = switch_mprintf("--imem-get=%ld", vlc_imem_get_callback);
|
|
|
|
opts[3] = switch_mprintf("--imem-release=%ld", vlc_imem_release_callback);
|
|
|
|
opts[4] = switch_mprintf("--imem-cat=%d", 4);
|
|
|
|
opts[5] = "--demux=rawaud";
|
|
|
|
opts[6] = "--rawaud-fourcc=s16l";
|
|
|
|
opts[7] = switch_mprintf("--rawaud-samplerate=%d", context->samplerate);
|
|
|
|
opts[8] = switch_mprintf("--imem-data=%ld", context);
|
2014-06-12 22:06:33 +05:00
|
|
|
//opts[9] = "--rawaud-channels=1";
|
2012-03-24 14:09:57 -07:00
|
|
|
|
|
|
|
/* Prepare to write to an output stream. */
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_DEBUG, "VLC open %s for writing\n", path);
|
|
|
|
|
|
|
|
/* load the vlc engine. */
|
|
|
|
context->inst_out = libvlc_new(opts_count, opts);
|
|
|
|
|
|
|
|
/* Tell VLC the audio will come from memory, and to use the callbacks to fetch it. */
|
|
|
|
context->m = libvlc_media_new_location(context->inst_out, "imem/rawaud://");
|
|
|
|
context->mp = libvlc_media_player_new_from_media(context->m);
|
|
|
|
context->samples = 0;
|
|
|
|
context->pts = 0;
|
|
|
|
context->playing = 1;
|
|
|
|
libvlc_media_player_play(context->mp);
|
2012-02-23 16:16:55 -08:00
|
|
|
} else {
|
2012-03-24 14:09:24 -07:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "VLC tried to open %s for unknown reason\n", path);
|
|
|
|
return SWITCH_STATUS_GENERR;
|
2012-02-23 16:16:55 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
handle->private_info = context;
|
|
|
|
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
static switch_status_t vlc_file_read(switch_file_handle_t *handle, void *data, size_t *len)
|
|
|
|
{
|
|
|
|
vlc_file_context_t *context = handle->private_info;
|
2014-06-12 22:06:33 +05:00
|
|
|
size_t bytes = *len * sizeof(int16_t) * handle->channels, read;
|
2012-02-23 16:16:55 -08:00
|
|
|
libvlc_state_t status;
|
|
|
|
|
|
|
|
if (!context) {
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "VLC read handle context is NULL\n");
|
|
|
|
return SWITCH_STATUS_GENERR;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (context->err) {
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "VLC error\n");
|
|
|
|
return SWITCH_STATUS_GENERR;
|
|
|
|
}
|
|
|
|
|
|
|
|
status = libvlc_media_get_state(context->m);
|
2012-03-24 14:09:24 -07:00
|
|
|
|
2012-11-01 11:47:38 -07:00
|
|
|
if (status == libvlc_Error) {
|
2012-02-23 16:16:55 -08:00
|
|
|
return SWITCH_STATUS_GENERR;
|
|
|
|
}
|
2012-03-24 14:09:24 -07:00
|
|
|
|
|
|
|
switch_mutex_lock(context->audio_mutex);
|
2012-11-01 11:47:38 -07:00
|
|
|
while (context->playing == 0 && status != libvlc_Ended && status != libvlc_Error) {
|
2012-03-24 14:09:24 -07:00
|
|
|
switch_thread_cond_wait(context->started, context->audio_mutex);
|
2012-11-01 11:47:38 -07:00
|
|
|
status = libvlc_media_get_state(context->m);
|
2012-03-24 14:09:24 -07:00
|
|
|
}
|
2012-11-01 11:47:38 -07:00
|
|
|
|
|
|
|
if (context->err == 1) {
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "VLC error\n");
|
|
|
|
return SWITCH_STATUS_FALSE;
|
|
|
|
}
|
|
|
|
|
2012-03-24 14:09:24 -07:00
|
|
|
switch_mutex_unlock(context->audio_mutex);
|
2012-02-23 16:16:55 -08:00
|
|
|
|
|
|
|
switch_mutex_lock(context->audio_mutex);
|
2012-03-24 14:09:24 -07:00
|
|
|
read = switch_buffer_read(context->audio_buffer, data, bytes);
|
2012-02-23 16:16:55 -08:00
|
|
|
switch_mutex_unlock(context->audio_mutex);
|
|
|
|
|
2014-06-12 22:06:33 +05:00
|
|
|
status = libvlc_media_get_state(context->m);
|
|
|
|
|
2012-11-01 11:47:38 -07:00
|
|
|
if (!read && (status == libvlc_Stopped || status == libvlc_Ended || status == libvlc_Error)) {
|
2012-03-24 14:09:24 -07:00
|
|
|
return SWITCH_STATUS_FALSE;
|
|
|
|
} else if (!read) {
|
2012-09-22 18:53:27 -07:00
|
|
|
read = 2;
|
|
|
|
memset(data, 0, read);
|
2012-03-24 14:09:24 -07:00
|
|
|
}
|
|
|
|
|
2014-06-12 22:06:33 +05:00
|
|
|
if (read) {
|
|
|
|
*len = read / 2 / handle->channels;
|
|
|
|
}
|
2012-03-24 14:09:24 -07:00
|
|
|
|
2012-02-23 16:16:55 -08:00
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2012-03-24 14:09:57 -07:00
|
|
|
static switch_status_t vlc_file_write(switch_file_handle_t *handle, void *data, size_t *len)
|
2012-02-23 16:16:55 -08:00
|
|
|
{
|
|
|
|
vlc_file_context_t *context = handle->private_info;
|
2014-06-12 22:06:33 +05:00
|
|
|
size_t bytes = *len * sizeof(int16_t) * handle->channels;
|
2012-03-24 14:09:57 -07:00
|
|
|
|
|
|
|
switch_mutex_lock(context->audio_mutex);
|
|
|
|
context->samples += *len;
|
|
|
|
switch_buffer_write(context->audio_buffer, data, bytes);
|
|
|
|
switch_mutex_unlock(context->audio_mutex);
|
|
|
|
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
2012-02-23 16:16:55 -08:00
|
|
|
|
2012-03-24 14:09:57 -07:00
|
|
|
static switch_status_t vlc_file_close(switch_file_handle_t *handle)
|
|
|
|
{
|
|
|
|
vlc_file_context_t *context = handle->private_info;
|
2012-11-01 11:50:15 -07:00
|
|
|
int sanity = 0;
|
2012-03-24 14:09:57 -07:00
|
|
|
|
2012-03-24 14:09:24 -07:00
|
|
|
context->playing = 0;
|
|
|
|
|
2012-03-24 14:09:57 -07:00
|
|
|
/* The clients need to empty the last of the audio buffer */
|
|
|
|
while ( switch_buffer_inuse(context->audio_buffer) > 0 ) {
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_DEBUG, "VLC waiting to close the files: %d \n", (int) switch_buffer_inuse(context->audio_buffer));
|
2012-11-01 11:50:15 -07:00
|
|
|
switch_yield(500000);
|
|
|
|
if (++sanity > 10) {
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_WARNING, "Giving up waiting for client to empty the audio buffer\n");
|
|
|
|
break;
|
|
|
|
}
|
2012-03-24 14:09:57 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Let the clients get the last of the audio stream */
|
2012-11-01 11:50:15 -07:00
|
|
|
sanity = 0;
|
2012-03-24 14:09:57 -07:00
|
|
|
while ( 3 == libvlc_media_get_state(context->m) ) {
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_DEBUG, "VLC waiting for clients: %d \n", libvlc_media_get_state(context->m));
|
2012-11-01 11:50:15 -07:00
|
|
|
switch_yield(500000);
|
|
|
|
if (++sanity > 10) {
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_WARNING, "Giving up waiting for client to get the last of the audio stream\n");
|
|
|
|
break;
|
|
|
|
}
|
2012-03-24 14:09:57 -07:00
|
|
|
}
|
|
|
|
|
2012-03-24 14:09:24 -07:00
|
|
|
if( context->mp )
|
|
|
|
libvlc_media_player_stop(context->mp);
|
|
|
|
|
|
|
|
if( context->m )
|
|
|
|
libvlc_media_release(context->m);
|
|
|
|
|
2012-03-24 14:09:57 -07:00
|
|
|
if ( context->inst_out != NULL )
|
|
|
|
libvlc_release(context->inst_out);
|
|
|
|
|
2012-02-23 16:16:55 -08:00
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Macro expands to: switch_status_t mod_vlc_load(switch_loadable_module_interface_t **module_interface, switch_memory_pool_t *pool) */
|
|
|
|
SWITCH_MODULE_LOAD_FUNCTION(mod_vlc_load)
|
|
|
|
{
|
|
|
|
switch_file_interface_t *file_interface;
|
|
|
|
|
|
|
|
/* connect my internal structure to the blank pointer passed to me */
|
|
|
|
*module_interface = switch_loadable_module_create_module_interface(pool, modname);
|
|
|
|
|
|
|
|
vlc_file_supported_formats[0] = "vlc";
|
|
|
|
|
|
|
|
file_interface = switch_loadable_module_create_interface(*module_interface, SWITCH_FILE_INTERFACE);
|
|
|
|
file_interface->interface_name = modname;
|
|
|
|
file_interface->extens = vlc_file_supported_formats;
|
|
|
|
file_interface->file_open = vlc_file_open;
|
|
|
|
file_interface->file_close = vlc_file_close;
|
|
|
|
file_interface->file_read = vlc_file_read;
|
2012-03-24 14:09:57 -07:00
|
|
|
file_interface->file_write = vlc_file_write;
|
2012-02-23 16:16:55 -08:00
|
|
|
|
|
|
|
/* load the vlc engine. */
|
2012-03-24 14:09:24 -07:00
|
|
|
read_inst = libvlc_new(1, &vlc_args);
|
2012-02-23 16:16:55 -08:00
|
|
|
|
2014-01-22 13:55:13 -08:00
|
|
|
if ( ! read_inst ) {
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CRIT, "FAILED TO LOAD\n");
|
|
|
|
return SWITCH_STATUS_GENERR;
|
|
|
|
}
|
|
|
|
|
2012-02-23 16:16:55 -08:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_DEBUG, "Initialized VLC instance\n");
|
|
|
|
|
|
|
|
/* indicate that the module should continue to be loaded */
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
Called when the system shuts down
|
|
|
|
Macro expands to: switch_status_t mod_vlc_shutdown() */
|
|
|
|
SWITCH_MODULE_SHUTDOWN_FUNCTION(mod_vlc_shutdown)
|
|
|
|
{
|
2012-03-24 14:09:24 -07:00
|
|
|
if ( read_inst != NULL )
|
|
|
|
libvlc_release(read_inst);
|
2012-02-23 16:16:55 -08:00
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* For Emacs:
|
|
|
|
* Local Variables:
|
|
|
|
* mode:c
|
|
|
|
* indent-tabs-mode:t
|
|
|
|
* tab-width:4
|
|
|
|
* c-basic-offset:4
|
|
|
|
* End:
|
|
|
|
* For VIM:
|
2013-06-25 11:50:17 -05:00
|
|
|
* vim:set softtabstop=4 shiftwidth=4 tabstop=4 noet
|
2012-02-23 16:16:55 -08:00
|
|
|
*/
|