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.
 
 
 
 
 
 

383 lines
11 KiB

/*****************************************************************************
* misc.c
*****************************************************************************
* Copyright (C) 2007-2008 the VideoLAN team
* $Id$
*
* Authors: Antoine Cellerier <dionoea at videolan tod org>
* Pierre d'Herbemont <pdherbemont # videolan.org>
* Rémi Duraffort <ivoire # videolan tod 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.
*****************************************************************************/
/*****************************************************************************
* Preamble
*****************************************************************************/
#ifndef _GNU_SOURCE
# define _GNU_SOURCE
#endif
#ifdef HAVE_CONFIG_H
# include "config.h"
#endif
#include <vlc_common.h>
#include <vlc_plugin.h>
#include <vlc_meta.h>
#include <vlc_aout.h>
#include <vlc_interface.h>
#include <vlc_keys.h>
#include <lua.h> /* Low level lua C API */
#include <lauxlib.h> /* Higher level C API */
#include <lualib.h> /* Lua libs */
#include "../vlc.h"
#include "../libs.h"
/*****************************************************************************
* Internal lua<->vlc utils
*****************************************************************************/
static void vlclua_set_object( lua_State *L, void *id, void *value )
{
lua_pushlightuserdata( L, id );
lua_pushlightuserdata( L, value );
lua_rawset( L, LUA_REGISTRYINDEX );
}
static void *vlclua_get_object( lua_State *L, void *id )
{
lua_pushlightuserdata( L, id );
lua_rawget( L, LUA_REGISTRYINDEX );
const void *p = lua_topointer( L, -1 );
lua_pop( L, 1 );
return (void *)p;
}
#undef vlclua_set_this
void vlclua_set_this( lua_State *L, vlc_object_t *p_this )
{
vlclua_set_object( L, vlclua_set_this, p_this );
}
vlc_object_t * vlclua_get_this( lua_State *L )
{
return vlclua_get_object( L, vlclua_set_this );
}
void vlclua_set_intf( lua_State *L, intf_sys_t *p_intf )
{
vlclua_set_object( L, vlclua_set_intf, p_intf );
}
static intf_sys_t * vlclua_get_intf( lua_State *L )
{
return vlclua_get_object( L, vlclua_set_intf );
}
/*****************************************************************************
* VLC error code translation
*****************************************************************************/
int vlclua_push_ret( lua_State *L, int i_error )
{
lua_pushnumber( L, i_error );
lua_pushstring( L, vlc_error( i_error ) );
return 2;
}
/*****************************************************************************
* Get the VLC version string
*****************************************************************************/
static int vlclua_version( lua_State *L )
{
lua_pushstring( L, VERSION_MESSAGE );
return 1;
}
/*****************************************************************************
* Get the VLC copyright
*****************************************************************************/
static int vlclua_copyright( lua_State *L )
{
lua_pushliteral( L, COPYRIGHT_MESSAGE );
return 1;
}
/*****************************************************************************
* Get the VLC license msg/disclaimer
*****************************************************************************/
static int vlclua_license( lua_State *L )
{
lua_pushstring( L, LICENSE_MSG );
return 1;
}
/*****************************************************************************
* Quit VLC
*****************************************************************************/
static int vlclua_quit( lua_State *L )
{
vlc_object_t *p_this = vlclua_get_this( L );
/* The rc.c code also stops the playlist ... not sure if this is needed
* though. */
libvlc_Quit( p_this->p_libvlc );
return 0;
}
/*****************************************************************************
* Global properties getters
*****************************************************************************/
static int vlclua_datadir( lua_State *L )
{
char *psz_data = config_GetDataDir( vlclua_get_this( L ) );
lua_pushstring( L, psz_data );
free( psz_data );
return 1;
}
static int vlclua_userdatadir( lua_State *L )
{
char *dir = config_GetUserDir( VLC_DATA_DIR );
lua_pushstring( L, dir );
free( dir );
return 1;
}
static int vlclua_homedir( lua_State *L )
{
char *home = config_GetUserDir( VLC_HOME_DIR );
lua_pushstring( L, home );
free( home );
return 1;
}
static int vlclua_configdir( lua_State *L )
{
char *dir = config_GetUserDir( VLC_CONFIG_DIR );
lua_pushstring( L, dir );
free( dir );
return 1;
}
static int vlclua_cachedir( lua_State *L )
{
char *dir = config_GetUserDir( VLC_CACHE_DIR );
lua_pushstring( L, dir );
free( dir );
return 1;
}
static int vlclua_datadir_list( lua_State *L )
{
const char *psz_dirname = luaL_checkstring( L, 1 );
char **ppsz_dir_list = NULL;
int i = 1;
if( vlclua_dir_list( vlclua_get_this( L ), psz_dirname, &ppsz_dir_list )
!= VLC_SUCCESS )
return 0;
lua_newtable( L );
for( char **ppsz_dir = ppsz_dir_list; *ppsz_dir; ppsz_dir++ )
{
lua_pushstring( L, *ppsz_dir );
lua_rawseti( L, -2, i );
i ++;
}
vlclua_dir_list_free( ppsz_dir_list );
return 1;
}
/*****************************************************************************
*
*****************************************************************************/
static int vlclua_lock_and_wait( lua_State *L )
{
intf_sys_t *p_sys = vlclua_get_intf( L );
vlc_mutex_lock( &p_sys->lock );
mutex_cleanup_push( &p_sys->lock );
while( !p_sys->exiting )
vlc_cond_wait( &p_sys->wait, &p_sys->lock );
vlc_cleanup_run();
lua_pushboolean( L, 1 );
return 1;
}
static int vlclua_mdate( lua_State *L )
{
lua_pushnumber( L, mdate() );
return 1;
}
static int vlclua_mwait( lua_State *L )
{
double f = luaL_checknumber( L, 1 );
mwait( (int64_t)f );
return 0;
}
static int vlclua_intf_should_die( lua_State *L )
{
intf_sys_t *p_sys = vlclua_get_intf( L );
lua_pushboolean( L, p_sys->exiting );
return 1;
}
static int vlclua_action_id( lua_State *L )
{
vlc_key_t i_key = vlc_GetActionId( luaL_checkstring( L, 1 ) );
if (i_key == 0)
return 0;
lua_pushnumber( L, i_key );
return 1;
}
/*****************************************************************************
* Timer functions
*****************************************************************************/
static int vlclua_timer_schedule( lua_State *L );
static int vlclua_timer_getoverrun( lua_State *L);
static const luaL_Reg vlclua_timer_reg[] = {
{ "schedule", vlclua_timer_schedule },
{ "getoverrun", vlclua_timer_getoverrun },
{ NULL, NULL }
};
typedef struct
{
lua_State *L;
vlc_timer_t timer;
char *psz_callback;
} vlclua_timer_t;
static int vlclua_timer_schedule( lua_State *L )
{
vlclua_timer_t **pp_timer = (vlclua_timer_t**)luaL_checkudata( L, 1, "timer" );
if( !pp_timer || !*pp_timer )
luaL_error( L, "Can't get pointer to timer" );
bool b_relative = luaL_checkboolean( L, 2 );
mtime_t i_value = luaL_checkinteger( L, 3 );
mtime_t i_interval = luaL_checkinteger( L, 4 );
vlc_timer_schedule( (*pp_timer)->timer, b_relative, i_value, i_interval );
return 0;
}
static int vlclua_timer_getoverrun( lua_State *L )
{
vlclua_timer_t **pp_timer = (vlclua_timer_t**)luaL_checkudata(L, 1, "timer" );
if( !pp_timer || !*pp_timer )
luaL_error( L, "Can't get pointer to timer" );
lua_pushinteger( L, vlc_timer_getoverrun( (*pp_timer)->timer ) );
return 1;
}
static void vlclua_timer_callback( void *data )
{
vlclua_timer_t *p_timer = (vlclua_timer_t*)data;
lua_State *L = p_timer->L;
lua_getglobal( L, p_timer->psz_callback );
if( lua_pcall( L, 0, 0, 0 ) )
{
const char *psz_err = lua_tostring( L, -1 );
msg_Err( vlclua_get_this( L ), "Error while running the timer callback: '%s'", psz_err );
lua_settop( L, 0 );
}
}
static int vlclua_timer_delete( lua_State *L )
{
vlclua_timer_t **pp_timer = (vlclua_timer_t**)luaL_checkudata( L, 1, "timer" );
if( !pp_timer || !*pp_timer )
luaL_error( L, "Can't get pointer to timer" );
vlc_timer_destroy( (*pp_timer)->timer );
free( (*pp_timer)->psz_callback );
free( (*pp_timer) );
return 0;
}
static int vlclua_timer_create( lua_State *L )
{
if( !lua_isstring( L, 1 ) )
return luaL_error( L, "timer(function_name)" );
vlclua_timer_t *p_timer = malloc( sizeof( vlclua_timer_t ) );
if( vlc_timer_create( &p_timer->timer, vlclua_timer_callback, p_timer ) )
{
free( p_timer );
return luaL_error( L, "Cannot initialize the timer" );
}
p_timer->L = L;
p_timer->psz_callback = strdup( luaL_checkstring( L, 1 ) );
vlclua_timer_t **pp_timer = lua_newuserdata( L, sizeof( vlclua_timer_t* ) );
*pp_timer = p_timer;
/* Create the object */
if( luaL_newmetatable( L, "timer" ) )
{
lua_newtable( L );
luaL_register( L, NULL, vlclua_timer_reg );
lua_setfield( L, -2, "__index" );
lua_pushcfunction( L, vlclua_timer_delete );
lua_setfield( L, -2, "__gc" );
}
lua_setmetatable( L, -2 );
return 1;
}
/*****************************************************************************
*
*****************************************************************************/
static const luaL_Reg vlclua_misc_reg[] = {
{ "version", vlclua_version },
{ "copyright", vlclua_copyright },
{ "license", vlclua_license },
{ "datadir", vlclua_datadir },
{ "userdatadir", vlclua_userdatadir },
{ "homedir", vlclua_homedir },
{ "configdir", vlclua_configdir },
{ "cachedir", vlclua_cachedir },
{ "datadir_list", vlclua_datadir_list },
{ "action_id", vlclua_action_id },
{ "mdate", vlclua_mdate },
{ "mwait", vlclua_mwait },
{ "lock_and_wait", vlclua_lock_and_wait },
{ "should_die", vlclua_intf_should_die },
{ "quit", vlclua_quit },
{ "timer", vlclua_timer_create },
{ NULL, NULL }
};
void luaopen_misc( lua_State *L )
{
lua_newtable( L );
luaL_register( L, NULL, vlclua_misc_reg );
lua_setfield( L, -2, "misc" );
}