Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in
Toggle navigation
V
vlc
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Redmine
Redmine
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Operations
Operations
Metrics
Environments
Analytics
Analytics
CI / CD
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
videolan
vlc
Commits
350148d3
Commit
350148d3
authored
Jun 04, 2008
by
Antoine Cellerier
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Use filter chain in video output core.
parent
9675983e
Changes
2
Show whitespace changes
Inline
Side-by-side
Showing
2 changed files
with
66 additions
and
197 deletions
+66
-197
include/vlc_vout.h
include/vlc_vout.h
+30
-40
src/video_output/video_output.c
src/video_output/video_output.c
+36
-157
No files found.
include/vlc_vout.h
View file @
350148d3
...
@@ -31,6 +31,7 @@
...
@@ -31,6 +31,7 @@
#define _VLC_VOUT_H_ 1
#define _VLC_VOUT_H_ 1
#include <vlc_es.h>
#include <vlc_es.h>
#include <vlc_filter.h>
/** Description of a planar graphic field */
/** Description of a planar graphic field */
typedef
struct
plane_t
typedef
struct
plane_t
...
@@ -365,9 +366,6 @@ VLC_EXPORT( int, __vout_AllocatePicture,( vlc_object_t *p_this, picture_t *p_pic
...
@@ -365,9 +366,6 @@ VLC_EXPORT( int, __vout_AllocatePicture,( vlc_object_t *p_this, picture_t *p_pic
* @{
* @{
*/
*/
/** Maximum numbers of video filters2 that can be attached to a vout */
#define MAX_VFILTERS 10
/**
/**
* Video output thread descriptor
* Video output thread descriptor
*
*
...
@@ -464,17 +462,9 @@ struct vout_thread_t
...
@@ -464,17 +462,9 @@ struct vout_thread_t
char
*
psz_filter_chain
;
char
*
psz_filter_chain
;
bool
b_filter_change
;
bool
b_filter_change
;
/* Video filter2 chain
/* Video filter2 chain */
* these are handled like in transcode.c
filter_chain_t
*
p_vf2_chain
;
* XXX: we might need to merge the two chains (v1 and v2 filters) */
char
*
psz_vf2
;
char
*
psz_vfilters
[
MAX_VFILTERS
];
config_chain_t
*
p_vfilters_cfg
[
MAX_VFILTERS
];
int
i_vfilters_cfg
;
filter_t
*
pp_vfilters
[
MAX_VFILTERS
];
int
i_vfilters
;
bool
b_vfilter_change
;
/* Misc */
/* Misc */
bool
b_snapshot
;
/**< take one snapshot on the next loop */
bool
b_snapshot
;
/**< take one snapshot on the next loop */
...
...
src/video_output/video_output.c
View file @
350148d3
...
@@ -84,25 +84,16 @@ static int VideoFilter2Callback( vlc_object_t *, char const *,
...
@@ -84,25 +84,16 @@ static int VideoFilter2Callback( vlc_object_t *, char const *,
/* From vout_intf.c */
/* From vout_intf.c */
int
vout_Snapshot
(
vout_thread_t
*
,
picture_t
*
);
int
vout_Snapshot
(
vout_thread_t
*
,
picture_t
*
);
/* Video filter2 parsing */
static
int
ParseVideoFilter2Chain
(
vout_thread_t
*
,
char
*
);
static
void
RemoveVideoFilters2
(
vout_thread_t
*
p_vout
);
/* Display media title in OSD */
/* Display media title in OSD */
static
void
DisplayTitleOnOSD
(
vout_thread_t
*
p_vout
);
static
void
DisplayTitleOnOSD
(
vout_thread_t
*
p_vout
);
/*****************************************************************************
/*****************************************************************************
* Video Filter2 functions
* Video Filter2 functions
*****************************************************************************/
*****************************************************************************/
struct
filter_owner_sys_t
{
vout_thread_t
*
p_vout
;
};
static
picture_t
*
video_new_buffer_filter
(
filter_t
*
p_filter
)
static
picture_t
*
video_new_buffer_filter
(
filter_t
*
p_filter
)
{
{
picture_t
*
p_picture
;
picture_t
*
p_picture
;
vout_thread_t
*
p_vout
=
p_filter
->
p_owner
->
p_vout
;
vout_thread_t
*
p_vout
=
(
vout_thread_t
*
)
p_filter
->
p_owner
;
p_picture
=
vout_CreatePicture
(
p_vout
,
0
,
0
,
0
);
p_picture
=
vout_CreatePicture
(
p_vout
,
0
,
0
,
0
);
...
@@ -111,7 +102,15 @@ static picture_t *video_new_buffer_filter( filter_t *p_filter )
...
@@ -111,7 +102,15 @@ static picture_t *video_new_buffer_filter( filter_t *p_filter )
static
void
video_del_buffer_filter
(
filter_t
*
p_filter
,
picture_t
*
p_pic
)
static
void
video_del_buffer_filter
(
filter_t
*
p_filter
,
picture_t
*
p_pic
)
{
{
vout_DestroyPicture
(
p_filter
->
p_owner
->
p_vout
,
p_pic
);
vout_DestroyPicture
(
(
vout_thread_t
*
)
p_filter
->
p_owner
,
p_pic
);
}
static
int
video_filter_buffer_allocation_init
(
filter_t
*
p_filter
,
void
*
p_data
)
{
p_filter
->
pf_vout_buffer_new
=
video_new_buffer_filter
;
p_filter
->
pf_vout_buffer_del
=
video_del_buffer_filter
;
p_filter
->
p_owner
=
p_data
;
/* p_vout */
return
VLC_SUCCESS
;
}
}
/*****************************************************************************
/*****************************************************************************
...
@@ -342,16 +341,12 @@ vout_thread_t * __vout_Create( vlc_object_t *p_parent, video_format_t *p_fmt )
...
@@ -342,16 +341,12 @@ vout_thread_t * __vout_Create( vlc_object_t *p_parent, video_format_t *p_fmt )
if
(
p_parent
->
i_object_type
!=
VLC_OBJECT_VOUT
)
if
(
p_parent
->
i_object_type
!=
VLC_OBJECT_VOUT
)
{
{
/* Look for the default filter configuration */
/* Look for the default filter configuration */
var_Create
(
p_vout
,
"vout-filter"
,
VLC_VAR_STRING
|
VLC_VAR_DOINHERIT
);
p_vout
->
psz_filter_chain
=
var_Get
(
p_vout
,
"vout-filter"
,
&
val
);
var_CreateGetStringCommand
(
p_vout
,
"vout-filter"
);
p_vout
->
psz_filter_chain
=
val
.
psz_string
;
/* Apply video filter2 objects on the first vout */
/* Apply video filter2 objects on the first vout */
var_Create
(
p_vout
,
"video-filter"
,
p_vout
->
psz_vf2
=
VLC_VAR_STRING
|
VLC_VAR_DOINHERIT
);
var_CreateGetStringCommand
(
p_vout
,
"video-filter"
);
var_Get
(
p_vout
,
"video-filter"
,
&
val
);
ParseVideoFilter2Chain
(
p_vout
,
val
.
psz_string
);
free
(
val
.
psz_string
);
}
}
else
else
{
{
...
@@ -367,13 +362,14 @@ vout_thread_t * __vout_Create( vlc_object_t *p_parent, video_format_t *p_fmt )
...
@@ -367,13 +362,14 @@ vout_thread_t * __vout_Create( vlc_object_t *p_parent, video_format_t *p_fmt )
free
(
psz_tmp
);
free
(
psz_tmp
);
/* Create a video filter2 var ... but don't inherit values */
/* Create a video filter2 var ... but don't inherit values */
var_Create
(
p_vout
,
"video-filter"
,
VLC_VAR_STRING
);
var_Create
(
p_vout
,
"video-filter"
,
ParseVideoFilter2Chain
(
p_vout
,
NULL
);
VLC_VAR_STRING
|
VLC_VAR_ISCOMMAND
);
p_vout
->
psz_vf2
=
var_GetString
(
p_vout
,
"video-filter"
);
}
}
var_AddCallback
(
p_vout
,
"video-filter"
,
VideoFilter2Callback
,
NULL
);
var_AddCallback
(
p_vout
,
"video-filter"
,
VideoFilter2Callback
,
NULL
);
p_vout
->
b_vfilter_change
=
true
;
p_vout
->
p_vf2_chain
=
filter_chain_New
(
p_vout
,
"video filter2"
,
p_vout
->
i_vfilters
=
0
;
false
,
video_filter_buffer_allocation_init
,
NULL
,
p_vout
)
;
/* Choose the video output module */
/* Choose the video output module */
if
(
!
p_vout
->
psz_filter_chain
||
!*
p_vout
->
psz_filter_chain
)
if
(
!
p_vout
->
psz_filter_chain
||
!*
p_vout
->
psz_filter_chain
)
...
@@ -956,90 +952,32 @@ static void RunThread( vout_thread_t *p_vout)
...
@@ -956,90 +952,32 @@ static void RunThread( vout_thread_t *p_vout)
}
}
/* Video Filter2 stuff */
/* Video Filter2 stuff */
if
(
p_vout
->
b_vfilter_change
==
true
)
if
(
p_vout
->
psz_vf2
)
{
{
int
i
;
es_format_t
fmt
;
vlc_mutex_lock
(
&
p_vout
->
vfilter_lock
);
RemoveVideoFilters2
(
p_vout
);
for
(
i
=
0
;
i
<
p_vout
->
i_vfilters_cfg
;
i
++
)
{
filter_t
*
p_vfilter
=
p_vout
->
pp_vfilters
[
p_vout
->
i_vfilters
]
=
vlc_object_create
(
p_vout
,
VLC_OBJECT_FILTER
);
vlc_object_attach
(
p_vfilter
,
p_vout
);
p_vfilter
->
pf_vout_buffer_new
=
video_new_buffer_filter
;
vlc_mutex_lock
(
&
p_vout
->
vfilter_lock
);
p_vfilter
->
pf_vout_buffer_del
=
video_del_buffer_filter
;
if
(
!
p_vout
->
i_vfilters
)
es_format_Init
(
&
fmt
,
VIDEO_ES
,
p_vout
->
fmt_render
.
i_chroma
);
{
fmt
.
video
=
p_vout
->
fmt_render
;
p_vfilter
->
fmt_in
.
video
=
p_vout
->
fmt_render
;
filter_chain_Reset
(
p_vout
->
p_vf2_chain
,
&
fmt
,
&
fmt
);
}
else
{
p_vfilter
->
fmt_in
.
video
=
(
p_vfilter
-
1
)
->
fmt_out
.
video
;
}
/* TODO: one day filters in the middle of the chain might
* have a different fmt_out.video than fmt_render ... */
p_vfilter
->
fmt_out
.
video
=
p_vout
->
fmt_render
;
p_vfilter
->
p_cfg
=
p_vout
->
p_vfilters_cfg
[
i
];
if
(
filter_chain_AppendFromString
(
p_vout
->
p_vf2_chain
,
p_vfilter
->
p_module
=
module_Need
(
p_vfilter
,
"video filter2"
,
p_vout
->
psz_vf2
)
<
0
)
p_vout
->
psz_vfilters
[
i
],
msg_Err
(
p_vout
,
"Video filter chain creation failed"
);
true
);
if
(
p_vfilter
->
p_module
)
{
p_vfilter
->
p_owner
=
malloc
(
sizeof
(
filter_owner_sys_t
)
);
p_vfilter
->
p_owner
->
p_vout
=
p_vout
;
p_vout
->
i_vfilters
++
;
msg_Dbg
(
p_vout
,
"video filter found (%s)"
,
p_vout
->
psz_vfilters
[
i
]
);
}
else
else
{
{
msg_Err
(
p_vout
,
"no video filter found (%s)"
,
free
(
p_vout
->
psz_vf2
);
p_vout
->
psz_vfilters
[
i
]
);
p_vout
->
psz_vf2
=
NULL
;
vlc_object_detach
(
p_vfilter
);
vlc_object_release
(
p_vfilter
);
}
}
}
p_vout
->
b_vfilter_change
=
false
;
vlc_mutex_unlock
(
&
p_vout
->
vfilter_lock
);
vlc_mutex_unlock
(
&
p_vout
->
vfilter_lock
);
}
}
if
(
p_picture
)
if
(
p_picture
)
{
{
int
i
;
p_picture
=
filter_chain_VideoFilter
(
p_vout
->
p_vf2_chain
,
for
(
i
=
0
;
i
<
p_vout
->
i_vfilters
;
i
++
)
p_picture
);
{
picture_t
*
p_old
=
p_picture
;
p_picture
=
p_vout
->
pp_vfilters
[
i
]
->
pf_video_filter
(
p_vout
->
pp_vfilters
[
i
],
p_picture
);
if
(
!
p_picture
)
{
break
;
}
/* FIXME: this is kind of wrong
* if you have 2 or more vfilters and the 2nd breaks,
* on the next loop the 1st one will be applied again */
/* if p_old and p_picture are the same (ie the filter
* worked on the old picture), then following code is
* still alright since i_status gets changed back to
* the right value */
if
(
p_old
->
i_refcount
)
{
p_old
->
i_status
=
DISPLAYED_PICTURE
;
}
else
{
p_old
->
i_status
=
DESTROYED_PICTURE
;
}
p_picture
->
i_status
=
READY_PICTURE
;
}
}
}
if
(
p_picture
&&
p_vout
->
b_snapshot
)
if
(
p_picture
&&
p_vout
->
b_snapshot
)
...
@@ -1293,7 +1231,7 @@ static void EndThread( vout_thread_t *p_vout )
...
@@ -1293,7 +1231,7 @@ static void EndThread( vout_thread_t *p_vout )
spu_Destroy
(
p_vout
->
p_spu
);
spu_Destroy
(
p_vout
->
p_spu
);
/* Destroy the video filters2 */
/* Destroy the video filters2 */
RemoveVideoFilters2
(
p_vout
);
filter_chain_Delete
(
p_vout
->
p_vf2_chain
);
/* Destroy translation tables */
/* Destroy translation tables */
p_vout
->
pf_end
(
p_vout
);
p_vout
->
pf_end
(
p_vout
);
...
@@ -1550,46 +1488,6 @@ static int FilterCallback( vlc_object_t *p_this, char const *psz_cmd,
...
@@ -1550,46 +1488,6 @@ static int FilterCallback( vlc_object_t *p_this, char const *psz_cmd,
/*****************************************************************************
/*****************************************************************************
* Video Filter2 stuff
* Video Filter2 stuff
*****************************************************************************/
*****************************************************************************/
static
int
ParseVideoFilter2Chain
(
vout_thread_t
*
p_vout
,
char
*
psz_vfilters
)
{
int
i
;
for
(
i
=
0
;
i
<
p_vout
->
i_vfilters_cfg
;
i
++
)
{
struct
config_chain_t
*
p_cfg
=
p_vout
->
p_vfilters_cfg
[
p_vout
->
i_vfilters_cfg
];
config_ChainDestroy
(
p_cfg
);
free
(
p_vout
->
psz_vfilters
[
p_vout
->
i_vfilters_cfg
]
);
p_vout
->
psz_vfilters
[
p_vout
->
i_vfilters_cfg
]
=
NULL
;
}
p_vout
->
i_vfilters_cfg
=
0
;
if
(
psz_vfilters
&&
*
psz_vfilters
)
{
char
*
psz_parser
=
psz_vfilters
;
while
(
psz_parser
&&
*
psz_parser
)
{
psz_parser
=
config_ChainCreate
(
&
p_vout
->
psz_vfilters
[
p_vout
->
i_vfilters_cfg
],
&
p_vout
->
p_vfilters_cfg
[
p_vout
->
i_vfilters_cfg
],
psz_parser
);
msg_Dbg
(
p_vout
,
"adding vfilter: %s"
,
p_vout
->
psz_vfilters
[
p_vout
->
i_vfilters_cfg
]
);
p_vout
->
i_vfilters_cfg
++
;
if
(
psz_parser
&&
*
psz_parser
)
{
if
(
p_vout
->
i_vfilters_cfg
==
MAX_VFILTERS
)
{
msg_Warn
(
p_vout
,
"maximum number of video filters reached.
\"
%s
\"
discarded"
,
psz_parser
);
break
;
}
}
}
}
return
VLC_SUCCESS
;
}
static
int
VideoFilter2Callback
(
vlc_object_t
*
p_this
,
char
const
*
psz_cmd
,
static
int
VideoFilter2Callback
(
vlc_object_t
*
p_this
,
char
const
*
psz_cmd
,
vlc_value_t
oldval
,
vlc_value_t
newval
,
void
*
p_data
)
vlc_value_t
oldval
,
vlc_value_t
newval
,
void
*
p_data
)
{
{
...
@@ -1597,31 +1495,12 @@ static int VideoFilter2Callback( vlc_object_t *p_this, char const *psz_cmd,
...
@@ -1597,31 +1495,12 @@ static int VideoFilter2Callback( vlc_object_t *p_this, char const *psz_cmd,
(
void
)
psz_cmd
;
(
void
)
oldval
;
(
void
)
p_data
;
(
void
)
psz_cmd
;
(
void
)
oldval
;
(
void
)
p_data
;
vlc_mutex_lock
(
&
p_vout
->
vfilter_lock
);
vlc_mutex_lock
(
&
p_vout
->
vfilter_lock
);
ParseVideoFilter2Chain
(
p_vout
,
newval
.
psz_string
);
p_vout
->
psz_vf2
=
strdup
(
newval
.
psz_string
);
p_vout
->
b_vfilter_change
=
true
;
vlc_mutex_unlock
(
&
p_vout
->
vfilter_lock
);
vlc_mutex_unlock
(
&
p_vout
->
vfilter_lock
);
return
VLC_SUCCESS
;
return
VLC_SUCCESS
;
}
}
static
void
RemoveVideoFilters2
(
vout_thread_t
*
p_vout
)
{
int
i
;
for
(
i
=
0
;
i
<
p_vout
->
i_vfilters
;
i
++
)
{
vlc_object_detach
(
p_vout
->
pp_vfilters
[
i
]
);
if
(
p_vout
->
pp_vfilters
[
i
]
->
p_module
)
{
module_Unneed
(
p_vout
->
pp_vfilters
[
i
],
p_vout
->
pp_vfilters
[
i
]
->
p_module
);
}
free
(
p_vout
->
pp_vfilters
[
i
]
->
p_owner
);
vlc_object_release
(
p_vout
->
pp_vfilters
[
i
]
);
}
p_vout
->
i_vfilters
=
0
;
}
static
void
DisplayTitleOnOSD
(
vout_thread_t
*
p_vout
)
static
void
DisplayTitleOnOSD
(
vout_thread_t
*
p_vout
)
{
{
input_thread_t
*
p_input
;
input_thread_t
*
p_input
;
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment