You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
1126 lines
32 KiB
1126 lines
32 KiB
/*****************************************************************************
|
|
* extension.c: Lua Extensions (meta data, web information, ...)
|
|
*****************************************************************************
|
|
* Copyright (C) 2009-2010 VideoLAN and authors
|
|
*
|
|
* Authors: Jean-Philippe André < jpeg # videolan.org >
|
|
*
|
|
* This program 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 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program 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 this program; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
|
|
*****************************************************************************/
|
|
|
|
#ifndef _GNU_SOURCE
|
|
# define _GNU_SOURCE
|
|
#endif
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
# include "config.h"
|
|
#endif
|
|
|
|
#include "vlc.h"
|
|
#include "libs.h"
|
|
#include "extension.h"
|
|
#include "assert.h"
|
|
|
|
#include <vlc_common.h>
|
|
#include <vlc_interface.h>
|
|
#include <vlc_dialog.h>
|
|
#include <vlc_player.h>
|
|
|
|
/* Functions to register */
|
|
static const luaL_Reg p_reg[] =
|
|
{
|
|
{ NULL, NULL }
|
|
};
|
|
|
|
/*
|
|
* Extensions capabilities
|
|
* Note: #define and ppsz_capabilities must be in sync
|
|
*/
|
|
static const char caps[][20] = {
|
|
"menu",
|
|
"trigger",
|
|
"input-listener",
|
|
"meta-listener",
|
|
"playing-listener",
|
|
};
|
|
|
|
static int ScanExtensions( extensions_manager_t *p_this );
|
|
static int ScanLuaCallback( vlc_object_t *p_this, const char *psz_script,
|
|
const struct luabatch_context_t * );
|
|
static int Control( extensions_manager_t *, int, extension_t *, va_list );
|
|
static int GetMenuEntries( extensions_manager_t *p_mgr, extension_t *p_ext,
|
|
char ***pppsz_titles, uint16_t **ppi_ids );
|
|
static lua_State* GetLuaState( extensions_manager_t *p_mgr,
|
|
extension_t *p_ext );
|
|
static int TriggerMenu( extension_t *p_ext, int id );
|
|
static int TriggerExtension( extensions_manager_t *p_mgr,
|
|
extension_t *p_ext );
|
|
static void WatchTimerCallback( void* );
|
|
|
|
static int vlclua_extension_deactivate( lua_State *L );
|
|
static int vlclua_extension_keep_alive( lua_State *L );
|
|
|
|
/* Interactions */
|
|
static int vlclua_extension_dialog_callback( vlc_object_t *p_this,
|
|
char const *psz_var,
|
|
vlc_value_t oldval,
|
|
vlc_value_t newval,
|
|
void *p_data );
|
|
|
|
/**
|
|
* Module entry-point
|
|
**/
|
|
int Open_Extension( vlc_object_t *p_this )
|
|
{
|
|
if( lua_Disabled( p_this ) )
|
|
return VLC_EGENERIC;
|
|
|
|
msg_Dbg( p_this, "Opening Lua Extension module" );
|
|
|
|
extensions_manager_t *p_mgr = ( extensions_manager_t* ) p_this;
|
|
|
|
p_mgr->pf_control = Control;
|
|
|
|
p_mgr->p_sys = NULL;
|
|
vlc_mutex_init( &p_mgr->lock );
|
|
|
|
/* Scan available Lua Extensions */
|
|
if( ScanExtensions( p_mgr ) != VLC_SUCCESS )
|
|
{
|
|
msg_Err( p_mgr, "Can't load extensions modules" );
|
|
return VLC_EGENERIC;
|
|
}
|
|
|
|
// Create the dialog-event variable
|
|
var_Create( p_this, "dialog-event", VLC_VAR_ADDRESS );
|
|
var_AddCallback( p_this, "dialog-event",
|
|
vlclua_extension_dialog_callback, NULL );
|
|
|
|
return VLC_SUCCESS;
|
|
}
|
|
|
|
/**
|
|
* Module unload function
|
|
**/
|
|
void Close_Extension( vlc_object_t *p_this )
|
|
{
|
|
extensions_manager_t *p_mgr = ( extensions_manager_t* ) p_this;
|
|
|
|
var_DelCallback( p_this, "dialog-event",
|
|
vlclua_extension_dialog_callback, NULL );
|
|
var_Destroy( p_mgr, "dialog-event" );
|
|
|
|
extension_t *p_ext = NULL;
|
|
|
|
/* Free extensions' memory */
|
|
ARRAY_FOREACH( p_ext, p_mgr->extensions )
|
|
{
|
|
if( !p_ext )
|
|
break;
|
|
struct lua_extension *sys = p_ext->p_sys;
|
|
|
|
vlc_mutex_lock(&sys->command_lock);
|
|
if (sys->b_activated && sys->p_progress_id == NULL &&
|
|
!sys->b_deactivating)
|
|
{
|
|
// QueueDeactivateCommand will signal the wait condition.
|
|
sys->b_exiting = true;
|
|
QueueDeactivateCommand( p_ext );
|
|
}
|
|
else
|
|
{
|
|
if (sys->L != NULL)
|
|
vlclua_fd_interrupt(&sys->dtable);
|
|
// however here we need to manually signal the wait cond, since no command is queued.
|
|
sys->b_exiting = true;
|
|
vlc_cond_signal(&sys->wait);
|
|
}
|
|
vlc_mutex_unlock(&sys->command_lock);
|
|
|
|
if (sys->b_thread_running)
|
|
vlc_join(sys->thread, NULL);
|
|
|
|
/* Clear Lua State */
|
|
if (sys->L)
|
|
{
|
|
lua_close(sys->L);
|
|
vlclua_fd_cleanup(&sys->dtable);
|
|
}
|
|
|
|
free( p_ext->psz_name );
|
|
free( p_ext->psz_title );
|
|
free( p_ext->psz_author );
|
|
free( p_ext->psz_description );
|
|
free( p_ext->psz_shortdescription );
|
|
free( p_ext->psz_url );
|
|
free( p_ext->psz_version );
|
|
free( p_ext->p_icondata );
|
|
|
|
vlc_timer_destroy(sys->timer);
|
|
|
|
free( p_ext->p_sys );
|
|
free( p_ext );
|
|
}
|
|
|
|
ARRAY_RESET( p_mgr->extensions );
|
|
}
|
|
|
|
/**
|
|
* Batch scan all Lua files in folder "extensions"
|
|
* @param p_mgr This extensions_manager_t object
|
|
**/
|
|
static int ScanExtensions( extensions_manager_t *p_mgr )
|
|
{
|
|
int i_ret =
|
|
vlclua_scripts_batch_execute( VLC_OBJECT( p_mgr ),
|
|
"extensions",
|
|
&ScanLuaCallback,
|
|
NULL );
|
|
|
|
if( !i_ret )
|
|
return VLC_EGENERIC;
|
|
|
|
return VLC_SUCCESS;
|
|
}
|
|
|
|
/**
|
|
* Dummy Lua function: does nothing
|
|
* @note This function can be used to replace "require" while scanning for
|
|
* extensions
|
|
* Even the built-in libraries are not loaded when calling descriptor()
|
|
**/
|
|
static int vlclua_dummy_require( lua_State *L )
|
|
{
|
|
(void) L;
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Replacement for "require", adding support for packaged extensions
|
|
* @note Loads modules in the modules/ folder of a package
|
|
* @note Try first with .luac and then with .lua
|
|
**/
|
|
static int vlclua_extension_require( lua_State *L )
|
|
{
|
|
const char *psz_module = luaL_checkstring( L, 1 );
|
|
vlc_object_t *p_this = vlclua_get_this( L );
|
|
extension_t *p_ext = vlclua_extension_get( L );
|
|
msg_Dbg( p_this, "loading module '%s' from extension package",
|
|
psz_module );
|
|
char *psz_fullpath, *psz_package, *sep;
|
|
psz_package = strdup( p_ext->psz_name );
|
|
sep = strrchr( psz_package, '/' );
|
|
if( !sep )
|
|
{
|
|
free( psz_package );
|
|
return luaL_error( L, "could not find package name" );
|
|
}
|
|
*sep = '\0';
|
|
if( -1 == asprintf( &psz_fullpath,
|
|
"%s/modules/%s.luac", psz_package, psz_module ) )
|
|
{
|
|
free( psz_package );
|
|
return 1;
|
|
}
|
|
int i_ret = vlclua_dofile( p_this, L, psz_fullpath );
|
|
if( i_ret != 0 )
|
|
{
|
|
// Remove trailing 'c' --> try with .lua script
|
|
psz_fullpath[ strlen( psz_fullpath ) - 1 ] = '\0';
|
|
i_ret = vlclua_dofile( p_this, L, psz_fullpath );
|
|
}
|
|
free( psz_fullpath );
|
|
free( psz_package );
|
|
if( i_ret != 0 )
|
|
{
|
|
return luaL_error( L, "unable to load module '%s' from package",
|
|
psz_module );
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static char *
|
|
GetStringFieldOrNull(lua_State *L, const char *name)
|
|
{
|
|
lua_getfield(L, -1, name);
|
|
char *value = luaL_strdupornull(L, -1);
|
|
lua_pop(L, 1);
|
|
return value;
|
|
}
|
|
|
|
/**
|
|
* Batch scan all Lua files in folder "extensions": callback
|
|
* @param p_this This extensions_manager_t object
|
|
* @param psz_filename Name of the script to run
|
|
* @param L Lua State, common to all scripts here
|
|
* @param dummy: unused
|
|
**/
|
|
int ScanLuaCallback( vlc_object_t *p_this, const char *psz_filename,
|
|
const struct luabatch_context_t *dummy )
|
|
{
|
|
VLC_UNUSED(dummy);
|
|
extensions_manager_t *p_mgr = ( extensions_manager_t* ) p_this;
|
|
|
|
msg_Dbg( p_mgr, "Scanning Lua script %s", psz_filename );
|
|
|
|
/* Experimental: read .vle packages (Zip archives) */
|
|
char *psz_script = NULL;
|
|
char *extension = strrchr(psz_filename, '.');
|
|
if (extension != NULL && strcmp(extension, ".vle") == 0)
|
|
{
|
|
msg_Dbg( p_this, "reading Lua script in a zip archive" );
|
|
if (asprintf(&psz_script, "zip://%s!/script.lua", psz_filename) == -1)
|
|
return VLC_ENOMEM;
|
|
}
|
|
else
|
|
{
|
|
psz_script = strdup( psz_filename );
|
|
if( !psz_script )
|
|
return 0;
|
|
}
|
|
|
|
/* Create new script descriptor */
|
|
extension_t *p_ext = calloc( 1, sizeof( extension_t ) );
|
|
if( !p_ext )
|
|
{
|
|
free( psz_script );
|
|
return 0;
|
|
}
|
|
|
|
p_ext->logger = vlc_object_logger(p_mgr);
|
|
p_ext->psz_name = psz_script;
|
|
struct lua_extension *sys
|
|
= p_ext->p_sys
|
|
= calloc(1, sizeof(*sys));
|
|
if (sys == NULL || !p_ext->psz_name)
|
|
{
|
|
free( p_ext->psz_name );
|
|
free(sys);
|
|
free( p_ext );
|
|
return 0;
|
|
}
|
|
sys->p_mgr = p_mgr;
|
|
|
|
/* Watch timer */
|
|
if( vlc_timer_create( &sys->timer, WatchTimerCallback, p_ext ) )
|
|
{
|
|
free( p_ext->psz_name );
|
|
free(sys);
|
|
free( p_ext );
|
|
return 0;
|
|
}
|
|
|
|
/* Mutexes and conditions */
|
|
vlc_mutex_init(&sys->command_lock);
|
|
vlc_mutex_init(&sys->running_lock);
|
|
vlc_cond_init(&sys->wait);
|
|
|
|
/* Prepare Lua state */
|
|
lua_State *L = luaL_newstate();
|
|
lua_register( L, "require", &vlclua_dummy_require );
|
|
|
|
/* Let's run it */
|
|
if( vlclua_dofile( p_this, L, psz_script ) ) // luaL_dofile
|
|
{
|
|
msg_Warn( p_mgr, "Error loading script %s: %s", psz_script,
|
|
lua_tostring( L, lua_gettop( L ) ) );
|
|
lua_pop( L, 1 );
|
|
goto discard;
|
|
}
|
|
|
|
/* Scan script for capabilities */
|
|
lua_getglobal( L, "descriptor" );
|
|
|
|
if( !lua_isfunction( L, -1 ) )
|
|
{
|
|
msg_Warn( p_mgr, "Error while running script %s, "
|
|
"function descriptor() not found", psz_script );
|
|
goto discard;
|
|
}
|
|
|
|
if( lua_pcall( L, 0, 1, 0 ) )
|
|
{
|
|
msg_Warn( p_mgr, "Error while running script %s, "
|
|
"function descriptor(): %s", psz_script,
|
|
lua_tostring( L, lua_gettop( L ) ) );
|
|
goto discard;
|
|
}
|
|
|
|
if (lua_gettop(L) == 0)
|
|
{
|
|
msg_Err(p_mgr, "Script %s went completely foobar", psz_script);
|
|
goto discard;
|
|
}
|
|
|
|
if (!lua_istable(L, -1))
|
|
{
|
|
msg_Warn(p_mgr, "In script %s, function descriptor() "
|
|
"did not return a table!", psz_script);
|
|
goto discard;
|
|
}
|
|
|
|
/* Get caps */
|
|
lua_getfield(L, -1, "capabilities");
|
|
if (lua_istable(L, -1))
|
|
{
|
|
lua_pushnil(L);
|
|
while (lua_next(L, -2) != 0)
|
|
{
|
|
/* Key is at index -2 and value at index -1. Discard key */
|
|
const char *psz_cap = luaL_checkstring(L, -1);
|
|
bool found = false;
|
|
/* Find this capability's flag */
|
|
for (size_t i = 0; i < ARRAY_SIZE(caps); i++)
|
|
{
|
|
if (!strcmp(caps[i], psz_cap))
|
|
{
|
|
/* Flag it! */
|
|
sys->i_capabilities |= 1 << i;
|
|
found = true;
|
|
break;
|
|
}
|
|
}
|
|
if (!found)
|
|
{
|
|
msg_Warn(p_mgr, "Extension capability '%s' unknown in"
|
|
" script %s", psz_cap, psz_script);
|
|
}
|
|
/* Removes 'value'; keeps 'key' for next iteration */
|
|
lua_pop(L, 1);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
msg_Warn(p_mgr, "In script %s, function descriptor() "
|
|
"did not return a table of capabilities.",
|
|
psz_script);
|
|
}
|
|
lua_pop(L, 1);
|
|
|
|
/* Get title */
|
|
lua_getfield(L, -1, "title");
|
|
if (lua_isstring(L, -1))
|
|
{
|
|
p_ext->psz_title = strdup(luaL_checkstring(L, -1));
|
|
}
|
|
else
|
|
{
|
|
msg_Dbg(p_mgr,"In script %s, function descriptor() "
|
|
"did not return a string as title.",
|
|
psz_script);
|
|
p_ext->psz_title = strdup(psz_script);
|
|
}
|
|
lua_pop(L, 1);
|
|
|
|
/* Get the fields from the extension manifest. */
|
|
p_ext->psz_author = GetStringFieldOrNull(L, "author");
|
|
p_ext->psz_description = GetStringFieldOrNull(L, "description");
|
|
p_ext->psz_shortdescription = GetStringFieldOrNull(L, "shortdesc");
|
|
p_ext->psz_url = GetStringFieldOrNull(L, "url");
|
|
p_ext->psz_version = GetStringFieldOrNull(L, "version");
|
|
/* Get icon data */
|
|
lua_getfield(L, -1, "icon");
|
|
if (!lua_isnil(L, -1) && lua_isstring(L, -1))
|
|
{
|
|
int len = lua_strlen(L, -1);
|
|
p_ext->p_icondata = malloc(len);
|
|
if (p_ext->p_icondata)
|
|
{
|
|
p_ext->i_icondata_size = len;
|
|
memcpy(p_ext->p_icondata, lua_tostring(L, -1), len);
|
|
}
|
|
}
|
|
lua_pop(L, 1);
|
|
|
|
|
|
msg_Dbg(p_mgr, "Script %s has the following capability flags: 0x%x",
|
|
psz_script, sys->i_capabilities);
|
|
|
|
lua_close( L );
|
|
/* Add the extension to the list of known extensions */
|
|
ARRAY_APPEND(p_mgr->extensions, p_ext);
|
|
|
|
/* Continue batch execution */
|
|
return VLC_EGENERIC;
|
|
discard:
|
|
lua_close(L);
|
|
free(p_ext->psz_name);
|
|
free(p_ext->psz_title);
|
|
free(p_ext->psz_url);
|
|
free(p_ext->psz_author);
|
|
free(p_ext->psz_description);
|
|
free(p_ext->psz_shortdescription);
|
|
free(p_ext->psz_version);
|
|
free(p_ext);
|
|
free(sys);
|
|
return VLC_EGENERIC;
|
|
}
|
|
|
|
static int Control(extensions_manager_t *p_mgr, int i_control,
|
|
extension_t *ext, va_list args)
|
|
{
|
|
struct lua_extension *sys = ext->p_sys;
|
|
|
|
bool *pb = NULL;
|
|
uint16_t **ppus = NULL;
|
|
char ***pppsz = NULL;
|
|
int i = 0;
|
|
|
|
switch( i_control )
|
|
{
|
|
case EXTENSION_ACTIVATE:
|
|
return Activate(ext);
|
|
|
|
case EXTENSION_DEACTIVATE:
|
|
return Deactivate(p_mgr, ext);
|
|
|
|
case EXTENSION_IS_ACTIVATED:
|
|
pb = va_arg( args, bool* );
|
|
vlc_mutex_lock(&sys->command_lock);
|
|
*pb = sys->b_activated;
|
|
vlc_mutex_unlock(&sys->command_lock);
|
|
break;
|
|
|
|
case EXTENSION_HAS_MENU:
|
|
pb = va_arg( args, bool* );
|
|
*pb = (sys->i_capabilities & EXT_HAS_MENU) ? 1 : 0;
|
|
break;
|
|
|
|
case EXTENSION_GET_MENU:
|
|
pppsz = va_arg( args, char*** );
|
|
ppus = va_arg( args, uint16_t** );
|
|
return GetMenuEntries(p_mgr, ext, pppsz, ppus);
|
|
|
|
case EXTENSION_TRIGGER_ONLY:
|
|
pb = va_arg( args, bool* );
|
|
*pb = (sys->i_capabilities & EXT_TRIGGER_ONLY) ? 1 : 0;
|
|
break;
|
|
|
|
case EXTENSION_TRIGGER:
|
|
return TriggerExtension(p_mgr, ext);
|
|
|
|
case EXTENSION_TRIGGER_MENU:
|
|
i = va_arg( args, int );
|
|
return TriggerMenu(ext, i);
|
|
|
|
case EXTENSION_SET_INPUT:
|
|
{
|
|
input_item_t *p_item = va_arg( args, struct input_item_t * );
|
|
|
|
vlc_mutex_lock(&sys->command_lock);
|
|
if (sys->b_exiting)
|
|
{
|
|
vlc_mutex_unlock(&sys->command_lock);
|
|
return VLC_EGENERIC;
|
|
}
|
|
vlc_mutex_unlock(&sys->command_lock);
|
|
|
|
vlc_mutex_lock(&sys->running_lock);
|
|
|
|
// Change input
|
|
input_item_t *old = sys->p_item;
|
|
if( old )
|
|
input_item_Release( old );
|
|
|
|
sys->p_item = p_item ? input_item_Hold(p_item) : NULL;
|
|
|
|
// Tell the script the input changed
|
|
if (sys->i_capabilities & EXT_INPUT_LISTENER)
|
|
{
|
|
PushCommandUnique(ext, CMD_SET_INPUT);
|
|
}
|
|
|
|
vlc_mutex_unlock(&sys->running_lock);
|
|
break;
|
|
}
|
|
case EXTENSION_PLAYING_CHANGED:
|
|
{
|
|
assert(ext->psz_name != NULL);
|
|
i = va_arg( args, int );
|
|
if (sys->i_capabilities & EXT_PLAYING_LISTENER)
|
|
{
|
|
PushCommand(ext, CMD_PLAYING_CHANGED, i);
|
|
}
|
|
break;
|
|
}
|
|
case EXTENSION_META_CHANGED:
|
|
{
|
|
PushCommand(ext, CMD_UPDATE_META);
|
|
break;
|
|
}
|
|
default:
|
|
msg_Warn( p_mgr, "Control '%d' not yet implemented in Extension",
|
|
i_control );
|
|
return VLC_EGENERIC;
|
|
}
|
|
|
|
return VLC_SUCCESS;
|
|
}
|
|
|
|
int lua_ExtensionActivate( extensions_manager_t *p_mgr, extension_t *p_ext )
|
|
{
|
|
assert( p_mgr != NULL && p_ext != NULL );
|
|
return lua_ExecuteFunction( p_mgr, p_ext, "activate", LUA_END );
|
|
}
|
|
|
|
int lua_ExtensionDeactivate( extensions_manager_t *p_mgr, extension_t *p_ext )
|
|
{
|
|
assert( p_mgr != NULL && p_ext != NULL );
|
|
struct lua_extension *sys = p_ext->p_sys;
|
|
|
|
vlclua_fd_interrupt(&sys->dtable);
|
|
|
|
// Unset and release input objects
|
|
if (sys->p_item)
|
|
{
|
|
input_item_Release(sys->p_item);
|
|
sys->p_item = NULL;
|
|
}
|
|
|
|
return lua_ExecuteFunction( p_mgr, p_ext, "deactivate", LUA_END );
|
|
}
|
|
|
|
int lua_ExtensionWidgetClick( extensions_manager_t *p_mgr,
|
|
extension_t *p_ext,
|
|
extension_widget_t *p_widget )
|
|
{
|
|
struct lua_extension *sys = p_ext->p_sys;
|
|
if (sys->L == NULL)
|
|
return VLC_SUCCESS;
|
|
|
|
lua_State *L = GetLuaState( p_mgr, p_ext );
|
|
lua_pushlightuserdata( L, p_widget );
|
|
lua_gettable( L, LUA_REGISTRYINDEX );
|
|
return lua_ExecuteFunction( p_mgr, p_ext, NULL, LUA_END );
|
|
}
|
|
|
|
|
|
/**
|
|
* Get the list of menu entries from an extension script
|
|
* @param p_mgr
|
|
* @param p_ext
|
|
* @param pppsz_titles Pointer to NULL. All strings must be freed by the caller
|
|
* @param ppi_ids Pointer to NULL. Must be freed by the caller.
|
|
* @note This function is allowed to run in the UI thread. This means
|
|
* that it MUST respond very fast.
|
|
* @todo Remove the menu() hook and provide a new function vlc.set_menu()
|
|
**/
|
|
static int GetMenuEntries( extensions_manager_t *p_mgr, extension_t *p_ext,
|
|
char ***pppsz_titles, uint16_t **ppi_ids )
|
|
{
|
|
assert( *pppsz_titles == NULL );
|
|
assert( *ppi_ids == NULL );
|
|
struct lua_extension *sys = p_ext->p_sys;
|
|
|
|
vlc_mutex_lock(&sys->command_lock);
|
|
if (!sys->b_activated || sys->b_exiting)
|
|
{
|
|
vlc_mutex_unlock(&sys->command_lock);
|
|
msg_Dbg( p_mgr, "Can't get menu of an unactivated/dying extension!" );
|
|
return VLC_EGENERIC;
|
|
}
|
|
vlc_mutex_unlock(&sys->command_lock);
|
|
|
|
vlc_mutex_lock(&sys->running_lock);
|
|
|
|
int i_ret = VLC_EGENERIC;
|
|
lua_State *L = GetLuaState( p_mgr, p_ext );
|
|
|
|
if ((sys->i_capabilities & EXT_HAS_MENU) == 0)
|
|
{
|
|
msg_Dbg( p_mgr, "can't get a menu from an extension without menu!" );
|
|
goto exit;
|
|
}
|
|
|
|
lua_getglobal( L, "menu" );
|
|
|
|
if( !lua_isfunction( L, -1 ) )
|
|
{
|
|
msg_Warn( p_mgr, "Error while running script %s, "
|
|
"function menu() not found", p_ext->psz_name );
|
|
goto exit;
|
|
}
|
|
|
|
if( lua_pcall( L, 0, 1, 0 ) )
|
|
{
|
|
msg_Warn( p_mgr, "Error while running script %s, "
|
|
"function menu(): %s", p_ext->psz_name,
|
|
lua_tostring( L, lua_gettop( L ) ) );
|
|
goto exit;
|
|
}
|
|
|
|
if (lua_gettop(L) == 0)
|
|
{
|
|
msg_Warn(p_mgr, "Script %s went completely foobar", p_ext->psz_name);
|
|
goto exit;
|
|
}
|
|
|
|
if (!lua_istable(L, -1))
|
|
{
|
|
msg_Warn(p_mgr, "Function menu() in script %s "
|
|
"did not return a table", p_ext->psz_name);
|
|
goto exit;
|
|
}
|
|
|
|
/* Get table size */
|
|
size_t i_size = lua_objlen( L, -1 );
|
|
*pppsz_titles = calloc(i_size+1, sizeof(**pppsz_titles));
|
|
*ppi_ids = calloc(i_size+1, sizeof(**ppi_ids));
|
|
|
|
/* Walk table */
|
|
size_t i_idx = 0;
|
|
lua_pushnil(L);
|
|
while (lua_next(L, -2) != 0)
|
|
{
|
|
assert(i_idx < i_size);
|
|
if(!lua_isstring(L, -1) || !lua_isnumber(L, -2))
|
|
{
|
|
msg_Warn(p_mgr, "In script %s, an entry in "
|
|
"the menu table is invalid!", p_ext->psz_name);
|
|
goto exit;
|
|
}
|
|
(*pppsz_titles)[i_idx] = strdup(luaL_checkstring(L, -1));
|
|
(*ppi_ids)[i_idx] = luaL_checkinteger(L, -2) & 0xFFFF;
|
|
i_idx++;
|
|
lua_pop(L, 1);
|
|
}
|
|
|
|
i_ret = VLC_SUCCESS;
|
|
|
|
exit:
|
|
vlc_mutex_unlock(&sys->running_lock);
|
|
if( i_ret != VLC_SUCCESS )
|
|
{
|
|
msg_Dbg( p_mgr, "Something went wrong in %s (%s:%d)",
|
|
__func__, __FILE__, __LINE__ );
|
|
}
|
|
return i_ret;
|
|
}
|
|
|
|
/* Must be entered with the Lock on Extension */
|
|
static lua_State* GetLuaState( extensions_manager_t *p_mgr,
|
|
extension_t *p_ext )
|
|
{
|
|
assert( p_ext != NULL );
|
|
struct lua_extension *sys = p_ext->p_sys;
|
|
|
|
if (sys->L != NULL)
|
|
return sys->L;
|
|
|
|
lua_State *L = luaL_newstate();
|
|
if (L == NULL)
|
|
{
|
|
msg_Err(p_mgr, "Could not create new Lua State");
|
|
return NULL;
|
|
}
|
|
vlclua_set_this(L, p_mgr);
|
|
intf_thread_t *intf = (intf_thread_t *)vlc_object_parent(p_mgr);
|
|
vlc_playlist_t *playlist = vlc_intf_GetMainPlaylist(intf);
|
|
vlclua_set_playlist_internal(L, playlist);
|
|
vlclua_extension_set(L, p_ext);
|
|
|
|
luaL_openlibs(L);
|
|
luaL_register_namespace(L, "vlc", p_reg);
|
|
luaopen_msg(L);
|
|
|
|
/* Load more libraries */
|
|
luaopen_config(L);
|
|
luaopen_dialog(L, p_ext);
|
|
luaopen_input(L);
|
|
luaopen_msg(L);
|
|
if (vlclua_fd_init(L, &sys->dtable))
|
|
{
|
|
lua_close(L);
|
|
return NULL;
|
|
}
|
|
luaopen_object(L);
|
|
luaopen_osd(L);
|
|
luaopen_playlist(L);
|
|
luaopen_stream(L);
|
|
luaopen_strings(L);
|
|
luaopen_variables(L);
|
|
luaopen_video(L);
|
|
luaopen_vlm(L);
|
|
luaopen_volume(L);
|
|
luaopen_xml(L);
|
|
luaopen_vlcio(L);
|
|
luaopen_errno(L);
|
|
luaopen_rand(L);
|
|
luaopen_rd(L);
|
|
luaopen_ml(L);
|
|
#if defined(_WIN32)
|
|
# if WINAPI_FAMILY_PARTITION(WINAPI_PARTITION_DESKTOP)
|
|
luaopen_win(L);
|
|
# endif
|
|
#endif
|
|
|
|
/* Register extension specific functions */
|
|
lua_getglobal(L, "vlc");
|
|
lua_pushcfunction(L, vlclua_extension_deactivate);
|
|
lua_setfield(L, -2, "deactivate");
|
|
lua_pushcfunction(L, vlclua_extension_keep_alive);
|
|
lua_setfield(L, -2, "keep_alive");
|
|
|
|
/* Setup the module search path */
|
|
if (!strncmp(p_ext->psz_name, "zip://", 6))
|
|
{
|
|
/* Load all required modules manually */
|
|
lua_register(L, "require", &vlclua_extension_require);
|
|
}
|
|
else if (vlclua_add_modules_path(L, p_ext->psz_name))
|
|
{
|
|
msg_Warn(p_mgr, "Error while setting the module "
|
|
"search path for %s", p_ext->psz_name);
|
|
vlclua_fd_cleanup(&sys->dtable);
|
|
lua_close(L);
|
|
return NULL;
|
|
}
|
|
|
|
/* Load and run the script(s) */
|
|
if (vlclua_dofile(VLC_OBJECT(p_mgr), L, p_ext->psz_name))
|
|
{
|
|
msg_Warn(p_mgr, "Error loading script %s: %s", p_ext->psz_name,
|
|
lua_tostring(L, lua_gettop(L)));
|
|
vlclua_fd_cleanup(&sys->dtable);
|
|
lua_close(L);
|
|
return NULL;
|
|
}
|
|
|
|
sys->L = L;
|
|
|
|
return L;
|
|
}
|
|
|
|
int lua_ExecuteFunction( extensions_manager_t *p_mgr, extension_t *p_ext,
|
|
const char *psz_function, ... )
|
|
{
|
|
va_list args;
|
|
va_start( args, psz_function );
|
|
int i_ret = lua_ExecuteFunctionVa( p_mgr, p_ext, psz_function, args );
|
|
va_end( args );
|
|
return i_ret;
|
|
}
|
|
|
|
/**
|
|
* Execute a function in a Lua script
|
|
* @param psz_function Name of global function to execute. If NULL, assume
|
|
* that the function object is already on top of the
|
|
* stack.
|
|
* @return < 0 in case of failure, >= 0 in case of success
|
|
* @note It's better to call this function from a dedicated thread
|
|
* (see extension_thread.c)
|
|
**/
|
|
int lua_ExecuteFunctionVa( extensions_manager_t *p_mgr, extension_t *p_ext,
|
|
const char *psz_function, va_list args )
|
|
{
|
|
int i_ret = VLC_SUCCESS;
|
|
int i_args = 0;
|
|
assert( p_mgr != NULL );
|
|
assert( p_ext != NULL );
|
|
|
|
lua_State *L = GetLuaState( p_mgr, p_ext );
|
|
if( !L )
|
|
return -1;
|
|
|
|
if( psz_function )
|
|
lua_getglobal( L, psz_function );
|
|
|
|
if( !lua_isfunction( L, -1 ) )
|
|
{
|
|
msg_Warn( p_mgr, "Error while running script %s, "
|
|
"function %s() not found", p_ext->psz_name, psz_function );
|
|
lua_pop( L, 1 );
|
|
goto exit;
|
|
}
|
|
|
|
lua_datatype_e type = LUA_END;
|
|
while( ( type = va_arg( args, int ) ) != LUA_END )
|
|
{
|
|
if( type == LUA_NUM )
|
|
{
|
|
lua_pushnumber( L , va_arg( args, int ) );
|
|
}
|
|
else if( type == LUA_TEXT )
|
|
{
|
|
lua_pushstring( L , va_arg( args, char* ) );
|
|
}
|
|
else
|
|
{
|
|
msg_Warn( p_mgr, "Undefined argument type %d to lua function %s"
|
|
"from script %s", type, psz_function, p_ext->psz_name );
|
|
if( i_args > 0 )
|
|
lua_pop( L, i_args );
|
|
goto exit;
|
|
}
|
|
i_args ++;
|
|
}
|
|
|
|
// Start actual call to Lua
|
|
if( lua_pcall( L, i_args, 1, 0 ) )
|
|
{
|
|
msg_Warn( p_mgr, "Error while running script %s, "
|
|
"function %s(): %s", p_ext->psz_name, psz_function,
|
|
lua_tostring( L, lua_gettop( L ) ) );
|
|
i_ret = VLC_EGENERIC;
|
|
}
|
|
|
|
if (i_ret != VLC_SUCCESS)
|
|
lua_DialogFlush( L );
|
|
else
|
|
i_ret = lua_DialogFlush( L );
|
|
|
|
exit:
|
|
return i_ret;
|
|
|
|
}
|
|
|
|
static inline int TriggerMenu( extension_t *p_ext, int i_id )
|
|
{
|
|
return PushCommand( p_ext, CMD_TRIGGERMENU, i_id );
|
|
}
|
|
|
|
int lua_ExtensionTriggerMenu( extensions_manager_t *p_mgr,
|
|
extension_t *p_ext, int id )
|
|
{
|
|
int i_ret = VLC_SUCCESS;
|
|
lua_State *L = GetLuaState( p_mgr, p_ext );
|
|
|
|
if( !L )
|
|
return VLC_EGENERIC;
|
|
|
|
luaopen_dialog( L, p_ext );
|
|
|
|
lua_getglobal( L, "trigger_menu" );
|
|
if( !lua_isfunction( L, -1 ) )
|
|
{
|
|
msg_Warn( p_mgr, "Error while running script %s, "
|
|
"function trigger_menu() not found", p_ext->psz_name );
|
|
return VLC_EGENERIC;
|
|
}
|
|
|
|
/* Pass id as unique argument to the function */
|
|
lua_pushinteger( L, id );
|
|
|
|
if( lua_pcall( L, 1, 1, 0 ) != 0 )
|
|
{
|
|
msg_Warn( p_mgr, "Error while running script %s, "
|
|
"function trigger_menu(): %s", p_ext->psz_name,
|
|
lua_tostring( L, lua_gettop( L ) ) );
|
|
i_ret = VLC_EGENERIC;
|
|
}
|
|
|
|
if (i_ret != VLC_SUCCESS)
|
|
lua_DialogFlush( L );
|
|
else
|
|
i_ret = lua_DialogFlush( L );
|
|
if( i_ret < VLC_SUCCESS )
|
|
{
|
|
msg_Dbg( p_mgr, "Something went wrong in %s (%s:%d)",
|
|
__func__, __FILE__, __LINE__ );
|
|
}
|
|
|
|
return i_ret;
|
|
}
|
|
|
|
/** Directly trigger an extension, without activating it
|
|
* This is NOT multithreaded, and this code runs in the UI thread
|
|
* @param p_mgr
|
|
* @param p_ext Extension to trigger
|
|
* @return Value returned by the lua function "trigger"
|
|
**/
|
|
static int TriggerExtension( extensions_manager_t *p_mgr,
|
|
extension_t *p_ext )
|
|
{
|
|
struct lua_extension *sys = p_ext->p_sys;
|
|
|
|
int i_ret = lua_ExecuteFunction( p_mgr, p_ext, "trigger", LUA_END );
|
|
|
|
/* Close lua state for trigger-only extensions */
|
|
if (sys->L)
|
|
{
|
|
vlclua_fd_cleanup(&sys->dtable);
|
|
lua_close(sys->L);
|
|
}
|
|
sys->L = NULL;
|
|
|
|
return i_ret;
|
|
}
|
|
|
|
/** Set extension associated to the current script
|
|
* @param L current lua_State
|
|
* @param p_ext the extension
|
|
*/
|
|
void vlclua_extension_set( lua_State *L, extension_t *p_ext )
|
|
{
|
|
lua_pushlightuserdata( L, vlclua_extension_set );
|
|
lua_pushlightuserdata( L, p_ext );
|
|
lua_rawset( L, LUA_REGISTRYINDEX );
|
|
}
|
|
|
|
/** Retrieve extension associated to the current script
|
|
* @param L current lua_State
|
|
* @return Extension pointer
|
|
**/
|
|
extension_t *vlclua_extension_get( lua_State *L )
|
|
{
|
|
lua_pushlightuserdata( L, vlclua_extension_set );
|
|
lua_rawget( L, LUA_REGISTRYINDEX );
|
|
extension_t *p_ext = (extension_t*) lua_topointer( L, -1 );
|
|
lua_pop( L, 1 );
|
|
return p_ext;
|
|
}
|
|
|
|
/** Deactivate an extension by order from the extension itself
|
|
* @param L lua_State
|
|
* @note This is an asynchronous call. A script calling vlc.deactivate() will
|
|
* be executed to the end before the last call to deactivate() is done.
|
|
**/
|
|
int vlclua_extension_deactivate( lua_State *L )
|
|
{
|
|
extension_t *p_ext = vlclua_extension_get( L );
|
|
struct lua_extension *sys = p_ext->p_sys;
|
|
|
|
vlc_mutex_lock(&sys->command_lock);
|
|
bool b_ret = QueueDeactivateCommand( p_ext );
|
|
vlc_mutex_unlock(&sys->command_lock);
|
|
return ( b_ret == true ) ? 1 : 0;
|
|
}
|
|
|
|
/** Keep an extension alive. This resets the watch timer to 0
|
|
* @param L lua_State
|
|
* @note This is the "vlc.keep_alive()" function
|
|
**/
|
|
int vlclua_extension_keep_alive( lua_State *L )
|
|
{
|
|
extension_t *p_ext = vlclua_extension_get( L );
|
|
struct lua_extension *sys = p_ext->p_sys;
|
|
|
|
vlc_mutex_lock(&sys->command_lock);
|
|
if (sys->p_progress_id != NULL)
|
|
{
|
|
vlc_dialog_release(sys->p_mgr, sys->p_progress_id);
|
|
sys->p_progress_id = NULL;
|
|
}
|
|
vlc_timer_schedule(sys->timer, false, WATCH_TIMER_PERIOD,
|
|
VLC_TIMER_FIRE_ONCE);
|
|
vlc_mutex_unlock(&sys->command_lock);
|
|
|
|
return 1;
|
|
}
|
|
|
|
/** Callback for the variable "dialog-event"
|
|
* @param p_this Current object owner of the extension and the dialog
|
|
* @param psz_var "dialog-event"
|
|
* @param oldval Unused
|
|
* @param newval Address of the dialog
|
|
* @param p_data Unused
|
|
**/
|
|
static int vlclua_extension_dialog_callback( vlc_object_t *p_this,
|
|
char const *psz_var,
|
|
vlc_value_t oldval,
|
|
vlc_value_t newval,
|
|
void *p_data )
|
|
{
|
|
/* psz_var == "dialog-event" */
|
|
( void ) psz_var;
|
|
( void ) oldval;
|
|
( void ) p_data;
|
|
|
|
extension_dialog_command_t *command = newval.p_address;
|
|
assert( command != NULL );
|
|
assert( command->p_dlg != NULL);
|
|
|
|
extension_t *p_ext = command->p_dlg->p_sys;
|
|
assert( p_ext != NULL );
|
|
|
|
extension_widget_t *p_widget = command->p_data;
|
|
|
|
switch( command->event )
|
|
{
|
|
case EXTENSION_EVENT_CLICK:
|
|
assert( p_widget != NULL );
|
|
PushCommandUnique( p_ext, CMD_CLICK, p_widget );
|
|
break;
|
|
case EXTENSION_EVENT_CLOSE:
|
|
PushCommandUnique( p_ext, CMD_CLOSE );
|
|
break;
|
|
default:
|
|
msg_Dbg( p_this, "Received unknown UI event %d, discarded",
|
|
command->event );
|
|
break;
|
|
}
|
|
|
|
return VLC_SUCCESS;
|
|
}
|
|
|
|
/** Watch timer callback
|
|
* The timer expired, Lua may be stuck, ask the user what to do now
|
|
**/
|
|
static void WatchTimerCallback( void *data )
|
|
{
|
|
extension_t *p_ext = data;
|
|
struct lua_extension *sys = p_ext->p_sys;
|
|
extensions_manager_t *p_mgr = sys->p_mgr;
|
|
|
|
vlc_mutex_lock(&sys->command_lock);
|
|
|
|
for( struct command_t *cmd = sys->command;
|
|
cmd != NULL;
|
|
cmd = cmd->next )
|
|
if( cmd->i_command == CMD_DEACTIVATE )
|
|
{ /* We have a pending Deactivate command... */
|
|
if (sys->p_progress_id != NULL)
|
|
{
|
|
vlc_dialog_release(p_mgr, sys->p_progress_id);
|
|
sys->p_progress_id = NULL;
|
|
}
|
|
KillExtension(p_ext);
|
|
vlc_mutex_unlock(&sys->command_lock);
|
|
return;
|
|
}
|
|
|
|
if (sys->p_progress_id == NULL)
|
|
{
|
|
sys->p_progress_id =
|
|
vlc_dialog_display_progress( p_mgr, true, 0.0,
|
|
_( "Yes" ),
|
|
_( "Extension not responding!" ),
|
|
_( "Extension '%s' does not respond.\n"
|
|
"Do you want to kill it now? " ),
|
|
p_ext->psz_title );
|
|
if (sys->p_progress_id == NULL)
|
|
{
|
|
KillExtension(p_ext);
|
|
vlc_mutex_unlock(&sys->command_lock);
|
|
return;
|
|
}
|
|
vlc_timer_schedule(sys->timer, false, VLC_TICK_FROM_MS(100),
|
|
VLC_TIMER_FIRE_ONCE);
|
|
}
|
|
else
|
|
{
|
|
if (vlc_dialog_is_cancelled(p_mgr, sys->p_progress_id))
|
|
{
|
|
vlc_dialog_release(p_mgr, sys->p_progress_id);
|
|
sys->p_progress_id = NULL;
|
|
KillExtension(p_ext);
|
|
vlc_mutex_unlock(&sys->command_lock);
|
|
return;
|
|
}
|
|
vlc_timer_schedule(sys->timer, false, VLC_TICK_FROM_MS(100),
|
|
VLC_TIMER_FIRE_ONCE);
|
|
}
|
|
vlc_mutex_unlock(&sys->command_lock);
|
|
}
|
|
|