demux.c
上传用户:kjfoods
上传日期:2020-07-06
资源大小:29949k
文件大小:9k
源码类别:

midi

开发平台:

Unix_Linux

  1. /*****************************************************************************
  2.  * demux.c :  Lua playlist demux module
  3.  *****************************************************************************
  4.  * Copyright (C) 2007-2008 the VideoLAN team
  5.  * $Id: d7f4445bf0ced0fdaf02b7567cdd43de7a4c8449 $
  6.  *
  7.  * Authors: Antoine Cellerier <dionoea at videolan tod org>
  8.  *
  9.  * This program is free software; you can redistribute it and/or modify
  10.  * it under the terms of the GNU General Public License as published by
  11.  * the Free Software Foundation; either version 2 of the License, or
  12.  * (at your option) any later version.
  13.  *
  14.  * This program is distributed in the hope that it will be useful,
  15.  * but WITHOUT ANY WARRANTY; without even the implied warranty of
  16.  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  17.  * GNU General Public License for more details.
  18.  *
  19.  * You should have received a copy of the GNU General Public License
  20.  * along with this program; if not, write to the Free Software
  21.  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
  22.  *****************************************************************************/
  23. /*****************************************************************************
  24.  * Preamble
  25.  *****************************************************************************/
  26. #ifdef HAVE_CONFIG_H
  27. # include "config.h"
  28. #endif
  29. #include <assert.h>
  30. #include <vlc_common.h>
  31. #include <vlc_demux.h>
  32. #include <vlc_url.h>
  33. #include <vlc_strings.h>
  34. #include <vlc_charset.h>
  35. #include <errno.h>                                                 /* ENOMEM */
  36. #ifdef HAVE_SYS_STAT_H
  37. #   include <sys/stat.h>
  38. #endif
  39. #include "vlc.h"
  40. #include "libs.h"
  41. #include "libs/playlist.h"
  42. /*****************************************************************************
  43.  * Local prototypes
  44.  *****************************************************************************/
  45. static int Demux( demux_t *p_demux );
  46. static int Control( demux_t *p_demux, int i_query, va_list args );
  47. /*****************************************************************************
  48.  * Demux specific functions
  49.  *****************************************************************************/
  50. struct demux_sys_t
  51. {
  52.     lua_State *L;
  53.     char *psz_filename;
  54. };
  55. static int vlclua_demux_peek( lua_State *L )
  56. {
  57.     demux_t *p_demux = (demux_t *)vlclua_get_this( L );
  58.     int n = luaL_checkint( L, 1 );
  59.     const uint8_t *p_peek;
  60.     int i_peek = stream_Peek( p_demux->s, &p_peek, n );
  61.     lua_pushlstring( L, (const char *)p_peek, i_peek );
  62.     return 1;
  63. }
  64. static int vlclua_demux_read( lua_State *L )
  65. {
  66.     demux_t *p_demux = (demux_t *)vlclua_get_this( L );
  67.     const uint8_t *p_read;
  68.     int n = luaL_checkint( L, 1 );
  69.     int i_read = stream_Peek( p_demux->s, &p_read, n );
  70.     lua_pushlstring( L, (const char *)p_read, i_read );
  71.     int i_seek = stream_Read( p_demux->s, NULL, i_read );
  72.     assert(i_read==i_seek);
  73.     return 1;
  74. }
  75. static int vlclua_demux_readline( lua_State *L )
  76. {
  77.     demux_t *p_demux = (demux_t *)vlclua_get_this( L );
  78.     char *psz_line = stream_ReadLine( p_demux->s );
  79.     if( psz_line )
  80.     {
  81.         lua_pushstring( L, psz_line );
  82.         free( psz_line );
  83.     }
  84.     else
  85.     {
  86.         lua_pushnil( L );
  87.     }
  88.     return 1;
  89. }
  90. /*****************************************************************************
  91.  *
  92.  *****************************************************************************/
  93. /* Functions to register */
  94. static const luaL_Reg p_reg[] =
  95. {
  96.     { "peek", vlclua_demux_peek },
  97.     { NULL, NULL }
  98. };
  99. /* Functions to register for parse() function call only */
  100. static const luaL_Reg p_reg_parse[] =
  101. {
  102.     { "read", vlclua_demux_read },
  103.     { "readline", vlclua_demux_readline },
  104.     { NULL, NULL }
  105. };
  106. /*****************************************************************************
  107.  * Called through lua_scripts_batch_execute to call 'probe' on
  108.  * the script pointed by psz_filename.
  109.  *****************************************************************************/
  110. static int probe_luascript( vlc_object_t *p_this, const char * psz_filename,
  111.                             lua_State * L, void * user_data )
  112. {
  113.     VLC_UNUSED(user_data);
  114.     demux_t * p_demux = (demux_t *)p_this;
  115.     p_demux->p_sys->psz_filename = strdup(psz_filename);
  116.     /* In lua, setting a variable's value to nil is equivalent to deleting it */
  117.     lua_pushnil( L );
  118.     lua_pushnil( L );
  119.     lua_setglobal( L, "probe" );
  120.     lua_setglobal( L, "parse" );
  121.     /* Load and run the script(s) */
  122.     if( luaL_dofile( L, psz_filename ) )
  123.     {
  124.         msg_Warn( p_demux, "Error loading script %s: %s", psz_filename,
  125.                   lua_tostring( L, lua_gettop( L ) ) );
  126.         lua_pop( L, 1 );
  127.         return VLC_EGENERIC;
  128.     }
  129.     lua_getglobal( L, "probe" );
  130.     if( !lua_isfunction( L, -1 ) )
  131.     {
  132.         msg_Warn( p_demux, "Error while runing script %s, "
  133.                   "function probe() not found", psz_filename );
  134.         lua_pop( L, 1 );
  135.         return VLC_EGENERIC;
  136.     }
  137.     if( lua_pcall( L, 0, 1, 0 ) )
  138.     {
  139.         msg_Warn( p_demux, "Error while runing script %s, "
  140.                   "function probe(): %s", psz_filename,
  141.                   lua_tostring( L, lua_gettop( L ) ) );
  142.         lua_pop( L, 1 );
  143.         return VLC_EGENERIC;
  144.     }
  145.     if( lua_gettop( L ) )
  146.     {
  147.         int i_ret = VLC_EGENERIC;
  148.         if( lua_toboolean( L, 1 ) )
  149.         {
  150.             msg_Dbg( p_demux, "Lua playlist script %s's "
  151.                      "probe() function was successful", psz_filename );
  152.             i_ret = VLC_SUCCESS;
  153.         }
  154.         lua_pop( L, 1 );
  155.         return i_ret;
  156.     }
  157.     return VLC_EGENERIC;
  158. }
  159. /*****************************************************************************
  160.  * Import_LuaPlaylist: main import function
  161.  *****************************************************************************/
  162. int Import_LuaPlaylist( vlc_object_t *p_this )
  163. {
  164.     demux_t *p_demux = (demux_t *)p_this;
  165.     lua_State *L;
  166.     int ret;
  167.     p_demux->p_sys = (demux_sys_t*)malloc( sizeof( demux_sys_t ) );
  168.     if( !p_demux->p_sys )
  169.     {
  170.         return VLC_ENOMEM;
  171.     }
  172.     p_demux->p_sys->psz_filename = NULL;
  173.     p_demux->pf_control = Control;
  174.     p_demux->pf_demux = Demux;
  175.     /* Initialise Lua state structure */
  176.     L = luaL_newstate();
  177.     if( !L )
  178.     {
  179.         msg_Err( p_demux, "Could not create new Lua State" );
  180.         free( p_demux->p_sys );
  181.         return VLC_EGENERIC;
  182.     }
  183.     p_demux->p_sys->L = L;
  184.     /* Load Lua libraries */
  185.     luaL_openlibs( L ); /* FIXME: Don't open all the libs? */
  186.     luaL_register( L, "vlc", p_reg );
  187.     luaopen_msg( L );
  188.     luaopen_strings( L );
  189.     lua_pushlightuserdata( L, p_demux );
  190.     lua_setfield( L, -2, "private" );
  191.     lua_pushstring( L, p_demux->psz_path );
  192.     lua_setfield( L, -2, "path" );
  193.     lua_pushstring( L, p_demux->psz_access );
  194.     lua_setfield( L, -2, "access" );
  195.     lua_pop( L, 1 );
  196.     ret = vlclua_scripts_batch_execute( p_this, "playlist",
  197.                                         &probe_luascript, L, NULL );
  198.     if( ret )
  199.         Close_LuaPlaylist( p_this );
  200.     return ret;
  201. }
  202. /*****************************************************************************
  203.  * Deactivate: frees unused data
  204.  *****************************************************************************/
  205. void Close_LuaPlaylist( vlc_object_t *p_this )
  206. {
  207.     demux_t *p_demux = (demux_t *)p_this;
  208.     lua_close( p_demux->p_sys->L );
  209.     free( p_demux->p_sys->psz_filename );
  210.     free( p_demux->p_sys );
  211. }
  212. static int Demux( demux_t *p_demux )
  213. {
  214.     lua_State *L = p_demux->p_sys->L;
  215.     char *psz_filename = p_demux->p_sys->psz_filename;
  216.     input_thread_t *p_input_thread = (input_thread_t *)
  217.         vlc_object_find( p_demux, VLC_OBJECT_INPUT, FIND_PARENT );
  218.     input_item_t *p_current_input = input_GetItem( p_input_thread );
  219.     playlist_t *p_playlist = pl_Hold( p_demux );
  220.     luaL_register( L, "vlc", p_reg_parse );
  221.     lua_getglobal( L, "parse" );
  222.     if( !lua_isfunction( L, -1 ) )
  223.     {
  224.         msg_Warn( p_demux, "Error while runing script %s, "
  225.                   "function parse() not found", psz_filename );
  226.         pl_Release( p_demux );
  227.         return VLC_EGENERIC;
  228.     }
  229.     if( lua_pcall( L, 0, 1, 0 ) )
  230.     {
  231.         msg_Warn( p_demux, "Error while runing script %s, "
  232.                   "function parse(): %s", psz_filename,
  233.                   lua_tostring( L, lua_gettop( L ) ) );
  234.         pl_Release( p_demux );
  235.         return VLC_EGENERIC;
  236.     }
  237.     if( lua_gettop( L ) )
  238.         vlclua_playlist_add_internal( p_demux, L, p_playlist,
  239.                                       p_current_input, 0 );
  240.     else
  241.         msg_Err( p_demux, "Script went completely foobar" );
  242.     vlc_object_release( p_input_thread );
  243.     vlclua_release_playlist_internal( p_playlist );
  244.     return -1; /* Needed for correct operation of go back */
  245. }
  246. static int Control( demux_t *p_demux, int i_query, va_list args )
  247. {
  248.     VLC_UNUSED(p_demux); VLC_UNUSED(i_query); VLC_UNUSED(args);
  249.     return VLC_EGENERIC;
  250. }