Difference between revisions of "Hacker Guide/Interfaces"

From VideoLAN Wiki
Jump to navigation Jump to search
 
(20 intermediate revisions by 6 users not shown)
Line 1: Line 1:
=VLC Interface=
+
{{Back to|Hacker Guide}}
 +
==VLC Interface==
  
==A typical VLC run course==
+
===A typical VLC run course===
  
This section describes what happens when you launch the vlc program. After the ELF dynamic loader blah blah blah, the main thread becomes the interface thread. It passes through the following steps :
+
This section describes what happens when you launch the vlc program. After the ELF dynamic loader blah blah blah, the main thread becomes the interface thread. It passes through the following steps:
  
#CPU detection : which CPU are we running on, what are its capabilities (MMX, MMXEXT, 3DNow, AltiVec...) ?
+
# CPU detection: which CPU are we running on, what are its capabilities (MMX, MMXEXT, 3DNow, AltiVec...) ?
#Message interface initialization ;
+
# Message interface initialization
#Command line options parsing ;
+
# Command line options parsing
#Playlist creation ;
+
# Playlist creation
#Module bank initialization ;
+
# Module bank initialization
#Interface opening ;
+
# Interface opening
#Signal handler installation : SIGHUP, SIGINT and SIGQUIT are caught to manage a clean quit (please note that the SDL library also catches SIGSEGV) ;
+
# Signal handler installation: <code>SIGHUP</code>, <code>SIGINT</code> and <code>SIGQUIT</code> are caught to manage a clean quit (please note that the SDL library also catches <code>SIGSEGV</code>)
#Audio output thread spawning ;
+
# Audio output thread spawning
#Video output thread spawning ;
+
# Video output thread spawning
#Main loop : events management ;
+
# Main loop: events management  
  
Following sections describe each of these steps in particular, and many more.
+
The following sections describe each of these steps in particular, and many more.
  
==The message interface==
+
===The message interface===
  
It is a know fact that printf() functions are not necessarily thread-safe. As a result, one thread interrupted in a printf() call, followed by another calls to it, will leave the program in an undetermined state. So an API must be set up to print messages without crashing.
+
It is a known fact that <code>printf()</code> functions are not necessarily thread-safe. As a result, one thread interrupted in a <code>printf()</code> call, followed by another call to it, will leave the program in an undetermined state. So an {{API}} must be set up to print messages without crashing.
  
This API is implemented in two ways. If INTF_MSG_QUEUE is defined in config.h , every printf-like (see below) call will queue the message into a chained list. This list will be printed and flushed by the interface thread once upon an event loop. If INTF_MSG_QUEUE is undefined, the calling thread will acquire the print lock (which prevents two print operations to occur at the same time) and print the message directly (default behaviour).
+
This API is implemented in two ways. If <var>INTF_MSG_QUEUE</var> is defined in <code>config.h</code>, every printf-like (see below) call will queue the message into a chained list. This list will be printed and flushed by the interface thread once upon an event loop. If <var>INTF_MSG_QUEUE</var> is undefined, the calling thread will acquire the print lock (which prevents two print operations to occur at the same time) and print the message directly (default behaviour).
  
Functions available to print messages are :
+
Functions available to print messages are:
  
*msg_Info ( p_this, ... ) : Print a message to stdout , plain and stupid (for instance "it works!").
+
; <code>msg_Info ( p_this, ... )</code>
*msg_Err( p_this, ... ) : Print an error message to stderr .
+
: Print a message to stdout, plain and stupid (for instance "it works!").
*msg_Warn( p_this, ... ) : Print a message to stderr if the warning level (determined by -v, -vv and -vvv) is low enough. ''Please note that the lower the level, the less important the message is.''
+
; <code>msg_Err( p_this, ... )</code>
*msg_Dbg( p_this, ... ) : This function is designed for optional checkpoint messages, such as "we are now entering function dvd_foo_thingy". It does  
+
: Print an error message to stderr.
nothing in non-trace mode. If the VLC is compiled with --enable-trace, the message is either written to the file vlc-trace.log (if TRACE_LOG is defined in config.h), or printed to stderr (otherwise).
+
; <code>msg_Warn( p_this, ... )</code>
*msg_Flush(p_this) : Flush the message queue, if it is in use.
+
: Print a message to stderr if the warning level (determined by <code>-v</code>, <code>-vv</code> and <code>-vvv</code>) is low enough. ''Please note that the lower the level, the less important the message is.''
 +
; <code>msg_Dbg( p_this, ... )</code>
 +
: This function is designed for optional checkpoint messages, such as "we are now entering function dvd_foo_thingy". It does nothing in non-trace mode. If VLC is compiled with <code>--enable-trace</code> (and <var>TRACE_LOG</var> is defined in <code>config.h</code>), the message will be written to the file <code>vlc-trace.log</code>; else the message will be printed on stderr.
 +
; <code>msg_Flush(p_this)</code>
 +
: Flush the message queue, if it is in use.
  
==Command line options==
+
===Command line options===
  
VLC uses GNU getopt to parse command line options. getopt structures are defined in src/extras/getopt.c and command line parsing is done in src/config/cmdline.c
+
VLC uses GNU getopt to parse command line options. getopt structures are defined in {{VLCSourceFile|src/config/getopt.c}} and command line parsing is done in {{VLCSourceFile|src/config/cmdline.c}}.
  
Most configuration directives are exchanged via the environment array, using main_Put*Variable and main_Get*Variable. As a result, ./vlc --height 240 is strictly equivalent to : vlc_height=240 ./vlc. That way configuration variables are available everywhere, including plugins.
+
Most configuration directives are exchanged via the environment array, using <code>main_Put*Variable</code> and <code>main_Get*Variable</code>. As a result, <code>./vlc --height 240</code> is strictly equivalent to: <code>vlc_height=240 ./vlc</code>. That way configuration variables are available everywhere, including plugins.
  
===Warning===
+
====Warning====
 +
Please note that for thread-safety issues, you should not use <code>main_Put*Variable</code> once the second thread has been spawned.
  
Please note that for thread-safety issues, you should not use main_Put*Variable once the second thread has been spawned
+
===Playlist management===
 +
The playlist is created on startup from files given on the command line. An appropriate interface plugin can then add or remove files from it. Functions to be used are described in {{VLCSourceFile|src/playlist/control.c}}.
  
==Playlist management==
+
How to write a plugin is described in the latter sections. Other threads can request a plugin descriptor with:
 +
module_Need ( module_bank_t * p_bank, int i_capabilities, void * p_data )
 +
<code>p_data</code> is an optional parameter (reserved for future use) for the <code>pf_probe()</code> function. The returned <code>module_t</code> structure contains pointers to the functions of the plug-in.<br />
 +
See {{VLCSourceFile|include/vlc_modules.h}} for more information.
  
The playlist is created on startup from files given in the command line. An appropriate interface plugin can then add or remove files from it. Functions to be used are described in src/playlist/control.c.
+
===The interface main loop===
  
==Module bank==
+
The interface thread will first look for a suitable interface plugin. Then it enters the main interface loop, with the plugin's <code>pf_run function</code>. This function will do what's appropriate, and every 100 ms will call (typically via a GUI timer callback) <code>InteractionManage</code>.
  
On startup, VLC creates a bank of all available .so files (plugins) in ., ./lib, /usr/local/lib/videolan/vlc (PLUGIN_PATH), and built-in plugins. Every plugin is checked with its capability, which is set by a call to set_capability(void* capability, int score). The different capabilities are:
+
InteractionManage cleans up the module bank by unloading unnecessary modules, manages the playlist, and flushes waiting messages (if the message queue is in use).
  
*interface : An interface plugin (in modules/control and modules/gui)
+
===How to write an interface module===
*access :
+
====API for the Module====
*access_filter :
+
Have a look at the files in directories {{VLCSourceFolder|modules/control}}, {{VLCSourceFolder|modules/access}}, or {{VLCSourceFolder|modules/gui}}. However the GUIs are not very easy to understand, since they are quite big. I suggest to start digging into a non-graphical interface modules first. For example {{VLCSourceFile|modules/control/hotkeys.c}}.
*sout access : stream output
 
*sout stream :
 
*audio filter :
 
*video filter :
 
*video filter2 :
 
*audio mixer :
 
*demux : démultiplexeur
 
*packetizer :
 
*meta reader : meta engine
 
*sout mux : multiplexeur streaming output
 
*services_discovery :
 
*chroma :
 
*decoder :  decoder. Pour définir un decoder audio ou video, utiliser set_subcategory( SUBCAT_INPUT_ACODEC ) ou set_subcategory( SUBCAT_INPUT_VCODEC )
 
*audio output : An audio output module
 
*video output : A video output module
 
*vizualisation :
 
*packetizer :
 
 
 
How to write a plugin is described in the latter sections. Other threads can request a plugin descriptor with module_Need ( module_bank_t * p_bank, int i_capabilities, void * p_data ). p_data is an optional parameter (reserved for future use) for the pf_probe() function. The returned module_t structure contains pointers to the functions of the plug-in. See include/modules.h for more information.
 
 
 
==The interface main loop==
 
 
 
The interface thread will first look for a suitable interface plugin. Then it enters the main interface loop, with the plugin's pf_run function. This function will do what's appropriate, and every 100 ms will call (typically via a GUI timer callback) intf_Manage.
 
 
 
intf_Manage cleans up the module bank by unloading unnecessary modules, manages the playlist, and flushes waiting messages (if the message queue is in use).
 
 
 
==How to write an interface plugin==
 
===API for the Module===
 
 
 
Have a look the files in directories modules/misc/control, modules/misc/dummy, modules/misc/access, or modules/gui. However the GUI interfaces are not very easy to understand, since they are quite big. I suggest to start digging into a non-graphical interface modules first. For example modules/control/hotkeys.c.
 
  
 
An interface module is made of 3 entry functions and a module description:
 
An interface module is made of 3 entry functions and a module description:
  
    *The module description is made of macros that declares the capabilities of the module (interface, in this case) with their priority, the module description as it will appear in the preferences of GUI modules that implement them, some configuration variables specific to the module, shortcuts, sub-modules, etc.
+
The module description is made of macros that declares the capabilities of the module (interface, in this case) with their priority, the module description as it will appear in the preferences of GUI modules that implement them, some configuration variables specific to the module, shortcuts, sub-modules, etc.
    *Open ( vlc_object_t* p_object ): This is called by VLC to initialize the module.
 
    *Run ( vlc_object_t* p_object ): really does the job of the interface module (waiting for user input and displaying info). It should check periodically that p_intf->b_die is not VLC_TRUE.
 
    *Close ( vcl_object_t * p_object ) function is called by VLC to uninitialize the module (basically, this consists in destroying whatever have been allocated by Open)
 
 
 
The above functions take a vlc_object_t* as argument, but that may need to be cast into a intf_thread_t* depending on your needs. This structure is often needed as a parameter for exported VLC functions, such as msg_Err(), msg_Warn(), ...
 
 
 
Define intf_sys_t to contain any variable you need (don't use static variables, they suck in a multi-threaded application :-).
 
 
 
If additional capabilities (such as Open button, playlist, menus, etc.) are needed, consult one of the GUI modules. One of the simpler GUI modules to consult might be modules/gui/ncurses/ncurses.c. It is a quite simple complete interface module with playlist interaction, and progress bar, among other things.
 
 
 
===Arranging for your Module to get Compiled===
 
 
 
If you create a new directory for your module, add a Modules.am file in it. In this file, put something like : SOURCES_yourmodule = myfile1.c myfile2.c
 
 
 
Then go to the main configure.ac file, and add in the AC_CONFIG_FILES section (towards the end of the file) a line similar to the others.
 
 
 
If you don't create a directory for your plugin (but instead just put it in an existing directory), you only have to add the two SOURCES_... lines to the existing Modules.am file
 
 
 
This declares your module; it does not arrange for it to be automatically compiled; automatic compilatoin is described further below.
 
  
You do not write a Makefile for your module. Instead this is done via the bootstrap and configuration process. So now run:
+
; <code>Open ( vlc_object_t* p_object )</code>
 +
: This is called by VLC to initialize the module.
 +
; <code>Run ( vlc_object_t* p_object )</code>
 +
: Really does the job of the interface module (waiting for user input and displaying info). It should check periodically that <code>p_intf->b_die</code> is not <code>VLC_TRUE</code>.
 +
; <code>Close ( vcl_object_t * p_object )</code>
 +
: This function is called by VLC to uninitialize the module (basically, this consists in destroying whatever have been allocated by Open)
  
./bootstrap
+
The above functions take a <code>vlc_object_t*</code> as argument, but that may need to be cast into a <code>intf_thread_t*</code> depending on your needs. This structure is often needed as a parameter for exported VLC functions, such as <code>msg_Err()</code>, <code>msg_Warn()</code>, ...
  
./configure configure-options
+
Define <code>intf_sys_t</code> to contain any variable you need (don't use static variables, they suck in a multi-threaded application <span title="smile">:-)</span>.
  
make
+
If additional capabilities (such as Open button, playlist, menus, etc.) are needed, consult one of the GUI modules. One of the simpler GUI modules to consult might be {{VLCSourceFile|modules/gui/ncurses.c}}. It is a quite simple complete interface module with playlist interaction, and progress bar, among other things.
  
To build the module manually, go to the directory it resides and type make libyourmodule_plugin.so (or .dll, or whatever the file type for a shared library is on your Operating System.)
+
{{Hacker Guide}}
  
To automatically have your module get built, you also set this in the configure.ac file; add your module name to the default modules section in one of the AX_ADD_PLUGINS directives.
+
[[Category:Interfaces|*]]

Latest revision as of 21:26, 30 May 2019

← Back to Hacker Guide

VLC Interface

A typical VLC run course

This section describes what happens when you launch the vlc program. After the ELF dynamic loader blah blah blah, the main thread becomes the interface thread. It passes through the following steps:

  1. CPU detection: which CPU are we running on, what are its capabilities (MMX, MMXEXT, 3DNow, AltiVec...) ?
  2. Message interface initialization
  3. Command line options parsing
  4. Playlist creation
  5. Module bank initialization
  6. Interface opening
  7. Signal handler installation: SIGHUP, SIGINT and SIGQUIT are caught to manage a clean quit (please note that the SDL library also catches SIGSEGV)
  8. Audio output thread spawning
  9. Video output thread spawning
  10. Main loop: events management

The following sections describe each of these steps in particular, and many more.

The message interface

It is a known fact that printf() functions are not necessarily thread-safe. As a result, one thread interrupted in a printf() call, followed by another call to it, will leave the program in an undetermined state. So an API must be set up to print messages without crashing.

This API is implemented in two ways. If INTF_MSG_QUEUE is defined in config.h, every printf-like (see below) call will queue the message into a chained list. This list will be printed and flushed by the interface thread once upon an event loop. If INTF_MSG_QUEUE is undefined, the calling thread will acquire the print lock (which prevents two print operations to occur at the same time) and print the message directly (default behaviour).

Functions available to print messages are:

msg_Info ( p_this, ... )
Print a message to stdout, plain and stupid (for instance "it works!").
msg_Err( p_this, ... )
Print an error message to stderr.
msg_Warn( p_this, ... )
Print a message to stderr if the warning level (determined by -v, -vv and -vvv) is low enough. Please note that the lower the level, the less important the message is.
msg_Dbg( p_this, ... )
This function is designed for optional checkpoint messages, such as "we are now entering function dvd_foo_thingy". It does nothing in non-trace mode. If VLC is compiled with --enable-trace (and TRACE_LOG is defined in config.h), the message will be written to the file vlc-trace.log; else the message will be printed on stderr.
msg_Flush(p_this)
Flush the message queue, if it is in use.

Command line options

VLC uses GNU getopt to parse command line options. getopt structures are defined in src/config/getopt.c and command line parsing is done in src/config/cmdline.c.

Most configuration directives are exchanged via the environment array, using main_Put*Variable and main_Get*Variable. As a result, ./vlc --height 240 is strictly equivalent to: vlc_height=240 ./vlc. That way configuration variables are available everywhere, including plugins.

Warning

Please note that for thread-safety issues, you should not use main_Put*Variable once the second thread has been spawned.

Playlist management

The playlist is created on startup from files given on the command line. An appropriate interface plugin can then add or remove files from it. Functions to be used are described in src/playlist/control.c.

How to write a plugin is described in the latter sections. Other threads can request a plugin descriptor with:

module_Need ( module_bank_t * p_bank, int i_capabilities, void * p_data )

p_data is an optional parameter (reserved for future use) for the pf_probe() function. The returned module_t structure contains pointers to the functions of the plug-in.
See include/vlc_modules.h for more information.

The interface main loop

The interface thread will first look for a suitable interface plugin. Then it enters the main interface loop, with the plugin's pf_run function. This function will do what's appropriate, and every 100 ms will call (typically via a GUI timer callback) InteractionManage.

InteractionManage cleans up the module bank by unloading unnecessary modules, manages the playlist, and flushes waiting messages (if the message queue is in use).

How to write an interface module

API for the Module

Have a look at the files in directories modules/control, modules/access, or modules/gui. However the GUIs are not very easy to understand, since they are quite big. I suggest to start digging into a non-graphical interface modules first. For example modules/control/hotkeys.c.

An interface module is made of 3 entry functions and a module description:

The module description is made of macros that declares the capabilities of the module (interface, in this case) with their priority, the module description as it will appear in the preferences of GUI modules that implement them, some configuration variables specific to the module, shortcuts, sub-modules, etc.

Open ( vlc_object_t* p_object )
This is called by VLC to initialize the module.
Run ( vlc_object_t* p_object )
Really does the job of the interface module (waiting for user input and displaying info). It should check periodically that p_intf->b_die is not VLC_TRUE.
Close ( vcl_object_t * p_object )
This function is called by VLC to uninitialize the module (basically, this consists in destroying whatever have been allocated by Open)

The above functions take a vlc_object_t* as argument, but that may need to be cast into a intf_thread_t* depending on your needs. This structure is often needed as a parameter for exported VLC functions, such as msg_Err(), msg_Warn(), ...

Define intf_sys_t to contain any variable you need (don't use static variables, they suck in a multi-threaded application :-).

If additional capabilities (such as Open button, playlist, menus, etc.) are needed, consult one of the GUI modules. One of the simpler GUI modules to consult might be modules/gui/ncurses.c. It is a quite simple complete interface module with playlist interaction, and progress bar, among other things.

This page is part of official VLC media player Documentation (User GuideStreaming HowToHacker GuideModules)
Please read the Documentation Editing Guidelines before you edit the documentation
Permission is granted to copy, distribute and/or modify this document 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.