mirror of
https://github.com/mpv-player/mpv
synced 2024-12-29 10:32:15 +00:00
a0caadd512
Simply clamp off the U/V components in the colormatrix, instead of doing something special in the shader. Also, since YA8/YA16 gave a plane_bits value of 16/32, and a colormatrix calculation overflowed with 32, add a component_bits field to the image format descriptor, which for YA8/YA16 returns 8/16 (the wrong value had no bad consequences otherwise).
256 lines
8.4 KiB
C
256 lines
8.4 KiB
C
/*
|
|
* This file is part of MPlayer.
|
|
*
|
|
* MPlayer is free software; you can redistribute it and/or modify
|
|
* it 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.
|
|
*
|
|
* MPlayer is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License along
|
|
* with MPlayer; if not, write to the Free Software Foundation, Inc.,
|
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
*
|
|
* You can alternatively redistribute this file and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
*/
|
|
|
|
#ifndef MPLAYER_CSPUTILS_H
|
|
#define MPLAYER_CSPUTILS_H
|
|
|
|
#include <stdbool.h>
|
|
#include <stdint.h>
|
|
|
|
/* NOTE: the csp and levels AUTO values are converted to specific ones
|
|
* above vf/vo level. At least vf_scale relies on all valid settings being
|
|
* nonzero at vf/vo level.
|
|
*/
|
|
|
|
enum mp_csp {
|
|
MP_CSP_AUTO,
|
|
MP_CSP_BT_601,
|
|
MP_CSP_BT_709,
|
|
MP_CSP_SMPTE_240M,
|
|
MP_CSP_BT_2020_NC,
|
|
MP_CSP_BT_2020_C,
|
|
MP_CSP_RGB,
|
|
MP_CSP_XYZ,
|
|
MP_CSP_YCGCO,
|
|
MP_CSP_COUNT
|
|
};
|
|
|
|
// Any enum mp_csp value is a valid index (except MP_CSP_COUNT)
|
|
extern const char *const mp_csp_names[MP_CSP_COUNT];
|
|
|
|
enum mp_csp_levels {
|
|
MP_CSP_LEVELS_AUTO,
|
|
MP_CSP_LEVELS_TV,
|
|
MP_CSP_LEVELS_PC,
|
|
MP_CSP_LEVELS_COUNT,
|
|
};
|
|
|
|
// Any enum mp_csp_levels value is a valid index (except MP_CSP_LEVELS_COUNT)
|
|
extern const char *const mp_csp_levels_names[MP_CSP_LEVELS_COUNT];
|
|
|
|
enum mp_csp_prim {
|
|
MP_CSP_PRIM_AUTO,
|
|
MP_CSP_PRIM_BT_601_525,
|
|
MP_CSP_PRIM_BT_601_625,
|
|
MP_CSP_PRIM_BT_709,
|
|
MP_CSP_PRIM_BT_2020,
|
|
MP_CSP_PRIM_COUNT
|
|
};
|
|
|
|
enum mp_csp_trc {
|
|
MP_CSP_TRC_NONE,
|
|
MP_CSP_TRC_BT_1886,
|
|
MP_CSP_TRC_SRGB
|
|
};
|
|
|
|
// Any enum mp_csp_prim value is a valid index (except MP_CSP_PRIM_COUNT)
|
|
extern const char *const mp_csp_prim_names[MP_CSP_PRIM_COUNT];
|
|
|
|
// These constants are based on the ICC specification (Table 23) and match
|
|
// up with the API of LittleCMS, which treats them as integers.
|
|
enum mp_render_intent {
|
|
MP_INTENT_PERCEPTUAL = 0,
|
|
MP_INTENT_RELATIVE_COLORIMETRIC = 1,
|
|
MP_INTENT_SATURATION = 2,
|
|
MP_INTENT_ABSOLUTE_COLORIMETRIC = 3
|
|
};
|
|
|
|
// The numeric values (except -1) match the Matroska StereoMode element value.
|
|
enum mp_stereo3d_mode {
|
|
MP_STEREO3D_INVALID = -1,
|
|
/* only modes explicitly referenced in the code are listed */
|
|
MP_STEREO3D_MONO = 0,
|
|
MP_STEREO3D_SBS2L = 1,
|
|
MP_STEREO3D_AB2R = 2,
|
|
MP_STEREO3D_AB2L = 3,
|
|
MP_STEREO3D_SBS2R = 11,
|
|
/* no explicit enum entries for most valid values */
|
|
MP_STEREO3D_COUNT = 13, // 12 is last valid mode
|
|
};
|
|
|
|
extern const char *const mp_stereo3d_names[MP_STEREO3D_COUNT];
|
|
|
|
#define MP_STEREO3D_NAME(x) \
|
|
((x) >= 0 && (x) < MP_STEREO3D_COUNT ? (char *)mp_stereo3d_names[(x)] : NULL)
|
|
#define MP_STEREO3D_NAME_DEF(x, def) \
|
|
(MP_STEREO3D_NAME(x) ? MP_STEREO3D_NAME(x) : (def))
|
|
|
|
struct mp_csp_params {
|
|
enum mp_csp colorspace;
|
|
enum mp_csp_levels levels_in; // encoded video
|
|
enum mp_csp_levels levels_out; // output device
|
|
float brightness;
|
|
float contrast;
|
|
float hue;
|
|
float saturation;
|
|
float rgamma;
|
|
float ggamma;
|
|
float bgamma;
|
|
// discard U/V components
|
|
bool gray;
|
|
// texture_bits/input_bits is for rescaling fixed point input to range [0,1]
|
|
int texture_bits;
|
|
int input_bits;
|
|
// for scaling integer input and output (if 0, assume range [0,1])
|
|
int int_bits_in;
|
|
int int_bits_out;
|
|
};
|
|
|
|
#define MP_CSP_PARAMS_DEFAULTS { \
|
|
.colorspace = MP_CSP_BT_601, \
|
|
.levels_in = MP_CSP_LEVELS_TV, \
|
|
.levels_out = MP_CSP_LEVELS_PC, \
|
|
.brightness = 0, .contrast = 1, .hue = 0, .saturation = 1, \
|
|
.rgamma = 1, .ggamma = 1, .bgamma = 1, \
|
|
.texture_bits = 8, .input_bits = 8}
|
|
|
|
struct mp_image_params;
|
|
void mp_csp_set_image_params(struct mp_csp_params *params,
|
|
const struct mp_image_params *imgparams);
|
|
|
|
enum mp_chroma_location {
|
|
MP_CHROMA_AUTO,
|
|
MP_CHROMA_LEFT, // mpeg2/4, h264
|
|
MP_CHROMA_CENTER, // mpeg1, jpeg
|
|
MP_CHROMA_COUNT,
|
|
};
|
|
|
|
extern const char *const mp_chroma_names[MP_CHROMA_COUNT];
|
|
|
|
enum mp_csp_equalizer_param {
|
|
MP_CSP_EQ_BRIGHTNESS,
|
|
MP_CSP_EQ_CONTRAST,
|
|
MP_CSP_EQ_HUE,
|
|
MP_CSP_EQ_SATURATION,
|
|
MP_CSP_EQ_GAMMA,
|
|
MP_CSP_EQ_COUNT,
|
|
};
|
|
|
|
#define MP_CSP_EQ_CAPS_COLORMATRIX \
|
|
( (1 << MP_CSP_EQ_BRIGHTNESS) \
|
|
| (1 << MP_CSP_EQ_CONTRAST) \
|
|
| (1 << MP_CSP_EQ_HUE) \
|
|
| (1 << MP_CSP_EQ_SATURATION) )
|
|
|
|
#define MP_CSP_EQ_CAPS_GAMMA (1 << MP_CSP_EQ_GAMMA)
|
|
#define MP_CSP_EQ_CAPS_BRIGHTNESS (1 << MP_CSP_EQ_BRIGHTNESS)
|
|
|
|
extern const char *const mp_csp_equalizer_names[MP_CSP_EQ_COUNT];
|
|
|
|
// Default initialization with 0 is enough, except for the capabilities field
|
|
struct mp_csp_equalizer {
|
|
// Bit field of capabilities. For example (1 << MP_CSP_EQ_HUE) means hue
|
|
// support is available.
|
|
int capabilities;
|
|
// Value for each property is in the range [-100, 100].
|
|
// 0 is default, meaning neutral or no change.
|
|
int values[MP_CSP_EQ_COUNT];
|
|
};
|
|
|
|
struct mp_csp_col_xy {
|
|
float x, y;
|
|
};
|
|
|
|
struct mp_csp_primaries {
|
|
struct mp_csp_col_xy red, green, blue, white;
|
|
};
|
|
|
|
void mp_csp_copy_equalizer_values(struct mp_csp_params *params,
|
|
const struct mp_csp_equalizer *eq);
|
|
|
|
int mp_csp_equalizer_set(struct mp_csp_equalizer *eq, const char *property,
|
|
int value);
|
|
|
|
int mp_csp_equalizer_get(struct mp_csp_equalizer *eq, const char *property,
|
|
int *out_value);
|
|
|
|
enum mp_csp avcol_spc_to_mp_csp(int avcolorspace);
|
|
|
|
enum mp_csp_levels avcol_range_to_mp_csp_levels(int avrange);
|
|
|
|
enum mp_csp_prim avcol_pri_to_mp_csp_prim(int avpri);
|
|
|
|
int mp_csp_to_avcol_spc(enum mp_csp colorspace);
|
|
|
|
int mp_csp_levels_to_avcol_range(enum mp_csp_levels range);
|
|
|
|
int mp_csp_prim_to_avcol_pri(enum mp_csp_prim prim);
|
|
|
|
enum mp_csp mp_csp_guess_colorspace(int width, int height);
|
|
enum mp_csp_prim mp_csp_guess_primaries(int width, int height);
|
|
|
|
enum mp_chroma_location avchroma_location_to_mp(int avloc);
|
|
int mp_chroma_location_to_av(enum mp_chroma_location mploc);
|
|
|
|
void mp_get_chroma_location(enum mp_chroma_location loc, int *x, int *y);
|
|
|
|
void mp_gen_gamma_map(unsigned char *map, int size, float gamma);
|
|
|
|
struct mp_csp_primaries mp_get_csp_primaries(enum mp_csp_prim csp);
|
|
|
|
/* Color conversion matrix: RGB = m * YUV + c
|
|
* m is in row-major matrix, with m[row][col], e.g.:
|
|
* [ a11 a12 a13 ] float m[3][3] = { { a11, a12, a13 },
|
|
* [ a21 a22 a23 ] { a21, a22, a23 },
|
|
* [ a31 a32 a33 ] { a31, a32, a33 } };
|
|
* This is accessed as e.g.: m[2-1][1-1] = a21
|
|
* In particular, each row contains all the coefficients for one of R, G, B,
|
|
* while each column contains all the coefficients for one of Y, U, V:
|
|
* m[r,g,b][y,u,v] = ...
|
|
* The matrix could also be viewed as group of 3 vectors, e.g. the 1st column
|
|
* is the Y vector (1, 1, 1), the 2nd is the U vector, the 3rd the V vector.
|
|
* The matrix might also be used for other conversions and colorspaces.
|
|
*/
|
|
struct mp_cmat {
|
|
float m[3][3];
|
|
float c[3];
|
|
};
|
|
|
|
void mp_apply_chromatic_adaptation(struct mp_csp_col_xy src,
|
|
struct mp_csp_col_xy dest, float m[3][3]);
|
|
void mp_get_cms_matrix(struct mp_csp_primaries src, struct mp_csp_primaries dest,
|
|
enum mp_render_intent intent, float cms_matrix[3][3]);
|
|
void mp_get_rgb2xyz_matrix(struct mp_csp_primaries space, float m[3][3]);
|
|
|
|
void mp_get_xyz2rgb_coeffs(struct mp_csp_params *params, struct mp_csp_primaries prim,
|
|
enum mp_render_intent intent, struct mp_cmat *xyz2rgb);
|
|
void mp_get_yuv2rgb_coeffs(struct mp_csp_params *params, struct mp_cmat *yuv2rgb);
|
|
void mp_gen_yuv2rgb_map(struct mp_csp_params *params, uint8_t *map, int size);
|
|
|
|
void mp_mul_matrix3x3(float a[3][3], float b[3][3]);
|
|
void mp_invert_matrix3x3(float m[3][3]);
|
|
void mp_invert_yuv2rgb(struct mp_cmat *out, struct mp_cmat *in);
|
|
void mp_map_int_color(struct mp_cmat *matrix, int clip_bits, int c[3]);
|
|
|
|
#endif /* MPLAYER_CSPUTILS_H */
|