Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Contribute to GitLab
Sign in / Register
Toggle navigation
F
ffmpeg.wasm-core
Project
Project
Details
Activity
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Board
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
Linshizhi
ffmpeg.wasm-core
Commits
d54014d1
Commit
d54014d1
authored
Apr 27, 2018
by
Paul B Mahol
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
avfilter/vf_overlay: add slice threading
Signed-off-by:
Paul B Mahol
<
onemda@gmail.com
>
parent
1322b000
Show whitespace changes
Inline
Side-by-side
Showing
1 changed file
with
191 additions
and
91 deletions
+191
-91
vf_overlay.c
libavfilter/vf_overlay.c
+191
-91
No files found.
libavfilter/vf_overlay.c
View file @
d54014d1
...
...
@@ -40,6 +40,10 @@
#include "framesync.h"
#include "video.h"
typedef
struct
ThreadData
{
AVFrame
*
dst
,
*
src
;
}
ThreadData
;
static
const
char
*
const
var_names
[]
=
{
"main_w"
,
"W"
,
///< width of the main video
"main_h"
,
"H"
,
///< height of the main video
...
...
@@ -124,7 +128,7 @@ typedef struct OverlayContext {
AVExpr
*
x_pexpr
,
*
y_pexpr
;
void
(
*
blend_image
)(
AVFilterContext
*
ctx
,
AVFrame
*
dst
,
const
AVFrame
*
src
,
int
x
,
int
y
);
int
(
*
blend_slice
)(
AVFilterContext
*
ctx
,
void
*
arg
,
int
jobnr
,
int
nb_jobs
);
}
OverlayContext
;
static
av_cold
void
uninit
(
AVFilterContext
*
ctx
)
...
...
@@ -403,10 +407,10 @@ static int config_output(AVFilterLink *outlink)
* Blend image in src to destination buffer dst at position (x, y).
*/
static
av_always_inline
void
blend_
imag
e_packed_rgb
(
AVFilterContext
*
ctx
,
static
av_always_inline
void
blend_
slic
e_packed_rgb
(
AVFilterContext
*
ctx
,
AVFrame
*
dst
,
const
AVFrame
*
src
,
int
main_has_alpha
,
int
x
,
int
y
,
int
is_straight
)
int
is_straight
,
int
jobnr
,
int
nb_jobs
)
{
OverlayContext
*
s
=
ctx
->
priv
;
int
i
,
imax
,
j
,
jmax
;
...
...
@@ -425,13 +429,19 @@ static av_always_inline void blend_image_packed_rgb(AVFilterContext *ctx,
const
int
sb
=
s
->
overlay_rgba_map
[
B
];
const
int
sa
=
s
->
overlay_rgba_map
[
A
];
const
int
sstep
=
s
->
overlay_pix_step
[
0
];
int
slice_start
,
slice_end
;
uint8_t
*
S
,
*
sp
,
*
d
,
*
dp
;
i
=
FFMAX
(
-
y
,
0
);
sp
=
src
->
data
[
0
]
+
i
*
src
->
linesize
[
0
];
dp
=
dst
->
data
[
0
]
+
(
y
+
i
)
*
dst
->
linesize
[
0
];
imax
=
FFMIN
(
-
y
+
dst_h
,
src_h
);
slice_start
=
(
imax
*
jobnr
)
/
nb_jobs
;
slice_end
=
(
imax
*
(
jobnr
+
1
))
/
nb_jobs
;
for
(
imax
=
FFMIN
(
-
y
+
dst_h
,
src_h
);
i
<
imax
;
i
++
)
{
sp
=
src
->
data
[
0
]
+
(
i
+
slice_start
)
*
src
->
linesize
[
0
];
dp
=
dst
->
data
[
0
]
+
(
y
+
i
+
slice_start
)
*
dst
->
linesize
[
0
];
for
(
i
=
i
+
slice_start
;
i
<
slice_end
;
i
++
)
{
j
=
FFMAX
(
-
x
,
0
);
S
=
sp
+
j
*
sstep
;
d
=
dp
+
(
x
+
j
)
*
dstep
;
...
...
@@ -495,7 +505,9 @@ static av_always_inline void blend_plane(AVFilterContext *ctx,
int
dst_offset
,
int
dst_step
,
int
straight
,
int
yuv
)
int
yuv
,
int
jobnr
,
int
nb_jobs
)
{
int
src_wp
=
AV_CEIL_RSHIFT
(
src_w
,
hsub
);
int
src_hp
=
AV_CEIL_RSHIFT
(
src_h
,
vsub
);
...
...
@@ -505,16 +517,22 @@ static av_always_inline void blend_plane(AVFilterContext *ctx,
int
xp
=
x
>>
hsub
;
uint8_t
*
s
,
*
sp
,
*
d
,
*
dp
,
*
dap
,
*
a
,
*
da
,
*
ap
;
int
jmax
,
j
,
k
,
kmax
;
int
slice_start
,
slice_end
;
j
=
FFMAX
(
-
yp
,
0
);
sp
=
src
->
data
[
i
]
+
j
*
src
->
linesize
[
i
];
jmax
=
FFMIN
(
-
yp
+
dst_hp
,
src_hp
);
slice_start
=
(
jmax
*
jobnr
)
/
nb_jobs
;
slice_end
=
(
jmax
*
(
jobnr
+
1
))
/
nb_jobs
;
sp
=
src
->
data
[
i
]
+
slice_start
*
src
->
linesize
[
i
];
dp
=
dst
->
data
[
dst_plane
]
+
(
yp
+
j
)
*
dst
->
linesize
[
dst_plane
]
+
(
yp
+
slice_start
)
*
dst
->
linesize
[
dst_plane
]
+
dst_offset
;
ap
=
src
->
data
[
3
]
+
(
j
<<
vsub
)
*
src
->
linesize
[
3
];
dap
=
dst
->
data
[
3
]
+
((
yp
+
j
)
<<
vsub
)
*
dst
->
linesize
[
3
];
ap
=
src
->
data
[
3
]
+
(
slice_start
<<
vsub
)
*
src
->
linesize
[
3
];
dap
=
dst
->
data
[
3
]
+
((
yp
+
slice_start
)
<<
vsub
)
*
dst
->
linesize
[
3
];
for
(
j
max
=
FFMIN
(
-
yp
+
dst_hp
,
src_hp
);
j
<
jmax
;
j
++
)
{
for
(
j
=
j
+
slice_start
;
j
<
slice_end
;
j
++
)
{
k
=
FFMAX
(
-
xp
,
0
);
d
=
dp
+
(
xp
+
k
)
*
dst_step
;
s
=
sp
+
k
;
...
...
@@ -577,17 +595,23 @@ static av_always_inline void blend_plane(AVFilterContext *ctx,
static
inline
void
alpha_composite
(
const
AVFrame
*
src
,
const
AVFrame
*
dst
,
int
src_w
,
int
src_h
,
int
dst_w
,
int
dst_h
,
int
x
,
int
y
)
int
x
,
int
y
,
int
jobnr
,
int
nb_jobs
)
{
uint8_t
alpha
;
///< the amount of overlay to blend on to main
uint8_t
*
s
,
*
sa
,
*
d
,
*
da
;
int
i
,
imax
,
j
,
jmax
;
int
slice_start
,
slice_end
;
imax
=
FFMIN
(
-
y
+
dst_h
,
src_h
);
slice_start
=
(
imax
*
jobnr
)
/
nb_jobs
;
slice_end
=
((
imax
*
(
jobnr
+
1
))
/
nb_jobs
);
i
=
FFMAX
(
-
y
,
0
);
sa
=
src
->
data
[
3
]
+
i
*
src
->
linesize
[
3
];
da
=
dst
->
data
[
3
]
+
(
y
+
i
)
*
dst
->
linesize
[
3
];
sa
=
src
->
data
[
3
]
+
(
i
+
slice_start
)
*
src
->
linesize
[
3
];
da
=
dst
->
data
[
3
]
+
(
y
+
i
+
slice_start
)
*
dst
->
linesize
[
3
];
for
(
i
max
=
FFMIN
(
-
y
+
dst_h
,
src_h
)
;
i
<
imax
;
i
++
)
{
for
(
i
=
i
+
slice_start
;
i
<
imax
;
i
++
)
{
j
=
FFMAX
(
-
x
,
0
);
s
=
sa
+
j
;
d
=
da
+
x
+
j
;
...
...
@@ -616,12 +640,13 @@ static inline void alpha_composite(const AVFrame *src, const AVFrame *dst,
}
}
static
av_always_inline
void
blend_
imag
e_yuv
(
AVFilterContext
*
ctx
,
static
av_always_inline
void
blend_
slic
e_yuv
(
AVFilterContext
*
ctx
,
AVFrame
*
dst
,
const
AVFrame
*
src
,
int
hsub
,
int
vsub
,
int
main_has_alpha
,
int
x
,
int
y
,
int
is_straight
)
int
is_straight
,
int
jobnr
,
int
nb_jobs
)
{
OverlayContext
*
s
=
ctx
->
priv
;
const
int
src_w
=
src
->
width
;
...
...
@@ -630,22 +655,27 @@ static av_always_inline void blend_image_yuv(AVFilterContext *ctx,
const
int
dst_h
=
dst
->
height
;
blend_plane
(
ctx
,
dst
,
src
,
src_w
,
src_h
,
dst_w
,
dst_h
,
0
,
0
,
0
,
x
,
y
,
main_has_alpha
,
s
->
main_desc
->
comp
[
0
].
plane
,
s
->
main_desc
->
comp
[
0
].
offset
,
s
->
main_desc
->
comp
[
0
].
step
,
is_straight
,
1
);
s
->
main_desc
->
comp
[
0
].
plane
,
s
->
main_desc
->
comp
[
0
].
offset
,
s
->
main_desc
->
comp
[
0
].
step
,
is_straight
,
1
,
jobnr
,
nb_jobs
);
blend_plane
(
ctx
,
dst
,
src
,
src_w
,
src_h
,
dst_w
,
dst_h
,
1
,
hsub
,
vsub
,
x
,
y
,
main_has_alpha
,
s
->
main_desc
->
comp
[
1
].
plane
,
s
->
main_desc
->
comp
[
1
].
offset
,
s
->
main_desc
->
comp
[
1
].
step
,
is_straight
,
1
);
s
->
main_desc
->
comp
[
1
].
plane
,
s
->
main_desc
->
comp
[
1
].
offset
,
s
->
main_desc
->
comp
[
1
].
step
,
is_straight
,
1
,
jobnr
,
nb_jobs
);
blend_plane
(
ctx
,
dst
,
src
,
src_w
,
src_h
,
dst_w
,
dst_h
,
2
,
hsub
,
vsub
,
x
,
y
,
main_has_alpha
,
s
->
main_desc
->
comp
[
2
].
plane
,
s
->
main_desc
->
comp
[
2
].
offset
,
s
->
main_desc
->
comp
[
2
].
step
,
is_straight
,
1
);
s
->
main_desc
->
comp
[
2
].
plane
,
s
->
main_desc
->
comp
[
2
].
offset
,
s
->
main_desc
->
comp
[
2
].
step
,
is_straight
,
1
,
jobnr
,
nb_jobs
);
if
(
main_has_alpha
)
alpha_composite
(
src
,
dst
,
src_w
,
src_h
,
dst_w
,
dst_h
,
x
,
y
);
alpha_composite
(
src
,
dst
,
src_w
,
src_h
,
dst_w
,
dst_h
,
x
,
y
,
jobnr
,
nb_jobs
);
}
static
av_always_inline
void
blend_
imag
e_planar_rgb
(
AVFilterContext
*
ctx
,
static
av_always_inline
void
blend_
slic
e_planar_rgb
(
AVFilterContext
*
ctx
,
AVFrame
*
dst
,
const
AVFrame
*
src
,
int
hsub
,
int
vsub
,
int
main_has_alpha
,
int
x
,
int
y
,
int
is_straight
)
int
is_straight
,
int
jobnr
,
int
nb_jobs
)
{
OverlayContext
*
s
=
ctx
->
priv
;
const
int
src_w
=
src
->
width
;
...
...
@@ -654,114 +684,177 @@ static av_always_inline void blend_image_planar_rgb(AVFilterContext *ctx,
const
int
dst_h
=
dst
->
height
;
blend_plane
(
ctx
,
dst
,
src
,
src_w
,
src_h
,
dst_w
,
dst_h
,
0
,
0
,
0
,
x
,
y
,
main_has_alpha
,
s
->
main_desc
->
comp
[
1
].
plane
,
s
->
main_desc
->
comp
[
1
].
offset
,
s
->
main_desc
->
comp
[
1
].
step
,
is_straight
,
0
);
s
->
main_desc
->
comp
[
1
].
plane
,
s
->
main_desc
->
comp
[
1
].
offset
,
s
->
main_desc
->
comp
[
1
].
step
,
is_straight
,
0
,
jobnr
,
nb_jobs
);
blend_plane
(
ctx
,
dst
,
src
,
src_w
,
src_h
,
dst_w
,
dst_h
,
1
,
hsub
,
vsub
,
x
,
y
,
main_has_alpha
,
s
->
main_desc
->
comp
[
2
].
plane
,
s
->
main_desc
->
comp
[
2
].
offset
,
s
->
main_desc
->
comp
[
2
].
step
,
is_straight
,
0
);
s
->
main_desc
->
comp
[
2
].
plane
,
s
->
main_desc
->
comp
[
2
].
offset
,
s
->
main_desc
->
comp
[
2
].
step
,
is_straight
,
0
,
jobnr
,
nb_jobs
);
blend_plane
(
ctx
,
dst
,
src
,
src_w
,
src_h
,
dst_w
,
dst_h
,
2
,
hsub
,
vsub
,
x
,
y
,
main_has_alpha
,
s
->
main_desc
->
comp
[
0
].
plane
,
s
->
main_desc
->
comp
[
0
].
offset
,
s
->
main_desc
->
comp
[
0
].
step
,
is_straight
,
0
);
s
->
main_desc
->
comp
[
0
].
plane
,
s
->
main_desc
->
comp
[
0
].
offset
,
s
->
main_desc
->
comp
[
0
].
step
,
is_straight
,
0
,
jobnr
,
nb_jobs
);
if
(
main_has_alpha
)
alpha_composite
(
src
,
dst
,
src_w
,
src_h
,
dst_w
,
dst_h
,
x
,
y
);
alpha_composite
(
src
,
dst
,
src_w
,
src_h
,
dst_w
,
dst_h
,
x
,
y
,
jobnr
,
nb_jobs
);
}
static
void
blend_image_yuv420
(
AVFilterContext
*
ctx
,
AVFrame
*
dst
,
const
AVFrame
*
src
,
int
x
,
int
y
)
static
int
blend_slice_yuv420
(
AVFilterContext
*
ctx
,
void
*
arg
,
int
jobnr
,
int
nb_jobs
)
{
blend_image_yuv
(
ctx
,
dst
,
src
,
1
,
1
,
0
,
x
,
y
,
1
);
OverlayContext
*
s
=
ctx
->
priv
;
ThreadData
*
td
=
arg
;
blend_slice_yuv
(
ctx
,
td
->
dst
,
td
->
src
,
1
,
1
,
0
,
s
->
x
,
s
->
y
,
1
,
jobnr
,
nb_jobs
);
return
0
;
}
static
void
blend_image_yuva420
(
AVFilterContext
*
ctx
,
AVFrame
*
dst
,
const
AVFrame
*
src
,
int
x
,
int
y
)
static
int
blend_slice_yuva420
(
AVFilterContext
*
ctx
,
void
*
arg
,
int
jobnr
,
int
nb_jobs
)
{
blend_image_yuv
(
ctx
,
dst
,
src
,
1
,
1
,
1
,
x
,
y
,
1
);
OverlayContext
*
s
=
ctx
->
priv
;
ThreadData
*
td
=
arg
;
blend_slice_yuv
(
ctx
,
td
->
dst
,
td
->
src
,
1
,
1
,
1
,
s
->
x
,
s
->
y
,
1
,
jobnr
,
nb_jobs
);
return
0
;
}
static
void
blend_image_yuv422
(
AVFilterContext
*
ctx
,
AVFrame
*
dst
,
const
AVFrame
*
src
,
int
x
,
int
y
)
static
int
blend_slice_yuv422
(
AVFilterContext
*
ctx
,
void
*
arg
,
int
jobnr
,
int
nb_jobs
)
{
blend_image_yuv
(
ctx
,
dst
,
src
,
1
,
0
,
0
,
x
,
y
,
1
);
OverlayContext
*
s
=
ctx
->
priv
;
ThreadData
*
td
=
arg
;
blend_slice_yuv
(
ctx
,
td
->
dst
,
td
->
src
,
1
,
0
,
0
,
s
->
x
,
s
->
y
,
1
,
jobnr
,
nb_jobs
);
return
0
;
}
static
void
blend_image_yuva422
(
AVFilterContext
*
ctx
,
AVFrame
*
dst
,
const
AVFrame
*
src
,
int
x
,
int
y
)
static
int
blend_slice_yuva422
(
AVFilterContext
*
ctx
,
void
*
arg
,
int
jobnr
,
int
nb_jobs
)
{
blend_image_yuv
(
ctx
,
dst
,
src
,
1
,
0
,
1
,
x
,
y
,
1
);
OverlayContext
*
s
=
ctx
->
priv
;
ThreadData
*
td
=
arg
;
blend_slice_yuv
(
ctx
,
td
->
dst
,
td
->
src
,
1
,
0
,
1
,
s
->
x
,
s
->
y
,
1
,
jobnr
,
nb_jobs
);
return
0
;
}
static
void
blend_image_yuv444
(
AVFilterContext
*
ctx
,
AVFrame
*
dst
,
const
AVFrame
*
src
,
int
x
,
int
y
)
static
int
blend_slice_yuv444
(
AVFilterContext
*
ctx
,
void
*
arg
,
int
jobnr
,
int
nb_jobs
)
{
blend_image_yuv
(
ctx
,
dst
,
src
,
0
,
0
,
0
,
x
,
y
,
1
);
OverlayContext
*
s
=
ctx
->
priv
;
ThreadData
*
td
=
arg
;
blend_slice_yuv
(
ctx
,
td
->
dst
,
td
->
src
,
0
,
0
,
0
,
s
->
x
,
s
->
y
,
1
,
jobnr
,
nb_jobs
);
return
0
;
}
static
void
blend_image_yuva444
(
AVFilterContext
*
ctx
,
AVFrame
*
dst
,
const
AVFrame
*
src
,
int
x
,
int
y
)
static
int
blend_slice_yuva444
(
AVFilterContext
*
ctx
,
void
*
arg
,
int
jobnr
,
int
nb_jobs
)
{
blend_image_yuv
(
ctx
,
dst
,
src
,
0
,
0
,
1
,
x
,
y
,
1
);
OverlayContext
*
s
=
ctx
->
priv
;
ThreadData
*
td
=
arg
;
blend_slice_yuv
(
ctx
,
td
->
dst
,
td
->
src
,
0
,
0
,
1
,
s
->
x
,
s
->
y
,
1
,
jobnr
,
nb_jobs
);
return
0
;
}
static
void
blend_image_gbrp
(
AVFilterContext
*
ctx
,
AVFrame
*
dst
,
const
AVFrame
*
src
,
int
x
,
int
y
)
static
int
blend_slice_gbrp
(
AVFilterContext
*
ctx
,
void
*
arg
,
int
jobnr
,
int
nb_jobs
)
{
blend_image_planar_rgb
(
ctx
,
dst
,
src
,
0
,
0
,
0
,
x
,
y
,
1
);
OverlayContext
*
s
=
ctx
->
priv
;
ThreadData
*
td
=
arg
;
blend_slice_planar_rgb
(
ctx
,
td
->
dst
,
td
->
src
,
0
,
0
,
0
,
s
->
x
,
s
->
y
,
1
,
jobnr
,
nb_jobs
);
return
0
;
}
static
void
blend_image_gbrap
(
AVFilterContext
*
ctx
,
AVFrame
*
dst
,
const
AVFrame
*
src
,
int
x
,
int
y
)
static
int
blend_slice_gbrap
(
AVFilterContext
*
ctx
,
void
*
arg
,
int
jobnr
,
int
nb_jobs
)
{
blend_image_planar_rgb
(
ctx
,
dst
,
src
,
0
,
0
,
1
,
x
,
y
,
1
);
OverlayContext
*
s
=
ctx
->
priv
;
ThreadData
*
td
=
arg
;
blend_slice_planar_rgb
(
ctx
,
td
->
dst
,
td
->
src
,
0
,
0
,
1
,
s
->
x
,
s
->
y
,
1
,
jobnr
,
nb_jobs
);
return
0
;
}
static
void
blend_image_yuv420_pm
(
AVFilterContext
*
ctx
,
AVFrame
*
dst
,
const
AVFrame
*
src
,
int
x
,
int
y
)
static
int
blend_slice_yuv420_pm
(
AVFilterContext
*
ctx
,
void
*
arg
,
int
jobnr
,
int
nb_jobs
)
{
blend_image_yuv
(
ctx
,
dst
,
src
,
1
,
1
,
0
,
x
,
y
,
0
);
OverlayContext
*
s
=
ctx
->
priv
;
ThreadData
*
td
=
arg
;
blend_slice_yuv
(
ctx
,
td
->
dst
,
td
->
src
,
1
,
1
,
0
,
s
->
x
,
s
->
y
,
0
,
jobnr
,
nb_jobs
);
return
0
;
}
static
void
blend_image_yuva420_pm
(
AVFilterContext
*
ctx
,
AVFrame
*
dst
,
const
AVFrame
*
src
,
int
x
,
int
y
)
static
int
blend_slice_yuva420_pm
(
AVFilterContext
*
ctx
,
void
*
arg
,
int
jobnr
,
int
nb_jobs
)
{
blend_image_yuv
(
ctx
,
dst
,
src
,
1
,
1
,
1
,
x
,
y
,
0
);
OverlayContext
*
s
=
ctx
->
priv
;
ThreadData
*
td
=
arg
;
blend_slice_yuv
(
ctx
,
td
->
dst
,
td
->
src
,
1
,
1
,
1
,
s
->
x
,
s
->
y
,
0
,
jobnr
,
nb_jobs
);
return
0
;
}
static
void
blend_image_yuv422_pm
(
AVFilterContext
*
ctx
,
AVFrame
*
dst
,
const
AVFrame
*
src
,
int
x
,
int
y
)
static
int
blend_slice_yuv422_pm
(
AVFilterContext
*
ctx
,
void
*
arg
,
int
jobnr
,
int
nb_jobs
)
{
blend_image_yuv
(
ctx
,
dst
,
src
,
1
,
0
,
0
,
x
,
y
,
0
);
OverlayContext
*
s
=
ctx
->
priv
;
ThreadData
*
td
=
arg
;
blend_slice_yuv
(
ctx
,
td
->
dst
,
td
->
src
,
1
,
0
,
0
,
s
->
x
,
s
->
y
,
0
,
jobnr
,
nb_jobs
);
return
0
;
}
static
void
blend_image_yuva422_pm
(
AVFilterContext
*
ctx
,
AVFrame
*
dst
,
const
AVFrame
*
src
,
int
x
,
int
y
)
static
int
blend_slice_yuva422_pm
(
AVFilterContext
*
ctx
,
void
*
arg
,
int
jobnr
,
int
nb_jobs
)
{
blend_image_yuv
(
ctx
,
dst
,
src
,
1
,
0
,
1
,
x
,
y
,
0
);
OverlayContext
*
s
=
ctx
->
priv
;
ThreadData
*
td
=
arg
;
blend_slice_yuv
(
ctx
,
td
->
dst
,
td
->
src
,
1
,
0
,
1
,
s
->
x
,
s
->
y
,
0
,
jobnr
,
nb_jobs
);
return
0
;
}
static
void
blend_image_yuv444_pm
(
AVFilterContext
*
ctx
,
AVFrame
*
dst
,
const
AVFrame
*
src
,
int
x
,
int
y
)
static
int
blend_slice_yuv444_pm
(
AVFilterContext
*
ctx
,
void
*
arg
,
int
jobnr
,
int
nb_jobs
)
{
blend_image_yuv
(
ctx
,
dst
,
src
,
0
,
0
,
0
,
x
,
y
,
0
);
OverlayContext
*
s
=
ctx
->
priv
;
ThreadData
*
td
=
arg
;
blend_slice_yuv
(
ctx
,
td
->
dst
,
td
->
src
,
0
,
0
,
0
,
s
->
x
,
s
->
y
,
0
,
jobnr
,
nb_jobs
);
return
0
;
}
static
void
blend_image_yuva444_pm
(
AVFilterContext
*
ctx
,
AVFrame
*
dst
,
const
AVFrame
*
src
,
int
x
,
int
y
)
static
int
blend_slice_yuva444_pm
(
AVFilterContext
*
ctx
,
void
*
arg
,
int
jobnr
,
int
nb_jobs
)
{
blend_image_yuv
(
ctx
,
dst
,
src
,
0
,
0
,
1
,
x
,
y
,
0
);
OverlayContext
*
s
=
ctx
->
priv
;
ThreadData
*
td
=
arg
;
blend_slice_yuv
(
ctx
,
td
->
dst
,
td
->
src
,
0
,
0
,
1
,
s
->
x
,
s
->
y
,
0
,
jobnr
,
nb_jobs
);
return
0
;
}
static
void
blend_image_gbrp_pm
(
AVFilterContext
*
ctx
,
AVFrame
*
dst
,
const
AVFrame
*
src
,
int
x
,
int
y
)
static
int
blend_slice_gbrp_pm
(
AVFilterContext
*
ctx
,
void
*
arg
,
int
jobnr
,
int
nb_jobs
)
{
blend_image_planar_rgb
(
ctx
,
dst
,
src
,
0
,
0
,
0
,
x
,
y
,
0
);
OverlayContext
*
s
=
ctx
->
priv
;
ThreadData
*
td
=
arg
;
blend_slice_planar_rgb
(
ctx
,
td
->
dst
,
td
->
src
,
0
,
0
,
0
,
s
->
x
,
s
->
y
,
0
,
jobnr
,
nb_jobs
);
return
0
;
}
static
void
blend_image_gbrap_pm
(
AVFilterContext
*
ctx
,
AVFrame
*
dst
,
const
AVFrame
*
src
,
int
x
,
int
y
)
static
int
blend_slice_gbrap_pm
(
AVFilterContext
*
ctx
,
void
*
arg
,
int
jobnr
,
int
nb_jobs
)
{
blend_image_planar_rgb
(
ctx
,
dst
,
src
,
0
,
0
,
1
,
x
,
y
,
0
);
OverlayContext
*
s
=
ctx
->
priv
;
ThreadData
*
td
=
arg
;
blend_slice_planar_rgb
(
ctx
,
td
->
dst
,
td
->
src
,
0
,
0
,
1
,
s
->
x
,
s
->
y
,
0
,
jobnr
,
nb_jobs
);
return
0
;
}
static
void
blend_image_rgb
(
AVFilterContext
*
ctx
,
AVFrame
*
dst
,
const
AVFrame
*
src
,
int
x
,
int
y
)
static
int
blend_slice_rgb
(
AVFilterContext
*
ctx
,
void
*
arg
,
int
jobnr
,
int
nb_jobs
)
{
blend_image_packed_rgb
(
ctx
,
dst
,
src
,
0
,
x
,
y
,
1
);
OverlayContext
*
s
=
ctx
->
priv
;
ThreadData
*
td
=
arg
;
blend_slice_packed_rgb
(
ctx
,
td
->
dst
,
td
->
src
,
0
,
s
->
x
,
s
->
y
,
1
,
jobnr
,
nb_jobs
);
return
0
;
}
static
void
blend_image_rgba
(
AVFilterContext
*
ctx
,
AVFrame
*
dst
,
const
AVFrame
*
src
,
int
x
,
int
y
)
static
int
blend_slice_rgba
(
AVFilterContext
*
ctx
,
void
*
arg
,
int
jobnr
,
int
nb_jobs
)
{
blend_image_packed_rgb
(
ctx
,
dst
,
src
,
1
,
x
,
y
,
1
);
OverlayContext
*
s
=
ctx
->
priv
;
ThreadData
*
td
=
arg
;
blend_slice_packed_rgb
(
ctx
,
td
->
dst
,
td
->
src
,
1
,
s
->
x
,
s
->
y
,
1
,
jobnr
,
nb_jobs
);
return
0
;
}
static
void
blend_image_rgb_pm
(
AVFilterContext
*
ctx
,
AVFrame
*
dst
,
const
AVFrame
*
src
,
int
x
,
int
y
)
static
int
blend_slice_rgb_pm
(
AVFilterContext
*
ctx
,
void
*
arg
,
int
jobnr
,
int
nb_jobs
)
{
blend_image_packed_rgb
(
ctx
,
dst
,
src
,
0
,
x
,
y
,
0
);
OverlayContext
*
s
=
ctx
->
priv
;
ThreadData
*
td
=
arg
;
blend_slice_packed_rgb
(
ctx
,
td
->
dst
,
td
->
src
,
0
,
s
->
x
,
s
->
y
,
0
,
jobnr
,
nb_jobs
);
return
0
;
}
static
void
blend_image_rgba_pm
(
AVFilterContext
*
ctx
,
AVFrame
*
dst
,
const
AVFrame
*
src
,
int
x
,
int
y
)
static
int
blend_slice_rgba_pm
(
AVFilterContext
*
ctx
,
void
*
arg
,
int
jobnr
,
int
nb_jobs
)
{
blend_image_packed_rgb
(
ctx
,
dst
,
src
,
1
,
x
,
y
,
0
);
OverlayContext
*
s
=
ctx
->
priv
;
ThreadData
*
td
=
arg
;
blend_slice_packed_rgb
(
ctx
,
td
->
dst
,
td
->
src
,
1
,
s
->
x
,
s
->
y
,
0
,
jobnr
,
nb_jobs
);
return
0
;
}
static
int
config_input_main
(
AVFilterLink
*
inlink
)
...
...
@@ -781,39 +874,39 @@ static int config_input_main(AVFilterLink *inlink)
s
->
main_has_alpha
=
ff_fmt_is_in
(
inlink
->
format
,
alpha_pix_fmts
);
switch
(
s
->
format
)
{
case
OVERLAY_FORMAT_YUV420
:
s
->
blend_
image
=
s
->
main_has_alpha
?
blend_image_yuva420
:
blend_imag
e_yuv420
;
s
->
blend_
slice
=
s
->
main_has_alpha
?
blend_slice_yuva420
:
blend_slic
e_yuv420
;
break
;
case
OVERLAY_FORMAT_YUV422
:
s
->
blend_
image
=
s
->
main_has_alpha
?
blend_image_yuva422
:
blend_imag
e_yuv422
;
s
->
blend_
slice
=
s
->
main_has_alpha
?
blend_slice_yuva422
:
blend_slic
e_yuv422
;
break
;
case
OVERLAY_FORMAT_YUV444
:
s
->
blend_
image
=
s
->
main_has_alpha
?
blend_image_yuva444
:
blend_imag
e_yuv444
;
s
->
blend_
slice
=
s
->
main_has_alpha
?
blend_slice_yuva444
:
blend_slic
e_yuv444
;
break
;
case
OVERLAY_FORMAT_RGB
:
s
->
blend_
image
=
s
->
main_has_alpha
?
blend_image_rgba
:
blend_imag
e_rgb
;
s
->
blend_
slice
=
s
->
main_has_alpha
?
blend_slice_rgba
:
blend_slic
e_rgb
;
break
;
case
OVERLAY_FORMAT_GBRP
:
s
->
blend_
image
=
s
->
main_has_alpha
?
blend_image_gbrap
:
blend_imag
e_gbrp
;
s
->
blend_
slice
=
s
->
main_has_alpha
?
blend_slice_gbrap
:
blend_slic
e_gbrp
;
break
;
case
OVERLAY_FORMAT_AUTO
:
switch
(
inlink
->
format
)
{
case
AV_PIX_FMT_YUVA420P
:
s
->
blend_
image
=
blend_imag
e_yuva420
;
s
->
blend_
slice
=
blend_slic
e_yuva420
;
break
;
case
AV_PIX_FMT_YUVA422P
:
s
->
blend_
image
=
blend_imag
e_yuva422
;
s
->
blend_
slice
=
blend_slic
e_yuva422
;
break
;
case
AV_PIX_FMT_YUVA444P
:
s
->
blend_
image
=
blend_imag
e_yuva444
;
s
->
blend_
slice
=
blend_slic
e_yuva444
;
break
;
case
AV_PIX_FMT_ARGB
:
case
AV_PIX_FMT_RGBA
:
case
AV_PIX_FMT_BGRA
:
case
AV_PIX_FMT_ABGR
:
s
->
blend_
image
=
blend_imag
e_rgba
;
s
->
blend_
slice
=
blend_slic
e_rgba
;
break
;
case
AV_PIX_FMT_GBRAP
:
s
->
blend_
image
=
blend_imag
e_gbrap
;
s
->
blend_
slice
=
blend_slic
e_gbrap
;
break
;
default:
av_assert0
(
0
);
...
...
@@ -827,39 +920,39 @@ static int config_input_main(AVFilterLink *inlink)
switch
(
s
->
format
)
{
case
OVERLAY_FORMAT_YUV420
:
s
->
blend_
image
=
s
->
main_has_alpha
?
blend_image_yuva420_pm
:
blend_imag
e_yuv420_pm
;
s
->
blend_
slice
=
s
->
main_has_alpha
?
blend_slice_yuva420_pm
:
blend_slic
e_yuv420_pm
;
break
;
case
OVERLAY_FORMAT_YUV422
:
s
->
blend_
image
=
s
->
main_has_alpha
?
blend_image_yuva422_pm
:
blend_imag
e_yuv422_pm
;
s
->
blend_
slice
=
s
->
main_has_alpha
?
blend_slice_yuva422_pm
:
blend_slic
e_yuv422_pm
;
break
;
case
OVERLAY_FORMAT_YUV444
:
s
->
blend_
image
=
s
->
main_has_alpha
?
blend_image_yuva444_pm
:
blend_imag
e_yuv444_pm
;
s
->
blend_
slice
=
s
->
main_has_alpha
?
blend_slice_yuva444_pm
:
blend_slic
e_yuv444_pm
;
break
;
case
OVERLAY_FORMAT_RGB
:
s
->
blend_
image
=
s
->
main_has_alpha
?
blend_image_rgba_pm
:
blend_imag
e_rgb_pm
;
s
->
blend_
slice
=
s
->
main_has_alpha
?
blend_slice_rgba_pm
:
blend_slic
e_rgb_pm
;
break
;
case
OVERLAY_FORMAT_GBRP
:
s
->
blend_
image
=
s
->
main_has_alpha
?
blend_image_gbrap_pm
:
blend_imag
e_gbrp_pm
;
s
->
blend_
slice
=
s
->
main_has_alpha
?
blend_slice_gbrap_pm
:
blend_slic
e_gbrp_pm
;
break
;
case
OVERLAY_FORMAT_AUTO
:
switch
(
inlink
->
format
)
{
case
AV_PIX_FMT_YUVA420P
:
s
->
blend_
image
=
blend_imag
e_yuva420_pm
;
s
->
blend_
slice
=
blend_slic
e_yuva420_pm
;
break
;
case
AV_PIX_FMT_YUVA422P
:
s
->
blend_
image
=
blend_imag
e_yuva422_pm
;
s
->
blend_
slice
=
blend_slic
e_yuva422_pm
;
break
;
case
AV_PIX_FMT_YUVA444P
:
s
->
blend_
image
=
blend_imag
e_yuva444_pm
;
s
->
blend_
slice
=
blend_slic
e_yuva444_pm
;
break
;
case
AV_PIX_FMT_ARGB
:
case
AV_PIX_FMT_RGBA
:
case
AV_PIX_FMT_BGRA
:
case
AV_PIX_FMT_ABGR
:
s
->
blend_
image
=
blend_imag
e_rgba_pm
;
s
->
blend_
slice
=
blend_slic
e_rgba_pm
;
break
;
case
AV_PIX_FMT_GBRAP
:
s
->
blend_
image
=
blend_imag
e_gbrap_pm
;
s
->
blend_
slice
=
blend_slic
e_gbrap_pm
;
break
;
default:
av_assert0
(
0
);
...
...
@@ -905,8 +998,14 @@ static int do_blend(FFFrameSync *fs)
}
if
(
s
->
x
<
mainpic
->
width
&&
s
->
x
+
second
->
width
>=
0
||
s
->
y
<
mainpic
->
height
&&
s
->
y
+
second
->
height
>=
0
)
s
->
blend_image
(
ctx
,
mainpic
,
second
,
s
->
x
,
s
->
y
);
s
->
y
<
mainpic
->
height
&&
s
->
y
+
second
->
height
>=
0
)
{
ThreadData
td
;
td
.
dst
=
mainpic
;
td
.
src
=
second
;
ctx
->
internal
->
execute
(
ctx
,
s
->
blend_slice
,
&
td
,
NULL
,
FFMIN
(
FFMIN
(
mainpic
->
height
-
s
->
y
,
second
->
height
),
ff_filter_get_nb_threads
(
ctx
)));
}
return
ff_filter_frame
(
ctx
->
outputs
[
0
],
mainpic
);
}
...
...
@@ -992,5 +1091,6 @@ AVFilter ff_vf_overlay = {
.
process_command
=
process_command
,
.
inputs
=
avfilter_vf_overlay_inputs
,
.
outputs
=
avfilter_vf_overlay_outputs
,
.
flags
=
AVFILTER_FLAG_SUPPORT_TIMELINE_INTERNAL
,
.
flags
=
AVFILTER_FLAG_SUPPORT_TIMELINE_INTERNAL
|
AVFILTER_FLAG_SLICE_THREADS
,
};
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