summaryrefslogtreecommitdiff
path: root/libavfilter/vf_colorlevels.c
diff options
context:
space:
mode:
authorPaul B Mahol <onemda@gmail.com>2022-03-06 13:57:34 +0100
committerPaul B Mahol <onemda@gmail.com>2022-03-06 14:00:26 +0100
commit10c2ef1ca41dbe7811f0588f4163c8cf7b8fda66 (patch)
treef05918442927e601c9e303266f62b198070dacf9 /libavfilter/vf_colorlevels.c
parent47c3b34506bac2effdde09e863fcfee0ea82ef56 (diff)
avfilter/vf_colorlevels: add planar rgb formats support
Diffstat (limited to 'libavfilter/vf_colorlevels.c')
-rw-r--r--libavfilter/vf_colorlevels.c232
1 files changed, 189 insertions, 43 deletions
diff --git a/libavfilter/vf_colorlevels.c b/libavfilter/vf_colorlevels.c
index c089740b0a..a8859795dd 100644
--- a/libavfilter/vf_colorlevels.c
+++ b/libavfilter/vf_colorlevels.c
@@ -44,6 +44,9 @@ typedef struct ColorLevelsContext {
int preserve_color;
int nb_comp;
+ int depth;
+ int max;
+ int planar;
int bpp;
int step;
uint8_t rgba_map[4];
@@ -85,8 +88,8 @@ static const AVOption colorlevels_options[] = {
AVFILTER_DEFINE_CLASS(colorlevels);
typedef struct ThreadData {
- const uint8_t *srcrow;
- uint8_t *dstrow;
+ const uint8_t *srcrow[4];
+ uint8_t *dstrow[4];
int dst_linesize;
int src_linesize;
@@ -98,8 +101,7 @@ typedef struct ThreadData {
int omin[4];
} ThreadData;
-#define DO_COMMON(type, clip, preserve) \
- ColorLevelsContext *s = ctx->priv; \
+#define DO_COMMON(type, clip, preserve, planar) \
const ThreadData *td = arg; \
const int linesize = s->linesize; \
const int step = s->step; \
@@ -108,12 +110,14 @@ typedef struct ThreadData {
const int slice_end = (process_h * (jobnr+1)) / nb_jobs; \
const int src_linesize = td->src_linesize / sizeof(type); \
const int dst_linesize = td->dst_linesize / sizeof(type); \
- const type *srcrow = (const type *)td->srcrow + src_linesize * slice_start; \
- type *dstrow = (type *)td->dstrow + dst_linesize * slice_start; \
- const uint8_t offset_r = s->rgba_map[R]; \
- const uint8_t offset_g = s->rgba_map[G]; \
- const uint8_t offset_b = s->rgba_map[B]; \
- const uint8_t offset_a = s->rgba_map[A]; \
+ const type *src_r = (const type *)(td->srcrow[R]) + src_linesize * slice_start; \
+ const type *src_g = (const type *)(td->srcrow[G]) + src_linesize * slice_start; \
+ const type *src_b = (const type *)(td->srcrow[B]) + src_linesize * slice_start; \
+ const type *src_a = (const type *)(td->srcrow[A]) + src_linesize * slice_start; \
+ type *dst_r = (type *)(td->dstrow[R]) + src_linesize * slice_start; \
+ type *dst_g = (type *)(td->dstrow[G]) + src_linesize * slice_start; \
+ type *dst_b = (type *)(td->dstrow[B]) + src_linesize * slice_start; \
+ type *dst_a = (type *)(td->dstrow[A]) + src_linesize * slice_start; \
const int imin_r = td->imin[R]; \
const int imin_g = td->imin[G]; \
const int imin_b = td->imin[B]; \
@@ -126,14 +130,6 @@ typedef struct ThreadData {
const float coeff_g = td->coeff[G]; \
const float coeff_b = td->coeff[B]; \
const float coeff_a = td->coeff[A]; \
- const type *src_r = srcrow + offset_r; \
- const type *src_g = srcrow + offset_g; \
- const type *src_b = srcrow + offset_b; \
- const type *src_a = srcrow + offset_a; \
- type *dst_r = dstrow + offset_r; \
- type *dst_g = dstrow + offset_g; \
- type *dst_b = dstrow + offset_b; \
- type *dst_a = dstrow + offset_a; \
\
for (int y = slice_start; y < slice_end; y++) { \
for (int x = 0; x < linesize; x += step) { \
@@ -142,7 +138,7 @@ typedef struct ThreadData {
ig = src_g[x]; \
ib = src_b[x]; \
if (preserve) { \
- float ratio, icolor, ocolor, max = (1<<(8*sizeof(type)))-1; \
+ float ratio, icolor, ocolor, max = s->max; \
\
or = (ir - imin_r) * coeff_r + omin_r; \
og = (ig - imin_g) * coeff_g + omin_g; \
@@ -158,18 +154,18 @@ typedef struct ThreadData {
ob *= ratio; \
} \
\
- dst_r[x] = clip(or); \
- dst_g[x] = clip(og); \
- dst_b[x] = clip(ob); \
+ dst_r[x] = clip(or, depth); \
+ dst_g[x] = clip(og, depth); \
+ dst_b[x] = clip(ob, depth); \
} else { \
- dst_r[x] = clip((ir - imin_r) * coeff_r + omin_r); \
- dst_g[x] = clip((ig - imin_g) * coeff_g + omin_g); \
- dst_b[x] = clip((ib - imin_b) * coeff_b + omin_b); \
+ dst_r[x] = clip((ir - imin_r) * coeff_r + omin_r, depth); \
+ dst_g[x] = clip((ig - imin_g) * coeff_g + omin_g, depth); \
+ dst_b[x] = clip((ib - imin_b) * coeff_b + omin_b, depth); \
} \
} \
\
for (int x = 0; x < linesize && s->nb_comp == 4; x += step) \
- dst_a[x] = clip((src_a[x] - imin_a) * coeff_a + omin_a); \
+ dst_a[x] = clip((src_a[x] - imin_a) * coeff_a + omin_a, depth); \
\
src_r += src_linesize; \
src_g += src_linesize; \
@@ -182,31 +178,126 @@ typedef struct ThreadData {
dst_a += dst_linesize; \
}
+#define CLIP8(x, depth) av_clip_uint8(x)
+#define CLIP16(x, depth) av_clip_uint16(x)
+
static int colorlevels_slice_8(AVFilterContext *ctx, void *arg, int jobnr, int nb_jobs)
{
- DO_COMMON(uint8_t, av_clip_uint8, 0)
-
+ ColorLevelsContext *s = ctx->priv;
+ DO_COMMON(uint8_t, CLIP8, 0, 0)
return 0;
}
static int colorlevels_slice_16(AVFilterContext *ctx, void *arg, int jobnr, int nb_jobs)
{
- DO_COMMON(uint16_t, av_clip_uint16, 0)
-
+ ColorLevelsContext *s = ctx->priv;\
+ DO_COMMON(uint16_t, CLIP16, 0, 0)
return 0;
}
static int colorlevels_preserve_slice_8(AVFilterContext *ctx, void *arg, int jobnr, int nb_jobs)
{
- DO_COMMON(uint8_t, av_clip_uint8, 1)
-
+ ColorLevelsContext *s = ctx->priv;
+ DO_COMMON(uint8_t, CLIP8, 1, 0)
return 0;
}
static int colorlevels_preserve_slice_16(AVFilterContext *ctx, void *arg, int jobnr, int nb_jobs)
{
- DO_COMMON(uint16_t, av_clip_uint16, 1)
+ ColorLevelsContext *s = ctx->priv;
+ DO_COMMON(uint16_t, CLIP16, 1, 0)
+ return 0;
+}
+
+static int colorlevels_slice_8_planar(AVFilterContext *ctx, void *arg, int jobnr, int nb_jobs)
+{
+ ColorLevelsContext *s = ctx->priv;
+ DO_COMMON(uint8_t, CLIP8, 0, 1)
+ return 0;
+}
+
+static int colorlevels_slice_9_planar(AVFilterContext *ctx, void *arg, int jobnr, int nb_jobs)
+{
+ ColorLevelsContext *s = ctx->priv;
+ const int depth = s->depth;
+ DO_COMMON(uint16_t, av_clip_uintp2, 0, 1)
+ return 0;
+}
+
+static int colorlevels_slice_10_planar(AVFilterContext *ctx, void *arg, int jobnr, int nb_jobs)
+{
+ ColorLevelsContext *s = ctx->priv;
+ const int depth = s->depth;
+ DO_COMMON(uint16_t, av_clip_uintp2, 0, 1)
+ return 0;
+}
+
+static int colorlevels_slice_12_planar(AVFilterContext *ctx, void *arg, int jobnr, int nb_jobs)
+{
+ ColorLevelsContext *s = ctx->priv;
+ const int depth = s->depth;
+ DO_COMMON(uint16_t, av_clip_uintp2, 0, 1)
+ return 0;
+}
+
+static int colorlevels_slice_14_planar(AVFilterContext *ctx, void *arg, int jobnr, int nb_jobs)
+{
+ ColorLevelsContext *s = ctx->priv;
+ const int depth = s->depth;
+ DO_COMMON(uint16_t, av_clip_uintp2, 0, 1)
+ return 0;
+}
+static int colorlevels_slice_16_planar(AVFilterContext *ctx, void *arg, int jobnr, int nb_jobs)
+{
+ ColorLevelsContext *s = ctx->priv;
+ DO_COMMON(uint16_t, CLIP16, 0, 1)
+ return 0;
+}
+
+static int colorlevels_preserve_slice_8_planar(AVFilterContext *ctx, void *arg, int jobnr, int nb_jobs)
+{
+ ColorLevelsContext *s = ctx->priv;
+ DO_COMMON(uint8_t, CLIP8, 1, 1)
+ return 0;
+}
+
+static int colorlevels_preserve_slice_9_planar(AVFilterContext *ctx, void *arg, int jobnr, int nb_jobs)
+{
+ ColorLevelsContext *s = ctx->priv;
+ const int depth = s->depth;
+ DO_COMMON(uint16_t, av_clip_uintp2, 1, 1)
+ return 0;
+}
+
+static int colorlevels_preserve_slice_10_planar(AVFilterContext *ctx, void *arg, int jobnr, int nb_jobs)
+{
+ ColorLevelsContext *s = ctx->priv;
+ const int depth = s->depth;
+ DO_COMMON(uint16_t, av_clip_uintp2, 1, 1)
+ return 0;
+}
+
+static int colorlevels_preserve_slice_12_planar(AVFilterContext *ctx, void *arg, int jobnr, int nb_jobs)
+{
+ ColorLevelsContext *s = ctx->priv;
+ const int depth = s->depth;
+ DO_COMMON(uint16_t, av_clip_uintp2, 1, 1)
+ return 0;
+}
+
+static int colorlevels_preserve_slice_14_planar(AVFilterContext *ctx, void *arg, int jobnr, int nb_jobs)
+{
+ ColorLevelsContext *s = ctx->priv;
+ const int depth = s->depth;
+ DO_COMMON(uint16_t, av_clip_uintp2, 1, 1)
+ return 0;
+}
+
+static int colorlevels_preserve_slice_16_planar(AVFilterContext *ctx, void *arg, int jobnr, int nb_jobs)
+{
+ ColorLevelsContext *s = ctx->priv;
+ DO_COMMON(uint16_t, CLIP16, 1, 1)
return 0;
}
@@ -217,16 +308,48 @@ static int config_input(AVFilterLink *inlink)
const AVPixFmtDescriptor *desc = av_pix_fmt_desc_get(inlink->format);
s->nb_comp = desc->nb_components;
- s->bpp = desc->comp[0].depth >> 3;
- s->step = av_get_padded_bits_per_pixel(desc) >> (3 + (s->bpp == 2));
+ s->planar = desc->flags & AV_PIX_FMT_FLAG_PLANAR;
+ s->depth = desc->comp[0].depth;
+ s->max = (1 << s->depth) - 1;
+ s->bpp = (desc->comp[0].depth + 7) >> 3;
+ s->step = s->planar ? 1 : av_get_padded_bits_per_pixel(desc) >> (3 + (s->bpp == 2));
s->linesize = inlink->w * s->step;
ff_fill_rgba_map(s->rgba_map, inlink->format);
- s->colorlevels_slice[0] = colorlevels_slice_8;
- s->colorlevels_slice[1] = colorlevels_preserve_slice_8;
- if (s->bpp == 2) {
- s->colorlevels_slice[0] = colorlevels_slice_16;
- s->colorlevels_slice[1] = colorlevels_preserve_slice_16;
+ if (!s->planar) {
+ s->colorlevels_slice[0] = colorlevels_slice_8;
+ s->colorlevels_slice[1] = colorlevels_preserve_slice_8;
+ if (s->bpp == 2) {
+ s->colorlevels_slice[0] = colorlevels_slice_16;
+ s->colorlevels_slice[1] = colorlevels_preserve_slice_16;
+ }
+ } else {
+ switch (s->depth) {
+ case 8:
+ s->colorlevels_slice[0] = colorlevels_slice_8_planar;
+ s->colorlevels_slice[1] = colorlevels_preserve_slice_8_planar;
+ break;
+ case 9:
+ s->colorlevels_slice[0] = colorlevels_slice_9_planar;
+ s->colorlevels_slice[1] = colorlevels_preserve_slice_9_planar;
+ break;
+ case 10:
+ s->colorlevels_slice[0] = colorlevels_slice_10_planar;
+ s->colorlevels_slice[1] = colorlevels_preserve_slice_10_planar;
+ break;
+ case 12:
+ s->colorlevels_slice[0] = colorlevels_slice_12_planar;
+ s->colorlevels_slice[1] = colorlevels_preserve_slice_12_planar;
+ break;
+ case 14:
+ s->colorlevels_slice[0] = colorlevels_slice_14_planar;
+ s->colorlevels_slice[1] = colorlevels_preserve_slice_14_planar;
+ break;
+ case 16:
+ s->colorlevels_slice[0] = colorlevels_slice_16_planar;
+ s->colorlevels_slice[1] = colorlevels_preserve_slice_16_planar;
+ break;
+ }
}
return 0;
@@ -255,8 +378,25 @@ static int filter_frame(AVFilterLink *inlink, AVFrame *in)
td.h = inlink->h;
td.dst_linesize = out->linesize[0];
td.src_linesize = in->linesize[0];
- td.srcrow = in->data[0];
- td.dstrow = out->data[0];
+ if (s->planar) {
+ td.srcrow[R] = in->data[2];
+ td.dstrow[R] = out->data[2];
+ td.srcrow[G] = in->data[0];
+ td.dstrow[G] = out->data[0];
+ td.srcrow[B] = in->data[1];
+ td.dstrow[B] = out->data[1];
+ td.srcrow[A] = in->data[3];
+ td.dstrow[A] = out->data[3];
+ } else {
+ td.srcrow[R] = in->data[0] + s->rgba_map[R] * s->bpp;
+ td.dstrow[R] = out->data[0] + s->rgba_map[R] * s->bpp;
+ td.srcrow[G] = in->data[0] + s->rgba_map[G] * s->bpp;
+ td.dstrow[G] = out->data[0] + s->rgba_map[G] * s->bpp;
+ td.srcrow[B] = in->data[0] + s->rgba_map[B] * s->bpp;
+ td.dstrow[B] = out->data[0] + s->rgba_map[B] * s->bpp;
+ td.srcrow[A] = in->data[0] + s->rgba_map[A] * s->bpp;
+ td.dstrow[A] = out->data[0] + s->rgba_map[A] * s->bpp;
+ }
switch (s->bpp) {
case 1:
@@ -378,7 +518,13 @@ const AVFilter ff_vf_colorlevels = {
AV_PIX_FMT_RGB24, AV_PIX_FMT_BGR24,
AV_PIX_FMT_RGB48, AV_PIX_FMT_BGR48,
AV_PIX_FMT_RGBA64, AV_PIX_FMT_BGRA64,
- AV_PIX_FMT_RGBA, AV_PIX_FMT_BGRA),
+ AV_PIX_FMT_RGBA, AV_PIX_FMT_BGRA,
+ AV_PIX_FMT_GBRP, AV_PIX_FMT_GBRAP,
+ AV_PIX_FMT_GBRP9,
+ AV_PIX_FMT_GBRP10, AV_PIX_FMT_GBRAP10,
+ AV_PIX_FMT_GBRP12, AV_PIX_FMT_GBRAP12,
+ AV_PIX_FMT_GBRP14,
+ AV_PIX_FMT_GBRP16, AV_PIX_FMT_GBRAP16),
.flags = AVFILTER_FLAG_SUPPORT_TIMELINE_GENERIC | AVFILTER_FLAG_SLICE_THREADS,
.process_command = ff_filter_process_command,
};