avfilter/vf_colormatrix: add bt.2020 colorspace
Signed-off-by: Thomas Mundt <loudmax@yahoo.de> Signed-off-by: Michael Niedermayer <michael@niedermayer.cc>
This commit is contained in:
parent
2e395bbccf
commit
a0a4a4b370
@ -5025,6 +5025,9 @@ SMPTE-240M
|
|||||||
|
|
||||||
@item fcc
|
@item fcc
|
||||||
FCC
|
FCC
|
||||||
|
|
||||||
|
@item bt2020
|
||||||
|
BT.2020
|
||||||
@end table
|
@end table
|
||||||
@end table
|
@end table
|
||||||
|
|
||||||
|
@ -40,11 +40,12 @@
|
|||||||
#define NS(n) ((n) < 0 ? (int)((n)*65536.0-0.5+DBL_EPSILON) : (int)((n)*65536.0+0.5))
|
#define NS(n) ((n) < 0 ? (int)((n)*65536.0-0.5+DBL_EPSILON) : (int)((n)*65536.0+0.5))
|
||||||
#define CB(n) av_clip_uint8(n)
|
#define CB(n) av_clip_uint8(n)
|
||||||
|
|
||||||
static const double yuv_coeff_luma[4][3] = {
|
static const double yuv_coeff_luma[5][3] = {
|
||||||
{ +0.7152, +0.0722, +0.2126 }, // Rec.709 (0)
|
{ +0.7152, +0.0722, +0.2126 }, // Rec.709 (0)
|
||||||
{ +0.5900, +0.1100, +0.3000 }, // FCC (1)
|
{ +0.5900, +0.1100, +0.3000 }, // FCC (1)
|
||||||
{ +0.5870, +0.1140, +0.2990 }, // Rec.601 (ITU-R BT.470-2/SMPTE 170M) (2)
|
{ +0.5870, +0.1140, +0.2990 }, // Rec.601 (ITU-R BT.470-2/SMPTE 170M) (2)
|
||||||
{ +0.7010, +0.0870, +0.2120 }, // SMPTE 240M (3)
|
{ +0.7010, +0.0870, +0.2120 }, // SMPTE 240M (3)
|
||||||
|
{ +0.6780, +0.0593, +0.2627 }, // Rec.2020 (4)
|
||||||
};
|
};
|
||||||
|
|
||||||
enum ColorMode {
|
enum ColorMode {
|
||||||
@ -53,12 +54,13 @@ enum ColorMode {
|
|||||||
COLOR_MODE_FCC,
|
COLOR_MODE_FCC,
|
||||||
COLOR_MODE_BT601,
|
COLOR_MODE_BT601,
|
||||||
COLOR_MODE_SMPTE240M,
|
COLOR_MODE_SMPTE240M,
|
||||||
|
COLOR_MODE_BT2020,
|
||||||
COLOR_MODE_COUNT
|
COLOR_MODE_COUNT
|
||||||
};
|
};
|
||||||
|
|
||||||
typedef struct {
|
typedef struct {
|
||||||
const AVClass *class;
|
const AVClass *class;
|
||||||
int yuv_convert[16][3][3];
|
int yuv_convert[25][3][3];
|
||||||
int interlaced;
|
int interlaced;
|
||||||
int source, dest; ///< ColorMode
|
int source, dest; ///< ColorMode
|
||||||
int mode;
|
int mode;
|
||||||
@ -89,6 +91,7 @@ static const AVOption colormatrix_options[] = {
|
|||||||
{ "bt470bg", "set BT.470 colorspace", 0, AV_OPT_TYPE_CONST, {.i64=COLOR_MODE_BT601}, .flags=FLAGS, .unit="color_mode" },
|
{ "bt470bg", "set BT.470 colorspace", 0, AV_OPT_TYPE_CONST, {.i64=COLOR_MODE_BT601}, .flags=FLAGS, .unit="color_mode" },
|
||||||
{ "smpte170m", "set SMTPE-170M colorspace", 0, AV_OPT_TYPE_CONST, {.i64=COLOR_MODE_BT601}, .flags=FLAGS, .unit="color_mode" },
|
{ "smpte170m", "set SMTPE-170M colorspace", 0, AV_OPT_TYPE_CONST, {.i64=COLOR_MODE_BT601}, .flags=FLAGS, .unit="color_mode" },
|
||||||
{ "smpte240m", "set SMPTE-240M colorspace", 0, AV_OPT_TYPE_CONST, {.i64=COLOR_MODE_SMPTE240M}, .flags=FLAGS, .unit="color_mode" },
|
{ "smpte240m", "set SMPTE-240M colorspace", 0, AV_OPT_TYPE_CONST, {.i64=COLOR_MODE_SMPTE240M}, .flags=FLAGS, .unit="color_mode" },
|
||||||
|
{ "bt2020", "set BT.2020 colorspace", 0, AV_OPT_TYPE_CONST, {.i64=COLOR_MODE_BT2020}, .flags=FLAGS, .unit="color_mode" },
|
||||||
{ NULL }
|
{ NULL }
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -140,13 +143,13 @@ static void solve_coefficients(double cm[3][3], double rgb[3][3], double yuv[3][
|
|||||||
static void calc_coefficients(AVFilterContext *ctx)
|
static void calc_coefficients(AVFilterContext *ctx)
|
||||||
{
|
{
|
||||||
ColorMatrixContext *color = ctx->priv;
|
ColorMatrixContext *color = ctx->priv;
|
||||||
double yuv_coeff[4][3][3];
|
double yuv_coeff[5][3][3];
|
||||||
double rgb_coeffd[4][3][3];
|
double rgb_coeffd[5][3][3];
|
||||||
double yuv_convertd[16][3][3];
|
double yuv_convertd[25][3][3];
|
||||||
double bscale, rscale;
|
double bscale, rscale;
|
||||||
int v = 0;
|
int v = 0;
|
||||||
int i, j, k;
|
int i, j, k;
|
||||||
for (i = 0; i < 4; i++) {
|
for (i = 0; i < 5; i++) {
|
||||||
yuv_coeff[i][0][0] = yuv_coeff_luma[i][0];
|
yuv_coeff[i][0][0] = yuv_coeff_luma[i][0];
|
||||||
yuv_coeff[i][0][1] = yuv_coeff_luma[i][1];
|
yuv_coeff[i][0][1] = yuv_coeff_luma[i][1];
|
||||||
yuv_coeff[i][0][2] = yuv_coeff_luma[i][2];
|
yuv_coeff[i][0][2] = yuv_coeff_luma[i][2];
|
||||||
@ -159,10 +162,10 @@ static void calc_coefficients(AVFilterContext *ctx)
|
|||||||
yuv_coeff[i][2][1] = rscale * yuv_coeff[i][0][1];
|
yuv_coeff[i][2][1] = rscale * yuv_coeff[i][0][1];
|
||||||
yuv_coeff[i][2][2] = 0.5;
|
yuv_coeff[i][2][2] = 0.5;
|
||||||
}
|
}
|
||||||
for (i = 0; i < 4; i++)
|
for (i = 0; i < 5; i++)
|
||||||
inverse3x3(rgb_coeffd[i], yuv_coeff[i]);
|
inverse3x3(rgb_coeffd[i], yuv_coeff[i]);
|
||||||
for (i = 0; i < 4; i++) {
|
for (i = 0; i < 5; i++) {
|
||||||
for (j = 0; j < 4; j++) {
|
for (j = 0; j < 5; j++) {
|
||||||
solve_coefficients(yuv_convertd[v], rgb_coeffd[i], yuv_coeff[j]);
|
solve_coefficients(yuv_convertd[v], rgb_coeffd[i], yuv_coeff[j]);
|
||||||
for (k = 0; k < 3; k++) {
|
for (k = 0; k < 3; k++) {
|
||||||
color->yuv_convert[v][k][0] = NS(yuv_convertd[v][k][0]);
|
color->yuv_convert[v][k][0] = NS(yuv_convertd[v][k][0]);
|
||||||
@ -178,7 +181,7 @@ static void calc_coefficients(AVFilterContext *ctx)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static const char * const color_modes[] = {"bt709", "fcc", "bt601", "smpte240m"};
|
static const char * const color_modes[] = {"bt709", "fcc", "bt601", "smpte240m", "bt2020"};
|
||||||
|
|
||||||
static av_cold int init(AVFilterContext *ctx)
|
static av_cold int init(AVFilterContext *ctx)
|
||||||
{
|
{
|
||||||
@ -441,20 +444,23 @@ static int filter_frame(AVFilterLink *link, AVFrame *in)
|
|||||||
case AVCOL_SPC_SMPTE240M : source = COLOR_MODE_SMPTE240M ; break;
|
case AVCOL_SPC_SMPTE240M : source = COLOR_MODE_SMPTE240M ; break;
|
||||||
case AVCOL_SPC_BT470BG : source = COLOR_MODE_BT601 ; break;
|
case AVCOL_SPC_BT470BG : source = COLOR_MODE_BT601 ; break;
|
||||||
case AVCOL_SPC_SMPTE170M : source = COLOR_MODE_BT601 ; break;
|
case AVCOL_SPC_SMPTE170M : source = COLOR_MODE_BT601 ; break;
|
||||||
|
case AVCOL_SPC_BT2020_NCL: source = COLOR_MODE_BT2020 ; break;
|
||||||
|
case AVCOL_SPC_BT2020_CL : source = COLOR_MODE_BT2020 ; break;
|
||||||
default :
|
default :
|
||||||
av_log(ctx, AV_LOG_ERROR, "Input frame does not specify a supported colorspace, and none has been specified as source either\n");
|
av_log(ctx, AV_LOG_ERROR, "Input frame does not specify a supported colorspace, and none has been specified as source either\n");
|
||||||
av_frame_free(&out);
|
av_frame_free(&out);
|
||||||
return AVERROR(EINVAL);
|
return AVERROR(EINVAL);
|
||||||
}
|
}
|
||||||
color->mode = source * 4 + color->dest;
|
color->mode = source * 5 + color->dest;
|
||||||
} else
|
} else
|
||||||
color->mode = color->source * 4 + color->dest;
|
color->mode = color->source * 5 + color->dest;
|
||||||
|
|
||||||
switch(color->dest) {
|
switch(color->dest) {
|
||||||
case COLOR_MODE_BT709 : av_frame_set_colorspace(out, AVCOL_SPC_BT709) ; break;
|
case COLOR_MODE_BT709 : av_frame_set_colorspace(out, AVCOL_SPC_BT709) ; break;
|
||||||
case COLOR_MODE_FCC : av_frame_set_colorspace(out, AVCOL_SPC_FCC) ; break;
|
case COLOR_MODE_FCC : av_frame_set_colorspace(out, AVCOL_SPC_FCC) ; break;
|
||||||
case COLOR_MODE_SMPTE240M: av_frame_set_colorspace(out, AVCOL_SPC_SMPTE240M) ; break;
|
case COLOR_MODE_SMPTE240M: av_frame_set_colorspace(out, AVCOL_SPC_SMPTE240M) ; break;
|
||||||
case COLOR_MODE_BT601 : av_frame_set_colorspace(out, AVCOL_SPC_BT470BG) ; break;
|
case COLOR_MODE_BT601 : av_frame_set_colorspace(out, AVCOL_SPC_BT470BG) ; break;
|
||||||
|
case COLOR_MODE_BT2020 : av_frame_set_colorspace(out, AVCOL_SPC_BT2020_NCL); break;
|
||||||
}
|
}
|
||||||
|
|
||||||
td.src = in;
|
td.src = in;
|
||||||
|
Loading…
x
Reference in New Issue
Block a user